rusefi/firmware/controllers/trigger/trigger_structure.cpp

456 lines
13 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file trigger_structure.cpp
*
* @date Jan 20, 2014
2017-01-03 03:05:22 -08:00
* @author Andrey Belomutskiy, (c) 2012-2017
2015-07-10 06:01:56 -07:00
*
* This file is part of rusEfi - see http://rusefi.com
*
* rusEfi is free software; you can redistribute it and/or modify it under the terms of
* the GNU General Public License as published by the Free Software Foundation; either
* version 3 of the License, or (at your option) any later version.
*
* rusEfi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
* even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with this program.
* If not, see <http://www.gnu.org/licenses/>.
*/
#include "main.h"
#include "trigger_structure.h"
#include "error_handling.h"
#include "trigger_decoder.h"
#include "engine_math.h"
2017-01-03 13:02:43 -08:00
#include "trigger_universal.h"
2015-07-10 06:01:56 -07:00
EXTERN_ENGINE;
trigger_shape_helper::trigger_shape_helper() {
memset(&pinStates, 0, sizeof(pinStates));
for (int i = 0; i < TRIGGER_CHANNEL_COUNT; i++) {
waves[i].init(pinStates[i]);
}
}
TriggerShape::TriggerShape() :
wave(switchTimesBuffer, NULL) {
2015-10-29 11:02:52 -07:00
initialize(OM_NONE, false);
2015-07-10 06:01:56 -07:00
wave.waves = h.waves;
2016-08-18 21:02:50 -07:00
2015-07-10 06:01:56 -07:00
memset(triggerIndexByAngle, 0, sizeof(triggerIndexByAngle));
}
2015-09-13 13:01:38 -07:00
void TriggerShape::calculateTriggerSynchPoint(TriggerState *state DECLARE_ENGINE_PARAMETER_S) {
2015-09-13 07:01:39 -07:00
#if EFI_PROD_CODE || defined(__DOXYGEN__)
efiAssertVoid(getRemainingStack(chThdSelf()) > 256, "calc s");
#endif
2015-07-10 06:01:56 -07:00
trigger_config_s const*triggerConfig = &engineConfiguration->trigger;
2015-09-13 13:01:38 -07:00
triggerShapeSynchPointIndex = findTriggerZeroEventIndex(state, this, triggerConfig PASS_ENGINE_PARAMETER);
2015-07-10 06:01:56 -07:00
engine->engineCycleEventCount = getLength();
float firstAngle = getAngle(triggerShapeSynchPointIndex);
int frontOnlyIndex = 0;
for (int eventIndex = 0; eventIndex < engine->engineCycleEventCount; eventIndex++) {
if (eventIndex == 0) {
// explicit check for zero to avoid issues where logical zero is not exactly zero due to float nature
eventAngles[0] = 0;
// this value would be used in case of front-only
eventAngles[1] = 0;
frontOnlyIndexes[0] = 0;
} else {
int triggerDefinitionCoordinate = (triggerShapeSynchPointIndex + eventIndex) % engine->engineCycleEventCount;
int triggerDefinitionIndex = triggerDefinitionCoordinate >= size ? triggerDefinitionCoordinate - size : triggerDefinitionCoordinate;
float angle = getAngle(triggerDefinitionCoordinate) - firstAngle;
2016-12-05 19:01:54 -08:00
fixAngle(angle, "trgSync");
2016-02-27 20:03:34 -08:00
if (engineConfiguration->useOnlyRisingEdgeForTrigger) {
2015-07-10 06:01:56 -07:00
if (isFrontEvent[triggerDefinitionIndex]) {
frontOnlyIndex += 2;
eventAngles[frontOnlyIndex] = angle;
eventAngles[frontOnlyIndex + 1] = angle;
}
} else {
eventAngles[eventIndex] = angle;
}
frontOnlyIndexes[eventIndex] = frontOnlyIndex;
}
}
2016-11-08 22:02:15 -08:00
int totalExpected = 0;
2016-12-13 11:01:43 -08:00
for (int i = 0;i < PWM_PHASE_MAX_WAVE_PER_PWM;i++) {
2016-11-08 22:02:15 -08:00
totalExpected += expectedEventCount[i];
}
if (totalExpected < 2) {
2016-12-13 11:01:43 -08:00
firmwareError(CUSTOM_ERR_TOO_FEW_EVENTS, "Too few trigger events");
2016-11-08 22:02:15 -08:00
}
2015-07-10 06:01:56 -07:00
}
2015-10-31 09:01:48 -07:00
void TriggerShape::initialize(operation_mode_e operationMode, bool needSecondTriggerInput) {
2016-08-17 21:02:22 -07:00
isSynchronizationNeeded = true; // that's default value
this->needSecondTriggerInput = needSecondTriggerInput;
memset(dutyCycle, 0, sizeof(dutyCycle));
memset(eventAngles, 0, sizeof(eventAngles));
// memset(triggerIndexByAngle, 0, sizeof(triggerIndexByAngle));
2016-08-18 18:01:47 -07:00
setTriggerSynchronizationGap(2);
2016-08-17 21:02:22 -07:00
2016-08-18 19:01:57 -07:00
secondSyncRatioFrom = 0.000001;
secondSyncRatioTo = 100000;
thirdSyncRatioFrom = 0.000001;
thirdSyncRatioTo = 100000;
2015-07-10 06:01:56 -07:00
tdcPosition = 0;
2016-08-19 19:02:21 -07:00
useOnlyPrimaryForSync = false;
2015-07-10 06:01:56 -07:00
useRiseEdge = true;
invertOnAdd = false;
gapBothDirections = false;
this->operationMode = operationMode;
size = 0;
triggerShapeSynchPointIndex = 0;
memset(initialState, 0, sizeof(initialState));
memset(switchTimesBuffer, 0, sizeof(switchTimesBuffer));
memset(expectedEventCount, 0, sizeof(expectedEventCount));
wave.reset();
previousAngle = 0;
2016-08-19 20:02:21 -07:00
memset(frontOnlyIndexes, 0, sizeof(frontOnlyIndexes));
memset(isFrontEvent, 0, sizeof(isFrontEvent));
2016-08-19 21:02:28 -07:00
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
memset(&triggerSignals, 0, sizeof(triggerSignals));
#endif
2015-07-10 06:01:56 -07:00
}
2016-08-20 06:02:06 -07:00
int TriggerShape::getSize() const {
return size;
}
int TriggerShape::getTriggerShapeSynchPointIndex() {
return triggerShapeSynchPointIndex;
}
2015-07-10 06:01:56 -07:00
int multi_wave_s::getChannelState(int channelIndex, int phaseIndex) const {
return waves[channelIndex].pinStates[phaseIndex];
}
int multi_wave_s::waveIndertionAngle(float angle, int size) const {
for (int i = size - 1; i >= 0; i--) {
if (angle > switchTimes[i])
return i + 1;
}
return 0;
}
int multi_wave_s::findAngleMatch(float angle, int size) const {
for (int i = 0; i < size; i++) {
if (isSameF(switchTimes[i], angle))
return i;
}
return EFI_ERROR_CODE;
}
void multi_wave_s::setSwitchTime(int index, float value) {
switchTimes[index] = value;
}
TriggerState::TriggerState() {
2015-09-13 07:01:39 -07:00
reset();
}
void TriggerState::reset() {
2015-07-10 06:01:56 -07:00
cycleCallback = NULL;
shaft_is_synchronized = false;
toothed_previous_time = 0;
toothed_previous_duration = 0;
2015-09-11 22:02:28 -07:00
durationBeforePrevious = 0;
2015-12-27 13:02:44 -08:00
thirdPreviousDuration = 0;
2015-07-10 06:01:56 -07:00
totalRevolutionCounter = 0;
totalTriggerErrorCounter = 0;
orderingErrorCounter = 0;
currentDuration = 0;
2016-02-27 20:03:34 -08:00
curSignal = SHAFT_PRIMARY_FALLING;
prevSignal = SHAFT_PRIMARY_FALLING;
2015-07-10 06:01:56 -07:00
prevCycleDuration = 0;
startOfCycleNt = 0;
resetRunningCounters();
2015-09-08 20:01:07 -07:00
resetCurrentCycleState();
2015-07-10 06:01:56 -07:00
memset(expectedTotalTime, 0, sizeof(expectedTotalTime));
2016-01-24 16:01:59 -08:00
memset(prevTotalTime, 0, sizeof(prevTotalTime));
2015-07-10 06:01:56 -07:00
totalEventCountBase = 0;
isFirstEvent = true;
}
int TriggerState::getCurrentIndex() {
2015-09-08 20:01:07 -07:00
return currentCycle.current_index;
2015-07-10 06:01:56 -07:00
}
efitime_t TriggerState::getStartOfRevolutionIndex() {
return totalEventCountBase;
}
void TriggerState::resetRunningCounters() {
runningRevolutionCounter = 0;
runningTriggerErrorCounter = 0;
runningOrderingErrorCounter = 0;
}
efitime_t TriggerState::getTotalEventCounter() {
2015-09-08 20:01:07 -07:00
return totalEventCountBase + currentCycle.current_index;
2015-07-10 06:01:56 -07:00
}
int TriggerState::getTotalRevolutionCounter() {
return totalRevolutionCounter;
}
2016-08-23 19:02:18 -07:00
void TriggerState::intTotalEventCounter() {
totalRevolutionCounter++;
}
bool TriggerState::isEvenRevolution() {
return totalRevolutionCounter & 1;
}
2015-09-08 20:01:07 -07:00
void TriggerState::resetCurrentCycleState() {
memset(currentCycle.eventCount, 0, sizeof(currentCycle.eventCount));
memset(currentCycle.timeOfPreviousEventNt, 0, sizeof(currentCycle.timeOfPreviousEventNt));
memset(currentCycle.totalTimeNt, 0, sizeof(currentCycle.totalTimeNt));
currentCycle.current_index = 0;
2015-07-10 06:01:56 -07:00
}
/**
2016-11-14 12:04:44 -08:00
* physical primary trigger duration
2015-07-10 06:01:56 -07:00
*/
2016-11-14 12:04:44 -08:00
angle_t TriggerShape::getCycleDuration() const {
2016-11-13 21:02:13 -08:00
switch (operationMode) {
case FOUR_STROKE_SYMMETRICAL_CRANK_SENSOR:
2016-11-14 12:04:44 -08:00
return 180;
2016-11-13 21:02:13 -08:00
case FOUR_STROKE_CRANK_SENSOR:
2016-11-14 12:04:44 -08:00
case TWO_STROKE:
return 360;
2016-11-13 21:02:13 -08:00
default:
2016-11-14 12:04:44 -08:00
return 720;
2016-11-13 21:02:13 -08:00
}
2015-07-10 06:01:56 -07:00
}
2016-11-14 12:04:44 -08:00
/**
* Trigger event count equals engine cycle event count if we have a cam sensor.
* Two trigger cycles make one engine cycle in case of a four stroke engine If we only have a cranksensor.
*/
uint32_t TriggerShape::getLength() const {
/**
* 4 for FOUR_STROKE_SYMMETRICAL_CRANK_SENSOR
* 2 for FOUR_STROKE_CRANK_SENSOR
* 1 otherwise
*/
int multiplier = getEngineCycle(operationMode) / getCycleDuration();
return multiplier * getSize();
}
2016-01-14 20:03:17 -08:00
angle_t TriggerShape::getAngle(int index) const {
2015-07-10 06:01:56 -07:00
// todo: why is this check here? looks like the code below could be used universally
if (operationMode == FOUR_STROKE_CAM_SENSOR) {
return getSwitchAngle(index);
}
/**
* FOUR_STROKE_CRANK_SENSOR magic:
* We have two crank shaft revolutions for each engine cycle
* See also trigger_central.cpp
* See also getEngineCycleEventCount()
*/
2016-11-13 21:02:13 -08:00
int crankCycle = index / size;
int remainder = index % size;
return getCycleDuration() * crankCycle + getSwitchAngle(remainder);
2015-07-10 06:01:56 -07:00
}
2016-10-31 19:02:12 -07:00
void TriggerShape::addEvent2(angle_t angle, trigger_wheel_e const waveIndex, trigger_value_e const stateParam, float filterLeft, float filterRight DECLARE_ENGINE_PARAMETER_S) {
2015-07-10 06:01:56 -07:00
if (angle > filterLeft && angle < filterRight)
2016-10-31 19:02:12 -07:00
addEvent2(angle, waveIndex, stateParam PASS_ENGINE_PARAMETER);
2015-07-10 06:01:56 -07:00
}
operation_mode_e TriggerShape::getOperationMode() {
return operationMode;
}
2015-09-11 22:02:28 -07:00
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
2015-09-10 20:01:32 -07:00
extern bool printTriggerDebug;
#endif
2015-10-10 13:01:25 -07:00
void TriggerShape::addEvent2(angle_t angle, trigger_wheel_e const waveIndex, trigger_value_e const stateParam DECLARE_ENGINE_PARAMETER_S) {
2015-07-10 06:01:56 -07:00
efiAssertVoid(operationMode != OM_NONE, "operationMode not set");
efiAssertVoid(waveIndex!= T_SECONDARY || needSecondTriggerInput, "secondary needed or not?");
2015-09-11 22:02:28 -07:00
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
2015-09-10 20:01:32 -07:00
if (printTriggerDebug) {
printf("addEvent %f\r\n", angle);
}
#endif
2015-07-10 06:01:56 -07:00
trigger_value_e state;
if (invertOnAdd) {
2015-10-31 13:02:10 -07:00
state = (stateParam == TV_FALL) ? TV_RISE : TV_FALL;
2015-07-10 06:01:56 -07:00
} else {
state = stateParam;
}
2015-08-30 11:01:28 -07:00
#if EFI_UNIT_TEST || defined(__DOXYGEN__)
2015-07-10 06:01:56 -07:00
int signal = waveIndex * 1000 + stateParam;
2015-08-30 11:01:28 -07:00
triggerSignals[size] = signal;
2015-07-10 06:01:56 -07:00
#endif
float engineCycle = getEngineCycle(operationMode);
/**
* While '720' value works perfectly it has not much sense for crank sensor-only scenario.
* todo: accept angle as a value in the 0..1 range?
*/
angle /= engineCycle;
2016-02-27 20:03:34 -08:00
if (!engineConfiguration->useOnlyRisingEdgeForTrigger || stateParam == TV_RISE) {
2015-07-10 06:01:56 -07:00
expectedEventCount[waveIndex]++;
}
efiAssertVoid(angle > 0, "angle should be positive");
if (size > 0) {
if (angle <= previousAngle) {
2016-10-10 11:02:17 -07:00
firmwareError(OBD_PCM_Processor_Fault, "invalid angle order: %f and %f, size=%d", angle, previousAngle, size);
2015-07-10 06:01:56 -07:00
return;
}
}
previousAngle = angle;
if (size == 0) {
size = 1;
for (int i = 0; i < PWM_PHASE_MAX_WAVE_PER_PWM; i++) {
single_wave_s *wave = &this->wave.waves[i];
if (wave->pinStates == NULL) {
2016-10-10 11:02:17 -07:00
firmwareError(OBD_PCM_Processor_Fault, "wave pinStates is NULL");
2015-07-10 06:01:56 -07:00
return;
}
wave->pinStates[0] = initialState[i];
}
2015-10-31 13:02:10 -07:00
isFrontEvent[0] = TV_RISE == stateParam;
2015-07-10 06:01:56 -07:00
wave.setSwitchTime(0, angle);
wave.waves[waveIndex].pinStates[0] = state;
return;
}
int exactMatch = wave.findAngleMatch(angle, size);
if (exactMatch != EFI_ERROR_CODE) {
2016-10-10 11:02:17 -07:00
firmwareError(OBD_PCM_Processor_Fault, "same angle: not supported");
2015-07-10 06:01:56 -07:00
return;
}
int index = wave.waveIndertionAngle(angle, size);
// shifting existing data
// todo: does this logic actually work? I think it does not!
for (int i = size - 1; i >= index; i--) {
for (int j = 0; j < PWM_PHASE_MAX_WAVE_PER_PWM; j++) {
wave.waves[j].pinStates[i + 1] = wave.getChannelState(j, index);
}
wave.setSwitchTime(i + 1, wave.getSwitchTime(i));
}
2015-10-31 13:02:10 -07:00
isFrontEvent[index] = TV_RISE == stateParam;
2015-07-10 06:01:56 -07:00
if (index != size) {
2016-10-10 11:02:17 -07:00
firmwareError(OBD_PCM_Processor_Fault, "are we ever here?");
2015-07-10 06:01:56 -07:00
}
// int index = size;
size++;
for (int i = 0; i < PWM_PHASE_MAX_WAVE_PER_PWM; i++) {
wave.waves[i].pinStates[index] = wave.getChannelState(i, index - 1);
}
wave.setSwitchTime(index, angle);
wave.waves[waveIndex].pinStates[index] = state;
}
2016-01-14 20:03:17 -08:00
angle_t TriggerShape::getSwitchAngle(int index) const {
2015-07-10 06:01:56 -07:00
return getCycleDuration() * wave.getSwitchTime(index);
}
void multi_wave_s::checkSwitchTimes(int size) {
checkSwitchTimes2(size, switchTimes);
}
2016-10-31 19:02:12 -07:00
void setVwConfiguration(TriggerShape *s DECLARE_ENGINE_PARAMETER_S) {
2015-07-10 06:01:56 -07:00
efiAssertVoid(s != NULL, "TriggerShape is NULL");
operation_mode_e operationMode = FOUR_STROKE_CRANK_SENSOR;
2016-08-17 21:02:22 -07:00
s->initialize(operationMode, false);
2015-07-10 06:01:56 -07:00
s->isSynchronizationNeeded = true;
int totalTeethCount = 60;
int skippedCount = 2;
float engineCycle = getEngineCycle(operationMode);
float toothWidth = 0.5;
addSkippedToothTriggerEvents(T_PRIMARY, s, 60, 2, toothWidth, 0, engineCycle,
2016-10-31 19:02:12 -07:00
NO_LEFT_FILTER, 690 PASS_ENGINE_PARAMETER);
2015-07-10 06:01:56 -07:00
float angleDown = engineCycle / totalTeethCount * (totalTeethCount - skippedCount - 1 + (1 - toothWidth) );
2016-10-31 19:02:12 -07:00
s->addEvent2(0 + angleDown + 12, T_PRIMARY, TV_RISE, NO_LEFT_FILTER, NO_RIGHT_FILTER PASS_ENGINE_PARAMETER);
s->addEvent2(0 + engineCycle, T_PRIMARY, TV_FALL, NO_LEFT_FILTER, NO_RIGHT_FILTER PASS_ENGINE_PARAMETER);
2015-07-10 06:01:56 -07:00
s->setTriggerSynchronizationGap2(1.6, 4);
}
void setToothedWheelConfiguration(TriggerShape *s, int total, int skipped,
2016-10-31 19:02:12 -07:00
operation_mode_e operationMode DECLARE_ENGINE_PARAMETER_S) {
2015-08-30 11:01:28 -07:00
#if EFI_ENGINE_CONTROL || defined(__DOXYGEN__)
2015-07-10 06:01:56 -07:00
s->useRiseEdge = true;
initializeSkippedToothTriggerShapeExt(s, total, skipped,
2016-10-31 19:02:12 -07:00
operationMode PASS_ENGINE_PARAMETER);
2015-07-10 06:01:56 -07:00
#endif
}
void TriggerShape::setTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
isSynchronizationNeeded = true;
this->syncRatioFrom = syncRatioFrom;
this->syncRatioTo = syncRatioTo;
}
2015-12-27 13:02:44 -08:00
void TriggerShape::setTriggerSynchronizationGap(float syncRatio) {
setTriggerSynchronizationGap2(syncRatio * 0.75f, syncRatio * 1.25f);
2015-09-12 12:02:40 -07:00
}
void TriggerShape::setSecondTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
isSynchronizationNeeded = true;
this->secondSyncRatioFrom = syncRatioFrom;
this->secondSyncRatioTo = syncRatioTo;
}
2015-12-27 13:02:44 -08:00
void TriggerShape::setThirdTriggerSynchronizationGap(float syncRatio) {
setThirdTriggerSynchronizationGap2(syncRatio * 0.75f, syncRatio * 1.25f);
}
void TriggerShape::setThirdTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
isSynchronizationNeeded = true;
this->thirdSyncRatioFrom = syncRatioFrom;
this->thirdSyncRatioTo = syncRatioTo;
}
void TriggerShape::setSecondTriggerSynchronizationGap(float syncRatio) {
setSecondTriggerSynchronizationGap2(syncRatio * 0.75f, syncRatio * 1.25f);
2015-07-10 06:01:56 -07:00
}