AP_Scripting: added SVFFI generator support

This commit is contained in:
Andrew Tridgell 2023-08-14 18:15:38 +10:00
parent fb87369c7e
commit 96f7cc2d38
2 changed files with 281 additions and 0 deletions

View File

@ -0,0 +1,246 @@
--[[
SVFFI serial protocol for generator support
See http://www.svffi.com/en/
--]]
local PARAM_TABLE_KEY = 42
local PARAM_TABLE_PREFIX = "EFI_SVF_"
local MAV_SEVERITY = {EMERGENCY=0, ALERT=1, CRITICAL=2, ERROR=3, WARNING=4, NOTICE=5, INFO=6, DEBUG=7}
-- bind a parameter to a variable given
local function bind_param(name)
local p = Parameter()
assert(p:init(name), string.format('could not find %s parameter', name))
return p
end
-- add a parameter and bind it to a variable
local function bind_add_param(name, idx, default_value)
assert(param:add_param(PARAM_TABLE_KEY, idx, name, default_value), string.format('could not add param %s', name))
return bind_param(PARAM_TABLE_PREFIX .. name)
end
-- setup script specific parameters
assert(param:add_table(PARAM_TABLE_KEY, PARAM_TABLE_PREFIX, 8), 'could not add param table')
--[[
// @Param: EFI_SVF_ENABLE
// @DisplayName: Generator SVFFI enable
// @Description: Enable SVFFI generator support
// @Values: 0:Disabled,1:Enabled
// @User: Standard
--]]
EFI_SVF_ENABLE = bind_add_param("ENABLE", 1, 0)
--[[
// @Param: EFI_SVF_ARMCHECK
// @DisplayName: Generator SVFFI arming check
// @Description: Check for Generator ARM state before arming
// @Values: 0:Disabled,1:Enabled
// @User: Standard
--]]
EFI_SVF_ARMCHECK = bind_add_param("ARMCHECK", 2, 1)
if EFI_SVF_ENABLE:get() ~= 1 then
return
end
local auth_id = arming:get_aux_auth_id()
arming:set_aux_auth_failed(auth_id, "GEN: not in ARM state")
local uart = serial:find_serial(0) -- first scripting serial
if not uart then
gcs:send_text(MAV_SEVERITY.ERROR, "GEN_SVF: unable to find serial port")
return
end
uart:begin(115200)
local efi_backend = efi:get_backend(0)
if not efi_backend then
gcs:send_text(MAV_SEVERITY.ERROR, "GEN_SVF: unable to find EFI backend")
return
end
local function read_bytes(n)
local ret = ""
for _ = 1, n do
ret = ret .. string.char(uart:read())
end
return ret
end
local auchCRCHi = {
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40
}
local auchCRCLo = {
0x00, 0xC0, 0xC1, 0x01, 0xC3, 0x03, 0x02, 0xC2, 0xC6, 0x06, 0x07, 0xC7,
0x05, 0xC5, 0xC4, 0x04, 0xCC, 0x0C, 0x0D, 0xCD, 0x0F, 0xCF, 0xCE, 0x0E,
0x0A, 0xCA, 0xCB, 0x0B, 0xC9, 0x09, 0x08, 0xC8, 0xD8, 0x18, 0x19, 0xD9,
0x1B, 0xDB, 0xDA, 0x1A, 0x1E, 0xDE, 0xDF, 0x1F, 0xDD, 0x1D, 0x1C, 0xDC,
0x14, 0xD4, 0xD5, 0x15, 0xD7, 0x17, 0x16, 0xD6, 0xD2, 0x12, 0x13, 0xD3,
0x11, 0xD1, 0xD0, 0x10, 0xF0, 0x30, 0x31, 0xF1, 0x33, 0xF3, 0xF2, 0x32,
0x36, 0xF6, 0xF7, 0x37, 0xF5, 0x35, 0x34, 0xF4, 0x3C, 0xFC, 0xFD, 0x3D,
0xFF, 0x3F, 0x3E, 0xFE, 0xFA, 0x3A, 0x3B, 0xFB, 0x39, 0xF9, 0xF8, 0x38,
0x28, 0xE8, 0xE9, 0x29, 0xEB, 0x2B, 0x2A, 0xEA, 0xEE, 0x2E, 0x2F, 0xEF,
0x2D, 0xED, 0xEC, 0x2C, 0xE4, 0x24, 0x25, 0xE5, 0x27, 0xE7, 0xE6, 0x26,
0x22, 0xE2, 0xE3, 0x23, 0xE1, 0x21, 0x20, 0xE0, 0xA0, 0x60, 0x61, 0xA1,
0x63, 0xA3, 0xA2, 0x62, 0x66, 0xA6, 0xA7, 0x67, 0xA5, 0x65, 0x64, 0xA4,
0x6C, 0xAC, 0xAD, 0x6D, 0xAF, 0x6F, 0x6E, 0xAE, 0xAA, 0x6A, 0x6B, 0xAB,
0x69, 0xA9, 0xA8, 0x68, 0x78, 0xB8, 0xB9, 0x79, 0xBB, 0x7B, 0x7A, 0xBA,
0xBE, 0x7E, 0x7F, 0xBF, 0x7D, 0xBD, 0xBC, 0x7C, 0xB4, 0x74, 0x75, 0xB5,
0x77, 0xB7, 0xB6, 0x76, 0x72, 0xB2, 0xB3, 0x73, 0xB1, 0x71, 0x70, 0xB0,
0x50, 0x90, 0x91, 0x51, 0x93, 0x53, 0x52, 0x92, 0x96, 0x56, 0x57, 0x97,
0x55, 0x95, 0x94, 0x54, 0x9C, 0x5C, 0x5D, 0x9D, 0x5F, 0x9F, 0x9E, 0x5E,
0x5A, 0x9A, 0x9B, 0x5B, 0x99, 0x59, 0x58, 0x98, 0x88, 0x48, 0x49, 0x89,
0x4B, 0x8B, 0x8A, 0x4A, 0x4E, 0x8E, 0x8F, 0x4F, 0x8D, 0x4D, 0x4C, 0x8C,
0x44, 0x84, 0x85, 0x45, 0x87, 0x47, 0x46, 0x86, 0x82, 0x42, 0x43, 0x83,
0x41, 0x81, 0x80, 0x40
}
--[[
calculate crc16
--]]
local function get_crc16(s)
local uchCRCHi = 0xFF
local uchCRCLo = 0xFF
for i = 1, #s do
local b = string.byte(string.sub(s, i, i))
local uIndex = uchCRCLo ~ b
--gcs:send_text(MAV_SEVERITY.INFO, string.format("uIndex=%u", uIndex))
uchCRCLo = uchCRCHi ~ auchCRCHi[uIndex+1]
uchCRCHi = auchCRCLo[uIndex+1]
end
return (uchCRCHi << 8 | uchCRCLo)
end
local state = {}
state.last_read_us = uint32_t(0)
state.last_status = -1
--[[
check for input and parse data
--]]
local function check_input()
local n_bytes = uart:available():toint()
--gcs:send_text(MAV_SEVERITY.INFO, string.format("n_bytes=%u %.2f", n_bytes, millis():tofloat()*0.001))
if n_bytes < 31 then
return
end
local s = read_bytes(n_bytes)
local prefix, len = string.unpack("<HB", s, 1)
if prefix ~= 0xa55a then
gcs:send_text(MAV_SEVERITY.INFO, string.format("bad prefix 0x%x", prefix))
return
end
if len+5 ~= n_bytes then
gcs:send_text(MAV_SEVERITY.INFO, string.format("bad len %u %u", n_bytes, len))
return
end
local crc = string.unpack("<H", s, 4+len)
local s2 = string.sub(s,1,n_bytes-2)
--gcs:send_text(MAV_SEVERITY.INFO, string.format("s2 n_bytes=%u len1=%u len2=%u", n_bytes, #s, #s2))
local crc2 = get_crc16(s2)
if crc ~= crc2 then
gcs:send_text(MAV_SEVERITY.INFO, string.format("bad crc %x %x", crc, crc2))
return
end
state.version, state.rpm, state.throttle = string.unpack("<BHH", string.sub(s,4,8))
state.voltage, state.current, state.runtime = string.unpack("<HHI", string.sub(s,9,16))
state.maint_time, state.lock_time, state.status = string.unpack("<HHB", string.sub(s,17,21))
state.alarm, state.fuellevel, state.cht1 = string.unpack("<HBH", string.sub(s,22,26))
state.cht2, state.pcb_temp = string.unpack("<HB", string.sub(s,27,29))
if state.status ~= state.last_status then
state.last_status = state.status
local states = {"STOP", "IDLE", "RUN", "3", "CHARGE" }
local name = states[state.status+1]
if not name then
name = string.format("Unknown%u", state.status)
end
gcs:send_text(MAV_SEVERITY.WARNING, string.format("Generator state: %s", name))
if name ~= "RUN" and EFI_SVF_ARMCHECK:get() == 1 then
arming:set_aux_auth_failed(auth_id, string.format("GEN: not in ARM state (%s)", name))
else
arming:set_aux_auth_passed(auth_id)
end
end
state.last_read_us = micros()
end
--[[
update EFI state
--]]
local function update_EFI()
if state.last_read_us == uint32_t(0) then
return
end
local cylinder_state = Cylinder_Status()
local efi_state = EFI_State()
local C_TO_KELVIN = 273.2
cylinder_state:cylinder_head_temperature(state.cht1+C_TO_KELVIN)
efi_state:engine_speed_rpm(state.rpm)
efi_state:throttle_position_percent(state.throttle)
efi_state:ignition_voltage(state.voltage*0.1)
efi_state:cylinder_status(cylinder_state)
efi_state:last_updated_ms(millis())
-- Set the EFI_State into the EFI scripting driver
efi_backend:handle_scripting(efi_state)
gcs:send_named_float('GEN_VOLT', state.voltage*0.1)
gcs:send_named_float('GEN_AMPS', state.current*0.1)
gcs:send_named_float('GEN_STAT', state.status)
logger.write('SVF','Curr,Volt,Status', 'ffB',
state.current*0.1,
state.voltage*0.1,
state.status)
end
--[[
main update function
--]]
local function update()
check_input()
update_EFI()
return update, 100
end
gcs:send_text(MAV_SEVERITY.INFO, "GEN_SVF: loaded")
return update()

View File

@ -0,0 +1,35 @@
# SVFFI Generator Driver
This driver implements support for the SVFFI generator serial protocol for
this system http://www.svffi.com/en/
# Parameters
The script used the following parameters:
## EFI_SVF_ENABLE
this must be set to 1 to enable the driver
# Operation
This driver should be loaded by placing the lua script in the
APM/SCRIPTS directory on the microSD card, which can be done either
directly or via MAVFTP. The following key parameters should be set:
- SCR_ENABLE should be set to 1
- EFI_TYPE should be set to 7
- EFI_SVF_ENABLE should be set to 1
- SERIALn_PROTOCOL should be set to 28 for the connected serial port
- RPM_TYPE1 should be set to 3
then the flight controller should rebooted and parameters should be
refreshed.
Once loaded the GEN_SVF parameters will appear and should be set
according to the parameter list above.
The GCS will receive EFI_STATUS MAVLink messages which includes RPM,
cylinder head temperature and throttle position. It will also receive
GEN_FUEL and GEN_AMPS named value float messages which give generator
fuel level percentage and current.