2021-04-28 19:41:25 -07:00
|
|
|
#include "lua.hpp"
|
|
|
|
#include "lua_hooks.h"
|
|
|
|
|
2021-04-29 19:22:04 -07:00
|
|
|
#include "engine.h"
|
2021-04-28 19:41:25 -07:00
|
|
|
#include "loggingcentral.h"
|
|
|
|
#include "sensor.h"
|
2021-04-29 19:22:04 -07:00
|
|
|
#include "adc_inputs.h"
|
|
|
|
#include "efilib.h"
|
2021-05-04 07:05:39 -07:00
|
|
|
#include "tunerstudio_outputs.h"
|
2021-07-15 21:49:14 -07:00
|
|
|
#include "fuel_math.h"
|
|
|
|
#include "airmass.h"
|
|
|
|
#include "lua_airmass.h"
|
2021-05-20 16:01:17 -07:00
|
|
|
#include "pwm_generator_logic.h"
|
2021-04-29 19:22:04 -07:00
|
|
|
|
|
|
|
// Some functions lean on existing FSIO implementation
|
|
|
|
#include "fsio_impl.h"
|
|
|
|
|
|
|
|
EXTERN_ENGINE;
|
2021-04-28 19:41:25 -07:00
|
|
|
|
|
|
|
static int lua_efi_print(lua_State* l) {
|
|
|
|
auto msg = luaL_checkstring(l, 1);
|
|
|
|
|
|
|
|
efiPrintf("LUA: %s", msg);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-04-29 19:22:04 -07:00
|
|
|
static int lua_getSensor(lua_State* l) {
|
2021-04-28 19:41:25 -07:00
|
|
|
auto sensorIndex = luaL_checkinteger(l, 1);
|
|
|
|
|
|
|
|
auto result = Sensor::get(static_cast<SensorType>(sensorIndex));
|
|
|
|
|
|
|
|
if (result) {
|
|
|
|
// return value if valid
|
|
|
|
lua_pushnumber(l, result.Value);
|
|
|
|
} else {
|
|
|
|
// return nil if invalid
|
|
|
|
lua_pushnil(l);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2021-04-29 19:22:04 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-07-15 21:49:14 -07:00
|
|
|
static LuaAirmass luaAirmass;
|
|
|
|
|
|
|
|
AirmassModelBase& getLuaAirmassModel() {
|
|
|
|
return luaAirmass;
|
|
|
|
}
|
|
|
|
|
2021-04-29 19:22:04 -07:00
|
|
|
#if !EFI_UNIT_TEST
|
2021-05-20 16:01:17 -07:00
|
|
|
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, 2);
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2021-06-28 05:39:03 -07:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-20 16:01:17 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-04-29 19:22:04 -07:00
|
|
|
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;
|
|
|
|
}
|
2021-05-04 07:05:39 -07:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2021-05-29 14:54:36 -07:00
|
|
|
|
2021-07-15 21:49:14 -07:00
|
|
|
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::get(SensorType::Rpm).value_or(0);
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-06-03 11:30:25 -07:00
|
|
|
static int lua_stopEngine(lua_State*) {
|
2021-05-29 14:54:36 -07:00
|
|
|
doScheduleStopEngine();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2021-04-29 19:22:04 -07:00
|
|
|
#endif // EFI_UNIT_TEST
|
|
|
|
|
2021-04-28 19:41:25 -07:00
|
|
|
void configureRusefiLuaHooks(lua_State* l) {
|
|
|
|
lua_register(l, "print", lua_efi_print);
|
2021-04-29 19:22:04 -07:00
|
|
|
lua_register(l, "getSensor", lua_getSensor);
|
|
|
|
lua_register(l, "getSensorRaw", lua_getSensorRaw);
|
|
|
|
lua_register(l, "hasSensor", lua_hasSensor);
|
|
|
|
lua_register(l, "table3d", lua_table3d);
|
|
|
|
|
|
|
|
#if !EFI_UNIT_TEST
|
2021-05-20 16:01:17 -07:00
|
|
|
lua_register(l, "startPwm", lua_startPwm);
|
|
|
|
lua_register(l, "setPwmDuty", lua_setPwmDuty);
|
|
|
|
lua_register(l, "setPwmFreq", lua_setPwmFreq);
|
|
|
|
|
2021-04-29 19:22:04 -07:00
|
|
|
lua_register(l, "getFan", lua_fan);
|
|
|
|
lua_register(l, "getDigital", lua_getDigital);
|
2021-05-04 07:05:39 -07:00
|
|
|
lua_register(l, "setDebug", lua_setDebug);
|
2021-07-15 21:49:14 -07:00
|
|
|
lua_register(l, "getAirmass", lua_getAirmass);
|
|
|
|
lua_register(l, "setAirmass", lua_setAirmass);
|
2021-05-29 14:54:36 -07:00
|
|
|
|
|
|
|
lua_register(l, "stopEngine", lua_stopEngine);
|
2021-04-29 19:22:04 -07:00
|
|
|
#endif
|
2021-04-28 19:41:25 -07:00
|
|
|
}
|