2021-03-14 16:31:46 -07:00
|
|
|
/*
|
|
|
|
* @file vvt.cpp
|
|
|
|
*
|
|
|
|
* @date Jun 26, 2016
|
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
|
|
|
*/
|
|
|
|
|
2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2021-03-14 16:31:46 -07:00
|
|
|
#include "local_version_holder.h"
|
|
|
|
#include "vvt.h"
|
|
|
|
|
|
|
|
#include "fsio_impl.h"
|
|
|
|
|
|
|
|
#define NO_PIN_PERIOD 500
|
|
|
|
|
|
|
|
#if defined(HAS_OS_ACCESS)
|
|
|
|
#error "Unexpected OS ACCESS HERE"
|
|
|
|
#endif /* HAS_OS_ACCESS */
|
|
|
|
|
2021-06-03 11:30:25 -07:00
|
|
|
static fsio8_Map3D_u8t vvtTable1;
|
|
|
|
static fsio8_Map3D_u8t vvtTable2;
|
2021-03-14 16:31:46 -07:00
|
|
|
|
2021-03-25 04:39:23 -07:00
|
|
|
void VvtController::init(int index, int bankIndex, int camIndex, const ValueProvider3D* targetMap) {
|
2021-03-14 16:31:46 -07:00
|
|
|
this->index = index;
|
2021-03-25 04:39:23 -07:00
|
|
|
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]));
|
|
|
|
|
2021-03-14 16:31:46 -07:00
|
|
|
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<angle_t> VvtController::observePlant() const {
|
2021-03-25 04:39:23 -07:00
|
|
|
return engine->triggerCentral.getVVTPosition(m_bank, m_cam);
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
expected<angle_t> VvtController::getSetpoint() const {
|
|
|
|
int rpm = GET_RPM();
|
|
|
|
float load = getFuelingLoad(PASS_ENGINE_PARAMETER_SIGNATURE);
|
|
|
|
return m_targetMap->getValue(rpm, load);
|
|
|
|
}
|
|
|
|
|
|
|
|
expected<percent_t> VvtController::getOpenLoop(angle_t target) const {
|
|
|
|
// TODO: could we do VVT open loop?
|
2021-03-15 07:23:19 -07:00
|
|
|
UNUSED(target);
|
2021-03-14 16:31:46 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-07-06 16:59:40 -07:00
|
|
|
expected<percent_t> VvtController::getClosedLoop(angle_t target, angle_t observation) {
|
|
|
|
float retVal = m_pid.getOutput(target, observation);
|
2021-03-14 16:31:46 -07:00
|
|
|
|
|
|
|
if (engineConfiguration->isVerboseAuxPid1) {
|
2021-04-20 11:09:41 -07:00
|
|
|
efiPrintf("aux duty: %.2f/value=%.2f/p=%.2f/i=%.2f/d=%.2f int=%.2f", retVal, observation,
|
2021-03-14 16:31:46 -07:00
|
|
|
m_pid.getP(), m_pid.getI(), m_pid.getD(), m_pid.getIntegration());
|
|
|
|
}
|
|
|
|
|
|
|
|
#if EFI_TUNER_STUDIO
|
2021-07-06 16:59:40 -07:00
|
|
|
static debug_mode_e debugModeByIndex[4] = {DBG_VVT_1_PID, DBG_VVT_2_PID, DBG_VVT_3_PID, DBG_VVT_4_PID};
|
|
|
|
|
|
|
|
if (engineConfiguration->debugMode == debugModeByIndex[index]) {
|
2021-03-14 16:31:46 -07:00
|
|
|
m_pid.postState(&tsOutputChannels);
|
2021-07-06 16:59:40 -07:00
|
|
|
tsOutputChannels.debugIntField3 = (int)(10 * target);
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
|
|
|
|
|
|
|
return retVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VvtController::setOutput(expected<percent_t> 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 <CAM_INPUTS_COUNT;i++) {
|
|
|
|
turnAuxPidOn(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void stopVvtControlPins() {
|
|
|
|
for (int i = 0;i < CAM_INPUTS_COUNT;i++) {
|
|
|
|
instances[i].m_pin.deInit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-20 11:09:41 -07:00
|
|
|
void initAuxPid() {
|
2021-03-14 16:31:46 -07:00
|
|
|
|
|
|
|
vvtTable1.init(config->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]);
|
|
|
|
|
2021-03-25 04:39:23 -07:00
|
|
|
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);
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
startVvtControlPins();
|
|
|
|
|
|
|
|
for (int i = 0;i < CAM_INPUTS_COUNT;i++) {
|
|
|
|
instances[i].Start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|