]> mj.ucw.cz Git - home-hw.git/blobdiff - lib/modbus.c
Power daemon: A new daemon for relaying power meter data to MQTT
[home-hw.git] / lib / modbus.c
index c6336460d307d1a8fe814e77b3159409ddaeb113..cd1d34eed28a5abd29f3c24f97f412159c4d4c70 100644 (file)
 #endif
 #endif
 
+// Debugging
+// #define MODBUS_DEBUG
+
+#ifdef MODBUS_DEBUG
+#define DEBUG debug_printf
+#else
+#define DEBUG(xxx, ...) do { } while (0)
+#endif
 /*** State ***/
 
 enum mb_state {
@@ -101,6 +109,7 @@ static void rx_init(void)
        rx_bad = 0;
        usart_set_mode(MODBUS_USART, USART_MODE_RX);
        usart_enable_rx_interrupt(MODBUS_USART);
+       modbus_ready_hook();
 }
 
 static void rx_done(void)
@@ -127,6 +136,8 @@ static void tx_done(void)
 
 void modbus_init(void)
 {
+       DEBUG("MODBUS: Init\n");
+
        timer_set_prescaler(MODBUS_TIMER, CPU_CLOCK_MHZ-1);     // 1 tick = 1 μs
        timer_set_mode(MODBUS_TIMER, TIM_CR1_CKD_CK_INT, TIM_CR1_CMS_EDGE, TIM_CR1_DIR_DOWN);
        timer_update_on_overflow(MODBUS_TIMER);
@@ -159,6 +170,8 @@ void MODBUS_USART_ISR(void)
                        if (status & (USART_SR_FE | USART_SR_ORE | USART_SR_NE)) {
                                rx_bad = 1;
                        } else if (rx_size < MODBUS_RX_BUFSIZE) {
+                               if (!rx_size)
+                                       modbus_frame_start_hook();
                                rx_buf[rx_size++] = ch;
                        } else {
                                // Frame too long
@@ -212,9 +225,12 @@ void modbus_loop(void)
                return;
        }
 
+       DEBUG("MODBUS: < dest=%02x func=%02x len=%u\n", rx_buf[0], rx_buf[1], rx_size);
+
        if (rx_buf[0] == MODBUS_OUR_ADDRESS) {
                // Frame addressed to us: process and reply
                process_frame();
+               DEBUG("MODBUS: > status=%02x len=%u\n", tx_buf[1], tx_size);
                tx_init();
        } else if (rx_buf[0] == 0x00) {
                // Broadcast frame: process, but do not reply
@@ -305,11 +321,13 @@ static bool check_frame(void)
 {
        if (rx_bad) {
                // FIXME: Error counters?
+               DEBUG("MODBUS: RX bad\n");
                return false;
        }
        
        if (rx_size < 4) {
                // FIXME: Error counters?
+               DEBUG("MODBUS: RX undersize\n");
                return false;
        }
 
@@ -317,6 +335,7 @@ static bool check_frame(void)
        u16 rx_crc = (rx_buf[rx_size-2] << 8) | rx_buf[rx_size-1];
        if (crc != rx_crc) {
                // FIXME: Error counters?
+               DEBUG("MODBUS: Bad CRC\n");
                return false;
        }
 
@@ -459,6 +478,9 @@ static void func_write_single_coil(void)
                return report_error(ERR_ILLEGAL_DATA_VALUE);
 
        modbus_set_coil(addr, value);
+
+       write_u16(addr);
+       write_u16(value);
 }
 
 static void func_write_single_register(void)
@@ -473,6 +495,9 @@ static void func_write_single_register(void)
                return report_error(ERR_ILLEGAL_DATA_ADDRESS);
 
        modbus_set_holding_register(addr, value);
+
+       write_u16(addr);
+       write_u16(value);
 }
 
 static void func_write_multiple_coils(void)
@@ -493,6 +518,9 @@ static void func_write_multiple_coils(void)
 
        for (u16 i = 0; i < count; i++)
                modbus_set_coil(start + i, rx_frame[i/8] & (1U << (i%8)));
+
+       write_u16(start);
+       write_u16(count);
 }
 
 static void func_write_multiple_registers(void)
@@ -513,6 +541,9 @@ static void func_write_multiple_registers(void)
 
        for (u16 i = 0; i < count; i++)
                modbus_set_holding_register(start + i, read_u16());
+
+       write_u16(start);
+       write_u16(count);
 }
 
 static void func_mask_write_register(void)
@@ -530,6 +561,10 @@ static void func_mask_write_register(void)
        u16 reg = modbus_get_holding_register(addr);
        reg = (reg & and_mask) | (or_mask & ~and_mask);
        modbus_set_holding_register(addr, reg);
+
+       write_u16(addr);
+       write_u16(and_mask);
+       write_u16(or_mask);
 }
 
 static void func_read_write_multiple_registers(void)