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"
|
2023-09-06 06:39:38 -07:00
|
|
|
#include "bench_test.h"
|
2021-03-14 16:31:46 -07:00
|
|
|
|
|
|
|
#define NO_PIN_PERIOD 500
|
|
|
|
|
2022-08-08 19:51:51 -07:00
|
|
|
using vvt_map_t = Map3D<SCRIPT_TABLE_8, SCRIPT_TABLE_8, int8_t, uint16_t, uint16_t>;
|
2021-12-20 18:58:53 -08:00
|
|
|
|
2024-05-03 19:13:30 -07:00
|
|
|
static vvt_map_t vvtTable1{"vvt1"};
|
|
|
|
static vvt_map_t vvtTable2{"vvt2"};
|
2021-03-14 16:31:46 -07:00
|
|
|
|
2023-11-01 10:02:20 -07:00
|
|
|
VvtController::VvtController(int p_index)
|
|
|
|
: index(p_index)
|
|
|
|
, m_bank(BANK_BY_INDEX(p_index))
|
|
|
|
, m_cam(CAM_BY_INDEX(p_index))
|
2023-07-19 22:16:29 -07:00
|
|
|
{
|
|
|
|
}
|
2021-03-25 04:39:23 -07:00
|
|
|
|
2023-07-19 22:16:29 -07:00
|
|
|
void VvtController::init(const ValueProvider3D* targetMap, IPwm* pwm) {
|
2021-03-25 04:39:23 -07:00
|
|
|
// Use the same settings for the Nth cam in every bank (ie, all exhaust cams use the same PID)
|
2023-10-07 09:54:45 -07:00
|
|
|
m_pid.initPidClass(&engineConfiguration->auxPid[m_cam]);
|
2021-03-25 04:39:23 -07:00
|
|
|
|
2021-03-14 16:31:46 -07:00
|
|
|
m_targetMap = targetMap;
|
2023-07-19 22:16:29 -07:00
|
|
|
m_pwm = pwm;
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
2023-07-19 22:16:29 -07:00
|
|
|
void VvtController::onFastCallback() {
|
|
|
|
if (!m_pwm || !m_targetMap) {
|
|
|
|
// not init yet
|
|
|
|
return;
|
|
|
|
}
|
2021-03-14 16:31:46 -07:00
|
|
|
|
2023-09-26 17:49:33 -07:00
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VvtController::onConfigurationChange(engine_configuration_s const * previousConfig) {
|
2024-05-03 22:10:42 -07:00
|
|
|
if (!previousConfig || !m_pid.isSame(&previousConfig->auxPid[m_cam])) {
|
2021-03-14 16:31:46 -07:00
|
|
|
m_pid.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-19 18:06:09 -07:00
|
|
|
expected<angle_t> VvtController::observePlant() {
|
2022-04-16 13:39:52 -07:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2021-03-25 04:39:23 -07:00
|
|
|
return engine->triggerCentral.getVVTPosition(m_bank, m_cam);
|
2022-04-16 13:39:52 -07:00
|
|
|
#else
|
|
|
|
return unexpected;
|
|
|
|
#endif // EFI_SHAFT_POSITION_INPUT
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
2021-11-23 12:52:43 -08:00
|
|
|
expected<angle_t> VvtController::getSetpoint() {
|
2022-01-20 19:31:07 -08:00
|
|
|
int rpm = Sensor::getOrZero(SensorType::Rpm);
|
2023-11-10 16:07:17 -08:00
|
|
|
bool enabled = rpm > engineConfiguration->vvtControlMinRpm
|
|
|
|
&& engine->rpmCalculator.getSecondsSinceEngineStart(getTimeNowNt()) > engineConfiguration->vvtActivationDelayMs / MS_PER_SECOND
|
|
|
|
;
|
2023-11-11 04:07:08 -08:00
|
|
|
if (!enabled) {
|
2023-11-10 16:07:17 -08:00
|
|
|
return unexpected;
|
2023-11-11 04:07:08 -08:00
|
|
|
}
|
2023-11-10 16:07:17 -08:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
float load = getFuelingLoad();
|
2021-11-02 12:03:57 -07:00
|
|
|
float target = m_targetMap->getValue(rpm, load);
|
|
|
|
|
|
|
|
#if EFI_TUNER_STUDIO
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.vvtTargets[index] = target;
|
2021-11-02 12:03:57 -07:00
|
|
|
#endif
|
|
|
|
|
2023-07-01 21:58:40 -07:00
|
|
|
vvtTarget = target;
|
|
|
|
|
2021-11-02 12:03:57 -07:00
|
|
|
return target;
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
2021-11-23 12:52:43 -08:00
|
|
|
expected<percent_t> VvtController::getOpenLoop(angle_t target) {
|
2021-03-14 16:31:46 -07:00
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2022-08-09 13:22:55 -07:00
|
|
|
static bool shouldInvertVvt(int camIndex) {
|
|
|
|
// grumble grumble, can't do an array of bits in c++
|
|
|
|
switch (camIndex) {
|
|
|
|
case 0: return engineConfiguration->invertVvtControlIntake;
|
|
|
|
case 1: return engineConfiguration->invertVvtControlExhaust;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-07-06 16:59:40 -07:00
|
|
|
expected<percent_t> VvtController::getClosedLoop(angle_t target, angle_t observation) {
|
2022-08-20 17:12:32 -07:00
|
|
|
// User labels say "advance" and "retard"
|
|
|
|
// "advance" means that additional solenoid duty makes indicated VVT position more positive
|
|
|
|
// "retard" means that additional solenoid duty makes indicated VVT position more negative
|
|
|
|
bool isInverted = shouldInvertVvt(m_cam);
|
|
|
|
m_pid.setErrorAmplification(isInverted ? -1.0f : 1.0f);
|
2024-03-04 14:58:24 -08:00
|
|
|
|
2021-07-06 16:59:40 -07:00
|
|
|
float retVal = m_pid.getOutput(target, observation);
|
2021-03-14 16:31:46 -07:00
|
|
|
|
|
|
|
#if EFI_TUNER_STUDIO
|
2022-08-16 21:47:31 -07:00
|
|
|
m_pid.postState(engine->outputChannels.vvtStatus[index]);
|
2021-03-14 16:31:46 -07:00
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
|
|
|
|
2022-08-20 17:12:32 -07:00
|
|
|
return retVal;
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void VvtController::setOutput(expected<percent_t> outputValue) {
|
2022-04-16 13:49:59 -07:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2023-07-01 21:58:40 -07:00
|
|
|
vvtOutput = outputValue.value_or(0);
|
|
|
|
|
2023-11-10 16:07:17 -08:00
|
|
|
if (outputValue) {
|
2023-07-19 22:16:29 -07:00
|
|
|
m_pwm->setSimplePwmDutyCycle(PERCENT_TO_DUTY(outputValue.Value));
|
2021-03-14 16:31:46 -07:00
|
|
|
} else {
|
2023-07-19 22:16:29 -07:00
|
|
|
m_pwm->setSimplePwmDutyCycle(0);
|
2021-03-14 16:31:46 -07:00
|
|
|
|
|
|
|
// we need to avoid accumulating iTerm while engine is not running
|
|
|
|
m_pid.reset();
|
|
|
|
}
|
2022-04-16 13:49:59 -07:00
|
|
|
#endif // EFI_SHAFT_POSITION_INPUT
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
2023-03-04 23:07:34 -08:00
|
|
|
#if EFI_VVT_PID
|
2021-03-14 16:31:46 -07:00
|
|
|
|
2022-11-20 21:10:35 -08:00
|
|
|
static const char *vvtOutputNames[CAM_INPUTS_COUNT] = {
|
2022-11-20 13:42:44 -08:00
|
|
|
"Vvt Output#1",
|
|
|
|
#if CAM_INPUTS_COUNT > 1
|
|
|
|
"Vvt Output#2",
|
|
|
|
#endif
|
|
|
|
#if CAM_INPUTS_COUNT > 2
|
|
|
|
"Vvt Output#3",
|
|
|
|
#endif
|
|
|
|
#if CAM_INPUTS_COUNT > 3
|
|
|
|
"Vvt Output#4",
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2023-07-19 22:16:29 -07:00
|
|
|
static OutputPin vvtPins[CAM_INPUTS_COUNT];
|
2024-08-13 14:36:51 -07:00
|
|
|
static SimplePwm vvtPwms[CAM_INPUTS_COUNT] = { "VVT1", "VVT2", "VVT3", "VVT4" };
|
2021-03-14 16:31:46 -07:00
|
|
|
|
2023-09-06 05:11:39 -07:00
|
|
|
OutputPin* getVvtOutputPin(int index) {
|
2023-07-19 22:16:29 -07:00
|
|
|
return &vvtPins[index];
|
2023-09-06 05:11:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void applyVvtPinState(int stateIndex, PwmConfig *state) /* pwm_gen_callback */ {
|
|
|
|
OutputPin *output = state->outputPins[0];
|
2023-09-06 06:39:38 -07:00
|
|
|
if (output == getOutputOnTheBenchTest()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
state->applyPwmValue(output, stateIndex);
|
2023-09-06 05:11:39 -07:00
|
|
|
}
|
|
|
|
|
2022-11-20 13:42:44 -08:00
|
|
|
static void turnVvtPidOn(int index) {
|
2021-10-28 20:51:46 -07:00
|
|
|
if (!isBrainPinValid(engineConfiguration->vvtPins[index])) {
|
2021-03-14 16:31:46 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-07-19 22:16:29 -07:00
|
|
|
startSimplePwmExt(&vvtPwms[index], vvtOutputNames[index],
|
2021-03-14 16:31:46 -07:00
|
|
|
&engine->executor,
|
2021-10-28 20:51:46 -07:00
|
|
|
engineConfiguration->vvtPins[index],
|
2023-09-06 05:11:39 -07:00
|
|
|
getVvtOutputPin(index),
|
2023-09-06 06:39:38 -07:00
|
|
|
engineConfiguration->vvtOutputFrequency, 0.1,
|
|
|
|
applyVvtPinState);
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void startVvtControlPins() {
|
|
|
|
for (int i = 0;i <CAM_INPUTS_COUNT;i++) {
|
2022-11-20 13:42:44 -08:00
|
|
|
turnVvtPidOn(i);
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void stopVvtControlPins() {
|
|
|
|
for (int i = 0;i < CAM_INPUTS_COUNT;i++) {
|
2023-09-06 05:11:39 -07:00
|
|
|
getVvtOutputPin(i)->deInit();
|
2021-03-14 16:31:46 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-27 19:21:18 -07:00
|
|
|
void initVvtActuators() {
|
2023-06-01 11:09:30 -07:00
|
|
|
if (engineConfiguration->vvtControlMinRpm < engineConfiguration->cranking.rpm) {
|
|
|
|
engineConfiguration->vvtControlMinRpm = engineConfiguration->cranking.rpm;
|
|
|
|
}
|
2021-03-14 16:31:46 -07:00
|
|
|
|
2024-03-04 14:58:24 -08:00
|
|
|
vvtTable1.initTable(config->vvtTable1, config->vvtTable1RpmBins, config->vvtTable1LoadBins);
|
|
|
|
vvtTable2.initTable(config->vvtTable2, config->vvtTable2RpmBins, config->vvtTable2LoadBins);
|
2021-03-14 16:31:46 -07:00
|
|
|
|
|
|
|
|
2023-07-19 22:16:29 -07:00
|
|
|
engine->module<VvtController1>()->init(&vvtTable1, &vvtPwms[0]);
|
|
|
|
engine->module<VvtController2>()->init(&vvtTable2, &vvtPwms[1]);
|
|
|
|
engine->module<VvtController3>()->init(&vvtTable1, &vvtPwms[2]);
|
|
|
|
engine->module<VvtController4>()->init(&vvtTable2, &vvtPwms[3]);
|
2021-03-14 16:31:46 -07:00
|
|
|
|
|
|
|
startVvtControlPins();
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|