2022-05-04 17:37:01 +00:00
|
|
|
local comms = require("scada-common.comms")
|
|
|
|
local types = require("scada-common.types")
|
|
|
|
|
|
|
|
local modbus = {}
|
2022-05-03 15:27:40 +00:00
|
|
|
|
|
|
|
local MODBUS_FCODE = types.MODBUS_FCODE
|
|
|
|
local MODBUS_EXCODE = types.MODBUS_EXCODE
|
2022-04-18 01:12:25 +00:00
|
|
|
|
2022-01-14 17:42:11 +00:00
|
|
|
-- new modbus comms handler object
|
2022-05-04 17:37:01 +00:00
|
|
|
modbus.new = function (rtu_dev, use_parallel_read)
|
2022-01-13 15:12:44 +00:00
|
|
|
local self = {
|
2022-05-03 15:27:40 +00:00
|
|
|
rtu = rtu_dev,
|
|
|
|
use_parallel = use_parallel_read
|
2022-01-13 15:12:44 +00:00
|
|
|
}
|
|
|
|
|
2022-01-14 17:42:11 +00:00
|
|
|
local _1_read_coils = function (c_addr_start, count)
|
2022-05-03 15:39:03 +00:00
|
|
|
local tasks = {}
|
2022-01-14 17:42:11 +00:00
|
|
|
local readings = {}
|
2022-04-18 01:12:25 +00:00
|
|
|
local access_fault = false
|
2022-01-14 17:42:11 +00:00
|
|
|
local _, coils, _, _ = self.rtu.io_count()
|
2022-05-03 15:27:40 +00:00
|
|
|
local return_ok = ((c_addr_start + count) <= coils) and (count > 0)
|
2022-01-14 17:42:11 +00:00
|
|
|
|
|
|
|
if return_ok then
|
2022-05-03 15:27:40 +00:00
|
|
|
for i = 1, count do
|
|
|
|
local addr = c_addr_start + i - 1
|
2022-05-03 15:39:03 +00:00
|
|
|
|
|
|
|
if self.use_parallel then
|
|
|
|
table.insert(tasks, function ()
|
|
|
|
local reading, fault = self.rtu.read_coil(addr)
|
|
|
|
if fault then access_fault = true else readings[i] = reading end
|
|
|
|
end)
|
|
|
|
else
|
|
|
|
readings[i], access_fault = self.rtu.read_coil(addr)
|
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- run parallel tasks if configured
|
|
|
|
if self.use_parallel then
|
|
|
|
parallel.waitForAll(table.unpack(tasks))
|
2022-04-18 01:12:25 +00:00
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
end
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
2022-04-18 01:12:25 +00:00
|
|
|
else
|
|
|
|
readings = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return return_ok, readings
|
|
|
|
end
|
|
|
|
|
|
|
|
local _2_read_discrete_inputs = function (di_addr_start, count)
|
2022-05-03 15:39:03 +00:00
|
|
|
local tasks = {}
|
2022-01-14 17:42:11 +00:00
|
|
|
local readings = {}
|
2022-04-18 01:12:25 +00:00
|
|
|
local access_fault = false
|
2022-01-14 17:42:11 +00:00
|
|
|
local discrete_inputs, _, _, _ = self.rtu.io_count()
|
2022-05-03 15:27:40 +00:00
|
|
|
local return_ok = ((di_addr_start + count) <= discrete_inputs) and (count > 0)
|
2022-01-14 17:42:11 +00:00
|
|
|
|
|
|
|
if return_ok then
|
2022-05-03 15:27:40 +00:00
|
|
|
for i = 1, count do
|
|
|
|
local addr = di_addr_start + i - 1
|
2022-05-03 15:39:03 +00:00
|
|
|
|
|
|
|
if self.use_parallel then
|
|
|
|
table.insert(tasks, function ()
|
|
|
|
local reading, fault = self.rtu.read_di(addr)
|
|
|
|
if fault then access_fault = true else readings[i] = reading end
|
|
|
|
end)
|
|
|
|
else
|
|
|
|
readings[i], access_fault = self.rtu.read_di(addr)
|
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- run parallel tasks if configured
|
|
|
|
if self.use_parallel then
|
|
|
|
parallel.waitForAll(table.unpack(tasks))
|
2022-04-18 01:12:25 +00:00
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
end
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
2022-04-18 01:12:25 +00:00
|
|
|
else
|
|
|
|
readings = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return return_ok, readings
|
|
|
|
end
|
|
|
|
|
|
|
|
local _3_read_multiple_holding_registers = function (hr_addr_start, count)
|
2022-05-03 15:39:03 +00:00
|
|
|
local tasks = {}
|
2022-01-14 17:42:11 +00:00
|
|
|
local readings = {}
|
2022-04-18 01:12:25 +00:00
|
|
|
local access_fault = false
|
2022-01-14 17:42:11 +00:00
|
|
|
local _, _, _, hold_regs = self.rtu.io_count()
|
2022-05-03 15:27:40 +00:00
|
|
|
local return_ok = ((hr_addr_start + count) <= hold_regs) and (count > 0)
|
2022-01-14 17:42:11 +00:00
|
|
|
|
|
|
|
if return_ok then
|
2022-05-03 15:27:40 +00:00
|
|
|
for i = 1, count do
|
|
|
|
local addr = hr_addr_start + i - 1
|
2022-05-03 15:39:03 +00:00
|
|
|
|
|
|
|
if self.use_parallel then
|
|
|
|
table.insert(tasks, function ()
|
|
|
|
local reading, fault = self.rtu.read_holding_reg(addr)
|
|
|
|
if fault then access_fault = true else readings[i] = reading end
|
|
|
|
end)
|
|
|
|
else
|
|
|
|
readings[i], access_fault = self.rtu.read_holding_reg(addr)
|
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- run parallel tasks if configured
|
|
|
|
if self.use_parallel then
|
|
|
|
parallel.waitForAll(table.unpack(tasks))
|
2022-04-18 01:12:25 +00:00
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
end
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
2022-04-18 01:12:25 +00:00
|
|
|
else
|
|
|
|
readings = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return return_ok, readings
|
|
|
|
end
|
|
|
|
|
|
|
|
local _4_read_input_registers = function (ir_addr_start, count)
|
2022-05-03 15:39:03 +00:00
|
|
|
local tasks = {}
|
2022-01-14 17:42:11 +00:00
|
|
|
local readings = {}
|
2022-04-18 01:12:25 +00:00
|
|
|
local access_fault = false
|
2022-01-14 17:42:11 +00:00
|
|
|
local _, _, input_regs, _ = self.rtu.io_count()
|
2022-05-03 15:27:40 +00:00
|
|
|
local return_ok = ((ir_addr_start + count) <= input_regs) and (count > 0)
|
2022-01-14 17:42:11 +00:00
|
|
|
|
|
|
|
if return_ok then
|
2022-05-03 15:27:40 +00:00
|
|
|
for i = 1, count do
|
|
|
|
local addr = ir_addr_start + i - 1
|
2022-05-03 15:39:03 +00:00
|
|
|
|
|
|
|
if self.use_parallel then
|
|
|
|
table.insert(tasks, function ()
|
|
|
|
local reading, fault = self.rtu.read_input_reg(addr)
|
|
|
|
if fault then access_fault = true else readings[i] = reading end
|
|
|
|
end)
|
|
|
|
else
|
|
|
|
readings[i], access_fault = self.rtu.read_input_reg(addr)
|
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- run parallel tasks if configured
|
|
|
|
if self.use_parallel then
|
|
|
|
parallel.waitForAll(table.unpack(tasks))
|
2022-04-18 01:12:25 +00:00
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
end
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
2022-04-18 01:12:25 +00:00
|
|
|
else
|
|
|
|
readings = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return return_ok, readings
|
|
|
|
end
|
|
|
|
|
|
|
|
local _5_write_single_coil = function (c_addr, value)
|
2022-04-18 01:12:25 +00:00
|
|
|
local response = nil
|
2022-01-14 17:42:11 +00:00
|
|
|
local _, coils, _, _ = self.rtu.io_count()
|
|
|
|
local return_ok = c_addr <= coils
|
2022-04-18 01:12:25 +00:00
|
|
|
|
2022-01-14 17:42:11 +00:00
|
|
|
if return_ok then
|
2022-04-18 01:12:25 +00:00
|
|
|
local access_fault = self.rtu.write_coil(c_addr, value)
|
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
end
|
|
|
|
else
|
|
|
|
response = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
2022-04-18 01:12:25 +00:00
|
|
|
return return_ok, response
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
local _6_write_single_holding_register = function (hr_addr, value)
|
2022-04-18 01:12:25 +00:00
|
|
|
local response = nil
|
2022-01-14 17:42:11 +00:00
|
|
|
local _, _, _, hold_regs = self.rtu.io_count()
|
|
|
|
local return_ok = hr_addr <= hold_regs
|
|
|
|
|
|
|
|
if return_ok then
|
2022-04-18 01:12:25 +00:00
|
|
|
local access_fault = self.rtu.write_holding_reg(hr_addr, value)
|
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
end
|
2022-05-03 15:27:40 +00:00
|
|
|
else
|
|
|
|
response = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return return_ok
|
2022-01-13 15:12:44 +00:00
|
|
|
end
|
|
|
|
|
2022-01-14 17:42:11 +00:00
|
|
|
local _15_write_multiple_coils = function (c_addr_start, values)
|
2022-04-18 01:12:25 +00:00
|
|
|
local response = nil
|
2022-01-14 17:42:11 +00:00
|
|
|
local _, coils, _, _ = self.rtu.io_count()
|
|
|
|
local count = #values
|
2022-05-03 15:27:40 +00:00
|
|
|
local return_ok = ((c_addr_start + count) <= coils) and (count > 0)
|
2022-01-14 17:42:11 +00:00
|
|
|
|
|
|
|
if return_ok then
|
2022-05-03 15:27:40 +00:00
|
|
|
for i = 1, count do
|
|
|
|
local addr = c_addr_start + i - 1
|
|
|
|
local access_fault = self.rtu.write_coil(addr, values[i])
|
2022-04-18 01:12:25 +00:00
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
break
|
|
|
|
end
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
2022-05-03 15:27:40 +00:00
|
|
|
else
|
|
|
|
response = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
2022-04-18 01:12:25 +00:00
|
|
|
return return_ok, response
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
local _16_write_multiple_holding_registers = function (hr_addr_start, values)
|
2022-04-18 01:12:25 +00:00
|
|
|
local response = nil
|
2022-01-14 17:42:11 +00:00
|
|
|
local _, _, _, hold_regs = self.rtu.io_count()
|
|
|
|
local count = #values
|
2022-05-03 15:27:40 +00:00
|
|
|
local return_ok = ((hr_addr_start + count) <= hold_regs) and (count > 0)
|
2022-01-14 17:42:11 +00:00
|
|
|
|
|
|
|
if return_ok then
|
2022-05-03 15:27:40 +00:00
|
|
|
for i = 1, count do
|
|
|
|
local addr = hr_addr_start + i - 1
|
|
|
|
local access_fault = self.rtu.write_coil(addr, values[i])
|
2022-04-18 01:12:25 +00:00
|
|
|
|
|
|
|
if access_fault then
|
|
|
|
return_ok = false
|
|
|
|
readings = MODBUS_EXCODE.SERVER_DEVICE_FAIL
|
|
|
|
break
|
|
|
|
end
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
2022-05-03 15:27:40 +00:00
|
|
|
else
|
|
|
|
response = MODBUS_EXCODE.ILLEGAL_DATA_ADDR
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
2022-04-18 01:12:25 +00:00
|
|
|
return return_ok, response
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
2022-04-29 17:19:01 +00:00
|
|
|
-- validate a request without actually executing it
|
|
|
|
local check_request = function (packet)
|
|
|
|
local return_code = true
|
|
|
|
local response = { MODBUS_EXCODE.ACKNOWLEDGE }
|
|
|
|
|
|
|
|
if #packet.data == 2 then
|
|
|
|
-- handle by function code
|
|
|
|
if packet.func_code == MODBUS_FCODE.READ_COILS then
|
|
|
|
elseif packet.func_code == MODBUS_FCODE.READ_DISCRETE_INPUTS then
|
|
|
|
elseif packet.func_code == MODBUS_FCODE.READ_MUL_HOLD_REGS then
|
|
|
|
elseif packet.func_code == MODBUS_FCODE.READ_INPUT_REGISTERS then
|
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_SINGLE_COIL then
|
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_SINGLE_HOLD_REG then
|
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_MUL_COILS then
|
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_MUL_HOLD_REGS then
|
|
|
|
else
|
|
|
|
-- unknown function
|
|
|
|
return_code = false
|
|
|
|
response = { MODBUS_EXCODE.ILLEGAL_FUNCTION }
|
|
|
|
end
|
|
|
|
else
|
|
|
|
-- invalid length
|
|
|
|
return_code = false
|
|
|
|
response = { MODBUS_EXCODE.NEG_ACKNOWLEDGE }
|
|
|
|
end
|
|
|
|
|
|
|
|
-- default is to echo back
|
|
|
|
local func_code = packet.func_code
|
|
|
|
if not return_code then
|
|
|
|
-- echo back with error flag
|
|
|
|
func_code = bit.bor(packet.func_code, MODBUS_FCODE.ERROR_FLAG)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- create reply
|
|
|
|
local reply = comms.modbus_packet()
|
|
|
|
reply.make(packet.txn_id, packet.unit_id, func_code, response)
|
|
|
|
|
|
|
|
return return_code, reply
|
|
|
|
end
|
|
|
|
|
2022-04-23 00:21:28 +00:00
|
|
|
-- handle a MODBUS TCP packet and generate a reply
|
2022-01-14 17:42:11 +00:00
|
|
|
local handle_packet = function (packet)
|
|
|
|
local return_code = true
|
2022-04-18 01:12:25 +00:00
|
|
|
local response = nil
|
2022-01-14 17:42:11 +00:00
|
|
|
|
|
|
|
if #packet.data == 2 then
|
|
|
|
-- handle by function code
|
|
|
|
if packet.func_code == MODBUS_FCODE.READ_COILS then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _1_read_coils(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
elseif packet.func_code == MODBUS_FCODE.READ_DISCRETE_INPUTS then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _2_read_discrete_inputs(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
elseif packet.func_code == MODBUS_FCODE.READ_MUL_HOLD_REGS then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _3_read_multiple_holding_registers(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
elseif packet.func_code == MODBUS_FCODE.READ_INPUT_REGISTERS then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _4_read_input_registers(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_SINGLE_COIL then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _5_write_single_coil(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_SINGLE_HOLD_REG then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _6_write_single_holding_register(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_MUL_COILS then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _15_write_multiple_coils(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
elseif packet.func_code == MODBUS_FCODE.WRITE_MUL_HOLD_REGS then
|
2022-04-18 01:12:25 +00:00
|
|
|
return_code, response = _16_write_multiple_holding_registers(packet.data[1], packet.data[2])
|
2022-01-14 17:42:11 +00:00
|
|
|
else
|
|
|
|
-- unknown function
|
|
|
|
return_code = false
|
2022-04-18 01:12:25 +00:00
|
|
|
response = MODBUS_EXCODE.ILLEGAL_FUNCTION
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
else
|
|
|
|
-- invalid length
|
|
|
|
return_code = false
|
|
|
|
end
|
|
|
|
|
2022-04-23 00:21:28 +00:00
|
|
|
-- default is to echo back
|
|
|
|
local func_code = packet.func_code
|
|
|
|
if not return_code then
|
2022-01-14 17:42:11 +00:00
|
|
|
-- echo back with error flag
|
2022-04-23 00:21:28 +00:00
|
|
|
func_code = bit.bor(packet.func_code, MODBUS_FCODE.ERROR_FLAG)
|
2022-04-18 01:12:25 +00:00
|
|
|
|
|
|
|
if type(response) == "nil" then
|
2022-04-23 00:21:28 +00:00
|
|
|
response = { }
|
2022-04-18 01:12:25 +00:00
|
|
|
elseif type(response) == "number" then
|
2022-04-23 00:21:28 +00:00
|
|
|
response = { response }
|
2022-04-18 01:12:25 +00:00
|
|
|
elseif type(response) == "table" then
|
2022-04-23 00:21:28 +00:00
|
|
|
response = response
|
2022-04-18 01:12:25 +00:00
|
|
|
end
|
2022-01-14 17:42:11 +00:00
|
|
|
end
|
|
|
|
|
2022-04-23 00:21:28 +00:00
|
|
|
-- create reply
|
|
|
|
local reply = comms.modbus_packet()
|
|
|
|
reply.make(packet.txn_id, packet.unit_id, func_code, response)
|
|
|
|
|
2022-04-18 01:12:25 +00:00
|
|
|
return return_code, reply
|
2022-01-13 15:12:44 +00:00
|
|
|
end
|
|
|
|
|
2022-04-29 17:19:01 +00:00
|
|
|
-- return a SERVER_DEVICE_BUSY error reply
|
|
|
|
local reply__srv_device_busy = function (packet)
|
|
|
|
-- reply back with error flag and exception code
|
|
|
|
local reply = comms.modbus_packet()
|
|
|
|
local fcode = bit.bor(packet.func_code, MODBUS_FCODE.ERROR_FLAG)
|
|
|
|
local data = { MODBUS_EXCODE.SERVER_DEVICE_BUSY }
|
|
|
|
reply.make(packet.txn_id, packet.unit_id, fcode, data)
|
|
|
|
return reply
|
|
|
|
end
|
|
|
|
|
2022-04-23 00:21:28 +00:00
|
|
|
-- return a NEG_ACKNOWLEDGE error reply
|
|
|
|
local reply__neg_ack = function (packet)
|
|
|
|
-- reply back with error flag and exception code
|
|
|
|
local reply = comms.modbus_packet()
|
|
|
|
local fcode = bit.bor(packet.func_code, MODBUS_FCODE.ERROR_FLAG)
|
|
|
|
local data = { MODBUS_EXCODE.NEG_ACKNOWLEDGE }
|
|
|
|
reply.make(packet.txn_id, packet.unit_id, fcode, data)
|
|
|
|
return reply
|
|
|
|
end
|
|
|
|
|
|
|
|
-- return a GATEWAY_PATH_UNAVAILABLE error reply
|
|
|
|
local reply__gw_unavailable = function (packet)
|
|
|
|
-- reply back with error flag and exception code
|
|
|
|
local reply = comms.modbus_packet()
|
|
|
|
local fcode = bit.bor(packet.func_code, MODBUS_FCODE.ERROR_FLAG)
|
|
|
|
local data = { MODBUS_EXCODE.GATEWAY_PATH_UNAVAILABLE }
|
|
|
|
reply.make(packet.txn_id, packet.unit_id, fcode, data)
|
|
|
|
return reply
|
|
|
|
end
|
|
|
|
|
2022-01-14 17:42:11 +00:00
|
|
|
return {
|
2022-04-29 17:19:01 +00:00
|
|
|
check_request = check_request,
|
2022-04-23 00:21:28 +00:00
|
|
|
handle_packet = handle_packet,
|
2022-04-29 17:19:01 +00:00
|
|
|
reply__srv_device_busy = reply__srv_device_busy,
|
2022-04-23 00:21:28 +00:00
|
|
|
reply__neg_ack = reply__neg_ack,
|
|
|
|
reply__gw_unavailable = reply__gw_unavailable
|
2022-01-14 17:42:11 +00:00
|
|
|
}
|
|
|
|
end
|
2022-05-04 17:37:01 +00:00
|
|
|
|
|
|
|
return modbus
|