2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2020-05-06 18:00:40 -07:00
|
|
|
#include "closed_loop_fuel_cell.h"
|
|
|
|
|
|
|
|
constexpr float integrator_dt = FAST_CALLBACK_PERIOD_MS * 0.001f;
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void ClosedLoopFuelCellBase::update(float lambdaDeadband, bool ignoreErrorMagnitude)
|
2020-05-06 18:00:40 -07:00
|
|
|
{
|
|
|
|
// Compute how far off target we are
|
2021-11-16 01:15:29 -08:00
|
|
|
float lambdaError = getLambdaError();
|
2020-05-06 18:00:40 -07:00
|
|
|
|
|
|
|
// If we're within the deadband, make no adjustment.
|
|
|
|
if (absF(lambdaError) < lambdaDeadband) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fixed magnitude - runs in constant adjustment rate mode
|
|
|
|
if (ignoreErrorMagnitude) {
|
|
|
|
if (lambdaError > 0) {
|
|
|
|
lambdaError = 0.1f;
|
|
|
|
} else {
|
|
|
|
lambdaError = -0.1f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Integrate
|
|
|
|
float adjust = getIntegratorGain() * lambdaError * integrator_dt
|
|
|
|
+ m_adjustment;
|
|
|
|
|
|
|
|
// Clamp to bounds
|
|
|
|
float minAdjust = getMinAdjustment();
|
|
|
|
float maxAdjust = getMaxAdjustment();
|
|
|
|
|
|
|
|
if (adjust > maxAdjust) {
|
|
|
|
adjust = maxAdjust;
|
|
|
|
} else if (adjust < minAdjust) {
|
|
|
|
adjust = minAdjust;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save state
|
|
|
|
m_adjustment = adjust;
|
|
|
|
}
|
|
|
|
|
|
|
|
float ClosedLoopFuelCellBase::getAdjustment() const {
|
|
|
|
return 1.0f + m_adjustment;
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
float ClosedLoopFuelCellImpl::getLambdaError() const {
|
2021-01-10 05:10:49 -08:00
|
|
|
auto lambda = Sensor::get(m_lambdaSensor);
|
2020-09-01 13:22:31 -07:00
|
|
|
|
|
|
|
// Failed sensor -> no error
|
|
|
|
if (!lambda) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-11-17 00:54:21 -08:00
|
|
|
return lambda.Value - engine->engineState.targetLambda;
|
2020-05-06 18:00:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#define MAX_ADJ (0.25f)
|
|
|
|
|
|
|
|
float ClosedLoopFuelCellImpl::getMaxAdjustment() const {
|
|
|
|
if (!m_config) {
|
|
|
|
// If no config, disallow adjustment.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
float raw = m_config->maxAdd * 0.01f;
|
|
|
|
// Don't allow maximum less than 0, or more than maximum adjustment
|
|
|
|
return minF(MAX_ADJ, maxF(raw, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
float ClosedLoopFuelCellImpl::getMinAdjustment() const {
|
|
|
|
if (!m_config) {
|
|
|
|
// If no config, disallow adjustment.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
float raw = m_config->maxRemove * 0.01f;
|
|
|
|
// Don't allow minimum more than 0, or more than maximum adjustment
|
|
|
|
return maxF(-MAX_ADJ, minF(raw, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
float ClosedLoopFuelCellImpl::getIntegratorGain() const {
|
|
|
|
if (!m_config) {
|
|
|
|
// If no config, disallow adjustment.
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clamp to reasonable limits - 100ms to 100s
|
2022-04-28 05:16:02 -07:00
|
|
|
float timeConstant = maxF(0.1f, minF(m_config->timeConstant, 100));
|
2020-05-06 18:00:40 -07:00
|
|
|
|
|
|
|
return 1 / timeConstant;
|
|
|
|
}
|