2020-03-23 20:20:54 -07:00
|
|
|
/*
|
|
|
|
* @file launch_control.cpp
|
|
|
|
*
|
|
|
|
* @date 10. sep. 2019
|
|
|
|
* Author: Ola Ruud
|
2024-03-09 05:39:58 -08:00
|
|
|
* Rework: Patryk Chmura
|
2020-03-23 20:20:54 -07:00
|
|
|
*/
|
|
|
|
|
2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
2020-03-23 20:20:54 -07:00
|
|
|
|
|
|
|
#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"
|
2021-11-25 19:40:19 -08:00
|
|
|
#include "tinymt32.h"
|
2020-03-23 20:20:54 -07:00
|
|
|
|
2020-11-19 18:14:38 -08:00
|
|
|
/**
|
|
|
|
* We can have active condition from switch or from clutch.
|
|
|
|
* In case we are dependent on VSS we just return true.
|
|
|
|
*/
|
2021-11-15 17:09:03 -08:00
|
|
|
bool LaunchControlBase::isInsideSwitchCondition() {
|
2022-01-24 17:58:21 -08:00
|
|
|
isSwitchActivated = engineConfiguration->launchActivationMode == SWITCH_INPUT_LAUNCH;
|
|
|
|
isClutchActivated = engineConfiguration->launchActivationMode == CLUTCH_INPUT_LAUNCH;
|
|
|
|
|
|
|
|
|
|
|
|
if (isSwitchActivated) {
|
2021-11-16 13:45:14 -08:00
|
|
|
#if !EFI_SIMULATOR
|
2021-11-17 00:54:21 -08:00
|
|
|
if (isBrainPinValid(engineConfiguration->launchActivatePin)) {
|
2022-04-19 17:43:41 -07:00
|
|
|
launchActivatePinState = engineConfiguration->launchActivateInverted ^ efiReadPin(engineConfiguration->launchActivatePin);
|
2020-03-23 20:20:54 -07:00
|
|
|
}
|
2021-11-16 13:31:35 -08:00
|
|
|
#endif // EFI_PROD_CODE
|
2021-11-15 17:09:03 -08:00
|
|
|
return launchActivatePinState;
|
2022-01-24 17:58:21 -08:00
|
|
|
} else if (isClutchActivated) {
|
2021-11-17 00:54:21 -08:00
|
|
|
if (isBrainPinValid(engineConfiguration->clutchDownPin)) {
|
2022-06-28 23:37:24 -07:00
|
|
|
return engine->engineState.clutchDownState;
|
2020-11-19 18:14:38 -08:00
|
|
|
} else {
|
|
|
|
return false;
|
2020-03-23 20:20:54 -07:00
|
|
|
}
|
2022-01-24 17:58:21 -08:00
|
|
|
} else {
|
2020-03-23 20:20:54 -07:00
|
|
|
// ALWAYS_ACTIVE_LAUNCH
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-19 18:14:38 -08:00
|
|
|
/**
|
2023-07-20 13:09:52 -07:00
|
|
|
* Returns True when Vehicle speed ALLOWS launch control
|
2024-02-29 19:33:37 -08:00
|
|
|
*/
|
2020-11-19 05:15:56 -08:00
|
|
|
bool LaunchControlBase::isInsideSpeedCondition() const {
|
2023-07-20 13:09:52 -07:00
|
|
|
if (engineConfiguration->launchSpeedThreshold == 0) {
|
|
|
|
return true; // allow launch, speed does not matter
|
|
|
|
}
|
|
|
|
|
2021-10-05 16:59:07 -07:00
|
|
|
int speed = Sensor::getOrZero(SensorType::VehicleSpeed);
|
2024-02-29 19:33:37 -08:00
|
|
|
|
2023-07-20 13:25:24 -07:00
|
|
|
return engineConfiguration->launchSpeedThreshold > speed;
|
2020-11-19 05:15:56 -08:00
|
|
|
}
|
|
|
|
|
2020-11-19 18:14:38 -08:00
|
|
|
/**
|
2021-11-16 13:46:54 -08:00
|
|
|
* Returns false if TPS is invalid or TPS > preset threshold
|
2020-11-19 18:14:38 -08:00
|
|
|
*/
|
2020-11-19 05:15:56 -08:00
|
|
|
bool LaunchControlBase::isInsideTpsCondition() const {
|
|
|
|
auto tps = Sensor::get(SensorType::DriverThrottleIntent);
|
|
|
|
|
|
|
|
// Disallow launch without valid TPS
|
|
|
|
if (!tps.Valid) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-07-20 15:48:55 -07:00
|
|
|
// todo: should this be 'launchTpsThreshold <= tps.Value' so that nicely calibrated TPS of zero does not prevent launch?
|
|
|
|
return engineConfiguration->launchTpsThreshold < tps.Value;
|
2020-11-19 05:15:56 -08:00
|
|
|
}
|
|
|
|
|
2024-05-21 00:06:17 -07:00
|
|
|
LaunchCondition LaunchControlBase::calculateRPMLaunchCondition(const int rpm) {
|
2024-05-20 12:54:05 -07:00
|
|
|
const int launchRpm = engineConfiguration->launchRpm;
|
2024-05-21 00:06:17 -07:00
|
|
|
const int preLaunchRpm = launchRpm - engineConfiguration->launchRpmWindow;
|
|
|
|
if (rpm < preLaunchRpm) {
|
2024-05-20 12:54:05 -07:00
|
|
|
return LaunchCondition::NotMet;
|
2024-05-21 00:06:17 -07:00
|
|
|
} else if (launchRpm <= rpm) {
|
2024-05-20 12:54:05 -07:00
|
|
|
return LaunchCondition::Launch;
|
2024-05-21 00:06:17 -07:00
|
|
|
} else {
|
|
|
|
return LaunchCondition::PreLaunch;
|
2024-05-20 12:54:05 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-21 00:06:17 -07:00
|
|
|
bool LaunchControlBase::isInsideRPMCondition(const int rpm) {
|
2024-05-20 12:54:05 -07:00
|
|
|
const LaunchCondition rpmLaunchCondition = calculateRPMLaunchCondition(rpm);
|
2024-05-21 00:06:17 -07:00
|
|
|
bool result = false;
|
|
|
|
switch (rpmLaunchCondition) {
|
|
|
|
case LaunchCondition::NotMet: {
|
2024-05-21 10:15:33 -07:00
|
|
|
isAfterLaunch = false;
|
2024-05-21 00:06:17 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LaunchCondition::PreLaunch: {
|
2024-05-21 10:15:33 -07:00
|
|
|
result = isAfterLaunch;
|
2024-05-21 00:06:17 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LaunchCondition::Launch: {
|
2024-05-21 10:15:33 -07:00
|
|
|
isAfterLaunch = true;
|
2024-05-21 00:06:17 -07:00
|
|
|
result = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
2020-11-19 18:14:38 -08:00
|
|
|
}
|
2020-11-19 05:15:56 -08:00
|
|
|
|
2024-05-20 12:54:05 -07:00
|
|
|
bool LaunchControlBase::isLaunchConditionMet(const int rpm) {
|
2021-12-16 13:19:33 -08:00
|
|
|
activateSwitchCondition = isInsideSwitchCondition();
|
2024-05-20 12:54:05 -07:00
|
|
|
rpmCondition = isInsideRPMCondition(rpm);
|
2021-12-16 13:19:33 -08:00
|
|
|
speedCondition = isInsideSpeedCondition();
|
|
|
|
tpsCondition = isInsideTpsCondition();
|
2020-03-23 20:20:54 -07:00
|
|
|
|
2024-05-20 12:54:05 -07:00
|
|
|
return speedCondition && activateSwitchCondition && rpmCondition && tpsCondition;
|
2020-11-19 05:15:56 -08:00
|
|
|
}
|
|
|
|
|
2021-12-16 11:47:00 -08:00
|
|
|
LaunchControlBase::LaunchControlBase() {
|
|
|
|
launchActivatePinState = false;
|
2024-02-29 20:11:25 -08:00
|
|
|
isLaunchPreCondition = false;
|
2021-12-16 11:47:00 -08:00
|
|
|
isLaunchCondition = false;
|
2024-05-27 05:18:24 -07:00
|
|
|
isSmoothRetardCondition = false;
|
2021-12-16 11:47:00 -08:00
|
|
|
}
|
|
|
|
|
2024-02-29 19:33:37 -08:00
|
|
|
float LaunchControlBase::getFuelCoefficient() const {
|
2022-11-13 17:23:19 -08:00
|
|
|
return 1 + (isLaunchCondition && engineConfiguration->launchControlEnabled ? engineConfiguration->launchFuelAdderPercent / 100.0 : 0);
|
|
|
|
}
|
|
|
|
|
2020-11-19 05:15:56 -08:00
|
|
|
void LaunchControlBase::update() {
|
2021-11-17 00:54:21 -08:00
|
|
|
if (!engineConfiguration->launchControlEnabled) {
|
2020-11-19 05:15:56 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-05-27 05:18:24 -07:00
|
|
|
const int rpm = Sensor::getOrZero(SensorType::Rpm);
|
2024-05-20 12:54:05 -07:00
|
|
|
combinedConditions = isLaunchConditionMet(rpm);
|
2020-11-19 18:14:38 -08:00
|
|
|
|
2021-12-16 11:47:00 -08:00
|
|
|
//and still recalculate in case user changed the values
|
2022-07-20 15:16:54 -07:00
|
|
|
retardThresholdRpm = engineConfiguration->launchRpm
|
2022-07-20 15:30:20 -07:00
|
|
|
/*
|
|
|
|
we never had UI for 'launchAdvanceRpmRange' so it was always zero. are we supposed to forget about this dead line
|
|
|
|
or it is supposed to be referencing 'launchTimingRpmRange'?
|
2022-07-20 15:16:54 -07:00
|
|
|
+ (engineConfiguration->enableLaunchRetard ? engineConfiguration->launchAdvanceRpmRange : 0)
|
2024-05-21 13:00:23 -07:00
|
|
|
*/;
|
2020-11-19 05:15:56 -08:00
|
|
|
|
|
|
|
if (!combinedConditions) {
|
|
|
|
// conditions not met, reset timer
|
2020-11-30 16:35:06 -08:00
|
|
|
m_launchTimer.reset();
|
2021-11-15 17:09:03 -08:00
|
|
|
isLaunchCondition = false;
|
2020-11-19 05:15:56 -08:00
|
|
|
} else {
|
|
|
|
// If conditions are met...
|
2021-11-17 00:54:21 -08:00
|
|
|
isLaunchCondition = m_launchTimer.hasElapsedSec(engineConfiguration->launchActivateDelay);
|
2020-03-23 20:20:54 -07:00
|
|
|
}
|
2024-05-27 05:18:24 -07:00
|
|
|
|
|
|
|
isSmoothRetardCondition = isSmoothRetardRpmCondition(rpm);
|
2020-11-19 05:15:56 -08:00
|
|
|
}
|
|
|
|
|
2021-11-15 16:48:55 -08:00
|
|
|
bool LaunchControlBase::isLaunchRpmRetardCondition() const {
|
2022-01-20 20:43:18 -08:00
|
|
|
return isLaunchCondition && (retardThresholdRpm < Sensor::getOrZero(SensorType::Rpm));
|
2021-11-15 16:48:55 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool LaunchControlBase::isLaunchSparkRpmRetardCondition() const {
|
|
|
|
return isLaunchRpmRetardCondition() && engineConfiguration->launchSparkCutEnable;
|
|
|
|
}
|
|
|
|
|
2024-05-27 05:18:24 -07:00
|
|
|
bool LaunchControlBase::isSmoothRetardRpmCondition(const int rpm) const {
|
|
|
|
const int smoothRetardStartRpm = engineConfiguration->launchRpm - engineConfiguration->launchRpmWindow;
|
|
|
|
const int smoothRetardEndRpm = engineConfiguration->launchRpm - engineConfiguration->smoothRetardEndRpm;
|
|
|
|
return (smoothRetardStartRpm <= rpm) && (rpm <= smoothRetardEndRpm);
|
|
|
|
}
|
|
|
|
|
2021-11-15 16:48:55 -08:00
|
|
|
bool LaunchControlBase::isLaunchFuelRpmRetardCondition() const {
|
|
|
|
return isLaunchRpmRetardCondition() && engineConfiguration->launchFuelCutEnable;
|
2020-03-23 20:20:54 -07:00
|
|
|
}
|
|
|
|
|
2023-11-01 07:09:04 -07:00
|
|
|
SoftSparkLimiter::SoftSparkLimiter(bool p_allowHardCut) {
|
|
|
|
this->allowHardCut = p_allowHardCut;
|
2023-05-25 15:52:19 -07:00
|
|
|
#if EFI_UNIT_TEST
|
|
|
|
initLaunchControl();
|
|
|
|
#endif // EFI_UNIT_TEST
|
2023-05-25 09:41:11 -07:00
|
|
|
}
|
|
|
|
|
2023-11-01 07:09:04 -07:00
|
|
|
void SoftSparkLimiter::setTargetSkipRatio(float p_targetSkipRatio) {
|
|
|
|
this->targetSkipRatio = p_targetSkipRatio;
|
2021-11-16 10:15:12 -08:00
|
|
|
}
|
|
|
|
|
2021-11-25 19:40:19 -08:00
|
|
|
static tinymt32_t tinymt;
|
|
|
|
|
2021-11-16 10:15:12 -08:00
|
|
|
bool SoftSparkLimiter::shouldSkip() {
|
2023-05-25 12:17:07 -07:00
|
|
|
if (targetSkipRatio == 0 || (!allowHardCut && wasJustSkipped)) {
|
2021-11-16 10:15:12 -08:00
|
|
|
wasJustSkipped = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-11-25 19:40:19 -08:00
|
|
|
float r = tinymt32_generate_float(&tinymt);
|
2023-05-25 12:17:07 -07:00
|
|
|
wasJustSkipped = r < (allowHardCut ? 1 : 2) * targetSkipRatio;
|
2021-11-16 10:15:12 -08:00
|
|
|
return wasJustSkipped;
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void initLaunchControl() {
|
2021-11-25 19:40:19 -08:00
|
|
|
tinymt32_init(&tinymt, 1345135);
|
2020-03-23 20:20:54 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* EFI_LAUNCH_CONTROL */
|