2022-08-28 16:12:30 +00:00
|
|
|
local comms = require("scada-common.comms")
|
|
|
|
local log = require("scada-common.log")
|
2022-12-01 04:31:14 +00:00
|
|
|
local mqueue = require("scada-common.mqueue")
|
2022-08-28 16:12:30 +00:00
|
|
|
local types = require("scada-common.types")
|
2022-09-19 02:25:59 +00:00
|
|
|
local util = require("scada-common.util")
|
2022-05-17 21:16:04 +00:00
|
|
|
|
|
|
|
local txnctrl = require("supervisor.session.rtu.txnctrl")
|
|
|
|
|
|
|
|
local unit_session = {}
|
|
|
|
|
2023-02-21 16:05:57 +00:00
|
|
|
local PROTOCOL = comms.PROTOCOL
|
2022-05-17 21:16:04 +00:00
|
|
|
local MODBUS_FCODE = types.MODBUS_FCODE
|
|
|
|
local MODBUS_EXCODE = types.MODBUS_EXCODE
|
|
|
|
|
2022-11-11 19:59:53 +00:00
|
|
|
local RTU_US_CMDS = {
|
|
|
|
}
|
|
|
|
|
|
|
|
local RTU_US_DATA = {
|
2023-02-20 05:49:37 +00:00
|
|
|
BUILD_CHANGED = 1
|
2022-11-11 19:59:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unit_session.RTU_US_CMDS = RTU_US_CMDS
|
|
|
|
unit_session.RTU_US_DATA = RTU_US_DATA
|
|
|
|
|
2022-05-17 21:16:04 +00:00
|
|
|
-- create a new unit session runner
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2024-08-16 19:53:43 +00:00
|
|
|
---@param session_id integer RTU gateway session ID
|
2022-05-18 17:49:04 +00:00
|
|
|
---@param unit_id integer MODBUS unit ID
|
|
|
|
---@param advert rtu_advertisement RTU advertisement for this unit
|
|
|
|
---@param out_queue mqueue send queue
|
|
|
|
---@param log_tag string logging tag
|
|
|
|
---@param txn_tags table transaction log tags
|
2022-11-12 06:35:31 +00:00
|
|
|
function unit_session.new(session_id, unit_id, advert, out_queue, log_tag, txn_tags)
|
2022-05-17 21:16:04 +00:00
|
|
|
local self = {
|
2022-05-18 17:49:04 +00:00
|
|
|
device_index = advert.index,
|
2022-05-17 21:16:04 +00:00
|
|
|
reactor = advert.reactor,
|
|
|
|
transaction_controller = txnctrl.new(),
|
|
|
|
connected = true,
|
|
|
|
device_fail = false
|
|
|
|
}
|
|
|
|
|
|
|
|
---@class _unit_session
|
2022-12-01 04:31:14 +00:00
|
|
|
local protected = {
|
|
|
|
in_q = mqueue.new()
|
|
|
|
}
|
2022-05-17 21:16:04 +00:00
|
|
|
|
|
|
|
---@class unit_session
|
|
|
|
local public = {}
|
|
|
|
|
|
|
|
-- PROTECTED FUNCTIONS --
|
|
|
|
|
|
|
|
-- send a MODBUS message, creating a transaction in the process
|
|
|
|
---@param txn_type integer transaction type
|
|
|
|
---@param f_code MODBUS_FCODE function code
|
|
|
|
---@param register_param table register range or register and values
|
2022-12-01 04:31:14 +00:00
|
|
|
---@return integer txn_id transaction ID of this transaction
|
2022-05-31 19:36:17 +00:00
|
|
|
function protected.send_request(txn_type, f_code, register_param)
|
2022-05-17 21:16:04 +00:00
|
|
|
local m_pkt = comms.modbus_packet()
|
|
|
|
local txn_id = self.transaction_controller.create(txn_type)
|
|
|
|
|
2023-02-25 04:36:16 +00:00
|
|
|
m_pkt.make(txn_id, unit_id, f_code, register_param)
|
2022-05-17 21:16:04 +00:00
|
|
|
|
2023-02-25 04:36:16 +00:00
|
|
|
out_queue.push_packet(m_pkt)
|
2022-12-01 04:31:14 +00:00
|
|
|
|
|
|
|
return txn_id
|
2022-05-17 21:16:04 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- try to resolve a MODBUS transaction
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-17 21:16:04 +00:00
|
|
|
---@param m_pkt modbus_frame MODBUS packet
|
2022-12-01 04:31:14 +00:00
|
|
|
---@return integer|false txn_type, integer txn_id transaction type or false on error/busy, transaction ID
|
2022-05-31 19:36:17 +00:00
|
|
|
function protected.try_resolve(m_pkt)
|
2023-02-21 16:05:57 +00:00
|
|
|
if m_pkt.scada_frame.protocol() == PROTOCOL.MODBUS_TCP then
|
2023-02-25 04:36:16 +00:00
|
|
|
if m_pkt.unit_id == unit_id then
|
2022-05-17 21:16:04 +00:00
|
|
|
local txn_type = self.transaction_controller.resolve(m_pkt.txn_id)
|
2023-09-03 21:54:39 +00:00
|
|
|
local txn_tag = util.c(" (", txn_tags[txn_type], ")")
|
|
|
|
|
|
|
|
if txn_type == nil then
|
|
|
|
-- couldn't find this transaction
|
|
|
|
log.debug(log_tag .. "MODBUS: expired or spurious transaction reply (txn_id " .. m_pkt.txn_id .. ")")
|
|
|
|
return false, m_pkt.txn_id
|
|
|
|
end
|
2022-05-17 21:16:04 +00:00
|
|
|
|
|
|
|
if bit.band(m_pkt.func_code, MODBUS_FCODE.ERROR_FLAG) ~= 0 then
|
|
|
|
-- transaction incomplete or failed
|
|
|
|
local ex = m_pkt.data[1]
|
|
|
|
if ex == MODBUS_EXCODE.ILLEGAL_FUNCTION then
|
|
|
|
log.error(log_tag .. "MODBUS: illegal function" .. txn_tag)
|
|
|
|
elseif ex == MODBUS_EXCODE.ILLEGAL_DATA_ADDR then
|
|
|
|
log.error(log_tag .. "MODBUS: illegal data address" .. txn_tag)
|
|
|
|
elseif ex == MODBUS_EXCODE.SERVER_DEVICE_FAIL then
|
|
|
|
if self.device_fail then
|
|
|
|
log.debug(log_tag .. "MODBUS: repeated device failure" .. txn_tag)
|
|
|
|
else
|
|
|
|
self.device_fail = true
|
|
|
|
log.warning(log_tag .. "MODBUS: device failure" .. txn_tag)
|
|
|
|
end
|
|
|
|
elseif ex == MODBUS_EXCODE.ACKNOWLEDGE then
|
|
|
|
-- will have to wait on reply, renew the transaction
|
|
|
|
self.transaction_controller.renew(m_pkt.txn_id, txn_type)
|
|
|
|
elseif ex == MODBUS_EXCODE.SERVER_DEVICE_BUSY then
|
|
|
|
-- will have to wait on reply, renew the transaction
|
|
|
|
self.transaction_controller.renew(m_pkt.txn_id, txn_type)
|
|
|
|
log.debug(log_tag .. "MODBUS: device busy" .. txn_tag)
|
|
|
|
elseif ex == MODBUS_EXCODE.NEG_ACKNOWLEDGE then
|
|
|
|
-- general failure
|
|
|
|
log.error(log_tag .. "MODBUS: negative acknowledge (bad request)" .. txn_tag)
|
|
|
|
elseif ex == MODBUS_EXCODE.GATEWAY_PATH_UNAVAILABLE then
|
|
|
|
-- RTU gateway has no known unit with the given ID
|
|
|
|
log.error(log_tag .. "MODBUS: gateway path unavailable (unknown unit)" .. txn_tag)
|
|
|
|
elseif ex ~= nil then
|
|
|
|
-- unsupported exception code
|
|
|
|
log.debug(log_tag .. "MODBUS: unsupported error " .. ex .. txn_tag)
|
|
|
|
else
|
|
|
|
-- nil exception code
|
|
|
|
log.debug(log_tag .. "MODBUS: nil exception code" .. txn_tag)
|
|
|
|
end
|
|
|
|
else
|
|
|
|
-- clear device fail flag
|
|
|
|
self.device_fail = false
|
|
|
|
|
|
|
|
-- no error, return the transaction type
|
2022-12-01 04:31:14 +00:00
|
|
|
return txn_type, m_pkt.txn_id
|
2022-05-17 21:16:04 +00:00
|
|
|
end
|
|
|
|
else
|
|
|
|
log.error(log_tag .. "wrong unit ID: " .. m_pkt.unit_id, true)
|
|
|
|
end
|
|
|
|
else
|
|
|
|
log.error(log_tag .. "illegal packet type " .. m_pkt.scada_frame.protocol(), true)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- error or transaction in progress, return false
|
2022-12-01 04:31:14 +00:00
|
|
|
return false, m_pkt.txn_id
|
2022-05-17 21:16:04 +00:00
|
|
|
end
|
|
|
|
|
2022-05-31 19:40:17 +00:00
|
|
|
-- post update tasks
|
|
|
|
function protected.post_update()
|
|
|
|
self.transaction_controller.cleanup()
|
|
|
|
end
|
|
|
|
|
2022-05-17 21:16:04 +00:00
|
|
|
-- get the public interface
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function protected.get() return public end
|
2022-05-17 21:16:04 +00:00
|
|
|
|
|
|
|
-- PUBLIC FUNCTIONS --
|
|
|
|
|
2024-08-16 19:53:43 +00:00
|
|
|
-- get the RTU gateway session ID
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-11-12 06:35:31 +00:00
|
|
|
function public.get_session_id() return session_id end
|
2022-05-17 21:16:04 +00:00
|
|
|
-- get the unit ID
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
|
|
|
function public.get_unit_id() return unit_id end
|
2022-05-18 17:49:04 +00:00
|
|
|
-- get the device index
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2023-11-06 15:21:42 +00:00
|
|
|
function public.get_device_idx() return self.device_index or 0 end
|
2022-05-17 21:16:04 +00:00
|
|
|
-- get the reactor ID
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.get_reactor() return self.reactor end
|
2022-12-01 04:31:14 +00:00
|
|
|
-- get the command queue
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-12-01 04:31:14 +00:00
|
|
|
function public.get_cmd_queue() return protected.in_q end
|
2022-05-17 21:16:04 +00:00
|
|
|
|
|
|
|
-- close this unit
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.close() self.connected = false end
|
2022-05-17 21:16:04 +00:00
|
|
|
-- check if this unit is connected
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.is_connected() return self.connected end
|
2022-05-17 21:16:04 +00:00
|
|
|
-- check if this unit is faulted
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.is_faulted() return self.device_fail end
|
2022-05-17 21:16:04 +00:00
|
|
|
|
2022-05-18 17:49:04 +00:00
|
|
|
-- PUBLIC TEMPLATE FUNCTIONS --
|
|
|
|
|
2023-05-05 18:07:15 +00:00
|
|
|
-- luacheck: no unused args
|
2023-05-05 18:02:25 +00:00
|
|
|
|
2022-05-18 17:49:04 +00:00
|
|
|
-- handle a packet
|
|
|
|
---@param m_pkt modbus_frame
|
|
|
|
---@diagnostic disable-next-line: unused-local
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.handle_packet(m_pkt)
|
2022-05-18 17:49:04 +00:00
|
|
|
log.debug("template unit_session.handle_packet() called", true)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- update this runner
|
|
|
|
---@param time_now integer milliseconds
|
|
|
|
---@diagnostic disable-next-line: unused-local
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.update(time_now)
|
2022-05-18 17:49:04 +00:00
|
|
|
log.debug("template unit_session.update() called", true)
|
|
|
|
end
|
|
|
|
|
2023-05-05 18:07:15 +00:00
|
|
|
-- luacheck: unused args
|
2023-05-05 18:02:25 +00:00
|
|
|
|
2022-11-11 19:59:53 +00:00
|
|
|
-- invalidate build cache
|
|
|
|
function public.invalidate_cache()
|
|
|
|
log.debug("template unit_session.invalidate_cache() called", true)
|
|
|
|
end
|
|
|
|
|
2022-05-21 16:24:43 +00:00
|
|
|
-- get the unit session database
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-12-01 04:31:14 +00:00
|
|
|
function public.get_db()
|
|
|
|
log.debug("template unit_session.get_db() called", true)
|
|
|
|
return {}
|
|
|
|
end
|
2022-05-21 16:24:43 +00:00
|
|
|
|
2022-05-17 21:16:04 +00:00
|
|
|
return protected
|
|
|
|
end
|
|
|
|
|
|
|
|
return unit_session
|