/** * @author Andrey Belomutskiy, (c) 2012-2017 */ #include "main.h" #include "tps.h" #include "engine_configuration.h" #include "interpolation.h" #include "adc_inputs.h" #include "allsensors.h" #if !EFI_PROD_CODE int mockTps; #endif EXTERN_ENGINE; /** * 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_S) { 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) { warning(OBD_Throttle_Position_Sensor_Circuit_Malfunction, "TPS too low: %f", result); } if (result > engineConfiguration->tpsErrorDetectionTooHigh) { warning(OBD_Throttle_Position_Sensor_Range_Performance_Problem, "TPS too high: %f", result); } // 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_F) { 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_F) { #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_F) { percent_t tpsValue = getTpsValue(getTPS12bitAdc(PASS_ENGINE_PARAMETER_F) PASS_ENGINE_PARAMETER); return tpsValue; } #define NO_TPS_MAGIC_VALUE 66.611 bool hasPedalPositionSensor(DECLARE_ENGINE_PARAMETER_F) { return engineConfiguration->pedalPositionChannel != EFI_ADC_NONE; } percent_t getPedalPosition(DECLARE_ENGINE_PARAMETER_F) { float voltage = getVoltageDivided("pPS", engineConfiguration->pedalPositionChannel); float result = interpolate(engineConfiguration->pedalPositionMin, 0, engineConfiguration->pedalPositionMax, 100, voltage); // this would put the value into the 0-100 range return maxF(0, minF(100, result)); } bool hasTpsSensor(DECLARE_ENGINE_PARAMETER_F) { return engineConfiguration->tpsAdcChannel != EFI_ADC_NONE; } percent_t getTPS(DECLARE_ENGINE_PARAMETER_F) { if (!hasTpsSensor(PASS_ENGINE_PARAMETER_F)) 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_F); } 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); }