2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file malfunction_indicator.cpp
|
|
|
|
* @brief We can blink out OBD-II error codes using Malfunction Indicator Light (MIL)
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @date Dec 20, 2013
|
|
|
|
* @author Konstantin Nikonenko
|
2018-01-20 17:55:31 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2018
|
2015-07-10 06:01:56 -07:00
|
|
|
* we show 4 digit error code - 1,5sec * (4xxx+1) digit + 0,4sec * (x3xxx+1) + ....
|
|
|
|
* ATTENTION!!! 0 = 1 blink, 1 = 2 blinks, ...., 9 = 10 blinks
|
|
|
|
* sequence is the constant!!!
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2018-09-16 19:26:57 -07:00
|
|
|
#include "global.h"
|
2019-04-09 20:00:17 -07:00
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_MALFUNCTION_INDICATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "io_pins.h"
|
|
|
|
#include "malfunction_central.h"
|
|
|
|
#include "malfunction_indicator.h"
|
2019-03-29 06:11:13 -07:00
|
|
|
#include "efi_gpio.h"
|
2019-07-08 00:35:41 -07:00
|
|
|
#include "periodic_thread_controller.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-02-10 20:54:41 -08:00
|
|
|
#define TEST_MIL_CODE FALSE
|
|
|
|
|
2016-12-27 21:02:03 -08:00
|
|
|
EXTERN_ENGINE;
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#define MFI_LONG_BLINK 1500
|
|
|
|
#define MFI_SHORT_BLINK 400
|
|
|
|
#define MFI_BLINK_SEPARATOR 400
|
|
|
|
#define MFI_CHECKENGINE_LIGHT 10000
|
|
|
|
|
|
|
|
static void blink_digits(int digit, int duration) {
|
|
|
|
for (int iter = 0; iter < digit; iter++) {
|
2019-08-18 12:53:38 -07:00
|
|
|
// todo: why we set LOW and then HIGH? not the other way around?
|
2016-09-13 22:01:57 -07:00
|
|
|
enginePins.checkEnginePin.setValue(0);
|
2015-07-10 06:01:56 -07:00
|
|
|
chThdSleepMilliseconds(duration);
|
2016-09-13 22:01:57 -07:00
|
|
|
enginePins.checkEnginePin.setValue(1);
|
2015-07-10 06:01:56 -07:00
|
|
|
chThdSleepMilliseconds(MFI_BLINK_SEPARATOR);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// calculate how many digits our code have
|
|
|
|
static int DigitLength(int digit) {
|
|
|
|
int i = 0;
|
|
|
|
while (digit > 0) {
|
|
|
|
digit = digit / 10;
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
// display code
|
|
|
|
static void DisplayErrorCode(int length, int code) {
|
|
|
|
// todo: I suggest we use 'itoa' method to simplify this logic
|
|
|
|
for (int iter = length - 1; iter >= 0; iter--) {
|
|
|
|
int ourDigit = (int) efiPow10(iter); // 10^0 = 1, 10^1 = 10, 10^2=100, 10^3 = 1000, ....
|
|
|
|
int digit = 1; // as we remember "0" we show as one blink
|
|
|
|
while (code >= ourDigit) {
|
|
|
|
code = code - ourDigit;
|
|
|
|
digit++;
|
|
|
|
}
|
|
|
|
if (iter % 2 == 0)
|
|
|
|
blink_digits(digit, MFI_SHORT_BLINK); // even 2,0 - long blink
|
|
|
|
else
|
|
|
|
blink_digits(digit, MFI_LONG_BLINK); // odd 3,1 - short blink
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-10 20:54:41 -08:00
|
|
|
class MILController : public PeriodicController<UTILITY_THREAD_STACK_SIZE> {
|
|
|
|
public:
|
|
|
|
MILController() : PeriodicController("MFIndicator") { }
|
|
|
|
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);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-12-21 18:18:38 -08:00
|
|
|
if (nowNt - engine->triggerCentral.triggerState.mostRecentSyncTime < MS2NT(500)) {
|
2019-08-18 12:53:38 -07:00
|
|
|
enginePins.checkEnginePin.setValue(1);
|
|
|
|
chThdSleepMilliseconds(500);
|
|
|
|
enginePins.checkEnginePin.setValue(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static error_codes_set_s localErrorCopy;
|
|
|
|
// todo: why do I not see this on a real vehicle? is this whole blinking logic not used?
|
2015-07-10 06:01:56 -07:00
|
|
|
getErrorCodes(&localErrorCopy);
|
|
|
|
for (int p = 0; p < localErrorCopy.count; p++) {
|
|
|
|
// Calculate how many digits in this integer and display error code from start to end
|
|
|
|
int code = localErrorCopy.error_codes[p];
|
|
|
|
DisplayErrorCode(DigitLength(code), code);
|
|
|
|
}
|
|
|
|
}
|
2019-02-10 20:54:41 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
static MILController instance;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-02-10 20:54:41 -08:00
|
|
|
#if TEST_MIL_CODE
|
2015-07-10 06:01:56 -07:00
|
|
|
static void testMil(void) {
|
|
|
|
addError(OBD_Engine_Coolant_Temperature_Circuit_Malfunction);
|
|
|
|
addError(OBD_Intake_Air_Temperature_Circuit_Malfunction);
|
|
|
|
}
|
2019-02-10 20:54:41 -08:00
|
|
|
#endif /* TEST_MIL_CODE */
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2016-12-27 21:02:03 -08:00
|
|
|
bool isMilEnabled() {
|
2019-12-11 14:48:55 -08:00
|
|
|
return CONFIG(malfunctionIndicatorPin) != GPIO_UNASSIGNED;
|
2016-12-27 21:02:03 -08:00
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
void initMalfunctionIndicator(void) {
|
2016-12-27 21:02:03 -08:00
|
|
|
if (!isMilEnabled()) {
|
|
|
|
return;
|
|
|
|
}
|
2019-08-18 12:53:38 -07:00
|
|
|
instance.setPeriod(10 /*ms*/);
|
2019-02-10 20:54:41 -08:00
|
|
|
instance.Start();
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-02-10 20:54:41 -08:00
|
|
|
#if TEST_MIL_CODE
|
2015-07-10 06:01:56 -07:00
|
|
|
addConsoleAction("testmil", testMil);
|
2019-02-10 20:54:41 -08:00
|
|
|
#endif /* TEST_MIL_CODE */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* EFI_MALFUNCTION_INDICATOR */
|