541 lines
12 KiB
C++
541 lines
12 KiB
C++
#include "pch.h"
|
|
|
|
#include "rusefi_lua.h"
|
|
#include "lua_hooks.h"
|
|
|
|
#include "fuel_math.h"
|
|
#include "airmass.h"
|
|
#include "lua_airmass.h"
|
|
#include "can_msg_tx.h"
|
|
#include "settings.h"
|
|
#include <new>
|
|
#include "luaaa.hpp"
|
|
#include "lua_hooks_util.h"
|
|
using namespace luaaa;
|
|
|
|
// Some functions lean on existing FSIO implementation
|
|
#include "fsio_impl.h"
|
|
|
|
#define HUMAN_OFFSET 1
|
|
|
|
#if EFI_UNIT_TEST
|
|
Engine *engineForLuaUnitTests;
|
|
#endif
|
|
|
|
static int lua_readpin(lua_State* l) {
|
|
auto msg = luaL_checkstring(l, 1);
|
|
#if EFI_PROD_CODE
|
|
brain_pin_e pin = parseBrainPin(msg);
|
|
if (pin == GPIO_INVALID) {
|
|
lua_pushnil(l);
|
|
} else {
|
|
int physicalValue = palReadPad(getHwPort("read", pin), getHwPin("read", pin));
|
|
lua_pushnumber(l, physicalValue);
|
|
}
|
|
#endif
|
|
return 1;
|
|
}
|
|
|
|
static int getSensor(lua_State* l, SensorType type) {
|
|
auto result = Sensor::get(type);
|
|
|
|
if (result) {
|
|
// return value if valid
|
|
lua_pushnumber(l, result.Value);
|
|
} else {
|
|
// return nil if invalid
|
|
lua_pushnil(l);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int lua_getAuxAnalog(lua_State* l) {
|
|
// todo: shall we use HUMAN_INDEX since UI goes from 1 and Lua loves going from 1?
|
|
auto sensorIndex = luaL_checkinteger(l, 1);
|
|
|
|
auto type = static_cast<SensorType>(sensorIndex + static_cast<int>(SensorType::Aux1));
|
|
|
|
return getSensor(l, type);
|
|
}
|
|
|
|
static int lua_getSensorByIndex(lua_State* l) {
|
|
auto sensorIndex = luaL_checkinteger(l, 1);
|
|
|
|
return getSensor(l, static_cast<SensorType>(sensorIndex));
|
|
}
|
|
|
|
static int lua_getSensorByName(lua_State* l) {
|
|
auto sensorName = luaL_checklstring(l, 1, nullptr);
|
|
SensorType type = findSensorTypeByName(sensorName);
|
|
|
|
return getSensor(l, type);
|
|
}
|
|
|
|
static int lua_getSensorRaw(lua_State* l) {
|
|
auto sensorIndex = luaL_checkinteger(l, 1);
|
|
|
|
lua_pushnumber(l, Sensor::getRaw(static_cast<SensorType>(sensorIndex)));
|
|
return 1;
|
|
}
|
|
|
|
static int lua_hasSensor(lua_State* l) {
|
|
auto sensorIndex = luaL_checkinteger(l, 1);
|
|
|
|
lua_pushboolean(l, Sensor::hasSensor(static_cast<SensorType>(sensorIndex)));
|
|
return 1;
|
|
}
|
|
|
|
static int lua_table3d(lua_State* l) {
|
|
auto tableIdx = luaL_checkinteger(l, 1);
|
|
auto x = luaL_checknumber(l, 2);
|
|
auto y = luaL_checknumber(l, 3);
|
|
|
|
// index table, compute table lookup
|
|
auto result = getFSIOTable(tableIdx)->getValue(x, y);
|
|
|
|
lua_pushnumber(l, result);
|
|
return 1;
|
|
}
|
|
|
|
static int lua_curve2d(lua_State* l) {
|
|
// index starting from 1
|
|
auto curveIdx = luaL_checkinteger(l, 1);
|
|
auto x = luaL_checknumber(l, 2);
|
|
|
|
#if EFI_UNIT_TEST
|
|
Engine *engine = engineForLuaUnitTests;
|
|
EXPAND_Engine;
|
|
#endif
|
|
|
|
auto result = getCurveValue(curveIdx - HUMAN_OFFSET, x PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
lua_pushnumber(l, result);
|
|
return 1;
|
|
}
|
|
|
|
static int lua_findCurveIndex(lua_State* l) {
|
|
#if EFI_UNIT_TEST
|
|
Engine *engine = engineForLuaUnitTests;
|
|
EXPAND_Engine;
|
|
#endif
|
|
auto name = luaL_checklstring(l, 1, nullptr);
|
|
auto result = getCurveIndexByName(name PASS_ENGINE_PARAMETER_SUFFIX);
|
|
if (result == EFI_ERROR_CODE) {
|
|
lua_pushnil(l);
|
|
} else {
|
|
// TS counts curve from 1 so convert indexing here
|
|
lua_pushnumber(l, result + HUMAN_OFFSET);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int lua_txCan(lua_State* l) {
|
|
auto channel = luaL_checkinteger(l, 1);
|
|
// TODO: support multiple channels
|
|
luaL_argcheck(l, channel == 1, 1, "only channel 1 currently supported");
|
|
|
|
auto id = luaL_checkinteger(l, 2);
|
|
auto ext = luaL_checkinteger(l, 3);
|
|
|
|
// Check that ID is valid based on std vs. ext
|
|
if (ext == 0) {
|
|
luaL_argcheck(l, id <= 0x7FF, 2, "ID specified is greater than max std ID");
|
|
} else {
|
|
luaL_argcheck(l, id <= 0x1FFF'FFFF, 2, "ID specified is greater than max ext ID");
|
|
}
|
|
|
|
luaL_checktype(l, 4, LUA_TTABLE);
|
|
|
|
// conform ext parameter to true/false
|
|
CanTxMessage msg(id, 8, ext == 0 ? false : true);
|
|
|
|
// Unfortunately there is no way to inspect the length of a table,
|
|
// so we have to just iterate until we run out of numbers
|
|
uint8_t dlc = 0;
|
|
|
|
while (true) {
|
|
lua_pushnumber(l, dlc + 1);
|
|
auto elementType = lua_gettable(l, 4);
|
|
auto val = lua_tonumber(l, -1);
|
|
lua_pop(l, 1);
|
|
|
|
if (elementType == LUA_TNIL) {
|
|
// we're done, this is the end of the array.
|
|
break;
|
|
}
|
|
|
|
if (elementType != LUA_TNUMBER) {
|
|
// We're not at the end, but this isn't a number!
|
|
luaL_error(l, "Unexpected CAN data at position %d: %s", dlc, lua_tostring(l, -1));
|
|
}
|
|
|
|
// This element is valid, increment DLC
|
|
dlc++;
|
|
|
|
if (dlc > 8) {
|
|
luaL_error(l, "CAN frame length cannot be longer than 8");
|
|
}
|
|
|
|
msg[dlc - 1] = val;
|
|
}
|
|
|
|
msg.setDlc(dlc);
|
|
|
|
// no return value
|
|
return 0;
|
|
}
|
|
|
|
static LuaAirmass luaAirmass;
|
|
|
|
AirmassModelBase& getLuaAirmassModel() {
|
|
return luaAirmass;
|
|
}
|
|
|
|
#if !EFI_UNIT_TEST
|
|
static SimplePwm pwms[LUA_PWM_COUNT];
|
|
static OutputPin pins[LUA_PWM_COUNT];
|
|
|
|
struct P {
|
|
SimplePwm& pwm;
|
|
lua_Integer idx;
|
|
};
|
|
|
|
static P luaL_checkPwmIndex(lua_State* l, int pos) {
|
|
auto channel = luaL_checkinteger(l, pos);
|
|
|
|
// Ensure channel is valid
|
|
if (channel < 0 || channel >= FSIO_COMMAND_COUNT) {
|
|
luaL_error(l, "setPwmDuty invalid channel %d", channel);
|
|
}
|
|
|
|
return { pwms[channel], channel };
|
|
}
|
|
|
|
static int lua_startPwm(lua_State* l) {
|
|
auto p = luaL_checkPwmIndex(l, 1);
|
|
auto freq = luaL_checknumber(l, 2);
|
|
auto duty = luaL_checknumber(l, 3);
|
|
|
|
// clamp to 1..1000 hz
|
|
freq = clampF(1, freq, 1000);
|
|
|
|
startSimplePwmExt(
|
|
&p.pwm, "lua", &engine->executor,
|
|
CONFIG(luaOutputPins[p.idx]), &pins[p.idx],
|
|
freq, duty
|
|
);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void luaDeInitPins() {
|
|
// Simply de-init all pins - when the script runs again, they will be re-init'd
|
|
for (size_t i = 0; i < efi::size(pins); i++) {
|
|
pins[i].deInit();
|
|
}
|
|
}
|
|
|
|
static int lua_setPwmDuty(lua_State* l) {
|
|
auto p = luaL_checkPwmIndex(l, 1);
|
|
auto duty = luaL_checknumber(l, 2);
|
|
|
|
// clamp to 0..1
|
|
duty = clampF(0, duty, 1);
|
|
|
|
p.pwm.setSimplePwmDutyCycle(duty);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int lua_setPwmFreq(lua_State* l) {
|
|
auto p = luaL_checkPwmIndex(l, 1);
|
|
auto freq = luaL_checknumber(l, 2);
|
|
|
|
// clamp to 1..1000 hz
|
|
freq = clampF(1, freq, 1000);
|
|
|
|
p.pwm.setFrequency(freq);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int lua_fan(lua_State* l) {
|
|
lua_pushboolean(l, enginePins.fanRelay.getLogicValue());
|
|
return 1;
|
|
}
|
|
|
|
static int lua_getDigital(lua_State* l) {
|
|
auto idx = luaL_checkinteger(l, 1);
|
|
|
|
bool state = false;
|
|
|
|
switch (idx) {
|
|
case 0: state = engine->clutchDownState; break;
|
|
case 1: state = engine->clutchUpState; break;
|
|
case 2: state = engine->brakePedalState; break;
|
|
case 3: state = engine->acSwitchState; break;
|
|
default:
|
|
// Return nil to indicate invalid parameter
|
|
lua_pushnil(l);
|
|
return 1;
|
|
}
|
|
|
|
lua_pushboolean(l, state);
|
|
return 1;
|
|
}
|
|
|
|
static int lua_setDebug(lua_State* l) {
|
|
// wrong debug mode, ignore
|
|
if (CONFIG(debugMode) != DBG_LUA) {
|
|
return 0;
|
|
}
|
|
|
|
auto idx = luaL_checkinteger(l, 1);
|
|
auto val = luaL_checknumber(l, 2);
|
|
|
|
// invalid index, ignore
|
|
if (idx < 1 || idx > 7) {
|
|
return 0;
|
|
}
|
|
|
|
auto firstDebugField = &tsOutputChannels.debugFloatField1;
|
|
firstDebugField[idx - 1] = val;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static auto lua_getAirmassResolveMode(lua_State* l) {
|
|
if (lua_gettop(l) == 0) {
|
|
// zero args, return configured mode
|
|
return CONFIG(fuelAlgorithm);
|
|
} else {
|
|
return static_cast<engine_load_mode_e>(luaL_checkinteger(l, 1));
|
|
}
|
|
}
|
|
|
|
static int lua_getAirmass(lua_State* l) {
|
|
auto airmassMode = lua_getAirmassResolveMode(l);
|
|
auto airmass = getAirmassModel(airmassMode);
|
|
|
|
if (!airmass) {
|
|
return luaL_error(l, "null airmass");
|
|
}
|
|
|
|
auto rpm = Sensor::getOrZero(SensorType::Rpm);
|
|
auto result = airmass->getAirmass(rpm).CylinderAirmass;
|
|
|
|
lua_pushnumber(l, result);
|
|
return 1;
|
|
}
|
|
|
|
static int lua_setAirmass(lua_State* l) {
|
|
float airmass = luaL_checknumber(l, 1);
|
|
float engineLoadPercent = luaL_checknumber(l, 2);
|
|
|
|
airmass = clampF(0, airmass, 10);
|
|
engineLoadPercent = clampF(0, engineLoadPercent, 1000);
|
|
|
|
luaAirmass.setAirmass({airmass, engineLoadPercent});
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int lua_stopEngine(lua_State*) {
|
|
doScheduleStopEngine();
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int lua_setTimingAdd(lua_State* l) {
|
|
ENGINE(engineState).luaAdjustments.ignitionTimingAdd = luaL_checknumber(l, 1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int lua_setTimingMult(lua_State* l) {
|
|
ENGINE(engineState).luaAdjustments.ignitionTimingMult = luaL_checknumber(l, 1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int lua_setFuelAdd(lua_State* l) {
|
|
ENGINE(engineState).luaAdjustments.fuelAdd = luaL_checknumber(l, 1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int lua_setFuelMult(lua_State* l) {
|
|
ENGINE(engineState).luaAdjustments.fuelMult = luaL_checknumber(l, 1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#endif // EFI_UNIT_TEST
|
|
|
|
#if EFI_CAN_SUPPORT
|
|
static int lua_canRxAdd(lua_State* l) {
|
|
auto eid = luaL_checkinteger(l, 1);
|
|
addLuaCanRxFilter(eid);
|
|
|
|
return 0;
|
|
}
|
|
#endif // EFI_CAN_SUPPORT
|
|
|
|
struct LuaSensor : public StoredValueSensor {
|
|
LuaSensor() : LuaSensor("Invalid") { }
|
|
|
|
~LuaSensor() {
|
|
unregister();
|
|
}
|
|
|
|
LuaSensor(const char* name)
|
|
: StoredValueSensor(findSensorTypeByName(name), MS2NT(100))
|
|
{
|
|
Register();
|
|
}
|
|
|
|
void set(float value) {
|
|
setValidValue(value, getTimeNowNt());
|
|
}
|
|
|
|
void invalidate() {
|
|
StoredValueSensor::invalidate();
|
|
}
|
|
|
|
void showInfo(const char*) const {}
|
|
};
|
|
|
|
struct LuaPid {
|
|
LuaPid
|
|
()
|
|
// todo (float kp, float ki, float kd, float min, float max)
|
|
: m_pid(&m_params)
|
|
{
|
|
m_params.pFactor = 0;
|
|
m_params.iFactor = 0;
|
|
m_params.dFactor = 0;
|
|
|
|
m_params.offset = 0;
|
|
m_params.periodMs = 0;
|
|
m_params.minValue = 0;
|
|
m_params.maxValue = 0;
|
|
|
|
m_lastUpdate.reset();
|
|
}
|
|
|
|
float get(float input) {
|
|
#if EFI_UNIT_TEST
|
|
extern int timeNowUs;
|
|
// this is how we avoid zero dt
|
|
timeNowUs += 1000;
|
|
#endif
|
|
float dt = m_lastUpdate.getElapsedSecondsAndReset(getTimeNowNt());
|
|
|
|
return m_pid.getOutput(target, input, dt);
|
|
}
|
|
|
|
void setTarget(float value) {
|
|
target = value;
|
|
}
|
|
|
|
void setP(float value) {
|
|
m_params.pFactor = value;
|
|
}
|
|
|
|
void setI(float value) {
|
|
m_params.iFactor = value;
|
|
}
|
|
|
|
void setD(float value) {
|
|
m_params.dFactor = value;
|
|
}
|
|
|
|
void setMinValue(float value) {
|
|
m_params.minValue = value;
|
|
}
|
|
|
|
void setMaxValue(float value) {
|
|
m_params.maxValue = value;
|
|
}
|
|
|
|
void reset() {
|
|
m_pid.reset();
|
|
}
|
|
|
|
private:
|
|
Pid m_pid;
|
|
Timer m_lastUpdate;
|
|
pid_s m_params;
|
|
// ugly as hell, a way to move forward while we wait for https://github.com/gengyong/luaaa/issues/7
|
|
float target;
|
|
};
|
|
|
|
void configureRusefiLuaHooks(lua_State* l) {
|
|
|
|
LuaClass<Timer> luaTimer(l, "Timer");
|
|
luaTimer
|
|
.ctor()
|
|
.fun("reset", static_cast<void (Timer::*)() >(&Timer::reset ))
|
|
.fun("getElapsedSeconds", static_cast<float(Timer::*)()const>(&Timer::getElapsedSeconds));
|
|
|
|
LuaClass<LuaSensor> luaSensor(l, "Sensor");
|
|
luaSensor
|
|
.ctor<const char*>()
|
|
.fun("set", &LuaSensor::set)
|
|
.fun("invalidate", &LuaSensor::invalidate);
|
|
|
|
// not enough Lua memory even to initialize Lua :(
|
|
#if defined(STM32F7) || defined(STM32H7) || EFI_UNIT_TEST
|
|
LuaClass<LuaPid> luaPid(l, "Pid");
|
|
luaPid
|
|
.ctor()
|
|
.fun("get", &LuaPid::get)
|
|
.fun("setTarget", &LuaPid::setTarget)
|
|
.fun("setP", &LuaPid::setP)
|
|
.fun("setI", &LuaPid::setI)
|
|
.fun("setD", &LuaPid::setD)
|
|
.fun("setMinValue", &LuaPid::setMinValue)
|
|
.fun("setMaxValue", &LuaPid::setMaxValue)
|
|
.fun("reset", &LuaPid::reset)
|
|
;
|
|
#endif
|
|
|
|
configureRusefiLuaUtilHooks(l);
|
|
|
|
lua_register(l, "readPin", lua_readpin);
|
|
lua_register(l, "getAuxAnalog", lua_getAuxAnalog);
|
|
lua_register(l, "getSensorByIndex", lua_getSensorByIndex);
|
|
lua_register(l, "getSensor", lua_getSensorByName);
|
|
lua_register(l, "getSensorRaw", lua_getSensorRaw);
|
|
lua_register(l, "hasSensor", lua_hasSensor);
|
|
lua_register(l, "table3d", lua_table3d);
|
|
lua_register(l, "curve", lua_curve2d);
|
|
lua_register(l, "findCurveIndex", lua_findCurveIndex);
|
|
lua_register(l, "txCan", lua_txCan);
|
|
|
|
#if !EFI_UNIT_TEST
|
|
lua_register(l, "startPwm", lua_startPwm);
|
|
lua_register(l, "setPwmDuty", lua_setPwmDuty);
|
|
lua_register(l, "setPwmFreq", lua_setPwmFreq);
|
|
|
|
lua_register(l, "getFan", lua_fan);
|
|
lua_register(l, "getDigital", lua_getDigital);
|
|
lua_register(l, "setDebug", lua_setDebug);
|
|
lua_register(l, "getAirmass", lua_getAirmass);
|
|
lua_register(l, "setAirmass", lua_setAirmass);
|
|
|
|
lua_register(l, "stopEngine", lua_stopEngine);
|
|
|
|
lua_register(l, "setTimingAdd", lua_setTimingAdd);
|
|
lua_register(l, "setTimingMult", lua_setTimingMult);
|
|
|
|
lua_register(l, "setFuelAdd", lua_setFuelAdd);
|
|
lua_register(l, "setFuelMult", lua_setFuelMult);
|
|
|
|
#if EFI_CAN_SUPPORT
|
|
lua_register(l, "canRxAdd", lua_canRxAdd);
|
|
#endif // EFI_CAN_SUPPORT
|
|
#endif // not EFI_UNIT_TEST
|
|
}
|