static byte tx_buf[MODBUS_TX_BUFSIZE];
static u16 tx_size;
static u16 tx_pos;
+static byte pending_error;
static bool check_frame(void);
static void process_frame(void);
ERR_ILLEGAL_FUNCTION = 0x01,
ERR_ILLEGAL_DATA_ADDRESS = 0x02,
ERR_ILLEGAL_DATA_VALUE = 0x03,
+ ERR_SLAVE_DEVICE_FAILURE = 0x04,
};
static uint read_remains(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)
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)
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)
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)
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)
tx_buf[0] = MODBUS_OUR_ADDRESS;
tx_buf[1] = rx_buf[1];
tx_size = 2;
+ pending_error = 0;
switch (func) {
case FUNC_READ_COILS:
report_error(ERR_ILLEGAL_FUNCTION);
}
+ // Is there a deferred error pending?
+ if (pending_error)
+ report_error(pending_error);
+
// Finish reply frame
write_u16(crc16(tx_buf, tx_size));
}
+
+void modbus_slave_error(void)
+{
+ pending_error = ERR_SLAVE_DEVICE_FAILURE;
+}