cc-mek-scada/rtu/startup.lua

506 lines
20 KiB
Lua
Raw Normal View History

2022-01-13 15:12:44 +00:00
--
-- RTU: Remote Terminal Unit
--
require("/initenv").init_env()
2023-04-21 00:40:28 +00:00
local comms = require("scada-common.comms")
2022-11-13 20:56:27 +00:00
local crash = require("scada-common.crash")
local log = require("scada-common.log")
local mqueue = require("scada-common.mqueue")
local ppm = require("scada-common.ppm")
local rsio = require("scada-common.rsio")
local types = require("scada-common.types")
local util = require("scada-common.util")
local config = require("rtu.config")
2023-04-21 00:40:28 +00:00
local databus = require("rtu.databus")
local modbus = require("rtu.modbus")
2023-04-21 00:40:28 +00:00
local renderer = require("rtu.renderer")
local rtu = require("rtu.rtu")
local threads = require("rtu.threads")
local boilerv_rtu = require("rtu.dev.boilerv_rtu")
local envd_rtu = require("rtu.dev.envd_rtu")
local imatrix_rtu = require("rtu.dev.imatrix_rtu")
local redstone_rtu = require("rtu.dev.redstone_rtu")
local sna_rtu = require("rtu.dev.sna_rtu")
local sps_rtu = require("rtu.dev.sps_rtu")
local turbinev_rtu = require("rtu.dev.turbinev_rtu")
2023-04-21 21:10:15 +00:00
local RTU_VERSION = "v1.0.4"
2022-03-15 16:02:31 +00:00
2023-02-21 17:27:16 +00:00
local RTU_UNIT_TYPE = types.RTU_UNIT_TYPE
2023-04-21 00:40:28 +00:00
local RTU_UNIT_HW_STATE = databus.RTU_UNIT_HW_STATE
2022-05-03 14:44:18 +00:00
local println = util.println
local println_ts = util.println_ts
2022-06-05 19:09:02 +00:00
----------------------------------------
-- config validation
----------------------------------------
local cfv = util.new_validator()
cfv.assert_port(config.SERVER_PORT)
cfv.assert_port(config.LISTEN_PORT)
cfv.assert_type_int(config.TRUSTED_RANGE)
cfv.assert_type_num(config.COMMS_TIMEOUT)
2023-02-13 17:29:59 +00:00
cfv.assert_min(config.COMMS_TIMEOUT, 2)
2022-06-05 19:09:02 +00:00
cfv.assert_type_str(config.LOG_PATH)
cfv.assert_type_int(config.LOG_MODE)
cfv.assert_type_table(config.RTU_DEVICES)
cfv.assert_type_table(config.RTU_REDSTONE)
assert(cfv.valid(), "bad config file: missing/invalid fields")
----------------------------------------
-- log init
----------------------------------------
2022-04-29 17:36:00 +00:00
log.init(config.LOG_PATH, config.LOG_MODE)
log.info("========================================")
log.info("BOOTING rtu.startup " .. RTU_VERSION)
log.info("========================================")
println(">> RTU GATEWAY " .. RTU_VERSION .. " <<")
2022-03-15 16:02:31 +00:00
2022-11-13 20:56:27 +00:00
crash.set_env("rtu", RTU_VERSION)
2022-03-23 19:41:08 +00:00
----------------------------------------
2022-11-13 20:56:27 +00:00
-- main application
2022-03-23 19:41:08 +00:00
----------------------------------------
2022-11-13 20:56:27 +00:00
local function main()
----------------------------------------
-- startup
----------------------------------------
2023-04-21 00:40:28 +00:00
-- record firmware versions and ID
databus.tx_versions(RTU_VERSION, comms.version)
2022-11-13 20:56:27 +00:00
-- mount connected devices
ppm.mount_all()
---@class rtu_shared_memory
local __shared_memory = {
-- RTU system state flags
---@class rtu_state
rtu_state = {
2023-04-21 00:40:28 +00:00
fp_ok = false,
2022-11-13 20:56:27 +00:00
linked = false,
shutdown = false
},
-- core RTU devices
rtu_dev = {
modem = ppm.get_wireless_modem()
},
-- system objects
rtu_sys = {
rtu_comms = nil, ---@type rtu_comms
conn_watchdog = nil, ---@type watchdog
units = {} ---@type table
},
-- message queues
q = {
mq_comms = mqueue.new()
}
2022-04-27 16:46:04 +00:00
}
2022-11-13 20:56:27 +00:00
local smem_dev = __shared_memory.rtu_dev
local smem_sys = __shared_memory.rtu_sys
2022-04-27 16:46:04 +00:00
2022-11-13 20:56:27 +00:00
-- get modem
if smem_dev.modem == nil then
println("boot> wireless modem not found")
log.fatal("no wireless modem on startup")
return
end
2022-03-15 16:02:31 +00:00
2023-04-21 00:40:28 +00:00
databus.tx_hw_modem(true)
2022-11-13 20:56:27 +00:00
----------------------------------------
-- interpret config and init units
----------------------------------------
2022-03-23 19:41:08 +00:00
2022-11-13 20:56:27 +00:00
local units = __shared_memory.rtu_sys.units
2022-04-27 16:46:04 +00:00
2022-11-13 20:56:27 +00:00
local rtu_redstone = config.RTU_REDSTONE
local rtu_devices = config.RTU_DEVICES
2022-11-13 20:56:27 +00:00
-- configure RTU gateway based on config file definitions
local function configure()
-- redstone interfaces
for entry_idx = 1, #rtu_redstone do
local rs_rtu = redstone_rtu.new()
local io_table = rtu_redstone[entry_idx].io ---@type table
local io_reactor = rtu_redstone[entry_idx].for_reactor ---@type integer
2022-11-13 20:56:27 +00:00
-- CHECK: reactor ID must be >= to 1
if (not util.is_int(io_reactor)) or (io_reactor < 0) then
2023-02-22 04:50:43 +00:00
local message = util.c("configure> redstone entry #", entry_idx, " : ", io_reactor, " isn't an integer >= 0")
println(message)
log.fatal(message)
2022-11-13 20:56:27 +00:00
return false
end
2022-03-23 19:41:08 +00:00
2022-11-13 20:56:27 +00:00
-- CHECK: io table exists
if type(io_table) ~= "table" then
2023-02-22 04:50:43 +00:00
local message = util.c("configure> redstone entry #", entry_idx, " no IO table found")
println(message)
log.fatal(message)
2022-11-13 20:56:27 +00:00
return false
end
2022-11-13 20:56:27 +00:00
local capabilities = {}
2022-05-18 18:30:48 +00:00
2022-11-13 20:56:27 +00:00
log.debug(util.c("configure> starting redstone RTU I/O linking for reactor ", io_reactor, "..."))
2022-03-23 19:41:08 +00:00
2022-11-13 20:56:27 +00:00
local continue = true
2022-05-18 18:30:48 +00:00
2023-02-22 04:50:43 +00:00
-- CHECK: no duplicate entries
2022-11-13 20:56:27 +00:00
for i = 1, #units do
local unit = units[i] ---@type rtu_unit_registry_entry
2023-02-21 17:27:16 +00:00
if unit.reactor == io_reactor and unit.type == RTU_UNIT_TYPE.REDSTONE then
2022-11-13 20:56:27 +00:00
-- duplicate entry
local message = util.c("configure> skipping definition block #", entry_idx, " for reactor ", io_reactor,
" with already defined redstone I/O")
println(message)
log.warning(message)
continue = false
break
end
2022-03-23 19:41:08 +00:00
end
2022-11-13 20:56:27 +00:00
-- not a duplicate
if continue then
for i = 1, #io_table do
local valid = false
local conf = io_table[i]
2022-11-13 20:56:27 +00:00
-- verify configuration
if rsio.is_valid_port(conf.port) and rsio.is_valid_side(conf.side) then
2022-11-13 20:56:27 +00:00
if conf.bundled_color then
valid = rsio.is_color(conf.bundled_color)
else
2022-11-13 20:56:27 +00:00
valid = true
end
2022-11-13 20:56:27 +00:00
end
if not valid then
local message = util.c("configure> invalid redstone definition at index ", i, " in definition block #", entry_idx,
" (for reactor ", io_reactor, ")")
println(message)
2023-02-22 04:50:43 +00:00
log.fatal(message)
2022-11-13 20:56:27 +00:00
return false
else
-- link redstone in RTU
local mode = rsio.get_io_mode(conf.port)
2022-11-13 20:56:27 +00:00
if mode == rsio.IO_MODE.DIGITAL_IN then
-- can't have duplicate inputs
if util.table_contains(capabilities, conf.port) then
local message = util.c("configure> skipping duplicate input for port ", rsio.to_string(conf.port), " on side ", conf.side)
2022-11-13 20:56:27 +00:00
println(message)
log.warning(message)
else
rs_rtu.link_di(conf.side, conf.bundled_color)
end
elseif mode == rsio.IO_MODE.DIGITAL_OUT then
rs_rtu.link_do(conf.side, conf.bundled_color)
2022-11-13 20:56:27 +00:00
elseif mode == rsio.IO_MODE.ANALOG_IN then
-- can't have duplicate inputs
if util.table_contains(capabilities, conf.port) then
local message = util.c("configure> skipping duplicate input for port ", rsio.to_string(conf.port), " on side ", conf.side)
2022-11-13 20:56:27 +00:00
println(message)
log.warning(message)
else
rs_rtu.link_ai(conf.side)
end
elseif mode == rsio.IO_MODE.ANALOG_OUT then
rs_rtu.link_ao(conf.side)
else
-- should be unreachable code, we already validated ports
2022-11-13 20:56:27 +00:00
log.error("configure> fell through if chain attempting to identify IO mode", true)
println("configure> encountered a software error, check logs")
return false
end
2022-05-18 18:30:48 +00:00
table.insert(capabilities, conf.port)
2022-05-18 18:30:48 +00:00
log.debug(util.c("configure> linked redstone ", #capabilities, ": ", rsio.to_string(conf.port),
2022-11-13 20:56:27 +00:00
" (", conf.side, ") for reactor ", io_reactor))
end
end
2022-11-13 20:56:27 +00:00
---@class rtu_unit_registry_entry
local unit = {
2023-02-21 17:27:16 +00:00
uid = 0, ---@type integer
name = "redstone_io", ---@type string
type = RTU_UNIT_TYPE.REDSTONE, ---@type RTU_UNIT_TYPE
index = entry_idx, ---@type integer
reactor = io_reactor, ---@type integer
device = capabilities, ---@type table use device field for redstone ports
is_multiblock = false, ---@type boolean
formed = nil, ---@type boolean|nil
rtu = rs_rtu, ---@type rtu_device|rtu_rs_device
2022-11-13 20:56:27 +00:00
modbus_io = modbus.new(rs_rtu, false),
2023-02-21 17:27:16 +00:00
pkt_queue = nil, ---@type mqueue|nil
thread = nil ---@type parallel_thread|nil
2022-11-13 20:56:27 +00:00
}
table.insert(units, unit)
2023-02-21 17:27:16 +00:00
local for_message = "facility"
if io_reactor > 0 then
for_message = util.c("reactor ", io_reactor)
end
2023-02-22 04:50:43 +00:00
log.info(util.c("configure> initialized RTU unit #", #units, ": redstone_io (redstone) [1] for ", for_message))
unit.uid = #units
2023-04-21 00:40:28 +00:00
databus.tx_unit_hw_status(unit.uid, RTU_UNIT_HW_STATE.OK)
2022-11-13 20:56:27 +00:00
end
end
2022-03-15 16:02:31 +00:00
2022-11-13 20:56:27 +00:00
-- mounted peripherals
for i = 1, #rtu_devices do
local name = rtu_devices[i].name
local index = rtu_devices[i].index
local for_reactor = rtu_devices[i].for_reactor
2022-11-13 20:56:27 +00:00
-- CHECK: name is a string
if type(name) ~= "string" then
2023-02-22 04:50:43 +00:00
local message = util.c("configure> device entry #", i, ": device ", name, " isn't a string")
println(message)
log.fatal(message)
2022-11-13 20:56:27 +00:00
return false
end
2022-03-15 16:02:31 +00:00
2022-11-13 20:56:27 +00:00
-- CHECK: index is an integer >= 1
if (not util.is_int(index)) or (index <= 0) then
2023-02-22 04:50:43 +00:00
local message = util.c("configure> device entry #", i, ": index ", index, " isn't an integer >= 1")
println(message)
log.fatal(message)
2022-11-13 20:56:27 +00:00
return false
end
2022-04-29 17:19:01 +00:00
-- CHECK: reactor is an integer >= 0
if (not util.is_int(for_reactor)) or (for_reactor < 0) then
2023-02-22 04:50:43 +00:00
local message = util.c("configure> device entry #", i, ": reactor ", for_reactor, " isn't an integer >= 0")
println(message)
log.fatal(message)
2022-11-13 20:56:27 +00:00
return false
end
2022-04-29 17:19:01 +00:00
2022-11-13 20:56:27 +00:00
local device = ppm.get_periph(name)
2022-03-23 19:41:08 +00:00
local type ---@type string|nil
local rtu_iface ---@type rtu_device
local rtu_type ---@type RTU_UNIT_TYPE
local is_multiblock = false ---@type boolean
local formed = nil ---@type boolean|nil
local faulted = nil ---@type boolean|nil
2022-11-13 20:56:27 +00:00
if device == nil then
local message = util.c("configure> '", name, "' not found, using placeholder")
println(message)
log.warning(message)
2022-11-13 20:56:27 +00:00
-- mount a virtual (placeholder) device
type, device = ppm.mount_virtual()
else
type = ppm.get_type(name)
end
2022-11-13 20:56:27 +00:00
if type == "boilerValve" then
-- boiler multiblock
2023-02-21 17:27:16 +00:00
rtu_type = RTU_UNIT_TYPE.BOILER_VALVE
rtu_iface, faulted = boilerv_rtu.new(device)
is_multiblock = true
2022-11-13 20:56:27 +00:00
formed = device.isFormed()
2022-11-13 20:56:27 +00:00
if formed == ppm.UNDEFINED_FIELD or formed == ppm.ACCESS_FAULT then
println_ts(util.c("configure> failed to check if '", name, "' is formed"))
log.fatal(util.c("configure> failed to check if '", name, "' is a formed boiler multiblock"))
return false
end
elseif type == "turbineValve" then
-- turbine multiblock
2023-02-21 17:27:16 +00:00
rtu_type = RTU_UNIT_TYPE.TURBINE_VALVE
rtu_iface, faulted = turbinev_rtu.new(device)
is_multiblock = true
2022-11-13 20:56:27 +00:00
formed = device.isFormed()
if formed == ppm.UNDEFINED_FIELD or formed == ppm.ACCESS_FAULT then
println_ts(util.c("configure> failed to check if '", name, "' is formed"))
log.fatal(util.c("configure> failed to check if '", name, "' is a formed turbine multiblock"))
return false
end
elseif type == "inductionPort" then
-- induction matrix multiblock
2023-02-21 17:27:16 +00:00
rtu_type = RTU_UNIT_TYPE.IMATRIX
rtu_iface, faulted = imatrix_rtu.new(device)
is_multiblock = true
2022-11-13 20:56:27 +00:00
formed = device.isFormed()
if formed == ppm.UNDEFINED_FIELD or formed == ppm.ACCESS_FAULT then
println_ts(util.c("configure> failed to check if '", name, "' is formed"))
log.fatal(util.c("configure> failed to check if '", name, "' is a formed induction matrix multiblock"))
return false
end
elseif type == "spsPort" then
-- SPS multiblock
2023-02-21 17:27:16 +00:00
rtu_type = RTU_UNIT_TYPE.SPS
rtu_iface, faulted = sps_rtu.new(device)
is_multiblock = true
2022-11-13 20:56:27 +00:00
formed = device.isFormed()
if formed == ppm.UNDEFINED_FIELD or formed == ppm.ACCESS_FAULT then
println_ts(util.c("configure> failed to check if '", name, "' is formed"))
log.fatal(util.c("configure> failed to check if '", name, "' is a formed SPS multiblock"))
return false
end
elseif type == "solarNeutronActivator" then
-- SNA
2023-02-21 17:27:16 +00:00
rtu_type = RTU_UNIT_TYPE.SNA
rtu_iface, faulted = sna_rtu.new(device)
2022-11-13 20:56:27 +00:00
elseif type == "environmentDetector" then
-- advanced peripherals environment detector
2023-02-21 17:27:16 +00:00
rtu_type = RTU_UNIT_TYPE.ENV_DETECTOR
rtu_iface, faulted = envd_rtu.new(device)
2022-11-13 20:56:27 +00:00
elseif type == ppm.VIRTUAL_DEVICE_TYPE then
-- placeholder device
2023-02-21 17:27:16 +00:00
rtu_type = RTU_UNIT_TYPE.VIRTUAL
2022-11-13 20:56:27 +00:00
rtu_iface = rtu.init_unit().interface()
else
local message = util.c("configure> device '", name, "' is not a known type (", type, ")")
println_ts(message)
log.fatal(message)
return false
end
if is_multiblock then
if not formed then
log.info(util.c("configure> device '", name, "' is not formed"))
elseif faulted then
-- sometimes there is a race condition on server boot where it reports formed, but
-- the other functions are not yet defined (that's the theory at least). mark as unformed to attempt connection later
formed = false
log.warning(util.c("configure> device '", name, "' is formed, but initialization had one or more faults: marked as unformed"))
end
end
2022-11-13 20:56:27 +00:00
---@class rtu_unit_registry_entry
local rtu_unit = {
2023-02-21 17:27:16 +00:00
uid = 0, ---@type integer
name = name, ---@type string
type = rtu_type, ---@type RTU_UNIT_TYPE
index = index, ---@type integer
reactor = for_reactor, ---@type integer
device = device, ---@type table
is_multiblock = is_multiblock, ---@type boolean
formed = formed, ---@type boolean|nil
rtu = rtu_iface, ---@type rtu_device|rtu_rs_device
2022-11-13 20:56:27 +00:00
modbus_io = modbus.new(rtu_iface, true),
pkt_queue = mqueue.new(), ---@type mqueue|nil
2023-02-21 17:27:16 +00:00
thread = nil ---@type parallel_thread|nil
2022-11-13 20:56:27 +00:00
}
2022-11-13 20:56:27 +00:00
rtu_unit.thread = threads.thread__unit_comms(__shared_memory, rtu_unit)
2022-11-13 20:56:27 +00:00
table.insert(units, rtu_unit)
local for_message = "facility"
if for_reactor > 0 then
for_message = util.c("reactor ", for_reactor)
end
2023-02-22 04:50:43 +00:00
log.info(util.c("configure> initialized RTU unit #", #units, ": ", name, " (", types.rtu_type_to_string(rtu_type), ") [", index, "] for ", for_message))
rtu_unit.uid = #units
2023-04-21 00:40:28 +00:00
-- report hardware status
if rtu_unit.type == RTU_UNIT_TYPE.VIRTUAL then
databus.tx_unit_hw_status(rtu_unit.uid, RTU_UNIT_HW_STATE.OFFLINE)
else
if rtu_unit.is_multiblock then
databus.tx_unit_hw_status(rtu_unit.uid, util.trinary(rtu_unit.formed == true, RTU_UNIT_HW_STATE.OK, RTU_UNIT_HW_STATE.UNFORMED))
elseif faulted then
databus.tx_unit_hw_status(rtu_unit.uid, RTU_UNIT_HW_STATE.FAULTED)
else
databus.tx_unit_hw_status(rtu_unit.uid, RTU_UNIT_HW_STATE.OK)
end
end
2022-11-13 20:56:27 +00:00
end
2022-11-13 20:56:27 +00:00
-- we made it through all that trusting-user-to-write-a-config-file chaos
return true
end
2022-03-15 16:02:31 +00:00
2022-11-13 20:56:27 +00:00
----------------------------------------
-- start system
----------------------------------------
2022-03-15 16:02:31 +00:00
2023-04-21 00:40:28 +00:00
local rtu_state = __shared_memory.rtu_state
2022-11-13 20:56:27 +00:00
log.debug("boot> running configure()")
2022-11-13 20:56:27 +00:00
if configure() then
2023-04-21 00:40:28 +00:00
-- start UI
local message
rtu_state.fp_ok, message = pcall(renderer.start_ui, units)
if not rtu_state.fp_ok then
renderer.close_ui()
println_ts(util.c("UI error: ", message))
println("init> running without front panel")
log.error(util.c("GUI crashed with error ", message))
log.info("init> running in headless mode without front panel")
end
2022-11-13 20:56:27 +00:00
-- start connection watchdog
smem_sys.conn_watchdog = util.new_watchdog(config.COMMS_TIMEOUT)
2023-02-22 04:50:43 +00:00
log.debug("startup> conn watchdog started")
2022-11-13 20:56:27 +00:00
-- setup comms
smem_sys.rtu_comms = rtu.comms(RTU_VERSION, smem_dev.modem, config.LISTEN_PORT, config.SERVER_PORT,
config.TRUSTED_RANGE, smem_sys.conn_watchdog)
2023-02-22 04:50:43 +00:00
log.debug("startup> comms init")
2022-04-11 21:27:57 +00:00
2022-11-13 20:56:27 +00:00
-- init threads
local main_thread = threads.thread__main(__shared_memory)
local comms_thread = threads.thread__comms(__shared_memory)
2022-11-13 20:56:27 +00:00
-- assemble thread list
local _threads = { main_thread.p_exec, comms_thread.p_exec }
for i = 1, #units do
if units[i].thread ~= nil then
table.insert(_threads, units[i].thread.p_exec)
end
end
2022-11-13 20:56:27 +00:00
2023-02-22 04:50:43 +00:00
log.info("startup> completed")
2022-11-13 20:56:27 +00:00
-- run threads
parallel.waitForAll(table.unpack(_threads))
else
println("configuration failed, exiting...")
2022-04-29 17:19:01 +00:00
end
2023-04-21 00:40:28 +00:00
renderer.close_ui()
2022-11-13 20:56:27 +00:00
println_ts("exited")
log.info("exited")
end
2023-04-21 01:19:16 +00:00
if not xpcall(main, crash.handler) then
pcall(renderer.close_ui)
crash.exit()
else
log.close()
end