/* * @file aux_pid.cpp * * This class is a copy-paste of alternator_controller.cpp TODO: do something about it? extract more common logic? * * @date Jun 26, 2016 * @author Andrey Belomutskiy, (c) 2012-2020 */ #include "aux_pid.h" #include "local_version_holder.h" #include "allsensors.h" #if EFI_AUX_PID #include "pwm_generator.h" #include "tunerstudio_configuration.h" #include "fsio_impl.h" #include "engine_math.h" #include "pin_repository.h" #include "periodic_task.h" #define NO_PIN_PERIOD 500 #if defined(HAS_OS_ACCESS) #error "Unexpected OS ACCESS HERE" #endif /* HAS_OS_ACCESS */ EXTERN_ENGINE; extern fsio8_Map3D_f32t fsioTable1; // todo: this is to some extent a copy-paste of alternator_controller. maybe same loop // for all PIDs? static Logging *logger; static bool isEnabled(int index) { // todo: implement bit arrays for configuration switch(index) { case 0: return engineConfiguration->activateAuxPid1; case 1: return engineConfiguration->activateAuxPid2; case 2: return engineConfiguration->activateAuxPid3; default: return engineConfiguration->activateAuxPid4; } } class AuxPidController : public PeriodicTimerController { public: SimplePwm auxPidPwm; OutputPin auxOutputPin; void init(int index) { this->index = index; auxPid.initPidClass(&persistentState.persistentConfiguration.engineConfiguration.auxPid[index]); table = getFSIOTable(index); } int getPeriodMs() override { return engineConfiguration->auxPidPins[index] == GPIO_UNASSIGNED ? NO_PIN_PERIOD : GET_PERIOD_LIMITED(&engineConfiguration->auxPid[index]); } void PeriodicTask() override { if (engine->auxParametersVersion.isOld(engine->getGlobalConfigurationVersion())) { auxPid.reset(); } float rpm = GET_RPM_VALUE; // todo: make this configurable? bool enabledAtCurrentRpm = rpm > engineConfiguration->cranking.rpm; if (!enabledAtCurrentRpm) { // we need to avoid accumulating iTerm while engine is not running auxPid.reset(); return; } float value = engine->triggerCentral.getVVTPosition(); float targetValue = table->getValue(rpm, getEngineLoadT(PASS_ENGINE_PARAMETER_SIGNATURE)); percent_t pwm = auxPid.getOutput(targetValue, value); if (engineConfiguration->isVerboseAuxPid1) { scheduleMsg(logger, "aux duty: %.2f/value=%.2f/p=%.2f/i=%.2f/d=%.2f int=%.2f", pwm, value, auxPid.getP(), auxPid.getI(), auxPid.getD(), auxPid.getIntegration()); } if (engineConfiguration->debugMode == DBG_AUX_PID_1) { #if EFI_TUNER_STUDIO auxPid.postState(&tsOutputChannels); tsOutputChannels.debugIntField3 = (int)(10 * targetValue); #endif /* EFI_TUNER_STUDIO */ } auxPidPwm.setSimplePwmDutyCycle(PERCENT_TO_DUTY(pwm)); } private: Pid auxPid; int index = 0; ValueProvider3D *table = nullptr; }; static AuxPidController instances[AUX_PID_COUNT]; static void turnAuxPidOn(int index) { if (!isEnabled(index)) { return; } if (engineConfiguration->auxPidPins[index] == GPIO_UNASSIGNED) { return; } startSimplePwmExt(&instances[index].auxPidPwm, "Aux PID", &engine->executor, engineConfiguration->auxPidPins[index], &instances[index].auxOutputPin, engineConfiguration->auxPidFrequency[index], 0.1); } void startAuxPins(void) { for (int i = 0;i