/* * @file vvt.cpp * * @date Jun 26, 2016 * @author Andrey Belomutskiy, (c) 2012-2020 */ #include "local_version_holder.h" #include "allsensors.h" #include "vvt.h" #include "tunerstudio_outputs.h" #include "fsio_impl.h" #include "engine_math.h" #include "pin_repository.h" #define NO_PIN_PERIOD 500 #if defined(HAS_OS_ACCESS) #error "Unexpected OS ACCESS HERE" #endif /* HAS_OS_ACCESS */ EXTERN_ENGINE; static fsio8_Map3D_u8t vvtTable1("vvt#1"); static fsio8_Map3D_u8t vvtTable2("vvt#2"); void VvtController::init(int index, int bankIndex, int camIndex, const ValueProvider3D* targetMap) { this->index = index; m_bank = bankIndex; m_cam = camIndex; // Use the same settings for the Nth cam in every bank (ie, all exhaust cams use the same PID) m_pid.initPidClass(&CONFIG(auxPid[camIndex])); m_targetMap = targetMap; } int VvtController::getPeriodMs() { return isBrainPinValid(engineConfiguration->auxPidPins[index]) ? GET_PERIOD_LIMITED(&engineConfiguration->auxPid[index]) : NO_PIN_PERIOD; } void VvtController::PeriodicTask() { if (engine->auxParametersVersion.isOld(engine->getGlobalConfigurationVersion())) { m_pid.reset(); } update(); } expected VvtController::observePlant() const { return engine->triggerCentral.getVVTPosition(m_bank, m_cam); } expected VvtController::getSetpoint() const { int rpm = GET_RPM(); float load = getFuelingLoad(PASS_ENGINE_PARAMETER_SIGNATURE); return m_targetMap->getValue(rpm, load); } expected VvtController::getOpenLoop(angle_t target) const { // TODO: could we do VVT open loop? UNUSED(target); return 0; } expected VvtController::getClosedLoop(angle_t setpoint, angle_t observation) { float retVal = m_pid.getOutput(setpoint, observation); if (engineConfiguration->isVerboseAuxPid1) { efiPrintf("aux duty: %.2f/value=%.2f/p=%.2f/i=%.2f/d=%.2f int=%.2f", retVal, observation, m_pid.getP(), m_pid.getI(), m_pid.getD(), m_pid.getIntegration()); } #if EFI_TUNER_STUDIO if (engineConfiguration->debugMode == DBG_AUX_PID_1) { m_pid.postState(&tsOutputChannels); tsOutputChannels.debugIntField3 = (int)(10 * setpoint); } #endif /* EFI_TUNER_STUDIO */ return retVal; } void VvtController::setOutput(expected outputValue) { float rpm = GET_RPM(); // todo: make this configurable? bool enabledAtCurrentRpm = rpm > engineConfiguration->cranking.rpm; if (outputValue && enabledAtCurrentRpm) { m_pwm.setSimplePwmDutyCycle(PERCENT_TO_DUTY(outputValue.Value)); } else { m_pwm.setSimplePwmDutyCycle(0); // we need to avoid accumulating iTerm while engine is not running m_pid.reset(); } } #if EFI_AUX_PID static VvtController instances[CAM_INPUTS_COUNT]; static void turnAuxPidOn(int index) { if (!isBrainPinValid(engineConfiguration->auxPidPins[index])) { return; } startSimplePwmExt(&instances[index].m_pwm, "Aux PID", &engine->executor, engineConfiguration->auxPidPins[index], &instances[index].m_pin, // todo: do we need two separate frequencies? engineConfiguration->auxPidFrequency[0], 0.1); } void startVvtControlPins() { for (int i = 0;i vvtTable1, config->vvtTable1LoadBins, config->vvtTable1RpmBins); vvtTable2.init(config->vvtTable2, config->vvtTable2LoadBins, config->vvtTable2RpmBins); for (int i = 0;i < CAM_INPUTS_COUNT;i++) { INJECT_ENGINE_REFERENCE(&instances[i]); int camIndex = i % CAMS_PER_BANK; int bankIndex = i / CAMS_PER_BANK; auto targetMap = camIndex == 0 ? &vvtTable1 : &vvtTable2; instances[i].init(i, bankIndex, camIndex, targetMap); } startVvtControlPins(); for (int i = 0;i < CAM_INPUTS_COUNT;i++) { instances[i].Start(); } } #endif