/* * @file dynoview.cpp * * @date Nov 29, 2020 * @author Alexandru Miculescu, (c) 2012-2020 */ #include "engine.h" #if EFI_DYNO_VIEW #include "dynoview.h" #include "vehicle_speed.h" static Logging *logger; #if EFI_TUNER_STUDIO #include "tunerstudio_outputs.h" extern TunerStudioOutputChannels tsOutputChannels; #endif /* EFI_TUNER_STUDIO */ EXTERN_ENGINE; DynoView dynoInstance; void DynoView::update(vssSrc src) { efitimeus_t timeNow, deltaTime = 0.0; float speed,deltaSpeed = 0.0; timeNow = getTimeNowUs(); speed = getVehicleSpeed(); if (src == ICU) { speed = efiRound(speed,1.0); } else { //use speed with 0.001 precision from source CAN speed = efiRound(speed,0.001); } if(timeStamp != 0) { if (vss != speed) { deltaTime = timeNow - timeStamp; if (vss > speed) { deltaSpeed = (vss - speed); direction = 1; //decceleration } else { deltaSpeed = speed - vss; direction = 0; //acceleration } //save data timeStamp = timeNow; vss = speed; } //updating here would display acceleration = 0 at constant speed updateAcceleration(deltaTime, deltaSpeed); #if EFI_TUNER_STUDIO if (CONFIG(debugMode) == DBG_LOGIC_ANALYZER) { tsOutputChannels.debugIntField1 = deltaTime; tsOutputChannels.debugFloatField1 = vss; tsOutputChannels.debugFloatField2 = speed; tsOutputChannels.debugFloatField3 = deltaSpeed; tsOutputChannels.debugFloatField4 = acceleration; } #endif /* EFI_TUNER_STUDIO */ updateHP(); } else { //ensure we grab init values timeStamp = timeNow; vss = speed; } } /** * input units: deltaSpeed in km/h * deltaTime in uS */ void DynoView::updateAcceleration(efitimeus_t deltaTime, float deltaSpeed) { if (deltaSpeed != 0.0) { acceleration = ((deltaSpeed / 3.6) / (deltaTime / 1000000.0)); if (direction) { //decceleration acceleration *= -1; } } else { acceleration = 0.0; } } /** * E = m*a * ex. 900 (kg) * 1.5 (m/s^2) = 1350N * P = F*V * 1350N * 35(m/s) = 47250Watt (35 m/s is the final velocity) * 47250 * (1HP/746W) = 63HP * https://www.youtube.com/watch?v=FnN2asvFmIs * we do not take resistence into account right now. */ void DynoView::updateHP() { //these are actually at the wheel //we would need final drive to calcualte the correct torque at the wheel if (acceleration != 0) { engineForce = CONFIG(vehicleWeight) * acceleration; enginePower = engineForce * (vss / 3.6); engineHP = enginePower / 746; if (isValidRpm(GET_RPM())) { engineTorque = ((engineHP * 5252) / GET_RPM()); } } else { //we should calculate static power } } #if EFI_UNIT_TEST void DynoView::setAcceleration(float a) { acceleration = a; } #endif float DynoView::getAcceleration() { return acceleration; } int DynoView::getEngineForce() { return engineForce; } int DynoView::getEnginePower() { return (enginePower/1000); } int DynoView::getEngineHP() { return engineHP; } int DynoView::getEngineTorque() { return (engineTorque/0.73756); } float getDynoviewAcceleration(DECLARE_ENGINE_PARAMETER_SIGNATURE) { return dynoInstance.getAcceleration(); } int getDynoviewPower(DECLARE_ENGINE_PARAMETER_SIGNATURE) { return dynoInstance.getEnginePower(); } /** * Periodic update function called from SlowCallback. * Only updates if we have Vss from input pin. */ void updateDynoView(DECLARE_ENGINE_PARAMETER_SIGNATURE) { if ((CONFIG(vehicleSpeedSensorInputPin) != GPIO_UNASSIGNED) && (!CONFIG(enableCanVss))) { dynoInstance.update(ICU); } } /** * This function is called after every CAN msg received, we process it * as soon as we can to be more acurate. */ void updateDynoViewCan(DECLARE_ENGINE_PARAMETER_SIGNATURE) { if (!CONFIG(enableCanVss)) { return; } dynoInstance.update(CAN); } void initDynoView(Logging *sharedLogger DECLARE_ENGINE_PARAMETER_SUFFIX) { logger = sharedLogger; } #endif /* EFI_DYNO_VIEW */