-- -- Reactor Programmable Logic Controller -- require("/initenv").init_env() local log = require("scada-common.log") local mqueue = require("scada-common.mqueue") local ppm = require("scada-common.ppm") local util = require("scada-common.util") local config = require("reactor-plc.config") local plc = require("reactor-plc.plc") local threads = require("reactor-plc.threads") local R_PLC_VERSION = "beta-v0.8.7" local print = util.print local println = util.println local print_ts = util.print_ts local println_ts = util.println_ts ---------------------------------------- -- config validation ---------------------------------------- local cfv = util.new_validator() cfv.assert_type_bool(config.NETWORKED) cfv.assert_type_int(config.REACTOR_ID) cfv.assert_port(config.SERVER_PORT) cfv.assert_port(config.LISTEN_PORT) cfv.assert_type_str(config.LOG_PATH) cfv.assert_type_int(config.LOG_MODE) assert(cfv.valid(), "bad config file: missing/invalid fields") ---------------------------------------- -- log init ---------------------------------------- log.init(config.LOG_PATH, config.LOG_MODE) log.info("========================================") log.info("BOOTING reactor-plc.startup " .. R_PLC_VERSION) log.info("========================================") println(">> Reactor PLC " .. R_PLC_VERSION .. " <<") ---------------------------------------- -- startup ---------------------------------------- -- mount connected devices ppm.mount_all() -- shared memory across threads ---@class plc_shared_memory local __shared_memory = { -- networked setting networked = config.NETWORKED, ---@type boolean -- PLC system state flags ---@class plc_state plc_state = { init_ok = true, shutdown = false, degraded = false, no_reactor = false, no_modem = false }, -- control setpoints ---@class setpoints setpoints = { burn_rate_en = false, burn_rate = 0.0 }, -- core PLC devices plc_dev = { reactor = ppm.get_fission_reactor(), modem = ppm.get_wireless_modem() }, -- system objects plc_sys = { rps = nil, ---@type rps plc_comms = nil, ---@type plc_comms conn_watchdog = nil ---@type watchdog }, -- message queues q = { mq_rps = mqueue.new(), mq_comms_tx = mqueue.new(), mq_comms_rx = mqueue.new() } } local smem_dev = __shared_memory.plc_dev local smem_sys = __shared_memory.plc_sys local plc_state = __shared_memory.plc_state -- we need a reactor and a modem if smem_dev.reactor == nil then println("boot> fission reactor not found"); log.warning("no reactor on startup") plc_state.init_ok = false plc_state.degraded = true plc_state.no_reactor = true end if __shared_memory.networked and smem_dev.modem == nil then println("boot> wireless modem not found") log.warning("no wireless modem on startup") if smem_dev.reactor ~= nil and smem_dev.reactor.getStatus() then smem_dev.reactor.scram() end plc_state.init_ok = false plc_state.degraded = true plc_state.no_modem = true end -- PLC init local function init() if plc_state.init_ok then -- just booting up, no fission allowed (neutrons stay put thanks) if smem_dev.reactor.getStatus() then smem_dev.reactor.scram() end -- init reactor protection system smem_sys.rps = plc.rps_init(smem_dev.reactor) log.debug("init> rps init") if __shared_memory.networked then -- comms watchdog, 3 second timeout smem_sys.conn_watchdog = util.new_watchdog(3) log.debug("init> conn watchdog started") -- start comms smem_sys.plc_comms = plc.comms(config.REACTOR_ID, R_PLC_VERSION, smem_dev.modem, config.LISTEN_PORT, config.SERVER_PORT, smem_dev.reactor, smem_sys.rps, smem_sys.conn_watchdog) log.debug("init> comms init") else println("boot> starting in offline mode"); log.debug("init> running without networking") end ---@diagnostic disable-next-line: undefined-field os.queueEvent("clock_start") println("boot> completed"); log.debug("init> boot completed") else println("boot> system in degraded state, awaiting devices...") log.warning("init> booted in a degraded state, awaiting peripheral connections...") end end ---------------------------------------- -- start system ---------------------------------------- -- initialize PLC init() -- init threads local main_thread = threads.thread__main(__shared_memory, init) local rps_thread = threads.thread__rps(__shared_memory) if __shared_memory.networked then -- init comms threads local comms_thread_tx = threads.thread__comms_tx(__shared_memory) local comms_thread_rx = threads.thread__comms_rx(__shared_memory) -- setpoint control only needed when networked local sp_ctrl_thread = threads.thread__setpoint_control(__shared_memory) -- run threads parallel.waitForAll(main_thread.p_exec, rps_thread.p_exec, comms_thread_tx.p_exec, comms_thread_rx.p_exec, sp_ctrl_thread.p_exec) if plc_state.init_ok then -- send status one last time after RPS shutdown smem_sys.plc_comms.send_status(plc_state.degraded) smem_sys.plc_comms.send_rps_status() -- close connection smem_sys.plc_comms.close() end else -- run threads, excluding comms parallel.waitForAll(main_thread.p_exec, rps_thread.p_exec) end println_ts("exited") log.info("exited")