2020-02-02 00:56:21 -08:00
|
|
|
/*
|
|
|
|
* boost_control.cpp
|
|
|
|
*
|
|
|
|
* Created on: 13. des. 2019
|
|
|
|
* Author: Ola Ruud
|
|
|
|
*/
|
|
|
|
#include "global.h"
|
|
|
|
|
|
|
|
#if EFI_BOOST_CONTROL
|
|
|
|
|
|
|
|
#if EFI_TUNER_STUDIO
|
2020-05-25 10:02:05 -07:00
|
|
|
#include "tunerstudio_outputs.h"
|
2020-02-02 00:56:21 -08:00
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
|
|
|
#include "engine.h"
|
|
|
|
#include "boost_control.h"
|
2020-04-05 16:33:33 -07:00
|
|
|
#include "sensor.h"
|
2020-02-02 00:56:21 -08:00
|
|
|
#include "map.h"
|
|
|
|
#include "io_pins.h"
|
|
|
|
#include "engine_configuration.h"
|
|
|
|
#include "pwm_generator_logic.h"
|
|
|
|
#include "engine_controller.h"
|
|
|
|
#include "pin_repository.h"
|
2020-04-26 14:40:12 -07:00
|
|
|
#include "pwm_generator_logic.h"
|
2020-02-02 00:56:21 -08:00
|
|
|
#include "pid_auto_tune.h"
|
|
|
|
#include "local_version_holder.h"
|
|
|
|
#define NO_PIN_PERIOD 500
|
|
|
|
|
|
|
|
#if defined(HAS_OS_ACCESS)
|
|
|
|
#error "Unexpected OS ACCESS HERE"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
EXTERN_ENGINE;
|
|
|
|
|
|
|
|
static Logging *logger;
|
2020-09-10 15:44:10 -07:00
|
|
|
static boostOpenLoop_Map3D_t boostMapOpen("boostmapopen");
|
|
|
|
static boostOpenLoop_Map3D_t boostMapClosed("boostmapclosed");
|
2020-02-02 00:56:21 -08:00
|
|
|
static SimplePwm boostPwmControl("boost");
|
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
void BoostController::init(SimplePwm* pwm, const ValueProvider3D* openLoopMap, const ValueProvider3D* closedLoopTargetMap, pid_s* pidParams) {
|
|
|
|
m_pwm = pwm;
|
|
|
|
m_openLoopMap = openLoopMap;
|
|
|
|
m_closedLoopTargetMap = closedLoopTargetMap;
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
m_pid.initPidClass(pidParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BoostController::reset() {
|
|
|
|
m_shouldResetPid = true;
|
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
void BoostController::onConfigurationChange(pid_s* previousConfiguration) {
|
|
|
|
if (!m_pid.isSame(previousConfiguration)) {
|
|
|
|
m_shouldResetPid = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int BoostController::getPeriodMs() {
|
|
|
|
return GET_PERIOD_LIMITED(&engineConfiguration->boostPid);
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
expected<float> BoostController::observePlant() const {
|
|
|
|
float map = getMap(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2020-03-23 20:06:52 -07:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
if (cisnan(map)) {
|
|
|
|
return unexpected;
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
2020-02-04 17:17:31 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
return map;
|
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
expected<float> BoostController::getSetpoint() const {
|
|
|
|
float rpm = GET_RPM();
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
auto tps = Sensor::get(SensorType::DriverThrottleIntent);
|
2020-02-04 17:17:31 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
if (!tps) {
|
|
|
|
return unexpected;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_closedLoopTargetMap) {
|
|
|
|
return unexpected;
|
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-09-10 15:44:10 -07:00
|
|
|
return m_closedLoopTargetMap->getValue(rpm / RPM_1_BYTE_PACKING_MULT, tps.Value / TPS_1_BYTE_PACKING_MULT);
|
2020-05-06 05:40:42 -07:00
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
expected<percent_t> BoostController::getOpenLoop(float target) const {
|
|
|
|
// Boost control open loop doesn't care about target - only MAP/RPM
|
|
|
|
UNUSED(target);
|
2020-04-05 16:33:33 -07:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
float rpm = GET_RPM();
|
|
|
|
auto tps = Sensor::get(SensorType::DriverThrottleIntent);
|
|
|
|
|
|
|
|
if (!tps) {
|
|
|
|
return unexpected;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_openLoopMap) {
|
|
|
|
return unexpected;
|
|
|
|
}
|
2020-02-04 17:17:31 -08:00
|
|
|
|
2020-09-10 15:44:10 -07:00
|
|
|
percent_t openLoop = m_openLoopMap->getValue(rpm / RPM_1_BYTE_PACKING_MULT, tps.Value / TPS_1_BYTE_PACKING_MULT);
|
2020-02-04 17:17:31 -08:00
|
|
|
|
|
|
|
#if EFI_TUNER_STUDIO
|
2020-05-06 05:40:42 -07:00
|
|
|
if (engineConfiguration->debugMode == DBG_BOOST) {
|
|
|
|
tsOutputChannels.debugFloatField1 = openLoop;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return openLoop;
|
|
|
|
}
|
2020-02-04 17:17:31 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
expected<percent_t> BoostController::getClosedLoop(float target, float manifoldPressure) {
|
|
|
|
// If we're in open loop only mode, make no closed loop correction.
|
|
|
|
if (engineConfiguration->boostType != CLOSED_LOOP) {
|
|
|
|
return 0;
|
2020-03-26 16:22:14 -07:00
|
|
|
}
|
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
// Reset PID if requested
|
|
|
|
if (m_shouldResetPid) {
|
|
|
|
m_pid.reset();
|
|
|
|
m_shouldResetPid = false;
|
2020-02-04 17:17:31 -08:00
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
// If the engine isn't running, don't correct.
|
|
|
|
if (GET_RPM() == 0) {
|
|
|
|
m_pid.reset();
|
|
|
|
return 0;
|
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
float closedLoop = m_pid.getOutput(target, manifoldPressure);
|
|
|
|
|
|
|
|
#if EFI_TUNER_STUDIO
|
|
|
|
if (engineConfiguration->debugMode == DBG_BOOST) {
|
|
|
|
m_pid.postState(&tsOutputChannels);
|
|
|
|
tsOutputChannels.debugFloatField2 = closedLoop;
|
|
|
|
}
|
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
|
|
|
|
|
|
|
return closedLoop;
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
void BoostController::setOutput(expected<float> output) {
|
|
|
|
// TODO: hook up safe duty cycle
|
|
|
|
float duty = PERCENT_TO_DUTY(output.value_or(/*CONFIG(boostControlSafeDutyCycle)*/ 0));
|
|
|
|
|
|
|
|
if (m_pwm) {
|
|
|
|
m_pwm->setSimplePwmDutyCycle(duty);
|
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
void BoostController::PeriodicTask() {
|
|
|
|
m_pid.iTermMin = -50;
|
|
|
|
m_pid.iTermMax = 50;
|
|
|
|
|
|
|
|
update();
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
2020-05-06 05:40:42 -07:00
|
|
|
|
|
|
|
BoostController boostController;
|
2020-02-02 00:56:21 -08:00
|
|
|
|
|
|
|
void setDefaultBoostParameters(DECLARE_CONFIG_PARAMETER_SIGNATURE) {
|
|
|
|
engineConfiguration->isBoostControlEnabled = true;
|
|
|
|
engineConfiguration->boostPwmFrequency = 55;
|
|
|
|
engineConfiguration->boostPid.offset = 0;
|
|
|
|
engineConfiguration->boostPid.pFactor = 0.5;
|
|
|
|
engineConfiguration->boostPid.iFactor = 0.3;
|
|
|
|
engineConfiguration->boostPid.periodMs = 100;
|
|
|
|
engineConfiguration->boostPid.maxValue = 99;
|
|
|
|
engineConfiguration->boostPid.minValue = -99;
|
|
|
|
engineConfiguration->boostControlPin = GPIO_UNASSIGNED;
|
|
|
|
engineConfiguration->boostControlPinMode = OM_DEFAULT;
|
|
|
|
|
|
|
|
setLinearCurve(config->boostRpmBins, 0, 8000 / RPM_1_BYTE_PACKING_MULT, 1);
|
|
|
|
setLinearCurve(config->boostTpsBins, 0, 100 / TPS_1_BYTE_PACKING_MULT, 1);
|
2020-05-06 05:40:42 -07:00
|
|
|
|
|
|
|
for (int loadIndex = 0; loadIndex < BOOST_LOAD_COUNT; loadIndex++) {
|
|
|
|
for (int rpmIndex = 0; rpmIndex < BOOST_RPM_COUNT; rpmIndex++) {
|
|
|
|
config->boostTableOpenLoop[loadIndex][rpmIndex] = config->boostTpsBins[loadIndex];
|
2020-02-04 17:17:31 -08:00
|
|
|
config->boostTableClosedLoop[loadIndex][rpmIndex] = config->boostTpsBins[loadIndex];
|
|
|
|
}
|
|
|
|
}
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
void startBoostPin() {
|
2020-03-23 20:06:52 -07:00
|
|
|
#if !EFI_UNIT_TEST
|
2020-02-02 00:56:21 -08:00
|
|
|
if (CONFIG(boostControlPin) == GPIO_UNASSIGNED){
|
|
|
|
return;
|
|
|
|
}
|
2020-02-04 17:17:31 -08:00
|
|
|
|
|
|
|
startSimplePwmExt(
|
|
|
|
&boostPwmControl,
|
|
|
|
"Boost",
|
|
|
|
&engine->executor,
|
|
|
|
CONFIG(boostControlPin),
|
|
|
|
&enginePins.boostPin,
|
|
|
|
engineConfiguration->boostPwmFrequency,
|
2020-03-25 15:14:09 -07:00
|
|
|
0.5f
|
2020-02-04 17:17:31 -08:00
|
|
|
);
|
2020-03-23 20:06:52 -07:00
|
|
|
#endif /* EFI_UNIT_TEST */
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
2020-05-06 05:40:42 -07:00
|
|
|
void stopBoostPin() {
|
2020-03-23 20:15:44 -07:00
|
|
|
#if !EFI_UNIT_TEST
|
2020-02-02 00:56:21 -08:00
|
|
|
brain_pin_markUnused(activeConfiguration.boostControlPin);
|
2020-03-23 20:15:44 -07:00
|
|
|
#endif /* EFI_UNIT_TEST */
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void onConfigurationChangeBoostCallback(engine_configuration_s *previousConfiguration) {
|
2020-05-06 05:40:42 -07:00
|
|
|
boostController.onConfigurationChange(&previousConfiguration->boostPid);
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
2020-02-04 17:17:31 -08:00
|
|
|
void initBoostCtrl(Logging *sharedLogger DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2020-03-23 20:06:52 -07:00
|
|
|
#if !EFI_UNIT_TEST
|
2020-02-04 17:17:31 -08:00
|
|
|
if (CONFIG(boostControlPin) == GPIO_UNASSIGNED){
|
|
|
|
return;
|
|
|
|
}
|
2020-03-23 20:51:51 -07:00
|
|
|
#endif
|
2020-02-04 17:17:31 -08:00
|
|
|
|
2020-02-02 00:56:21 -08:00
|
|
|
logger = sharedLogger;
|
2020-05-06 05:40:42 -07:00
|
|
|
|
|
|
|
// Set up open & closed loop tables
|
|
|
|
boostMapOpen.init(config->boostTableOpenLoop, config->boostTpsBins, config->boostRpmBins);
|
2020-02-02 00:56:21 -08:00
|
|
|
boostMapClosed.init(config->boostTableClosedLoop, config->boostTpsBins, config->boostRpmBins);
|
2020-05-06 05:40:42 -07:00
|
|
|
|
|
|
|
// Set up boost controller instance
|
|
|
|
boostController.init(&boostPwmControl, &boostMapOpen, &boostMapClosed, &engineConfiguration->boostPid);
|
|
|
|
|
2020-03-23 20:51:51 -07:00
|
|
|
#if !EFI_UNIT_TEST
|
2020-02-02 00:56:21 -08:00
|
|
|
startBoostPin();
|
2020-05-06 05:40:42 -07:00
|
|
|
boostController.Start();
|
2020-03-23 20:06:52 -07:00
|
|
|
#endif
|
2020-02-02 00:56:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|