2021-10-24 05:37:04 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2022-08-21 12:22:22 -07:00
|
|
|
#include "can_filter.h"
|
2021-10-24 05:37:04 -07:00
|
|
|
|
|
|
|
|
2022-08-21 12:22:22 -07:00
|
|
|
#if EFI_CAN_SUPPORT
|
2021-10-24 05:37:04 -07:00
|
|
|
|
2022-08-21 12:22:22 -07:00
|
|
|
#include "rusefi_lua.h"
|
2021-10-24 05:37:04 -07:00
|
|
|
|
2022-01-31 15:37:42 -08:00
|
|
|
// Stores information about one received CAN frame: which bus, plus the actual frame
|
|
|
|
struct CanFrameData {
|
|
|
|
uint8_t BusIndex;
|
2022-07-27 23:27:18 -07:00
|
|
|
int Callback;
|
2022-01-31 15:37:42 -08:00
|
|
|
CANRxFrame Frame;
|
|
|
|
};
|
|
|
|
|
2021-10-24 05:37:04 -07:00
|
|
|
constexpr size_t canFrameCount = 32;
|
2022-01-31 15:37:42 -08:00
|
|
|
static CanFrameData canFrames[canFrameCount];
|
2021-10-24 05:37:04 -07:00
|
|
|
// CAN frame buffers that are not in use
|
2022-01-31 15:37:42 -08:00
|
|
|
chibios_rt::Mailbox<CanFrameData*, canFrameCount> freeBuffers;
|
2021-10-24 05:37:04 -07:00
|
|
|
// CAN frame buffers that are waiting to be processed by the lua thread
|
2022-01-31 15:37:42 -08:00
|
|
|
chibios_rt::Mailbox<CanFrameData*, canFrameCount> filledBuffers;
|
2021-10-24 05:37:04 -07:00
|
|
|
|
2021-12-20 22:17:02 -08:00
|
|
|
void processLuaCan(const size_t busIndex, const CANRxFrame& frame) {
|
2022-08-21 12:22:22 -07:00
|
|
|
auto filter = getFilterForId(busIndex, CAN_ID(frame));
|
2022-07-27 23:27:18 -07:00
|
|
|
|
2021-10-24 05:37:04 -07:00
|
|
|
// Filter the frame if we aren't listening for it
|
2022-07-27 23:27:18 -07:00
|
|
|
if (!filter) {
|
2021-10-24 05:37:04 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-31 15:37:42 -08:00
|
|
|
CanFrameData* frameBuffer;
|
2021-10-24 05:37:04 -07:00
|
|
|
msg_t msg;
|
|
|
|
|
|
|
|
{
|
|
|
|
// Acquire a buffer under lock
|
|
|
|
chibios_rt::CriticalSectionLocker csl;
|
|
|
|
msg = freeBuffers.fetchI(&frameBuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg != MSG_OK) {
|
|
|
|
// all buffers are already in use, this frame will be dropped!
|
|
|
|
// TODO: warn the user
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy the frame in to the buffer
|
2022-01-31 15:37:42 -08:00
|
|
|
frameBuffer->BusIndex = busIndex;
|
|
|
|
frameBuffer->Frame = frame;
|
2022-07-27 23:27:18 -07:00
|
|
|
frameBuffer->Callback = filter->Callback;
|
2021-10-24 05:37:04 -07:00
|
|
|
|
|
|
|
{
|
|
|
|
// Push the frame in to the queue under lock
|
|
|
|
chibios_rt::CriticalSectionLocker csl;
|
|
|
|
filledBuffers.postI(frameBuffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-31 15:37:42 -08:00
|
|
|
static void handleCanFrame(LuaHandle& ls, CanFrameData* data) {
|
2022-08-21 19:46:49 -07:00
|
|
|
if (data->Callback == NO_CALLBACK) {
|
2022-07-27 23:27:18 -07:00
|
|
|
// No callback, use catch-all function
|
|
|
|
lua_getglobal(ls, "onCanRx");
|
|
|
|
} else {
|
|
|
|
// Push the specified callback on to the stack
|
|
|
|
lua_rawgeti(ls, LUA_REGISTRYINDEX, data->Callback);
|
|
|
|
}
|
|
|
|
|
2021-10-24 05:37:04 -07:00
|
|
|
if (lua_isnil(ls, -1)) {
|
|
|
|
// no rx function, ignore
|
|
|
|
efiPrintf("LUA CAN rx missing function onCanRx");
|
|
|
|
lua_settop(ls, 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-31 15:37:42 -08:00
|
|
|
auto dlc = data->Frame.DLC;
|
2021-10-26 13:48:57 -07:00
|
|
|
|
|
|
|
// Push bus, ID and DLC
|
2022-01-31 15:37:42 -08:00
|
|
|
lua_pushinteger(ls, data->BusIndex); // TODO: support multiple busses!
|
2022-07-05 10:37:31 -07:00
|
|
|
lua_pushinteger(ls, CAN_ID(data->Frame));
|
2021-10-26 13:48:57 -07:00
|
|
|
lua_pushinteger(ls, dlc);
|
|
|
|
|
|
|
|
// Build table for data
|
|
|
|
lua_newtable(ls);
|
|
|
|
for (size_t i = 0; i < dlc; i++) {
|
2022-01-31 15:37:42 -08:00
|
|
|
lua_pushinteger(ls, data->Frame.data8[i]);
|
2021-10-28 05:18:56 -07:00
|
|
|
|
|
|
|
// index is i+1 because Lua "arrays" (tables) are 1-indexed
|
2021-10-29 13:32:41 -07:00
|
|
|
lua_rawseti(ls, -2, i + 1);
|
2021-10-26 13:48:57 -07:00
|
|
|
}
|
2021-10-24 05:37:04 -07:00
|
|
|
|
2021-10-26 13:48:57 -07:00
|
|
|
// Perform the actual function call
|
|
|
|
int status = lua_pcall(ls, 4, 0, 0);
|
2021-10-24 05:37:04 -07:00
|
|
|
|
|
|
|
if (0 != status) {
|
|
|
|
// error calling CAN rx hook function
|
|
|
|
auto errMsg = lua_tostring(ls, -1);
|
|
|
|
efiPrintf("LUA CAN RX error %s", errMsg);
|
|
|
|
lua_pop(ls, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
lua_settop(ls, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool doOneLuaCanRx(LuaHandle& ls) {
|
2022-01-31 15:37:42 -08:00
|
|
|
CanFrameData* data;
|
2021-10-24 05:37:04 -07:00
|
|
|
|
2022-01-31 15:37:42 -08:00
|
|
|
msg_t msg = filledBuffers.fetch(&data, TIME_IMMEDIATE);
|
2021-10-24 05:37:04 -07:00
|
|
|
|
|
|
|
if (msg == MSG_TIMEOUT) {
|
|
|
|
// No new CAN messages rx'd, nothing more to do.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg != MSG_OK) {
|
|
|
|
// Message was otherwise not OK
|
|
|
|
// TODO: what do here?
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We've accepted the frame, process it in Lua.
|
2022-01-31 15:37:42 -08:00
|
|
|
handleCanFrame(ls, data);
|
2021-10-24 05:37:04 -07:00
|
|
|
|
|
|
|
// We're done, return this frame to the free list
|
2022-01-31 15:37:42 -08:00
|
|
|
msg = freeBuffers.post(data, TIME_IMMEDIATE);
|
2021-11-22 14:52:03 -08:00
|
|
|
efiAssert(OBD_PCM_Processor_Fault, msg == MSG_OK, "lua can post to free buffer fail", false);
|
2021-10-24 05:37:04 -07:00
|
|
|
|
|
|
|
// We processed a frame so we should check again
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void doLuaCanRx(LuaHandle& ls) {
|
|
|
|
// While it processed a frame, continue checking
|
|
|
|
while (doOneLuaCanRx(ls)) ;
|
|
|
|
}
|
|
|
|
|
|
|
|
void initLuaCanRx() {
|
|
|
|
// Push all CAN frames in to the free buffer
|
|
|
|
for (size_t i = 0; i < canFrameCount; i++) {
|
|
|
|
freeBuffers.post(&canFrames[i], TIME_INFINITE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // EFI_CAN_SUPPORT
|