rusefi/firmware/controllers/algo/launch_control.cpp

230 lines
6.6 KiB
C++

/*
* @file launch_control.cpp
*
* @date 10. sep. 2019
* Author: Ola Ruud
* Rework: Patryk Chmura
*/
#include "pch.h"
#if EFI_LAUNCH_CONTROL
#include "boost_control.h"
#include "launch_control.h"
#include "periodic_task.h"
#include "advance_map.h"
#include "engine_state.h"
#include "advance_map.h"
#include "tinymt32.h"
/**
* We can have active condition from switch or from clutch.
* In case we are dependent on VSS we just return true.
*/
bool LaunchControlBase::isInsideSwitchCondition() {
isSwitchActivated = engineConfiguration->launchActivationMode == SWITCH_INPUT_LAUNCH;
isClutchActivated = engineConfiguration->launchActivationMode == CLUTCH_INPUT_LAUNCH;
if (isSwitchActivated) {
#if !EFI_SIMULATOR
if (isBrainPinValid(engineConfiguration->launchActivatePin)) {
launchActivatePinState = engineConfiguration->launchActivateInverted ^ efiReadPin(engineConfiguration->launchActivatePin);
}
#endif // EFI_PROD_CODE
return launchActivatePinState;
} else if (isClutchActivated) {
if (isBrainPinValid(engineConfiguration->clutchDownPin)) {
return engine->engineState.clutchDownState;
} else {
return false;
}
} else {
// ALWAYS_ACTIVE_LAUNCH
return true;
}
}
/**
* Returns True when Vehicle speed ALLOWS launch control
*/
bool LaunchControlBase::isInsideSpeedCondition() const {
if (engineConfiguration->launchSpeedThreshold == 0) {
return true; // allow launch, speed does not matter
}
int speed = Sensor::getOrZero(SensorType::VehicleSpeed);
return engineConfiguration->launchSpeedThreshold > speed;
}
/**
* Returns false if TPS is invalid or TPS > preset threshold
*/
bool LaunchControlBase::isInsideTpsCondition() const {
auto tps = Sensor::get(SensorType::DriverThrottleIntent);
// Disallow launch without valid TPS
if (!tps.Valid) {
return false;
}
// todo: should this be 'launchTpsThreshold <= tps.Value' so that nicely calibrated TPS of zero does not prevent launch?
return engineConfiguration->launchTpsThreshold < tps.Value;
}
LaunchCondition LaunchControlBase::calculateRPMLaunchCondition(const int rpm) {
const int launchRpm = engineConfiguration->launchRpm;
const int preLaunchRpm = launchRpm - engineConfiguration->launchRpmWindow;
if (rpm < preLaunchRpm) {
return LaunchCondition::NotMet;
} else if (launchRpm <= rpm) {
return LaunchCondition::Launch;
} else {
return LaunchCondition::PreLaunch;
}
}
bool LaunchControlBase::isInsideRPMCondition(const int rpm) {
const LaunchCondition rpmLaunchCondition = calculateRPMLaunchCondition(rpm);
bool result = false;
switch (rpmLaunchCondition) {
case LaunchCondition::NotMet: {
isAfterLaunch = false;
break;
}
case LaunchCondition::PreLaunch: {
result = isAfterLaunch;
break;
}
case LaunchCondition::Launch: {
isAfterLaunch = true;
result = true;
break;
}
}
return result;
}
bool LaunchControlBase::isLaunchConditionMet(const int rpm) {
activateSwitchCondition = isInsideSwitchCondition();
rpmCondition = isInsideRPMCondition(rpm);
speedCondition = isInsideSpeedCondition();
tpsCondition = isInsideTpsCondition();
return speedCondition && activateSwitchCondition && rpmCondition && tpsCondition;
}
LaunchControlBase::LaunchControlBase() {
launchActivatePinState = false;
isLaunchPreCondition = false;
isLaunchCondition = false;
}
float LaunchControlBase::getFuelCoefficient() const {
return 1 + (isLaunchCondition && engineConfiguration->launchControlEnabled ? engineConfiguration->launchFuelAdderPercent / 100.0 : 0);
}
void LaunchControlBase::update() {
if (!engineConfiguration->launchControlEnabled) {
return;
}
const int rpm = Sensor::getOrZero(SensorType::Rpm);
combinedConditions = isLaunchConditionMet(rpm);
//and still recalculate in case user changed the values
retardThresholdRpm = engineConfiguration->launchRpm;
if (!combinedConditions) {
// conditions not met, reset timer
m_launchTimer.reset();
isLaunchCondition = false;
} else {
// If conditions are met...
isLaunchCondition = m_launchTimer.hasElapsedSec(engineConfiguration->launchActivateDelay);
}
sparkSkipRatio = calculateSparkSkipRatio(rpm);
}
bool LaunchControlBase::isLaunchRpmRetardCondition() const {
return isLaunchCondition && engineConfiguration->launchControlEnabled && (retardThresholdRpm < Sensor::getOrZero(SensorType::Rpm));
}
bool LaunchControlBase::isLaunchSparkRpmRetardCondition() const {
return isLaunchRpmRetardCondition() && engineConfiguration->launchSparkCutEnable;
}
bool LaunchControlBase::isLaunchFuelRpmRetardCondition() const {
return isLaunchRpmRetardCondition() && engineConfiguration->launchFuelCutEnable;
}
float LaunchControlBase::calculateSparkSkipRatio(const int rpm) const {
float result = 0.0f;
if (engineConfiguration->launchControlEnabled && engineConfiguration->launchSparkCutEnable) {
if (isLaunchCondition) {
result = 1.0f;
} else {
const int launchRpm = engineConfiguration->launchRpm;
const int sparkSkipStartRpm = launchRpm - engineConfiguration->launchRpmWindow;
if (sparkSkipStartRpm <= rpm) {
const float initialIgnitionCutRatio = engineConfiguration->initialIgnitionCutPercent / 100.0f;
const int sparkSkipEndRpm = launchRpm - engineConfiguration->launchCorrectionsEndRpm;
const float finalIgnitionCutRatio = engineConfiguration->finalIgnitionCutPercentBeforeLaunch / 100.0f;
result = interpolateClamped(sparkSkipStartRpm, initialIgnitionCutRatio, sparkSkipEndRpm, finalIgnitionCutRatio, rpm);
}
}
}
return result;
}
SoftSparkLimiter::SoftSparkLimiter(const bool p_allowHardCut)
: allowHardCut(p_allowHardCut) {
#if EFI_UNIT_TEST
initLaunchControl();
#endif // EFI_UNIT_TEST
}
void SoftSparkLimiter::updateTargetSkipRatio(
const float luaSparkSkip,
const float tractionControlSparkSkip,
const float launchControllerSparkSkipRatio
) {
targetSkipRatio = luaSparkSkip;
if (engineConfiguration->useHardSkipInTraction) {
if (allowHardCut) {
targetSkipRatio += tractionControlSparkSkip;
}
} else if (!allowHardCut) {
targetSkipRatio += tractionControlSparkSkip;
}
if (allowHardCut) {
/*
* We are applying launch controller spark skip ratio only for hard skip limiter (see
* https://github.com/rusefi/rusefi/issues/6566#issuecomment-2153149902).
*/
targetSkipRatio += launchControllerSparkSkipRatio;
}
}
static tinymt32_t tinymt;
bool SoftSparkLimiter::shouldSkip() {
if (targetSkipRatio == 0 || (!allowHardCut && wasJustSkipped)) {
wasJustSkipped = false;
return false;
}
float r = tinymt32_generate_float(&tinymt);
wasJustSkipped = r < (allowHardCut ? 1 : 2) * targetSkipRatio;
return wasJustSkipped;
}
void initLaunchControl() {
tinymt32_init(&tinymt, 1345135);
}
#endif /* EFI_LAUNCH_CONTROL */