176 lines
5.0 KiB
C++
176 lines
5.0 KiB
C++
/**
|
|
* @author Andrey Belomutskiy, (c) 2012-2018
|
|
*/
|
|
#include "engine.h"
|
|
#include "tps.h"
|
|
#include "interpolation.h"
|
|
#include "analog_input.h"
|
|
|
|
EXTERN_ENGINE;
|
|
|
|
#if !EFI_PROD_CODE
|
|
static int mockTps;
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
static percent_t mockPedalPosition = MOCK_UNDEFINED;
|
|
|
|
/**
|
|
* this allows unit tests to simulate TPS position
|
|
*/
|
|
void setMockTpsPosition(percent_t tpsPosition) {
|
|
#if !EFI_PROD_CODE
|
|
mockTps = TPS_TS_CONVERSION * tpsPosition;
|
|
#endif
|
|
}
|
|
|
|
void setMockPedalPosition(percent_t value) {
|
|
mockPedalPosition = value;
|
|
}
|
|
|
|
/**
|
|
* We are using one instance for read and another for modification, this is how we get lock-free thread-safety
|
|
*
|
|
*/
|
|
static tps_roc_s states[2];
|
|
|
|
// todo if TPS_FAST_ADC
|
|
//int tpsFastAdc = 0;
|
|
|
|
static volatile int tpsRocIndex = 0;
|
|
|
|
/**
|
|
* this method is lock-free thread-safe if invoked only from one thread
|
|
*/
|
|
void saveTpsState(efitimeus_t now, float curValue) {
|
|
int tpsNextIndex = (tpsRocIndex + 1) % 2;
|
|
tps_roc_s *cur = &states[tpsRocIndex];
|
|
tps_roc_s *next = &states[tpsNextIndex];
|
|
|
|
next->prevTime = cur->curTime;
|
|
next->prevValue = cur->curValue;
|
|
next->curTime = now;
|
|
next->curValue = curValue;
|
|
|
|
int diffSysticks = overflowDiff(now, cur->curTime);
|
|
float diffSeconds = 0;// TODO: do we need this? diffSysticks * 1.0 / CH_FREQUENCY;
|
|
next->rateOfChange = (curValue - cur->curValue) / diffSeconds;
|
|
|
|
// here we update volatile index
|
|
tpsRocIndex = tpsNextIndex;
|
|
}
|
|
|
|
/**
|
|
* this read-only method is lock-free thread-safe
|
|
*/
|
|
float getTpsRateOfChange(void) {
|
|
return states[tpsRocIndex].rateOfChange;
|
|
}
|
|
|
|
/*
|
|
* Return current TPS position based on configured ADC levels, and adc
|
|
*
|
|
* */
|
|
percent_t getTpsValue(int adc DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
if (engineConfiguration->tpsMin == engineConfiguration->tpsMax) {
|
|
warning(CUSTOM_INVALID_TPS_SETTING, "Invalid TPS configuration: same value %d", engineConfiguration->tpsMin);
|
|
return NAN;
|
|
}
|
|
float result = interpolate(TPS_TS_CONVERSION * engineConfiguration->tpsMax, 100, TPS_TS_CONVERSION * engineConfiguration->tpsMin, 0, adc);
|
|
if (result < engineConfiguration->tpsErrorDetectionTooLow) {
|
|
#if EFI_PROD_CODE || defined(__DOXYGEN__)
|
|
// too much noise with simulator
|
|
warning(OBD_Throttle_Position_Sensor_Circuit_Malfunction, "TPS too low: %f", result);
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
if (result > engineConfiguration->tpsErrorDetectionTooHigh) {
|
|
#if EFI_PROD_CODE || defined(__DOXYGEN__)
|
|
// too much noise with simulator
|
|
warning(OBD_Throttle_Position_Sensor_Range_Performance_Problem, "TPS too high: %f", result);
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
// this would put the value into the 0-100 range
|
|
return maxF(0, minF(100, result));
|
|
}
|
|
|
|
/*
|
|
* Return voltage on TPS AND channel
|
|
* */
|
|
float getTPSVoltage(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
return getVoltageDivided("tps", engineConfiguration->tpsAdcChannel);
|
|
}
|
|
|
|
/*
|
|
* Return TPS ADC readings.
|
|
* We need ADC value because TunerStudio has a nice TPS configuration wizard, and this wizard
|
|
* wants a TPS value.
|
|
*/
|
|
int getTPS12bitAdc(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
#if !EFI_PROD_CODE
|
|
if (mockTps != MOCK_UNDEFINED) {
|
|
return mockTps;
|
|
}
|
|
#endif
|
|
if (engineConfiguration->tpsAdcChannel == EFI_ADC_NONE)
|
|
return -1;
|
|
#if EFI_PROD_CODE
|
|
|
|
return getAdcValue("tps10", engineConfiguration->tpsAdcChannel);
|
|
// return tpsFastAdc / 4;
|
|
#else
|
|
return 0;
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
/**
|
|
* @brief Position on physical primary TPS
|
|
*/
|
|
static percent_t getPrimatyRawTPS(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
percent_t tpsValue = getTpsValue(getTPS12bitAdc(PASS_ENGINE_PARAMETER_SIGNATURE) PASS_ENGINE_PARAMETER_SUFFIX);
|
|
return tpsValue;
|
|
}
|
|
|
|
#define NO_TPS_MAGIC_VALUE 66.611
|
|
|
|
bool hasPedalPositionSensor(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
return engineConfiguration->pedalPositionChannel != EFI_ADC_NONE;
|
|
}
|
|
|
|
percent_t getPedalPosition(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
if (mockPedalPosition != MOCK_UNDEFINED) {
|
|
return mockPedalPosition;
|
|
}
|
|
float voltage = getVoltageDivided("pPS", engineConfiguration->pedalPositionChannel);
|
|
float result = interpolate(engineConfiguration->throttlePedalUpVoltage, 0, engineConfiguration->throttlePedalWOTVoltage, 100, voltage);
|
|
|
|
// this would put the value into the 0-100 range
|
|
return maxF(0, minF(100, result));
|
|
}
|
|
|
|
bool hasTpsSensor(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
return engineConfiguration->tpsAdcChannel != EFI_ADC_NONE;
|
|
}
|
|
|
|
percent_t getTPS(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
if (!hasTpsSensor(PASS_ENGINE_PARAMETER_SIGNATURE))
|
|
return NO_TPS_MAGIC_VALUE;
|
|
// todo: if (config->isDualTps)
|
|
// todo: blah blah
|
|
// todo: if two TPS do not match - show OBD code via malfunction_central.c
|
|
|
|
return getPrimatyRawTPS(PASS_ENGINE_PARAMETER_SIGNATURE);
|
|
}
|
|
|
|
void setBosch0280750009(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
// see http://rusefi.com/wiki/index.php?title=Vehicle:VW_Passat_2002_1.8
|
|
engineConfiguration->tpsMin = 159;
|
|
engineConfiguration->tpsMax = 957;
|
|
|
|
// todo: add 2nd TPS sensor calibration
|
|
}
|
|
|
|
int convertVoltageTo10bitADC(float voltage) {
|
|
// divided by 2 because of voltage divider, then converted into 10bit ADC value (TunerStudio format)
|
|
return (int) (voltage / 2 * 1024 / 3.3);
|
|
}
|