[fluksod + nixio] add { ctrl, delta, uart } dispatching
This commit is contained in:
parent
0d3f076b5f
commit
2fa9a53766
|
@ -1,6 +1,6 @@
|
||||||
--[[
|
--[[
|
||||||
|
|
||||||
spi.lua - Lua 5.1 flukso module for spidev message processing
|
spi.lua - Lua 5.1 flukso module for spidev message processing and dispatching
|
||||||
|
|
||||||
Copyright (C) 2011 Bart Van Der Meerssche <bart.vandermeerssche@flukso.net>
|
Copyright (C) 2011 Bart Van Der Meerssche <bart.vandermeerssche@flukso.net>
|
||||||
|
|
||||||
|
@ -30,7 +30,11 @@ local getfenv, setmetatable =
|
||||||
module (...)
|
module (...)
|
||||||
local modenv = getfenv()
|
local modenv = getfenv()
|
||||||
|
|
||||||
local SPI_MAX_READ_BYTES = 256
|
local SPI_END_OF_MESSAGE = '.'
|
||||||
|
local SPI_FORWARD_TO_UART_PORT = 'u'
|
||||||
|
local SPI_FORWARD_TO_CTRL_PORT = 'l' -- 'l'ocal port
|
||||||
|
|
||||||
|
local SPI_MAX_READ_BYTES = 1024
|
||||||
|
|
||||||
--- Create a new spi message object.
|
--- Create a new spi message object.
|
||||||
--
|
--
|
||||||
|
@ -64,7 +68,7 @@ end
|
||||||
|
|
||||||
function encode(msg)
|
function encode(msg)
|
||||||
if msg.to == 'uart' then
|
if msg.to == 'uart' then
|
||||||
msg.encoded = nixio.bin.hexlify(msg.body)
|
msg.encoded = nixio.bin.hexlify(msg.body or '')
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -97,39 +101,39 @@ function encode(msg)
|
||||||
elseif msg.parsed.cmd == 'ct' then
|
elseif msg.parsed.cmd == 'ct' then
|
||||||
|
|
||||||
else
|
else
|
||||||
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
msg.encoded = msg.encoded .. nixio.bin.numtohex(nixio.bin.dow_crc(msg.encoded), 1)
|
msg.encoded = msg.encoded .. nixio.bin.numtohex(nixio.bin.dow_crc(msg.encoded), 1)
|
||||||
end
|
end
|
||||||
|
|
||||||
function tx(msg, cdev)
|
function tx(msg, cdev)
|
||||||
if msg.to == 'ctrl' or msg.to == 'delta' then
|
if msg.encoded then
|
||||||
cdev:write('l' .. msg.encoded .. '.')
|
if (msg.to == 'ctrl' or msg.to == 'delta') then
|
||||||
elseif msg.to == 'uart' then
|
cdev:write(SPI_FORWARD_TO_CTRL_PORT .. msg.encoded .. SPI_END_OF_MESSAGE)
|
||||||
cdev:write('u' .. msg.encoded)
|
elseif msg.to == 'uart' then
|
||||||
|
cdev:write(SPI_FORWARD_TO_UART_PORT .. msg.encoded)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
function rx(msg, cdev)
|
function rx(msg, cdev)
|
||||||
msg.received = {}
|
msg.received = {}
|
||||||
msg.received.raw = cdev:read(SPI_MAX_READ_BYTES)
|
msg.received.raw = cdev:read(SPI_MAX_READ_BYTES)
|
||||||
msg.received.l, msg.received.u = msg.received.raw:match('^l(%w*)%.?u(%w*)%.?$')
|
|
||||||
-- protect against nil values when match should fail
|
|
||||||
-- TODO error handling when no reply due to:
|
|
||||||
-- * sensor board not operational
|
|
||||||
-- * state machine not synced
|
|
||||||
msg.received.l, msg.received.u = msg.received.l or '', msg.received.u or ''
|
|
||||||
|
|
||||||
if msg.received.l ~= '' then
|
if msg.received.raw == '' then -- state machine not synced
|
||||||
|
msg.received.raw = cdev:read(SPI_MAX_READ_BYTES)
|
||||||
|
end
|
||||||
|
|
||||||
|
msg.received.l, msg.received.u = msg.received.raw:match('^l(%w*)%.?u(%w*)%.?$')
|
||||||
|
|
||||||
|
if msg.received.l then
|
||||||
msg.received.crc = msg.received.l:sub(-2, -1)
|
msg.received.crc = msg.received.l:sub(-2, -1)
|
||||||
msg.received.l = msg.received.l:sub( 1, -3)
|
msg.received.l = msg.received.l:sub( 1, -3)
|
||||||
|
|
||||||
if nixio.bin.dow_crc(msg.received.l) ~= nixio.bin.hextonum(msg.received.crc) then
|
if nixio.bin.dow_crc(msg.received.l) ~= nixio.bin.hextonum(msg.received.crc) then
|
||||||
--> TODO implement near-end crc error counter
|
--> TODO implement near-end crc error counter
|
||||||
msg.received.l = ''
|
msg.received.l = nil
|
||||||
else
|
|
||||||
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -137,11 +141,11 @@ end
|
||||||
function decode(msg)
|
function decode(msg)
|
||||||
msg.decoded = {}
|
msg.decoded = {}
|
||||||
|
|
||||||
if msg.received.u ~= '' then
|
if msg.received.u then
|
||||||
msg.decoded.u = nixio.bin.unhexlify(msg.received.u)
|
msg.decoded.uart = nixio.bin.unhexlify(msg.received.u)
|
||||||
end
|
end
|
||||||
|
|
||||||
if msg.received.l ~= '' then
|
if msg.received.l then
|
||||||
msg.decoded.cmd = msg.received.l:sub(1, 2)
|
msg.decoded.cmd = msg.received.l:sub(1, 2)
|
||||||
msg.decoded.args = msg.received.l:sub(3, -1)
|
msg.decoded.args = msg.received.l:sub(3, -1)
|
||||||
|
|
||||||
|
@ -154,34 +158,38 @@ function decode(msg)
|
||||||
msg.decoded[(i-1)*3 + 3] =
|
msg.decoded[(i-1)*3 + 3] =
|
||||||
nixio.bin.hextonum(msg.decoded.args:sub((i-1)*18 + 11, (i-1)*18 + 18))
|
nixio.bin.hextonum(msg.decoded.args:sub((i-1)*18 + 11, (i-1)*18 + 18))
|
||||||
end
|
end
|
||||||
elseif msg.parsed.cmd == 'gv' then
|
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'gp' then
|
msg.decoded.delta = msg.decoded.cmd .. ' ' .. table.concat(msg.decoded, ' ')
|
||||||
|
elseif msg.decoded.cmd == 'gv' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'gc' then
|
elseif msg.decoded.cmd == 'gp' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'gm' then
|
elseif msg.decoded.cmd == 'gc' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'gw' then
|
elseif msg.decoded.cmd == 'gm' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'gb' then
|
elseif msg.decoded.cmd == 'gw' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'sv' then
|
elseif msg.decoded.cmd == 'gb' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'sp' then
|
elseif msg.decoded.cmd == 'sv' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'sc' then
|
elseif msg.decoded.cmd == 'sp' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'sm' then
|
elseif msg.decoded.cmd == 'sc' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'sw' then
|
elseif msg.decoded.cmd == 'sm' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'sb' then
|
elseif msg.decoded.cmd == 'sw' then
|
||||||
|
|
||||||
elseif msg.parsed.cmd == 'ct' then
|
elseif msg.decoded.cmd == 'sb' then
|
||||||
|
|
||||||
|
elseif msg.decoded.cmd == 'ct' then
|
||||||
|
|
||||||
elseif msg.decoded.cmd == 'zz' then
|
elseif msg.decoded.cmd == 'zz' then
|
||||||
--> TODO implement far-end crc error counter
|
--> TODO implement far-end crc error counter
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
return { ctrl = msg.decoded.ctrl, delta = msg.decoded.delta, uart = msg.decoded.uart }
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
--[[
|
--[[
|
||||||
|
|
||||||
fluksod.lua - Lua part of fluksod
|
fluksod.lua - Lua part of the Flukso daemon
|
||||||
|
|
||||||
Copyright (C) 2011 Bart Van Der Meerssche <bart.vandermeerssche@flukso.net>
|
Copyright (C) 2011 Bart Van Der Meerssche <bart.vandermeerssche@flukso.net>
|
||||||
|
|
||||||
|
@ -21,30 +21,37 @@
|
||||||
|
|
||||||
]]--
|
]]--
|
||||||
|
|
||||||
require 'dbg'
|
local dbg = require 'dbg'
|
||||||
require 'nixio'
|
local spi = require 'flukso.spi'
|
||||||
require 'nixio.fs'
|
local nixio = require 'nixio'
|
||||||
local spi = require 'flukso.spi'
|
nixio.fs = require 'nixio.fs'
|
||||||
|
|
||||||
DAEMON = os.getenv('DAEMON') or 'fluksod'
|
local SPI_DEV = '/dev/spidev0.0'
|
||||||
DAEMON_PATH = os.getenv('DAEMON_PATH') or '/var/run/' .. DAEMON
|
local SPI_MAX_CLK_SPEED_HZ = 10e6
|
||||||
|
local SPI_MIN_BYTE_DELAY_US = 250
|
||||||
|
local SPI_TX_RX_DELAY_NS = 10e7
|
||||||
|
local POLL_TIMEOUT_MS = 100
|
||||||
|
|
||||||
|
local DAEMON = os.getenv('DAEMON') or 'fluksod'
|
||||||
|
local DAEMON_PATH = os.getenv('DAEMON_PATH') or '/var/run/' .. DAEMON
|
||||||
|
|
||||||
|
local O_RDWR_NONBLOCK = nixio.open_flags('rdwr', 'nonblock')
|
||||||
|
local POLLIN = nixio.poll_flags('in')
|
||||||
|
|
||||||
O_RDWR_NONBLOCK = nixio.open_flags('rdwr', 'nonblock')
|
|
||||||
POLLIN = nixio.poll_flags('in')
|
|
||||||
|
|
||||||
function mkfifos(input)
|
function mkfifos(input)
|
||||||
local path = string.format('%s/%s/', DAEMON_PATH, input)
|
local path = string.format('%s/%s/', DAEMON_PATH, input)
|
||||||
|
|
||||||
nixio.fs.mkdirr(path)
|
nixio.fs.mkdirr(path)
|
||||||
nixio.fs.unlink(path .. 'in') --> clean up mess from previous run
|
nixio.fs.unlink(path .. 'in') -- clean up mess from previous run
|
||||||
nixio.fs.unlink(path .. 'out') --> idem
|
nixio.fs.unlink(path .. 'out') -- idem
|
||||||
nixio.fs.mkfifo(path .. 'in', '644')
|
nixio.fs.mkfifo(path .. 'in', '644')
|
||||||
nixio.fs.mkfifo(path .. 'out', '644')
|
nixio.fs.mkfifo(path .. 'out', '644')
|
||||||
|
|
||||||
local fdin = nixio.open(path .. 'in', O_RDWR_NONBLOCK)
|
local fdin = nixio.open(path .. 'in', O_RDWR_NONBLOCK)
|
||||||
local fdout = nixio.open(path .. 'out', O_RDWR_NONBLOCK)
|
local fdout = nixio.open(path .. 'out', O_RDWR_NONBLOCK)
|
||||||
|
|
||||||
return { fd = fdin, --> need this entry for nixio.poll
|
return { fd = fdin, -- need this entry for nixio.poll
|
||||||
fdin = fdin,
|
fdin = fdin,
|
||||||
fdout = fdout,
|
fdout = fdout,
|
||||||
events = POLLIN,
|
events = POLLIN,
|
||||||
|
@ -58,21 +65,49 @@ local delta = mkfifos('delta')
|
||||||
|
|
||||||
local fds = { uart, ctrl, delta }
|
local fds = { uart, ctrl, delta }
|
||||||
|
|
||||||
local spidev = nixio.open('/dev/spidev0.0', O_RDWR_NONBLOCK)
|
local spidev = nixio.open(SPI_DEV, O_RDWR_NONBLOCK)
|
||||||
|
nixio.spi.setspeed(spidev, SPI_MAX_CLK_SPEED_HZ, SPI_MIN_BYTE_DELAY_US)
|
||||||
|
|
||||||
while true do
|
while true do
|
||||||
if nixio.poll(fds, -1) then
|
local msg
|
||||||
if delta.revents == POLLIN then
|
local poll = nixio.poll(fds, POLL_TIMEOUT_MS)
|
||||||
--> TODO flush the delta fd after each POLLIN
|
|
||||||
local msg = spi.new_msg('delta', delta.line())
|
if poll == 0 then -- poll timed out, so time to 'poll' the spi bus
|
||||||
|
msg = spi.new_msg('', '')
|
||||||
|
elseif poll > 0 then
|
||||||
|
if ctrl.revents == POLLIN then
|
||||||
|
msg = spi.new_msg('ctrl', ctrl.line())
|
||||||
msg:parse()
|
msg:parse()
|
||||||
msg:encode()
|
|
||||||
msg:tx(spidev)
|
elseif delta.revents == POLLIN then
|
||||||
nixio.nanosleep(0, 10e7) --> 10ms
|
-- TODO flush the delta fd after each POLLIN
|
||||||
msg:rx(spidev)
|
msg = spi.new_msg('delta', delta.line())
|
||||||
msg:decode()
|
msg:parse()
|
||||||
--> dbg.vardump(msg)
|
|
||||||
delta.fdout:write(msg.decoded.cmd .. ' ' .. table.concat(msg.decoded, ' ') .. '\n')
|
elseif uart.revents == POLLIN then
|
||||||
|
msg = spi.new_msg('uart', uart.line())
|
||||||
|
end
|
||||||
|
|
||||||
|
msg:encode()
|
||||||
|
msg:tx(spidev)
|
||||||
|
nixio.nanosleep(0, SPI_TX_RX_DELAY_NS)
|
||||||
|
end
|
||||||
|
|
||||||
|
if poll >= 0 then
|
||||||
|
msg:rx(spidev)
|
||||||
|
local dispatch = msg:decode()
|
||||||
|
-- dbg.vardump(msg)
|
||||||
|
|
||||||
|
if dispatch.ctrl then
|
||||||
|
ctrl.fdout:write(dispatch.ctrl .. '\n')
|
||||||
|
end
|
||||||
|
|
||||||
|
if dispatch.delta then
|
||||||
|
delta.fdout:write(dispatch.delta .. '\n')
|
||||||
|
end
|
||||||
|
|
||||||
|
if dispatch.uart then
|
||||||
|
uart.fdout:write(dispatch.uart .. '\n')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -169,13 +169,10 @@ static int nixio_poll(lua_State *L) {
|
||||||
|
|
||||||
status = poll(fds, (nfds_t)len, timeout);
|
status = poll(fds, (nfds_t)len, timeout);
|
||||||
|
|
||||||
if (status == 0) {
|
if (status <= 0) {
|
||||||
free(fds);
|
free(fds);
|
||||||
lua_pushboolean(L, 0);
|
lua_pushinteger(L, status);
|
||||||
return 1;
|
return 1;
|
||||||
} else if (status < 0) {
|
|
||||||
free(fds);
|
|
||||||
return nixio__perror(L);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < len; i++) {
|
for (i = 0; i < len; i++) {
|
||||||
|
|
Loading…
Reference in New Issue