2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
2020-12-16 05:28:53 -08:00
|
|
|
|
2021-03-09 15:54:01 -08:00
|
|
|
#if EFI_WIDEBAND_FIRMWARE_UPDATE && EFI_CAN_SUPPORT
|
2020-12-16 05:28:53 -08:00
|
|
|
|
|
|
|
#include "ch.h"
|
|
|
|
#include "can_msg_tx.h"
|
2021-07-13 04:32:41 -07:00
|
|
|
#include "rusefi_wideband.h"
|
2020-12-16 05:28:53 -08:00
|
|
|
|
|
|
|
// This file contains an array called build_wideband_noboot_bin
|
|
|
|
// This array contains the firmware image for the wideband contoller
|
|
|
|
#include "wideband_firmware/for_rusefi/wideband_image.h"
|
|
|
|
|
|
|
|
#define EVT_BOOTLOADER_ACK EVENT_MASK(0)
|
|
|
|
|
|
|
|
static thread_t* waitingBootloaderThread = nullptr;
|
|
|
|
|
|
|
|
void handleWidebandBootloaderAck() {
|
|
|
|
auto t = waitingBootloaderThread;
|
|
|
|
if (t) {
|
|
|
|
chEvtSignal(t, EVT_BOOTLOADER_ACK);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool waitAck() {
|
|
|
|
return chEvtWaitAnyTimeout(EVT_BOOTLOADER_ACK, TIME_MS2I(1000)) != 0;
|
|
|
|
}
|
|
|
|
|
2021-04-18 17:02:32 -07:00
|
|
|
void updateWidebandFirmware() {
|
2020-12-16 05:28:53 -08:00
|
|
|
// Clear any pending acks for this thread
|
|
|
|
chEvtGetAndClearEvents(EVT_BOOTLOADER_ACK);
|
|
|
|
|
|
|
|
// Send messages to the current thread when acks come in
|
|
|
|
waitingBootloaderThread = chThdGetSelfX();
|
|
|
|
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("***************************************");
|
|
|
|
efiPrintf(" WIDEBAND FIRMWARE UPDATE");
|
|
|
|
efiPrintf("***************************************");
|
|
|
|
efiPrintf("Wideband Update: Rebooting to bootloader...");
|
2020-12-16 05:28:53 -08:00
|
|
|
|
|
|
|
// The first request will reboot the chip (if necessary), and the second one will enable bootloader mode
|
|
|
|
// If the chip was already in bootloader (aka manual mode), then that's ok - the second request will
|
|
|
|
// just be safely ignored (but acked)
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
{
|
|
|
|
// Send bootloader entry command
|
|
|
|
CanTxMessage m(0xEF0'0000, 0, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!waitAck()) {
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("Wideband Update ERROR: Expected ACK from entry to bootloader, didn't get one.");
|
2020-12-16 05:28:53 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Let the controller reboot (and show blinky lights for a second before the update begins)
|
|
|
|
chThdSleepMilliseconds(200);
|
|
|
|
}
|
|
|
|
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("Wideband Update: in update mode, erasing flash...");
|
2020-12-16 05:28:53 -08:00
|
|
|
|
|
|
|
{
|
|
|
|
// Erase flash - opcode 1, magic value 0x5A5A
|
|
|
|
CanTxMessage m(0xEF1'5A5A, 0, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!waitAck()) {
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("Wideband Update ERROR: Expected ACK from flash erase command, didn't get one.");
|
2020-12-16 05:28:53 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t totalSize = sizeof(build_wideband_image_bin);
|
|
|
|
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("Wideband Update: Flash erased! Sending %d bytes...", totalSize);
|
2020-12-16 05:28:53 -08:00
|
|
|
|
|
|
|
// Send flash data 8 bytes at a time
|
|
|
|
for (size_t i = 0; i < totalSize; i += 8) {
|
|
|
|
{
|
|
|
|
CanTxMessage m(0xEF2'0000 + i, 8, true);
|
|
|
|
memcpy(&m[0], build_wideband_image_bin + i, 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!waitAck()) {
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("Wideband Update ERROR: Expected ACK from data write, didn't get one.");
|
2020-12-16 05:28:53 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("Wideband Update: Update complete! Rebooting controller.");
|
2020-12-16 05:28:53 -08:00
|
|
|
|
|
|
|
{
|
|
|
|
// Reboot to firmware!
|
|
|
|
CanTxMessage m(0xEF3'0000, 0, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
waitAck();
|
|
|
|
|
|
|
|
waitingBootloaderThread = nullptr;
|
|
|
|
}
|
|
|
|
|
2021-04-18 17:02:32 -07:00
|
|
|
void setWidebandOffset(uint8_t index) {
|
2021-03-14 14:20:50 -07:00
|
|
|
// Clear any pending acks for this thread
|
|
|
|
chEvtGetAndClearEvents(EVT_BOOTLOADER_ACK);
|
|
|
|
|
|
|
|
// Send messages to the current thread when acks come in
|
|
|
|
waitingBootloaderThread = chThdGetSelfX();
|
|
|
|
|
2021-04-18 17:02:32 -07:00
|
|
|
efiPrintf("***************************************");
|
|
|
|
efiPrintf(" WIDEBAND INDEX SET");
|
|
|
|
efiPrintf("***************************************");
|
|
|
|
efiPrintf("Setting all connected widebands to index %d...", index);
|
2021-03-14 14:20:50 -07:00
|
|
|
|
|
|
|
{
|
|
|
|
CanTxMessage m(0xEF4'0000, 1, true);
|
|
|
|
m[0] = index;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!waitAck()) {
|
|
|
|
firmwareError(OBD_PCM_Processor_Fault, "Wideband index set failed: no controller detected!");
|
|
|
|
}
|
|
|
|
|
|
|
|
waitingBootloaderThread = nullptr;
|
|
|
|
}
|
|
|
|
|
2021-07-13 04:32:41 -07:00
|
|
|
void sendWidebandInfo() {
|
|
|
|
CanTxMessage m(0xEF5'0000, 2, true);
|
|
|
|
|
2021-10-05 16:59:07 -07:00
|
|
|
float vbatt = Sensor::getOrZero(SensorType::BatteryVoltage) * 10;
|
2021-07-13 04:32:41 -07:00
|
|
|
|
|
|
|
m[0] = vbatt;
|
|
|
|
|
2021-07-16 11:06:52 -07:00
|
|
|
// Offset 1 bit 0 = heater enable
|
|
|
|
m[1] = enginePins.o2heater.getLogicValue() ? 0x01 : 0x00;
|
2021-07-13 04:32:41 -07:00
|
|
|
}
|
|
|
|
|
2021-02-08 13:24:38 -08:00
|
|
|
#endif // EFI_WIDEBAND_FIRMWARE_UPDATE && HAL_USE_CAN
|