-- -- SCADA System Access on a Pocket Computer -- require("/initenv").init_env() local crash = require("scada-common.crash") local log = require("scada-common.log") local network = require("scada-common.network") local ppm = require("scada-common.ppm") local tcd = require("scada-common.tcd") local util = require("scada-common.util") local core = require("graphics.core") local config = require("pocket.config") local iocontrol = require("pocket.iocontrol") local pocket = require("pocket.pocket") local renderer = require("pocket.renderer") local POCKET_VERSION = "v0.6.0-alpha" local println = util.println local println_ts = util.println_ts ---------------------------------------- -- config validation ---------------------------------------- local cfv = util.new_validator() cfv.assert_channel(config.SVR_CHANNEL) cfv.assert_channel(config.CRD_CHANNEL) cfv.assert_channel(config.PKT_CHANNEL) cfv.assert_type_int(config.TRUSTED_RANGE) cfv.assert_type_num(config.COMMS_TIMEOUT) cfv.assert_min(config.COMMS_TIMEOUT, 2) 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, config.LOG_DEBUG == true) log.info("========================================") log.info("BOOTING pocket.startup " .. POCKET_VERSION) log.info("========================================") crash.set_env("pocket", POCKET_VERSION) ---------------------------------------- -- main application ---------------------------------------- local function main() ---------------------------------------- -- system startup ---------------------------------------- -- mount connected devices ppm.mount_all() ---------------------------------------- -- setup communications & clocks ---------------------------------------- -- message authentication init if type(config.AUTH_KEY) == "string" then network.init_mac(config.AUTH_KEY) end iocontrol.report_link_state(iocontrol.LINK_STATE.UNLINKED) -- get the communications modem local modem = ppm.get_wireless_modem() if modem == nil then println("startup> wireless modem not found: please craft the pocket computer with a wireless modem") log.fatal("startup> no wireless modem on startup") return end -- create connection watchdogs local conn_wd = { sv = util.new_watchdog(config.COMMS_TIMEOUT), api = util.new_watchdog(config.COMMS_TIMEOUT) } conn_wd.sv.cancel() conn_wd.api.cancel() log.debug("startup> conn watchdogs created") -- create network interface then setup comms local nic = network.nic(modem) local pocket_comms = pocket.comms(POCKET_VERSION, nic, config.PKT_CHANNEL, config.SVR_CHANNEL, config.CRD_CHANNEL, config.TRUSTED_RANGE, conn_wd.sv, conn_wd.api) log.debug("startup> comms init") -- base loop clock (2Hz, 10 ticks) local MAIN_CLOCK = 0.5 local loop_clock = util.new_clock(MAIN_CLOCK) -- init I/O control iocontrol.init_core(pocket_comms) ---------------------------------------- -- start the UI ---------------------------------------- local ui_ok, message = pcall(renderer.start_ui) if not ui_ok then renderer.close_ui() println(util.c("UI error: ", message)) log.error(util.c("startup> GUI render failed with error ", message)) else -- start clock loop_clock.start() end ---------------------------------------- -- main event loop ---------------------------------------- if ui_ok then -- start connection watchdogs conn_wd.sv.feed() conn_wd.api.feed() log.debug("startup> conn watchdog started") local io_db = iocontrol.get_db() local nav = io_db.nav -- main event loop while true do local event, param1, param2, param3, param4, param5 = util.pull_event() -- handle event if event == "timer" then if loop_clock.is_clock(param1) then -- main loop tick -- relink if necessary pocket_comms.link_update() -- update any tasks for the active page if (type(nav.tasks[nav.page]) == "table") then for i = 1, #nav.tasks[nav.page] do nav.tasks[nav.page][i]() end end loop_clock.start() elseif conn_wd.sv.is_timer(param1) then -- supervisor watchdog timeout log.info("supervisor server timeout") pocket_comms.close_sv() elseif conn_wd.api.is_timer(param1) then -- coordinator watchdog timeout log.info("coordinator api server timeout") pocket_comms.close_api() else -- a non-clock/main watchdog timer event -- notify timer callback dispatcher tcd.handle(param1) end elseif event == "modem_message" then -- got a packet local packet = pocket_comms.parse_packet(param1, param2, param3, param4, param5) pocket_comms.handle_packet(packet) elseif event == "mouse_click" or event == "mouse_up" or event == "mouse_drag" or event == "mouse_scroll" then -- handle a monitor touch event renderer.handle_mouse(core.events.new_mouse_event(event, param1, param2, param3)) end -- check for termination request if event == "terminate" or ppm.should_terminate() then log.info("terminate requested, closing server connections...") pocket_comms.close() log.info("connections closed") break end end renderer.close_ui() end println_ts("exited") log.info("exited") end if not xpcall(main, crash.handler) then pcall(renderer.close_ui) crash.exit() else log.close() end