2017-03-24 10:41:36 -07:00
|
|
|
/**
|
|
|
|
* @file status_loop.cpp
|
|
|
|
* @brief Human-readable protocol status messages
|
|
|
|
*
|
2019-05-02 14:52:48 -07:00
|
|
|
* http://rusefi.com/forum/viewtopic.php?t=263 rusEfi console overview
|
2017-03-24 10:41:36 -07:00
|
|
|
* http://rusefi.com/forum/viewtopic.php?t=210 Commands overview
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @date Mar 15, 2013
|
2020-01-13 18:57:43 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2017-03-24 10:41:36 -07:00
|
|
|
*
|
|
|
|
* This file is part of rusEfi - see http://rusefi.com
|
|
|
|
*
|
|
|
|
* rusEfi is free software; you can redistribute it and/or modify it under the terms of
|
|
|
|
* the GNU General Public License as published by the Free Software Foundation; either
|
|
|
|
* version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* rusEfi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
|
|
|
|
* even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with this program.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
2017-03-24 10:41:36 -07:00
|
|
|
#include "status_loop.h"
|
2019-04-01 14:04:49 -07:00
|
|
|
#include "hip9011_logic.h"
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-12-03 22:11:10 -08:00
|
|
|
#if EFI_LOGIC_ANALYZER
|
|
|
|
#include "logic_analyzer.h"
|
|
|
|
#endif /* EFI_LOGIC_ANALYZER */
|
2017-03-24 10:41:36 -07:00
|
|
|
|
|
|
|
#include "trigger_central.h"
|
2019-09-21 11:33:38 -07:00
|
|
|
#include "sensor_reader.h"
|
2017-03-24 10:41:36 -07:00
|
|
|
#include "mmc_card.h"
|
|
|
|
#include "console_io.h"
|
|
|
|
#include "malfunction_central.h"
|
|
|
|
#include "speed_density.h"
|
|
|
|
|
|
|
|
#include "advance_map.h"
|
|
|
|
#include "tunerstudio.h"
|
|
|
|
#include "fuel_math.h"
|
|
|
|
#include "main_trigger_callback.h"
|
|
|
|
#include "spark_logic.h"
|
|
|
|
#include "idle_thread.h"
|
|
|
|
#include "svnversion.h"
|
|
|
|
#include "lcd_controller.h"
|
2018-01-29 16:41:39 -08:00
|
|
|
#include "can_hw.h"
|
2019-07-08 00:35:41 -07:00
|
|
|
#include "periodic_thread_controller.h"
|
2019-01-03 21:16:08 -08:00
|
|
|
#include "cdm_ion_sense.h"
|
2020-05-21 12:52:58 -07:00
|
|
|
#include "binary_logging.h"
|
2020-10-14 17:06:05 -07:00
|
|
|
#include "buffered_writer.h"
|
2020-12-04 17:28:48 -08:00
|
|
|
#include "dynoview.h"
|
2022-05-14 05:39:08 -07:00
|
|
|
#include "frequency_sensor.h"
|
2022-08-23 16:45:28 -07:00
|
|
|
#include "digital_input_exti.h"
|
2017-03-24 10:41:36 -07:00
|
|
|
|
|
|
|
extern bool main_loop_started;
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_PROD_CODE
|
2017-03-24 10:41:36 -07:00
|
|
|
// todo: move this logic to algo folder!
|
|
|
|
#include "rtc_helper.h"
|
2021-05-16 03:01:00 -07:00
|
|
|
#include "HD44780.h"
|
2017-03-24 10:41:36 -07:00
|
|
|
#include "rusefi.h"
|
|
|
|
#include "pin_repository.h"
|
|
|
|
#include "max31855.h"
|
2019-03-29 06:11:13 -07:00
|
|
|
#include "single_timer_executor.h"
|
2019-07-13 07:24:23 -07:00
|
|
|
#include "periodic_task.h"
|
2017-12-16 21:17:55 -08:00
|
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
|
2022-09-23 05:47:50 -07:00
|
|
|
#if EFI_INTERNAL_FLASH
|
|
|
|
#include "flash_main.h"
|
|
|
|
#endif
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_CJ125
|
2019-04-01 14:04:49 -07:00
|
|
|
#include "cj125.h"
|
2018-11-03 08:44:57 -07:00
|
|
|
#endif /* EFI_CJ125 */
|
|
|
|
|
2018-02-06 13:21:41 -08:00
|
|
|
#if EFI_MAP_AVERAGING
|
|
|
|
#include "map_averaging.h"
|
|
|
|
#endif
|
|
|
|
|
2019-10-31 13:06:34 -07:00
|
|
|
#if (BOARD_TLE8888_COUNT > 0)
|
|
|
|
#include "tle8888.h"
|
|
|
|
#endif /* BOARD_TLE8888_COUNT */
|
|
|
|
|
2019-07-13 07:36:31 -07:00
|
|
|
#if EFI_ENGINE_SNIFFER
|
|
|
|
#include "engine_sniffer.h"
|
|
|
|
extern WaveChart waveChart;
|
|
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
|
|
|
|
2021-04-12 11:05:52 -07:00
|
|
|
#include "sensor_chart.h"
|
|
|
|
|
2021-02-06 09:22:57 -08:00
|
|
|
extern pin_output_mode_e DEFAULT_OUTPUT;
|
|
|
|
extern pin_output_mode_e INVERTED_OUTPUT;
|
|
|
|
|
|
|
|
#ifndef LED_WARNING_BRAIN_PIN_MODE
|
|
|
|
#define LED_WARNING_BRAIN_PIN_MODE DEFAULT_OUTPUT
|
|
|
|
#endif
|
|
|
|
#ifndef LED_RUNING_BRAIN_PIN_MODE
|
|
|
|
#define LED_RUNING_BRAIN_PIN_MODE DEFAULT_OUTPUT
|
|
|
|
#endif
|
2021-02-17 07:18:46 -08:00
|
|
|
#ifndef LED_COMMUNICATION_BRAIN_PIN_MODE
|
|
|
|
#define LED_COMMUNICATION_BRAIN_PIN_MODE DEFAULT_OUTPUT
|
|
|
|
#endif
|
2021-02-06 09:22:57 -08:00
|
|
|
|
2017-03-24 10:41:36 -07:00
|
|
|
int warningEnabled = true;
|
|
|
|
|
2021-12-01 22:06:40 -08:00
|
|
|
extern int maxTriggerReentrant;
|
2018-09-17 18:42:04 -07:00
|
|
|
extern uint32_t maxLockedDuration;
|
2019-12-21 17:35:13 -08:00
|
|
|
|
2017-03-24 10:41:36 -07:00
|
|
|
static void setWarningEnabled(int value) {
|
|
|
|
warningEnabled = value;
|
|
|
|
}
|
|
|
|
|
2021-11-23 08:42:45 -08:00
|
|
|
/**
|
|
|
|
* This is useful if we are changing engine mode dynamically
|
|
|
|
* For example http://rusefi.com/forum/viewtopic.php?f=5&t=1085
|
|
|
|
*/
|
|
|
|
static int packEngineMode() {
|
|
|
|
return (engineConfiguration->fuelAlgorithm << 4) +
|
|
|
|
(engineConfiguration->injectionMode << 2) +
|
|
|
|
engineConfiguration->ignitionMode;
|
|
|
|
}
|
|
|
|
|
2017-03-24 10:41:36 -07:00
|
|
|
static int prevCkpEventCounter = -1;
|
|
|
|
|
|
|
|
/**
|
2022-02-09 03:53:16 -08:00
|
|
|
* Time when the firmware version was last reported
|
2017-03-24 10:41:36 -07:00
|
|
|
* TODO: implement a request/response instead of just constantly sending this out
|
|
|
|
*/
|
2022-02-09 03:53:16 -08:00
|
|
|
static Timer printVersionTimer;
|
|
|
|
|
|
|
|
static void printRusefiVersion(const char *engineTypeName, const char *firmwareBuildId) {
|
|
|
|
// VersionChecker in rusEFI console is parsing these version string, please follow the expected format
|
|
|
|
efiPrintfProto(PROTOCOL_VERSION_TAG, "%d@%s %s %s %d",
|
|
|
|
getRusEfiVersion(), VCS_VERSION,
|
|
|
|
firmwareBuildId,
|
|
|
|
engineTypeName,
|
2022-09-11 10:06:03 -07:00
|
|
|
getTimeNowS());
|
2022-02-09 03:53:16 -08:00
|
|
|
}
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2022-02-09 03:53:16 -08:00
|
|
|
// Inform the console about the mapping between a pin's logical name (for example, injector 3)
|
|
|
|
// and the physical MCU pin backing that function (for example, PE3)
|
2017-03-24 10:41:36 -07:00
|
|
|
static void printOutPin(const char *pinName, brain_pin_e hwPin) {
|
2022-04-28 14:32:39 -07:00
|
|
|
if (hwPin == Gpio::Unassigned || hwPin == Gpio::Invalid) {
|
2022-01-01 09:10:40 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const char *hwPinName;
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(hwPin)) {
|
2022-01-01 09:10:40 -08:00
|
|
|
hwPinName = hwPortname(hwPin);
|
|
|
|
} else {
|
|
|
|
hwPinName = "smart";
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
2019-07-13 07:36:31 -07:00
|
|
|
|
2022-02-09 03:53:16 -08:00
|
|
|
efiPrintfProto(PROTOCOL_OUTPIN, "%s@%s", pinName, hwPinName);
|
|
|
|
}
|
2021-04-12 11:05:52 -07:00
|
|
|
|
2022-02-09 03:53:16 -08:00
|
|
|
// Print out the current mapping between logical and physical pins that
|
|
|
|
// the engine sniffer cares about, so we can display a physical pin
|
|
|
|
// in each engine sniffer row
|
|
|
|
static void printEngineSnifferPinMappings() {
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_PROD_CODE
|
2021-11-17 00:54:21 -08:00
|
|
|
printOutPin(PROTOCOL_CRANK1, engineConfiguration->triggerInputPins[0]);
|
|
|
|
printOutPin(PROTOCOL_CRANK2, engineConfiguration->triggerInputPins[1]);
|
2021-04-25 06:18:35 -07:00
|
|
|
for (int i = 0;i<CAM_INPUTS_COUNT;i++) {
|
|
|
|
extern const char *vvtNames[];
|
|
|
|
printOutPin(vvtNames[i], engineConfiguration->camInputs[i]);
|
|
|
|
}
|
2021-11-17 00:54:21 -08:00
|
|
|
printOutPin(PROTOCOL_HIP_NAME, engineConfiguration->hip9011IntHoldPin);
|
|
|
|
printOutPin(PROTOCOL_TACH_NAME, engineConfiguration->tachOutputPin);
|
2019-12-03 22:11:10 -08:00
|
|
|
#if EFI_LOGIC_ANALYZER
|
2021-11-17 00:54:21 -08:00
|
|
|
printOutPin(PROTOCOL_WA_CHANNEL_1, engineConfiguration->logicAnalyzerPins[0]);
|
|
|
|
printOutPin(PROTOCOL_WA_CHANNEL_2, engineConfiguration->logicAnalyzerPins[1]);
|
2019-12-03 22:11:10 -08:00
|
|
|
#endif /* EFI_LOGIC_ANALYZER */
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2021-11-17 00:54:21 -08:00
|
|
|
int cylCount = minI(engineConfiguration->specs.cylindersCount, MAX_CYLINDER_COUNT);
|
2021-03-17 06:24:13 -07:00
|
|
|
for (int i = 0; i < cylCount; i++) {
|
2021-11-17 00:54:21 -08:00
|
|
|
printOutPin(enginePins.coils[i].getShortName(), engineConfiguration->ignitionPins[i]);
|
|
|
|
printOutPin(enginePins.trailingCoils[i].getShortName(), engineConfiguration->trailingCoilPins[i]);
|
|
|
|
printOutPin(enginePins.injectors[i].getShortName(), engineConfiguration->injectionPins[i]);
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
2017-11-27 18:49:58 -08:00
|
|
|
for (int i = 0; i < AUX_DIGITAL_VALVE_COUNT;i++) {
|
2019-08-18 12:27:10 -07:00
|
|
|
printOutPin(enginePins.auxValve[i].getShortName(), engineConfiguration->auxValves[i]);
|
2017-11-27 18:49:58 -08:00
|
|
|
}
|
2017-03-24 10:41:36 -07:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2022-02-09 03:53:16 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void printOverallStatus() {
|
|
|
|
#if EFI_ENGINE_SNIFFER
|
|
|
|
waveChart.publishIfFull();
|
|
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
|
|
|
|
|
|
|
#if EFI_SENSOR_CHART
|
|
|
|
publishSensorChartIfFull();
|
|
|
|
#endif // EFI_SENSOR_CHART
|
|
|
|
|
|
|
|
/**
|
|
|
|
* we report the version every second - this way the console does not need to
|
|
|
|
* request it and we will display it pretty soon
|
|
|
|
*/
|
|
|
|
if (printVersionTimer.hasElapsedSec(1)) {
|
|
|
|
// we're sending, reset the timer
|
|
|
|
printVersionTimer.reset();
|
|
|
|
|
|
|
|
// Output the firmware version, board type, git hash, uptime in seconds, etc
|
|
|
|
printRusefiVersion(getEngine_type_e(engineConfiguration->engineType), FIRMWARE_ID);
|
|
|
|
|
|
|
|
// Output the current engine sniffer pin mappings
|
|
|
|
printEngineSnifferPinMappings();
|
|
|
|
}
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static systime_t timeOfPreviousReport = (systime_t) -1;
|
|
|
|
|
2022-02-09 03:53:16 -08:00
|
|
|
#if !defined(LOGIC_ANALYZER_BUFFER_SIZE)
|
|
|
|
// TODO: how small can this be?
|
|
|
|
#define LOGIC_ANALYZER_BUFFER_SIZE 1000
|
|
|
|
#endif /* LOGIC_ANALYZER_BUFFER_SIZE */
|
|
|
|
|
|
|
|
#if EFI_LOGIC_ANALYZER
|
|
|
|
static char logicAnalyzerBuffer[LOGIC_ANALYZER_BUFFER_SIZE];
|
|
|
|
static Logging logicAnalyzerLogger("logic analyzer", logicAnalyzerBuffer, sizeof(logicAnalyzerBuffer));
|
|
|
|
#endif // EFI_LOGIC_ANALYZER
|
|
|
|
|
2017-03-24 10:41:36 -07:00
|
|
|
/**
|
2019-05-02 14:52:48 -07:00
|
|
|
* @brief Sends all pending data to rusEfi console
|
2018-04-01 20:29:00 -07:00
|
|
|
*
|
|
|
|
* This method is periodically invoked by the main loop
|
2020-06-13 19:46:10 -07:00
|
|
|
* todo: is this mostly dead code?
|
2017-03-24 10:41:36 -07:00
|
|
|
*/
|
2017-04-09 19:26:11 -07:00
|
|
|
void updateDevConsoleState(void) {
|
2018-04-01 20:29:00 -07:00
|
|
|
// todo: make SWO work
|
|
|
|
// char *msg = "hello\r\n";
|
|
|
|
// for(int i=0;i<strlen(msg);i++) {
|
|
|
|
// ITM_SendChar(msg[i]);
|
|
|
|
// }
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_PROD_CODE
|
2017-03-24 10:41:36 -07:00
|
|
|
// todo: unify with simulator!
|
|
|
|
if (hasFirmwareError()) {
|
2022-04-16 14:35:59 -07:00
|
|
|
efiPrintf("%s error: %s", CRITICAL_PREFIX, getCriticalErrorMessage());
|
2017-03-24 10:41:36 -07:00
|
|
|
warningEnabled = false;
|
|
|
|
return;
|
|
|
|
}
|
2019-06-16 14:13:11 -07:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-06-16 14:13:11 -07:00
|
|
|
#if HAL_USE_ADC
|
2021-05-03 13:44:19 -07:00
|
|
|
printFullAdcReportIfNeeded();
|
2019-06-16 14:13:11 -07:00
|
|
|
#endif /* HAL_USE_ADC */
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2022-09-11 10:06:03 -07:00
|
|
|
systime_t nowSeconds = getTimeNowS();
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT
|
2019-02-02 23:47:20 -08:00
|
|
|
int currentCkpEventCounter = engine->triggerCentral.triggerState.getTotalEventCounter();
|
2017-03-24 10:41:36 -07:00
|
|
|
if (prevCkpEventCounter == currentCkpEventCounter && timeOfPreviousReport == nowSeconds) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
timeOfPreviousReport = nowSeconds;
|
|
|
|
|
|
|
|
prevCkpEventCounter = currentCkpEventCounter;
|
|
|
|
#else
|
|
|
|
chThdSleepMilliseconds(200);
|
|
|
|
#endif
|
|
|
|
|
2019-12-03 22:11:10 -08:00
|
|
|
#if EFI_LOGIC_ANALYZER
|
2022-02-09 03:53:16 -08:00
|
|
|
printWave(&logicAnalyzerLogger);
|
|
|
|
scheduleLogging(&logicAnalyzerLogger);
|
2019-12-03 22:11:10 -08:00
|
|
|
#endif /* EFI_LOGIC_ANALYZER */
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
|
|
|
|
2019-11-05 17:07:55 -08:00
|
|
|
static OutputPin *leds[] = { &enginePins.warningLedPin, &enginePins.runningLedPin,
|
2018-07-26 12:51:06 -07:00
|
|
|
&enginePins.errorLedPin, &enginePins.communicationLedPin, &enginePins.checkEnginePin };
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2021-11-15 04:02:34 -08:00
|
|
|
static void initStatusLeds() {
|
2021-07-20 14:09:52 -07:00
|
|
|
enginePins.communicationLedPin.initPin("led: comm status", engineConfiguration->communicationLedPin, &LED_COMMUNICATION_BRAIN_PIN_MODE, true);
|
2020-11-05 15:47:45 -08:00
|
|
|
// checkEnginePin is already initialized by the time we get here
|
2017-04-21 14:50:28 -07:00
|
|
|
|
2021-07-20 14:09:52 -07:00
|
|
|
enginePins.warningLedPin.initPin("led: warning status", engineConfiguration->warningLedPin, &LED_WARNING_BRAIN_PIN_MODE, true);
|
|
|
|
enginePins.runningLedPin.initPin("led: running status", engineConfiguration->runningLedPin, &LED_RUNING_BRAIN_PIN_MODE, true);
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_PROD_CODE
|
2017-03-24 10:41:36 -07:00
|
|
|
|
|
|
|
static bool isTriggerErrorNow() {
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT
|
2022-05-10 13:55:28 -07:00
|
|
|
bool justHadError = engine->triggerCentral.triggerState.someSortOfTriggerError();
|
2021-11-15 17:22:05 -08:00
|
|
|
return justHadError || engine->triggerCentral.isTriggerDecoderError();
|
2019-01-31 14:55:23 -08:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif /* EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT */
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
extern bool consoleByteArrived;
|
|
|
|
|
2019-07-13 08:06:03 -07:00
|
|
|
class CommunicationBlinkingTask : public PeriodicTimerController {
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-07-13 07:24:23 -07:00
|
|
|
int getPeriodMs() override {
|
|
|
|
return counter % 2 == 0 ? onTimeMs : offTimeMs;
|
|
|
|
}
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-07-13 07:24:23 -07:00
|
|
|
void setAllLeds(int value) {
|
2020-03-28 18:28:32 -07:00
|
|
|
// make sure we do not turn the critical LED off if already have
|
|
|
|
// critical error by now
|
2019-07-13 11:08:08 -07:00
|
|
|
for (uint32_t i = 0; !hasFirmwareError() && i < sizeof(leds) / sizeof(leds[0]); i++) {
|
2019-07-13 07:24:23 -07:00
|
|
|
leds[i]->setValue(value);
|
|
|
|
}
|
|
|
|
}
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-07-13 07:24:23 -07:00
|
|
|
void PeriodicTask() override {
|
|
|
|
counter++;
|
2020-05-12 22:17:48 -07:00
|
|
|
|
2021-10-05 16:59:07 -07:00
|
|
|
bool lowVBatt = Sensor::getOrZero(SensorType::BatteryVoltage) < LOW_VBATT;
|
2020-05-12 22:17:48 -07:00
|
|
|
|
2019-07-13 07:24:23 -07:00
|
|
|
if (counter == 1) {
|
|
|
|
// first invocation of BlinkingTask
|
|
|
|
setAllLeds(1);
|
|
|
|
} else if (counter == 2) {
|
|
|
|
// second invocation of BlinkingTask
|
|
|
|
setAllLeds(0);
|
|
|
|
} else if (counter % 2 == 0) {
|
|
|
|
enginePins.communicationLedPin.setValue(0);
|
2021-04-22 18:23:20 -07:00
|
|
|
#if HW_CHECK_SD
|
|
|
|
extern int totalLoggedBytes;
|
2021-04-22 21:45:36 -07:00
|
|
|
if (totalLoggedBytes > 2000) {
|
2021-04-22 18:23:20 -07:00
|
|
|
enginePins.communicationLedPin.setValue(1);
|
|
|
|
}
|
|
|
|
#endif // HW_CHECK_SD
|
|
|
|
|
2021-04-22 19:04:39 -07:00
|
|
|
//#if HW_CHECK_MODE
|
|
|
|
// // we have to do anything possible to help users notice FACTORY MODE
|
|
|
|
// enginePins.errorLedPin.setValue(1);
|
|
|
|
// enginePins.runningLedPin.setValue(1);
|
|
|
|
//#endif // HW_CHECK_MODE
|
2020-05-12 22:17:48 -07:00
|
|
|
if (!lowVBatt) {
|
|
|
|
enginePins.warningLedPin.setValue(0);
|
|
|
|
}
|
2019-07-13 07:24:23 -07:00
|
|
|
} else {
|
2020-06-13 10:23:34 -07:00
|
|
|
#define BLINKING_PERIOD_MS 33
|
|
|
|
|
2019-07-13 07:24:23 -07:00
|
|
|
if (hasFirmwareError()) {
|
2020-03-28 18:28:32 -07:00
|
|
|
// special behavior in case of critical error - not equal on/off time
|
2022-04-09 14:44:38 -07:00
|
|
|
// this special behavior helps to notice that something is not right, also
|
2020-03-28 18:28:32 -07:00
|
|
|
// differentiates software firmware error from critical interrupt error with CPU halt.
|
2019-07-13 07:24:23 -07:00
|
|
|
offTimeMs = 50;
|
|
|
|
onTimeMs = 450;
|
2020-06-13 10:23:34 -07:00
|
|
|
} else if (consoleByteArrived) {
|
|
|
|
offTimeMs = 100;
|
|
|
|
onTimeMs = 33;
|
|
|
|
#if EFI_INTERNAL_FLASH
|
|
|
|
} else if (getNeedToWriteConfiguration()) {
|
|
|
|
offTimeMs = onTimeMs = 500;
|
|
|
|
#endif // EFI_INTERNAL_FLASH
|
2019-07-13 07:24:23 -07:00
|
|
|
} else {
|
2020-09-09 13:25:50 -07:00
|
|
|
onTimeMs =
|
|
|
|
#if EFI_USB_SERIAL
|
|
|
|
is_usb_serial_ready() ? 3 * BLINKING_PERIOD_MS :
|
|
|
|
#endif // EFI_USB_SERIAL
|
|
|
|
BLINKING_PERIOD_MS;
|
2020-06-13 10:23:34 -07:00
|
|
|
offTimeMs = 0.6 * onTimeMs;
|
2019-07-13 07:24:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
enginePins.communicationLedPin.setValue(1);
|
2021-04-22 19:04:39 -07:00
|
|
|
//#if HW_CHECK_MODE
|
|
|
|
// // we have to do anything possible to help users notice FACTORY MODE
|
|
|
|
// enginePins.errorLedPin.setValue(0);
|
|
|
|
// enginePins.runningLedPin.setValue(0);
|
|
|
|
//#endif // HW_CHECK_MODE
|
2020-05-20 19:15:47 -07:00
|
|
|
|
2019-07-13 07:24:23 -07:00
|
|
|
#if EFI_ENGINE_CONTROL
|
2020-06-13 10:23:34 -07:00
|
|
|
if (lowVBatt || isTriggerErrorNow() || isIgnitionTimingError()) {
|
|
|
|
// todo: at the moment warning codes do not affect warning LED?!
|
2019-07-13 07:24:23 -07:00
|
|
|
enginePins.warningLedPin.setValue(1);
|
|
|
|
}
|
|
|
|
#endif /* EFI_ENGINE_CONTROL */
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
2019-07-13 07:24:23 -07:00
|
|
|
}
|
2019-02-05 18:29:55 -08:00
|
|
|
|
2019-07-13 07:24:23 -07:00
|
|
|
private:
|
|
|
|
int counter = 0;
|
|
|
|
int onTimeMs = 100;
|
|
|
|
int offTimeMs = 100;
|
|
|
|
};
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-07-13 08:06:03 -07:00
|
|
|
static CommunicationBlinkingTask communicationsBlinkingTask;
|
2017-03-24 10:41:36 -07:00
|
|
|
|
|
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_LCD
|
2019-02-11 12:09:24 -08:00
|
|
|
class LcdController : public PeriodicController<UTILITY_THREAD_STACK_SIZE> {
|
|
|
|
public:
|
2019-12-14 06:12:37 -08:00
|
|
|
LcdController() : PeriodicController("LCD") { }
|
2019-02-11 12:09:24 -08:00
|
|
|
private:
|
2019-12-21 18:11:09 -08:00
|
|
|
void PeriodicTask(efitick_t nowNt) override {
|
2019-02-21 02:44:45 -08:00
|
|
|
UNUSED(nowNt);
|
2020-06-13 19:46:10 -07:00
|
|
|
setPeriod(NOT_TOO_OFTEN(10 /* ms */, 300));
|
2019-12-11 14:48:55 -08:00
|
|
|
if (engineConfiguration->useLcdScreen) {
|
2017-03-24 10:41:36 -07:00
|
|
|
#if EFI_HD44780_LCD
|
2017-05-08 05:15:46 -07:00
|
|
|
updateHD44780lcd();
|
2017-03-24 10:41:36 -07:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2019-02-11 12:09:24 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
static LcdController lcdInstance;
|
2019-03-29 07:29:01 -07:00
|
|
|
#endif /* EFI_LCD */
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_HIP_9011
|
2019-01-07 20:55:56 -08:00
|
|
|
extern HIP9011 instance;
|
2017-04-09 19:07:41 -07:00
|
|
|
#endif /* EFI_HIP_9011 */
|
|
|
|
|
2019-09-20 16:38:19 -07:00
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_TUNER_STUDIO
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2021-10-03 13:24:34 -07:00
|
|
|
static void updateTempSensors() {
|
2020-04-14 22:12:07 -07:00
|
|
|
SensorResult clt = Sensor::get(SensorType::Clt);
|
2022-07-30 14:08:48 -07:00
|
|
|
engine->outputChannels.coolant = clt.value_or(0);
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.isCltError = !clt.Valid;
|
2020-04-14 22:12:07 -07:00
|
|
|
|
|
|
|
SensorResult iat = Sensor::get(SensorType::Iat);
|
2022-07-30 14:08:48 -07:00
|
|
|
engine->outputChannels.intake = iat.value_or(0);
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.isIatError = !iat.Valid;
|
2020-02-07 01:05:16 -08:00
|
|
|
|
2020-04-07 13:07:09 -07:00
|
|
|
SensorResult auxTemp1 = Sensor::get(SensorType::AuxTemp1);
|
2022-07-30 14:08:48 -07:00
|
|
|
engine->outputChannels.auxTemp1 = auxTemp1.value_or(0);
|
2020-04-07 13:07:09 -07:00
|
|
|
|
|
|
|
SensorResult auxTemp2 = Sensor::get(SensorType::AuxTemp2);
|
2022-07-30 14:08:48 -07:00
|
|
|
engine->outputChannels.auxTemp2 = auxTemp2.value_or(0);
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
2020-04-07 13:07:09 -07:00
|
|
|
|
2021-10-03 13:24:34 -07:00
|
|
|
static void updateThrottles() {
|
2020-02-27 00:28:49 -08:00
|
|
|
SensorResult tps1 = Sensor::get(SensorType::Tps1);
|
2022-07-30 14:08:48 -07:00
|
|
|
engine->outputChannels.TPSValue = tps1.value_or(0);
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.isTpsError = !tps1.Valid;
|
|
|
|
engine->outputChannels.tpsADC = convertVoltageTo10bitADC(Sensor::getRaw(SensorType::Tps1Primary));
|
2020-02-07 01:05:16 -08:00
|
|
|
|
2020-02-27 00:28:49 -08:00
|
|
|
SensorResult tps2 = Sensor::get(SensorType::Tps2);
|
2022-07-30 14:08:48 -07:00
|
|
|
engine->outputChannels.TPS2Value = tps2.value_or(0);
|
2020-12-04 21:01:24 -08:00
|
|
|
// If we don't have a TPS2 at all, don't turn on the failure light
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.isTps2Error = !tps2.Valid && Sensor::hasSensor(SensorType::Tps2Primary);
|
2020-02-27 00:28:49 -08:00
|
|
|
|
2020-03-22 14:09:46 -07:00
|
|
|
SensorResult pedal = Sensor::get(SensorType::AcceleratorPedal);
|
2022-07-30 14:08:48 -07:00
|
|
|
engine->outputChannels.throttlePedalPosition = pedal.value_or(0);
|
2020-04-23 21:10:19 -07:00
|
|
|
// Only report fail if you have one (many people don't)
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.isPedalError = !pedal.Valid && Sensor::hasSensor(SensorType::AcceleratorPedalPrimary);
|
2021-12-21 15:09:40 -08:00
|
|
|
|
|
|
|
// TPS 1 pri/sec split
|
|
|
|
engine->outputChannels.tps1Split = Sensor::getOrZero(SensorType::Tps1Primary) - Sensor::getOrZero(SensorType::Tps1Secondary);
|
|
|
|
// TPS 2 pri/sec split
|
|
|
|
engine->outputChannels.tps2Split = Sensor::getOrZero(SensorType::Tps2Primary) - Sensor::getOrZero(SensorType::Tps2Secondary);
|
|
|
|
// TPS1 - TPS2 split
|
|
|
|
engine->outputChannels.tps12Split = Sensor::getOrZero(SensorType::Tps1) - Sensor::getOrZero(SensorType::Tps2);
|
|
|
|
// Pedal pri/sec split
|
|
|
|
engine->outputChannels.accPedalSplit = Sensor::getOrZero(SensorType::AcceleratorPedalPrimary) - Sensor::getOrZero(SensorType::AcceleratorPedalSecondary);
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
2017-08-28 19:32:29 -07:00
|
|
|
|
2021-10-03 13:24:34 -07:00
|
|
|
static void updateLambda() {
|
2021-10-05 16:59:07 -07:00
|
|
|
float lambdaValue = Sensor::getOrZero(SensorType::Lambda1);
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.lambdaValue = lambdaValue;
|
2022-09-02 11:38:03 -07:00
|
|
|
engine->outputChannels.AFRValue = lambdaValue * engine->fuelComputer->stoichiometricRatio;
|
2020-09-01 13:22:31 -07:00
|
|
|
|
2021-10-05 16:59:07 -07:00
|
|
|
float lambda2Value = Sensor::getOrZero(SensorType::Lambda2);
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.lambdaValue2 = lambda2Value;
|
2022-09-02 11:38:03 -07:00
|
|
|
engine->outputChannels.AFRValue2 = lambda2Value * engine->fuelComputer->stoichiometricRatio;
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void updateFuelSensors() {
|
|
|
|
// Low pressure is directly in kpa
|
2022-08-01 12:27:06 -07:00
|
|
|
engine->outputChannels.lowFuelPressure = Sensor::getOrZero(SensorType::FuelPressureLow);
|
2021-10-03 13:24:34 -07:00
|
|
|
// High pressure is in bar, aka 100 kpa
|
2022-08-01 12:27:06 -07:00
|
|
|
engine->outputChannels.highFuelPressure = KPA2BAR(Sensor::getOrZero(SensorType::FuelPressureHigh));
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2022-08-01 12:27:06 -07:00
|
|
|
engine->outputChannels.flexPercent = Sensor::getOrZero(SensorType::FuelEthanolPercent);
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.fuelTankLevel = Sensor::getOrZero(SensorType::FuelLevel);
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void updateVvtSensors() {
|
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
|
|
|
// 248
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.vvtPositionB1I = engine->triggerCentral.getVVTPosition(/*bankIndex*/0, /*camIndex*/0);
|
|
|
|
engine->outputChannels.vvtPositionB1E = engine->triggerCentral.getVVTPosition(/*bankIndex*/0, /*camIndex*/1);
|
|
|
|
engine->outputChannels.vvtPositionB2I = engine->triggerCentral.getVVTPosition(/*bankIndex*/1, /*camIndex*/0);
|
|
|
|
engine->outputChannels.vvtPositionB2E = engine->triggerCentral.getVVTPosition(/*bankIndex*/1, /*camIndex*/1);
|
2021-10-03 13:24:34 -07:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2022-04-07 06:21:11 -07:00
|
|
|
static void updateVehicleSpeed() {
|
2021-10-03 13:24:34 -07:00
|
|
|
#if EFI_VEHICLE_SPEED
|
2022-04-06 13:15:58 -07:00
|
|
|
engine->outputChannels.vehicleSpeedKph = Sensor::getOrZero(SensorType::VehicleSpeed);
|
|
|
|
engine->outputChannels.speedToRpmRatio = engine->module<GearDetector>()->getGearboxRatio();
|
2022-04-07 06:21:11 -07:00
|
|
|
engine->outputChannels.detectedGear = engine->module<GearDetector>()->getCurrentGear();
|
2021-10-03 13:24:34 -07:00
|
|
|
#endif /* EFI_VEHICLE_SPEED */
|
|
|
|
}
|
|
|
|
|
|
|
|
static void updateRawSensors() {
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.rawTps1Primary = Sensor::getRaw(SensorType::Tps1Primary);
|
|
|
|
engine->outputChannels.rawTps1Secondary = Sensor::getRaw(SensorType::Tps1Secondary);
|
|
|
|
engine->outputChannels.rawTps2Primary = Sensor::getRaw(SensorType::Tps2Primary);
|
|
|
|
engine->outputChannels.rawTps2Secondary = Sensor::getRaw(SensorType::Tps2Secondary);
|
|
|
|
engine->outputChannels.rawPpsPrimary = Sensor::getRaw(SensorType::AcceleratorPedalPrimary);
|
|
|
|
engine->outputChannels.rawPpsSecondary = Sensor::getRaw(SensorType::AcceleratorPedalSecondary);
|
2022-08-11 07:54:54 -07:00
|
|
|
engine->outputChannels.rawBattery = Sensor::getRaw(SensorType::BatteryVoltage);
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.rawClt = Sensor::getRaw(SensorType::Clt);
|
|
|
|
engine->outputChannels.rawIat = Sensor::getRaw(SensorType::Iat);
|
|
|
|
engine->outputChannels.rawOilPressure = Sensor::getRaw(SensorType::OilPressure);
|
|
|
|
engine->outputChannels.rawLowFuelPressure = Sensor::getRaw(SensorType::FuelPressureLow);
|
|
|
|
engine->outputChannels.rawHighFuelPressure = Sensor::getRaw(SensorType::FuelPressureHigh);
|
2022-01-12 12:41:35 -08:00
|
|
|
engine->outputChannels.rawMaf = Sensor::getRaw(SensorType::Maf);
|
2022-01-21 12:53:16 -08:00
|
|
|
engine->outputChannels.rawMap = Sensor::getRaw(SensorType::MapSlow);
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.rawWastegatePosition = Sensor::getRaw(SensorType::WastegatePosition);
|
|
|
|
engine->outputChannels.rawIdlePositionSensor = Sensor::getRaw(SensorType::IdlePosition);
|
2022-01-21 12:53:16 -08:00
|
|
|
|
2022-10-18 12:52:32 -07:00
|
|
|
engine->outputChannels.luaGauges[0] = Sensor::getOrZero(SensorType::LuaGauge1);
|
|
|
|
engine->outputChannels.luaGauges[1] = Sensor::getOrZero(SensorType::LuaGauge2);
|
2022-10-17 17:01:46 -07:00
|
|
|
|
2022-07-15 03:37:30 -07:00
|
|
|
for (int i = 0;i<AUX_ANALOG_INPUT_COUNT;i++) {
|
|
|
|
adc_channel_e ch = engineConfiguration->auxAnalogInputs[i];
|
|
|
|
if (ch != EFI_ADC_NONE) {
|
2022-07-18 05:26:17 -07:00
|
|
|
engine->outputChannels.rawAnalogInput[i] = getVoltageDivided("raw aux", ch);
|
2022-07-15 03:37:30 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-21 12:53:16 -08:00
|
|
|
// TODO: transition AFR to new sensor model
|
2022-04-03 14:39:29 -07:00
|
|
|
engine->outputChannels.rawAfr = (engineConfiguration->afr.hwChannel == EFI_ADC_NONE) ? 0 : getVoltageDivided("ego", engineConfiguration->afr.hwChannel);
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
2021-10-05 16:06:17 -07:00
|
|
|
static void updatePressures() {
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.baroPressure = Sensor::getOrZero(SensorType::BarometricPressure);
|
|
|
|
engine->outputChannels.MAPValue = Sensor::getOrZero(SensorType::Map);
|
2022-08-01 12:27:06 -07:00
|
|
|
engine->outputChannels.oilPressure = Sensor::getOrZero(SensorType::OilPressure);
|
2022-04-15 04:22:52 -07:00
|
|
|
|
2022-08-01 12:27:06 -07:00
|
|
|
engine->outputChannels.auxLinear1 = Sensor::getOrZero(SensorType::AuxLinear1);
|
|
|
|
engine->outputChannels.auxLinear2 = Sensor::getOrZero(SensorType::AuxLinear2);
|
2021-10-05 16:06:17 -07:00
|
|
|
}
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2021-10-05 16:06:17 -07:00
|
|
|
static void updateMiscSensors() {
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.VBatt = Sensor::getOrZero(SensorType::BatteryVoltage);
|
2021-10-05 16:06:17 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.idlePositionSensor = Sensor::getOrZero(SensorType::IdlePosition);
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.wastegatePositionSensor = Sensor::getOrZero(SensorType::WastegatePosition);
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2022-04-21 06:07:52 -07:00
|
|
|
engine->outputChannels.ISSValue = Sensor::getOrZero(SensorType::InputShaftSpeed);
|
2022-09-20 20:00:42 -07:00
|
|
|
engine->outputChannels.auxSpeed1 = Sensor::getOrZero(SensorType::AuxSpeed1);
|
|
|
|
engine->outputChannels.auxSpeed2 = Sensor::getOrZero(SensorType::AuxSpeed2);
|
2022-04-21 06:07:52 -07:00
|
|
|
|
2021-10-03 13:24:34 -07:00
|
|
|
#if HAL_USE_ADC
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.internalMcuTemperature = getMCUInternalTemperature();
|
2021-10-03 13:24:34 -07:00
|
|
|
#endif /* HAL_USE_ADC */
|
|
|
|
|
|
|
|
// tCharge depends on the previous state, so we should use the stored value.
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.tCharge = engine->engineState.sd.tCharge;
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
2020-12-09 17:26:23 -08:00
|
|
|
|
2022-04-07 06:21:11 -07:00
|
|
|
static void updateSensors() {
|
2021-10-05 16:06:17 -07:00
|
|
|
updateTempSensors();
|
|
|
|
updateThrottles();
|
|
|
|
updateRawSensors();
|
|
|
|
updateLambda();
|
|
|
|
updateFuelSensors();
|
|
|
|
updateVvtSensors();
|
2022-04-07 06:21:11 -07:00
|
|
|
updateVehicleSpeed();
|
2021-10-05 16:06:17 -07:00
|
|
|
updatePressures();
|
|
|
|
updateMiscSensors();
|
|
|
|
}
|
|
|
|
|
2021-10-03 13:24:34 -07:00
|
|
|
static void updateFuelCorrections() {
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.fuelPidCorrection[0] = 100.0f * (engine->stftCorrection[0] - 1.0f);
|
|
|
|
engine->outputChannels.fuelPidCorrection[1] = 100.0f * (engine->stftCorrection[1] - 1.0f);
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2022-09-01 07:58:16 -07:00
|
|
|
// get rid of this, have gauges use injector model info directly
|
|
|
|
engine->outputChannels.injectorLagMs = engine->module<InjectorModel>()->getDeadtime();
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void updateFuelLoads() {
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.veTableYAxis = engine->engineState.currentVeLoad;
|
2022-09-02 11:38:03 -07:00
|
|
|
engine->outputChannels.afrTableYAxis = engine->fuelComputer->currentAfrLoad;
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
2020-02-07 01:05:16 -08:00
|
|
|
|
2021-10-03 13:24:34 -07:00
|
|
|
static void updateFuelResults() {
|
2022-09-03 04:44:24 -07:00
|
|
|
// todo: kill outputChannel while taking care of gauge name and scale!
|
|
|
|
engine->outputChannels.chargeAirMass = engine->fuelComputer->sdAirMassInOneCylinder;
|
2021-02-07 15:54:41 -08:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.baseFuel = engine->engineState.baseFuel * 1000; // Convert grams to mg
|
|
|
|
engine->outputChannels.fuelRunning = engine->engineState.running.fuel;
|
2017-05-18 12:57:55 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.fuelFlowRate = engine->engineState.fuelConsumption.getConsumptionGramPerSecond();
|
|
|
|
engine->outputChannels.totalFuelConsumption = engine->engineState.fuelConsumption.getConsumedGrams();
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void updateFuelInfo() {
|
|
|
|
updateFuelCorrections();
|
|
|
|
updateFuelLoads();
|
|
|
|
updateFuelResults();
|
|
|
|
|
2021-11-17 00:54:21 -08:00
|
|
|
const auto& wallFuel = engine->injectionEvents.elements[0].wallFuel;
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.wallFuelAmount = wallFuel.getWallFuel() * 1000; // Convert grams to mg
|
|
|
|
engine->outputChannels.wallFuelCorrection = wallFuel.wallFuelCorrection * 1000; // Convert grams to mg
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.injectionOffset = engine->engineState.injectionOffset;
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.veValue = engine->engineState.currentVe;
|
2022-09-02 11:38:03 -07:00
|
|
|
engine->outputChannels.currentTargetAfr = engine->fuelComputer->targetAFR;
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.crankingFuelMs = engine->engineState.cranking.fuel;
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void updateIgnition(int rpm) {
|
2021-12-06 18:19:37 -08:00
|
|
|
float timing = engine->engineState.timingAdvance[0];
|
2021-11-06 21:09:17 -07:00
|
|
|
// that's weird logic. also seems broken for two stroke?
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.ignitionAdvance = timing > FOUR_STROKE_CYCLE_DURATION / 2 ? timing - FOUR_STROKE_CYCLE_DURATION : timing;
|
2021-10-03 13:24:34 -07:00
|
|
|
// 60
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.sparkDwellValue = engine->engineState.sparkDwell;
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.coilDutyCycle = getCoilDutyCycle(rpm);
|
2021-11-01 20:33:59 -07:00
|
|
|
|
2022-04-18 05:03:16 -07:00
|
|
|
engine->outputChannels.knockCount = engine->module<KnockController>()->getKnockCount();
|
|
|
|
engine->outputChannels.knockRetard = engine->module<KnockController>()->getKnockRetard();
|
2021-10-03 13:24:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void updateFlags() {
|
2022-02-04 14:25:05 -08:00
|
|
|
#if EFI_USB_SERIAL
|
|
|
|
engine->outputChannels.isUsbConnected = is_usb_serial_ready();
|
|
|
|
#endif // EFI_USB_SERIAL
|
|
|
|
|
2022-02-01 21:45:11 -08:00
|
|
|
engine->outputChannels.isMainRelayOn = enginePins.mainRelay.getLogicValue();
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.isFanOn = enginePins.fanRelay.getLogicValue();
|
|
|
|
engine->outputChannels.isFan2On = enginePins.fanRelay2.getLogicValue();
|
|
|
|
engine->outputChannels.isO2HeaterOn = enginePins.o2heater.getLogicValue();
|
2022-08-23 19:25:08 -07:00
|
|
|
// todo: eliminate state copy logic by giving DfcoController it's owm xxx.txt and leveraging LiveData
|
2022-01-23 16:44:41 -08:00
|
|
|
engine->outputChannels.dfcoActive = engine->module<DfcoController>()->cutFuel();
|
2021-10-03 13:24:34 -07:00
|
|
|
|
|
|
|
#if EFI_LAUNCH_CONTROL
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.launchTriggered = engine->launchController.isLaunchCondition;
|
2021-10-03 13:24:34 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.isTriggerError = isTriggerErrorNow();
|
2021-10-03 13:24:34 -07:00
|
|
|
#endif // EFI_PROD_CODE
|
|
|
|
|
|
|
|
#if EFI_INTERNAL_FLASH
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.needBurn = getNeedToWriteConfiguration();
|
2021-10-03 13:24:34 -07:00
|
|
|
#endif /* EFI_INTERNAL_FLASH */
|
|
|
|
}
|
|
|
|
|
2021-10-05 12:30:51 -07:00
|
|
|
// weird thing: one of the reasons for this to be a separate method is stack usage reduction in non-optimized build
|
|
|
|
// see https://github.com/rusefi/rusefi/issues/3302 and linked tickets
|
2021-10-05 12:27:08 -07:00
|
|
|
static void updateTpsDebug() {
|
|
|
|
// TPS 1 pri/sec ratio - useful for ford ETB that has partial-range second channel
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.debugFloatField5 = 100 * Sensor::getOrZero(SensorType::Tps1Primary) / Sensor::getOrZero(SensorType::Tps1Secondary);
|
2021-10-05 12:27:08 -07:00
|
|
|
}
|
|
|
|
|
2022-03-14 20:02:00 -07:00
|
|
|
// sensor state for EFI Analytics Tuner Studio
|
2022-08-23 19:25:08 -07:00
|
|
|
// todo: the 'let's copy internal state for external consumers' approach is DEPRECATED
|
|
|
|
// As of 2022 it's preferred to leverage LiveData where all state is exposed
|
2021-12-15 17:11:08 -08:00
|
|
|
void updateTunerStudioState() {
|
|
|
|
TunerStudioOutputChannels *tsOutputChannels = &engine->outputChannels;
|
2021-10-03 13:24:34 -07:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2022-07-30 14:08:48 -07:00
|
|
|
int rpm = Sensor::get(SensorType::Rpm).value_or(0);
|
2021-10-03 13:24:34 -07:00
|
|
|
#else /* EFI_SHAFT_POSITION_INPUT */
|
|
|
|
int rpm = 0;
|
|
|
|
#endif /* EFI_SHAFT_POSITION_INPUT */
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE
|
|
|
|
executorStatistics();
|
|
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
|
|
|
|
// header
|
|
|
|
tsOutputChannels->tsConfigVersion = TS_FILE_VERSION;
|
2022-06-01 08:07:28 -07:00
|
|
|
static_assert(offsetof (TunerStudioOutputChannels, tsConfigVersion) == TS_FILE_VERSION_OFFSET);
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2022-04-16 14:35:59 -07:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
|
|
|
|
2021-10-03 13:24:34 -07:00
|
|
|
// offset 0
|
2021-11-26 18:54:31 -08:00
|
|
|
tsOutputChannels->RPMValue = rpm;
|
2022-01-12 12:41:35 -08:00
|
|
|
auto instantRpm = engine->triggerCentral.triggerState.getInstantRpm();
|
|
|
|
tsOutputChannels->instantRpm = instantRpm;
|
2021-10-03 13:24:34 -07:00
|
|
|
|
2022-04-07 06:21:11 -07:00
|
|
|
updateSensors();
|
2021-10-03 13:24:34 -07:00
|
|
|
updateFuelInfo();
|
|
|
|
updateIgnition(rpm);
|
|
|
|
updateFlags();
|
2019-09-20 16:38:19 -07:00
|
|
|
|
2019-09-22 04:37:36 -07:00
|
|
|
// 104
|
2019-09-22 04:49:26 -07:00
|
|
|
tsOutputChannels->rpmAcceleration = engine->rpmCalculator.getRpmAcceleration();
|
2022-02-01 21:45:11 -08:00
|
|
|
|
2022-01-12 12:41:35 -08:00
|
|
|
// Output both the estimated air flow, and measured air flow (if available)
|
|
|
|
tsOutputChannels->mafMeasured = Sensor::getOrZero(SensorType::Maf);
|
|
|
|
tsOutputChannels->mafEstimate = engine->engineState.airflowEstimate;
|
|
|
|
|
2019-09-20 16:38:19 -07:00
|
|
|
// offset 116
|
|
|
|
// TPS acceleration
|
|
|
|
tsOutputChannels->deltaTps = engine->tpsAccelEnrichment.getMaxDelta();
|
2021-06-20 23:11:36 -07:00
|
|
|
|
2019-12-25 12:57:47 -08:00
|
|
|
tsOutputChannels->totalTriggerErrorCounter = engine->triggerCentral.triggerState.totalTriggerErrorCounter;
|
2021-06-20 23:11:36 -07:00
|
|
|
|
2019-12-25 12:57:47 -08:00
|
|
|
tsOutputChannels->orderingErrorCounter = engine->triggerCentral.triggerState.orderingErrorCounter;
|
2022-04-16 14:35:59 -07:00
|
|
|
#endif // EFI_SHAFT_POSITION_INPUT
|
|
|
|
|
|
|
|
|
2019-12-25 12:57:47 -08:00
|
|
|
// 68
|
2019-09-20 16:38:19 -07:00
|
|
|
// 140
|
2020-02-26 15:16:35 -08:00
|
|
|
#if EFI_ENGINE_CONTROL
|
2021-11-16 01:15:29 -08:00
|
|
|
tsOutputChannels->injectorDutyCycle = getInjectorDutyCycle(rpm);
|
2020-02-26 15:16:35 -08:00
|
|
|
#endif
|
2021-06-20 23:11:36 -07:00
|
|
|
|
2019-09-20 16:38:19 -07:00
|
|
|
// 224
|
2022-09-11 10:06:03 -07:00
|
|
|
efitimesec_t timeSeconds = getTimeNowS();
|
2021-11-27 10:05:44 -08:00
|
|
|
tsOutputChannels->seconds = timeSeconds;
|
2020-02-26 15:16:35 -08:00
|
|
|
|
2019-09-20 16:38:19 -07:00
|
|
|
// 252
|
2021-11-16 01:15:29 -08:00
|
|
|
tsOutputChannels->engineMode = packEngineMode();
|
2019-12-25 11:15:44 -08:00
|
|
|
// 120
|
2019-09-20 16:38:19 -07:00
|
|
|
tsOutputChannels->firmwareVersion = getRusEfiVersion();
|
2021-05-08 14:41:50 -07:00
|
|
|
|
2019-12-16 19:45:46 -08:00
|
|
|
// 276
|
|
|
|
tsOutputChannels->accelerationX = engine->sensors.accelerometer.x;
|
|
|
|
// 278
|
|
|
|
tsOutputChannels->accelerationY = engine->sensors.accelerometer.y;
|
2021-10-06 12:05:29 -07:00
|
|
|
tsOutputChannels->accelerationZ = engine->sensors.accelerometer.z;
|
2021-10-05 21:19:33 -07:00
|
|
|
tsOutputChannels->accelerationRoll = engine->sensors.accelerometer.roll;
|
|
|
|
tsOutputChannels->accelerationYaw = engine->sensors.accelerometer.yaw;
|
2019-09-20 16:38:19 -07:00
|
|
|
|
2020-12-04 17:28:48 -08:00
|
|
|
#if EFI_DYNO_VIEW
|
2021-11-16 01:15:29 -08:00
|
|
|
tsOutputChannels->VssAcceleration = getDynoviewAcceleration();
|
2020-12-04 17:28:48 -08:00
|
|
|
#endif
|
|
|
|
|
2021-11-04 16:43:22 -07:00
|
|
|
tsOutputChannels->turboSpeed = Sensor::getOrZero(SensorType::TurbochargerSpeed);
|
2022-05-14 05:39:08 -07:00
|
|
|
extern FrequencySensor inputShaftSpeedSensor;
|
|
|
|
tsOutputChannels->issEdgeCounter = inputShaftSpeedSensor.eventCounter;
|
|
|
|
extern FrequencySensor vehicleSpeedSensor;
|
|
|
|
tsOutputChannels->vssEdgeCounter = vehicleSpeedSensor.eventCounter;
|
2021-11-04 16:43:22 -07:00
|
|
|
|
2020-05-20 19:15:47 -07:00
|
|
|
#if HW_CHECK_MODE
|
|
|
|
tsOutputChannels->hasCriticalError = 1;
|
|
|
|
#else
|
2020-03-28 18:28:32 -07:00
|
|
|
tsOutputChannels->hasCriticalError = hasFirmwareError();
|
2020-05-20 19:15:47 -07:00
|
|
|
#endif // HW_CHECK_MODE
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2022-08-16 22:12:25 -07:00
|
|
|
tsOutputChannels->isWarnNow = engine->engineState.warnings.isWarningNow();
|
2021-05-08 15:43:55 -07:00
|
|
|
#if EFI_HIP_9011_DEBUG
|
2021-03-30 07:56:25 -07:00
|
|
|
tsOutputChannels->isKnockChipOk = (instance.invalidResponsesCount == 0);
|
2018-05-30 21:32:44 -07:00
|
|
|
#endif /* EFI_HIP_9011 */
|
2017-03-24 10:41:36 -07:00
|
|
|
|
2019-09-20 16:38:19 -07:00
|
|
|
tsOutputChannels->tpsAccelFuel = engine->engineState.tpsAccelEnrich;
|
2021-10-04 15:33:10 -07:00
|
|
|
|
2019-09-20 16:38:19 -07:00
|
|
|
tsOutputChannels->checkEngine = hasErrorCodes();
|
|
|
|
|
|
|
|
#if EFI_MAX_31855
|
|
|
|
for (int i = 0; i < EGT_CHANNEL_COUNT; i++)
|
2022-08-22 15:07:59 -07:00
|
|
|
tsOutputChannels->egt[i] = getMax31855EgtValue(i);
|
2019-09-20 16:38:19 -07:00
|
|
|
#endif /* EFI_MAX_31855 */
|
|
|
|
|
|
|
|
#if EFI_IDLE_CONTROL
|
2021-11-27 09:28:59 -08:00
|
|
|
tsOutputChannels->idleAirValvePosition = getIdlePosition();
|
2019-09-20 16:38:19 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
tsOutputChannels->warningCounter = engine->engineState.warnings.warningCounter;
|
|
|
|
tsOutputChannels->lastErrorCode = engine->engineState.warnings.lastErrorCode;
|
|
|
|
for (int i = 0; i < 8;i++) {
|
2022-08-16 22:12:25 -07:00
|
|
|
tsOutputChannels->recentErrorCode[i] = engine->engineState.warnings.recentWarnings.get(i).Code;
|
2019-09-20 16:38:19 -07:00
|
|
|
}
|
|
|
|
|
2021-11-29 07:16:45 -08:00
|
|
|
tsOutputChannels->starterState = enginePins.starterControl.getLogicValue();
|
|
|
|
tsOutputChannels->starterRelayDisable = enginePins.starterRelayDisable.getLogicValue();
|
|
|
|
|
2022-09-15 11:55:49 -07:00
|
|
|
tsOutputChannels->mapFast = Sensor::getOrZero(SensorType::MapFast);
|
|
|
|
|
|
|
|
|
2021-11-29 07:16:45 -08:00
|
|
|
tsOutputChannels->revolutionCounterSinceStart = engine->rpmCalculator.getRevolutionCounterSinceStart();
|
|
|
|
#if EFI_CAN_SUPPORT
|
2021-12-20 21:48:13 -08:00
|
|
|
postCanState();
|
2021-11-29 07:16:45 -08:00
|
|
|
#endif /* EFI_CAN_SUPPORT */
|
|
|
|
|
|
|
|
#if EFI_CLOCK_LOCKS
|
2022-09-13 11:07:25 -07:00
|
|
|
tsOutputChannels->maxLockedDuration = NT2US(maxLockedDuration);
|
2021-12-01 22:06:40 -08:00
|
|
|
tsOutputChannels->maxTriggerReentrant = maxTriggerReentrant;
|
2021-11-29 07:16:45 -08:00
|
|
|
#endif /* EFI_CLOCK_LOCKS */
|
|
|
|
|
2022-04-16 14:35:59 -07:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2021-12-06 10:08:41 -08:00
|
|
|
tsOutputChannels->triggerPrimaryFall = engine->triggerCentral.getHwEventCounter((int)SHAFT_PRIMARY_FALLING);
|
|
|
|
tsOutputChannels->triggerPrimaryRise = engine->triggerCentral.getHwEventCounter((int)SHAFT_PRIMARY_RISING);
|
|
|
|
|
|
|
|
tsOutputChannels->triggerSecondaryFall = engine->triggerCentral.getHwEventCounter((int)SHAFT_SECONDARY_FALLING);
|
|
|
|
tsOutputChannels->triggerSecondaryRise = engine->triggerCentral.getHwEventCounter((int)SHAFT_SECONDARY_RISING);
|
|
|
|
|
|
|
|
tsOutputChannels->triggerVvtRise = engine->triggerCentral.vvtEventRiseCounter[0];
|
|
|
|
tsOutputChannels->triggerVvtFall = engine->triggerCentral.vvtEventFallCounter[0];
|
2022-04-16 14:35:59 -07:00
|
|
|
#endif // EFI_SHAFT_POSITION_INPUT
|
2021-12-06 10:08:41 -08:00
|
|
|
|
2022-08-23 16:45:28 -07:00
|
|
|
#if HAL_USE_PAL && EFI_PROD_CODE
|
|
|
|
tsOutputChannels->extiOverflowCount = getExtiOverflowCounter();
|
|
|
|
#endif
|
2021-12-06 10:08:41 -08:00
|
|
|
|
2017-12-03 04:42:44 -08:00
|
|
|
switch (engineConfiguration->debugMode) {
|
|
|
|
case DBG_TPS_ACCEL:
|
2017-03-24 10:41:36 -07:00
|
|
|
tsOutputChannels->debugIntField1 = engine->tpsAccelEnrichment.cb.getSize();
|
2017-12-03 04:42:44 -08:00
|
|
|
break;
|
|
|
|
case DBG_SR5_PROTOCOL: {
|
|
|
|
const int _10_6 = 100000;
|
2017-03-26 19:45:19 -07:00
|
|
|
tsOutputChannels->debugIntField1 = tsState.textCommandCounter * _10_6 + tsState.totalCounter;
|
|
|
|
tsOutputChannels->debugIntField2 = tsState.outputChannelsCommandCounter * _10_6 + tsState.writeValueCommandCounter;
|
|
|
|
tsOutputChannels->debugIntField3 = tsState.readPageCommandsCounter * _10_6 + tsState.burnCommandCounter;
|
2017-12-03 04:42:44 -08:00
|
|
|
break;
|
|
|
|
}
|
2019-08-07 19:20:16 -07:00
|
|
|
case DBG_TRIGGER_COUNTERS:
|
2020-04-10 10:19:54 -07:00
|
|
|
|
2022-04-16 14:35:59 -07:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2020-04-10 10:19:54 -07:00
|
|
|
tsOutputChannels->debugIntField4 = engine->triggerCentral.triggerState.currentCycle.eventCount[0];
|
|
|
|
tsOutputChannels->debugIntField5 = engine->triggerCentral.triggerState.currentCycle.eventCount[1];
|
2022-04-16 14:35:59 -07:00
|
|
|
#endif // EFI_SHAFT_POSITION_INPUT
|
2020-04-10 10:19:54 -07:00
|
|
|
|
2017-12-03 04:42:44 -08:00
|
|
|
break;
|
2021-05-08 15:43:55 -07:00
|
|
|
#if EFI_HIP_9011_DEBUG
|
2017-12-03 04:42:44 -08:00
|
|
|
case DBG_KNOCK:
|
2019-01-07 20:55:56 -08:00
|
|
|
// todo: maybe extract hipPostState(tsOutputChannels);
|
|
|
|
tsOutputChannels->debugIntField1 = instance.correctResponsesCount;
|
2021-03-30 07:56:25 -07:00
|
|
|
tsOutputChannels->debugIntField2 = instance.invalidResponsesCount;
|
2017-12-03 04:42:44 -08:00
|
|
|
break;
|
2017-04-09 19:07:41 -07:00
|
|
|
#endif /* EFI_HIP_9011 */
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_CJ125 && HAL_USE_SPI
|
2018-01-28 10:04:19 -08:00
|
|
|
case DBG_CJ125:
|
|
|
|
cjPostState(tsOutputChannels);
|
|
|
|
break;
|
2019-02-01 20:16:34 -08:00
|
|
|
#endif /* EFI_CJ125 && HAL_USE_SPI */
|
2018-02-06 13:21:41 -08:00
|
|
|
#if EFI_MAP_AVERAGING
|
2021-11-24 11:26:41 -08:00
|
|
|
case DBG_MAP:
|
2018-02-06 13:21:41 -08:00
|
|
|
postMapState(tsOutputChannels);
|
|
|
|
break;
|
|
|
|
#endif /* EFI_MAP_AVERAGING */
|
2019-04-25 05:32:20 -07:00
|
|
|
case DBG_ANALOG_INPUTS:
|
2021-11-16 01:15:29 -08:00
|
|
|
tsOutputChannels->debugFloatField4 = isAdcChannelValid(engineConfiguration->map.sensor.hwChannel) ? getVoltageDivided("map", engineConfiguration->map.sensor.hwChannel) : 0.0f;
|
|
|
|
tsOutputChannels->debugFloatField7 = isAdcChannelValid(engineConfiguration->afr.hwChannel) ? getVoltageDivided("ego", engineConfiguration->afr.hwChannel) : 0.0f;
|
2017-12-03 04:42:44 -08:00
|
|
|
break;
|
2019-04-25 05:32:20 -07:00
|
|
|
case DBG_ANALOG_INPUTS2:
|
2021-10-05 12:27:08 -07:00
|
|
|
updateTpsDebug();
|
2019-04-25 05:32:20 -07:00
|
|
|
break;
|
2017-12-16 21:17:55 -08:00
|
|
|
case DBG_INSTANT_RPM:
|
2017-12-05 20:49:39 -08:00
|
|
|
{
|
2022-04-16 14:35:59 -07:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2022-01-20 19:27:48 -08:00
|
|
|
tsOutputChannels->debugFloatField2 = instantRpm / Sensor::getOrZero(SensorType::Rpm);
|
2022-04-16 14:35:59 -07:00
|
|
|
#endif // EFI_SHAFT_POSITION_INPUT
|
2021-07-17 13:12:54 -07:00
|
|
|
|
2017-12-05 20:49:39 -08:00
|
|
|
}
|
|
|
|
break;
|
2019-01-03 21:16:08 -08:00
|
|
|
case DBG_ION:
|
2019-05-05 14:21:36 -07:00
|
|
|
#if EFI_CDM_INTEGRATION
|
|
|
|
ionPostState(tsOutputChannels);
|
|
|
|
#endif /* EFI_CDM_INTEGRATION */
|
2019-01-03 21:16:08 -08:00
|
|
|
break;
|
2019-10-31 13:06:34 -07:00
|
|
|
case DBG_TLE8888:
|
|
|
|
#if (BOARD_TLE8888_COUNT > 0)
|
2022-01-27 16:21:05 -08:00
|
|
|
tle8888PostState();
|
2019-10-31 13:06:34 -07:00
|
|
|
#endif /* BOARD_TLE8888_COUNT */
|
|
|
|
break;
|
2020-12-15 11:52:48 -08:00
|
|
|
case DBG_LOGIC_ANALYZER:
|
|
|
|
#if EFI_LOGIC_ANALYZER
|
|
|
|
reportLogicAnalyzerToTS();
|
|
|
|
#endif /* EFI_LOGIC_ANALYZER */
|
|
|
|
break;
|
2017-12-03 04:42:44 -08:00
|
|
|
default:
|
|
|
|
;
|
|
|
|
}
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
|
|
|
|
2017-04-09 19:26:11 -07:00
|
|
|
void initStatusLoop(void) {
|
2017-03-24 10:41:36 -07:00
|
|
|
addConsoleActionI("warn", setWarningEnabled);
|
|
|
|
}
|
|
|
|
|
2017-04-09 19:26:11 -07:00
|
|
|
void startStatusThreads(void) {
|
2017-03-24 10:41:36 -07:00
|
|
|
// todo: refactoring needed, this file should probably be split into pieces
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_PROD_CODE
|
2017-03-24 10:41:36 -07:00
|
|
|
initStatusLeds();
|
2022-07-21 12:17:32 -07:00
|
|
|
communicationsBlinkingTask.start();
|
2017-03-24 10:41:36 -07:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2019-07-13 07:24:23 -07:00
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_LCD
|
2022-07-21 12:17:32 -07:00
|
|
|
lcdInstance.start();
|
2019-03-29 07:29:01 -07:00
|
|
|
#endif /* EFI_LCD */
|
2017-03-24 10:41:36 -07:00
|
|
|
}
|