/** * @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); }