custom-board-bundle-sample-.../firmware/controllers/trigger/decoders/trigger_structure.cpp

617 lines
18 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file trigger_structure.cpp
*
* @date Jan 20, 2014
2018-01-20 17:55:31 -08:00
* @author Andrey Belomutskiy, (c) 2012-2018
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/>.
*/
2018-12-08 13:31:46 -08:00
2018-09-16 19:26:57 -07:00
#include "global.h"
#include "os_access.h"
2019-07-05 17:03:32 -07:00
#include "engine.h"
#include "trigger_bmw.h"
#include "trigger_chrysler.h"
#include "trigger_gm.h"
#include "trigger_nissan.h"
#include "trigger_mazda.h"
#include "trigger_misc.h"
#include "trigger_mitsubishi.h"
#include "trigger_subaru.h"
2015-07-10 06:01:56 -07:00
#include "trigger_structure.h"
#include "trigger_toyota.h"
#include "trigger_rover.h"
#include "trigger_honda.h"
#include "trigger_vw.h"
2017-01-03 13:02:43 -08:00
#include "trigger_universal.h"
2019-04-12 19:07:03 -07:00
#if EFI_SENSOR_CHART
2017-05-18 13:39:04 -07:00
#include "sensor_chart.h"
2017-05-19 10:04:16 -07:00
#endif /* EFI_SENSOR_CHART */
2015-07-10 06:01:56 -07:00
#include "engine_configuration.h"
extern persistent_config_container_s persistentState;
2015-07-10 06:01:56 -07:00
trigger_shape_helper::trigger_shape_helper() {
memset(&pinStates, 0, sizeof(pinStates));
2018-12-25 07:13:00 -08:00
for (int channelIndex = 0; channelIndex < TRIGGER_CHANNEL_COUNT; channelIndex++) {
channels[channelIndex].init(pinStates[channelIndex]);
2015-07-10 06:01:56 -07:00
}
}
TriggerShape::TriggerShape() :
wave(switchTimesBuffer, NULL) {
2015-10-29 11:02:52 -07:00
initialize(OM_NONE, false);
2018-12-25 07:13:00 -08:00
wave.channels = h.channels;
2016-08-18 21:02:50 -07:00
2015-07-10 06:01:56 -07:00
memset(triggerIndexByAngle, 0, sizeof(triggerIndexByAngle));
}
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
bothFrontsRequired = false;
2016-08-17 21:02:22 -07:00
this->needSecondTriggerInput = needSecondTriggerInput;
2018-02-05 14:16:34 -08:00
memset(expectedDutyCycle, 0, sizeof(expectedDutyCycle));
2016-08-17 21:02:22 -07:00
memset(eventAngles, 0, sizeof(eventAngles));
// memset(triggerIndexByAngle, 0, sizeof(triggerIndexByAngle));
2016-08-18 19:01:57 -07:00
2018-10-21 10:58:54 -07:00
setTriggerSynchronizationGap(2);
2018-12-25 07:20:13 -08:00
for (int gapIndex = 1; gapIndex < GAP_TRACKING_LENGTH ; gapIndex++) {
2018-10-21 10:58:54 -07:00
// NaN means do not use this gap ratio
2018-12-25 07:20:13 -08:00
setTriggerSynchronizationGap3(gapIndex, NAN, 100000);
2018-10-21 10:58:54 -07:00
}
2016-08-18 19:01:57 -07:00
2015-07-10 06:01:56 -07:00
tdcPosition = 0;
2017-02-23 10:53:27 -08:00
shapeDefinitionError = useOnlyPrimaryForSync = false;
2015-07-10 06:01:56 -07:00
useRiseEdge = true;
2017-03-18 17:59:52 -07:00
gapBothDirections = false;
2015-07-10 06:01:56 -07:00
invertOnAdd = false;
this->operationMode = operationMode;
privateTriggerDefinitionSize = 0;
2015-07-10 06:01:56 -07:00
triggerShapeSynchPointIndex = 0;
memset(initialState, 0, sizeof(initialState));
memset(switchTimesBuffer, 0, sizeof(switchTimesBuffer));
memset(expectedEventCount, 0, sizeof(expectedEventCount));
wave.reset();
previousAngle = 0;
memset(riseOnlyIndexes, 0, sizeof(riseOnlyIndexes));
memset(isRiseEvent, 0, sizeof(isRiseEvent));
2019-04-12 19:07:03 -07:00
#if EFI_UNIT_TEST
2016-08-19 21:02:28 -07:00
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 privateTriggerDefinitionSize;
2016-08-20 06:02:06 -07:00
}
int TriggerShape::getTriggerShapeSynchPointIndex() const {
2016-08-20 06:02:06 -07:00
return triggerShapeSynchPointIndex;
}
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()
*/
efiAssert(CUSTOM_ERR_ASSERT, privateTriggerDefinitionSize != 0, "shapeSize=0", NAN);
int crankCycle = index / privateTriggerDefinitionSize;
int remainder = index % privateTriggerDefinitionSize;
2016-11-13 21:02:13 -08:00
return getCycleDuration() * crankCycle + getSwitchAngle(remainder);
2015-07-10 06:01:56 -07:00
}
2019-05-02 15:05:33 -07:00
void TriggerShape::addEventClamped(angle_t angle, trigger_wheel_e const channelIndex, trigger_value_e const stateParam, float filterLeft, float filterRight) {
2019-07-23 16:22:40 -07:00
if (angle > filterLeft && angle < filterRight) {
#if EFI_UNIT_TEST
// printf("addEventClamped %f %s\r\n", angle, getTrigger_value_e(stateParam));
#endif /* EFI_UNIT_TEST */
2019-02-02 22:34:20 -08:00
addEvent(angle / getEngineCycle(operationMode), channelIndex, stateParam);
2019-07-23 16:22:40 -07:00
}
2015-07-10 06:01:56 -07:00
}
operation_mode_e TriggerShape::getOperationMode() const {
2015-07-10 06:01:56 -07:00
return operationMode;
}
2019-04-12 19:07:03 -07:00
#if EFI_UNIT_TEST
2015-09-10 20:01:32 -07:00
extern bool printTriggerDebug;
#endif
2018-12-24 20:16:33 -08:00
void TriggerShape::calculateExpectedEventCounts(bool useOnlyRisingEdgeForTrigger) {
UNUSED(useOnlyRisingEdgeForTrigger);
2018-12-24 20:16:33 -08:00
// todo: move the following logic from below here
// if (!useOnlyRisingEdgeForTrigger || stateParam == TV_RISE) {
2018-12-25 07:13:00 -08:00
// expectedEventCount[channelIndex]++;
2018-12-24 20:16:33 -08:00
// }
}
2018-12-25 07:33:28 -08:00
void TriggerShape::addEvent720(angle_t angle, trigger_wheel_e const channelIndex, trigger_value_e const stateParam) {
2019-02-02 22:34:20 -08:00
addEvent(angle / 720, channelIndex, stateParam);
2018-12-25 07:20:13 -08:00
}
2019-02-02 22:34:20 -08:00
void TriggerShape::addEvent(angle_t angle, trigger_wheel_e const channelIndex, trigger_value_e const stateParam) {
2018-09-10 19:10:55 -07:00
efiAssertVoid(CUSTOM_OMODE_UNDEF, operationMode != OM_NONE, "operationMode not set");
2015-07-10 06:01:56 -07:00
2018-12-25 07:13:00 -08:00
efiAssertVoid(CUSTOM_ERR_6598, channelIndex!= T_SECONDARY || needSecondTriggerInput, "secondary needed or not?");
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_UNIT_TEST
2015-09-10 20:01:32 -07:00
if (printTriggerDebug) {
2018-12-25 07:13:00 -08:00
printf("addEvent2 %.2f i=%d r/f=%d\r\n", angle, channelIndex, stateParam);
2015-09-10 20:01:32 -07:00
}
#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;
}
2019-04-12 19:07:03 -07:00
#if EFI_UNIT_TEST
2018-12-25 07:13:00 -08:00
int signal = channelIndex * 1000 + stateParam;
triggerSignals[privateTriggerDefinitionSize] = signal;
2015-07-10 06:01:56 -07:00
#endif
2019-02-02 22:34:20 -08:00
// todo: the whole 'useOnlyRisingEdgeForTrigger' parameter and logic should not be here
// todo: see calculateExpectedEventCounts
// related calculation should be done once trigger is initialized outside of trigger shape scope
if (!useOnlyRisingEdgeForTriggerTemp || stateParam == TV_RISE) {
2018-12-25 07:13:00 -08:00
expectedEventCount[channelIndex]++;
2015-07-10 06:01:56 -07:00
}
2018-07-25 20:03:04 -07:00
efiAssertVoid(CUSTOM_ERR_6599, angle > 0, "angle should be positive");
if (privateTriggerDefinitionSize > 0) {
2015-07-10 06:01:56 -07:00
if (angle <= previousAngle) {
warning(CUSTOM_ERR_TRG_ANGLE_ORDER, "invalid angle order: new=%.2f and prev=%.2f, size=%d", angle, previousAngle, privateTriggerDefinitionSize);
setShapeDefinitionError(true);
2015-07-10 06:01:56 -07:00
return;
}
}
previousAngle = angle;
if (privateTriggerDefinitionSize == 0) {
privateTriggerDefinitionSize = 1;
2015-07-10 06:01:56 -07:00
for (int i = 0; i < PWM_PHASE_MAX_WAVE_PER_PWM; i++) {
2018-12-25 07:13:00 -08:00
SingleWave *wave = &this->wave.channels[i];
2015-07-10 06:01:56 -07:00
if (wave->pinStates == NULL) {
2017-03-12 19:46:01 -07:00
warning(CUSTOM_ERR_STATE_NULL, "wave pinStates is NULL");
setShapeDefinitionError(true);
2015-07-10 06:01:56 -07:00
return;
}
2019-02-02 22:04:24 -08:00
wave->setState(/* switchIndex */ 0, /* value */ initialState[i]);
2015-07-10 06:01:56 -07:00
}
isRiseEvent[0] = TV_RISE == stateParam;
2015-07-10 06:01:56 -07:00
wave.setSwitchTime(0, angle);
2018-12-25 07:13:00 -08:00
wave.channels[channelIndex].setState(/* channelIndex */ 0, /* value */ state);
2015-07-10 06:01:56 -07:00
return;
}
int exactMatch = wave.findAngleMatch(angle, privateTriggerDefinitionSize);
2015-07-10 06:01:56 -07:00
if (exactMatch != EFI_ERROR_CODE) {
2017-04-19 19:03:14 -07:00
warning(CUSTOM_ERR_SAME_ANGLE, "same angle: not supported");
setShapeDefinitionError(true);
2015-07-10 06:01:56 -07:00
return;
}
int index = wave.findInsertionAngle(angle, privateTriggerDefinitionSize);
2015-07-10 06:01:56 -07:00
/**
* todo: it would be nice to be able to provide trigger angles without sorting them externally
* The idea here is to shift existing data - including handling high vs low state of the signals
*/
2018-02-05 14:16:34 -08:00
// todo: does this logic actually work? I think it does not! due to broken state handling
/*
2015-07-10 06:01:56 -07:00
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));
}
*/
isRiseEvent[index] = TV_RISE == stateParam;
2015-07-10 06:01:56 -07:00
if ((unsigned)index != privateTriggerDefinitionSize) {
2017-05-29 16:23:15 -07:00
firmwareError(ERROR_TRIGGER_DRAMA, "are we ever here?");
2015-07-10 06:01:56 -07:00
}
privateTriggerDefinitionSize++;
2015-07-10 06:01:56 -07:00
for (int i = 0; i < PWM_PHASE_MAX_WAVE_PER_PWM; i++) {
2019-02-02 22:27:47 -08:00
pin_state_t value = wave.getChannelState(/* channelIndex */i, index - 1);
2018-12-25 07:13:00 -08:00
wave.channels[i].setState(index, value);
2015-07-10 06:01:56 -07:00
}
wave.setSwitchTime(index, angle);
2018-12-25 07:13:00 -08:00
wave.channels[channelIndex].setState(index, state);
2015-07-10 06:01:56 -07:00
}
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 setToothedWheelConfiguration(TriggerShape *s, int total, int skipped,
2018-12-25 17:09:35 -08:00
operation_mode_e operationMode) {
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL
2015-07-10 06:01:56 -07:00
s->useRiseEdge = true;
initializeSkippedToothTriggerShapeExt(s, total, skipped,
2018-12-25 17:09:35 -08:00
operationMode);
2015-07-10 06:01:56 -07:00
#endif
}
void TriggerShape::setTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
2018-12-25 07:20:13 -08:00
setTriggerSynchronizationGap3(/*gapIndex*/0, syncRatioFrom, syncRatioTo);
2018-10-21 10:41:01 -07:00
}
2018-12-25 07:20:13 -08:00
void TriggerShape::setTriggerSynchronizationGap3(int gapIndex, float syncRatioFrom, float syncRatioTo) {
2015-07-10 06:01:56 -07:00
isSynchronizationNeeded = true;
2018-12-25 07:20:13 -08:00
this->syncronizationRatioFrom[gapIndex] = syncRatioFrom;
this->syncronizationRatioTo[gapIndex] = syncRatioTo;
if (gapIndex == 0) {
2018-10-21 10:41:01 -07:00
// we have a special case here - only sync with one gap has this feature
this->syncRatioAvg = (int)efiRound((syncRatioFrom + syncRatioTo) * 0.5f, 1.0f);
}
2019-04-12 19:07:03 -07:00
#if EFI_UNIT_TEST
2017-03-04 16:47:15 -08:00
if (printTriggerDebug) {
2018-12-25 07:20:13 -08:00
printf("setTriggerSynchronizationGap3 %d %.2f %.2f\r\n", gapIndex, syncRatioFrom, syncRatioTo);
2017-03-04 16:47:15 -08:00
}
#endif /* EFI_UNIT_TEST */
2018-10-21 10:41:01 -07:00
2015-07-10 06:01:56 -07:00
}
2018-12-25 09:27:34 -08:00
/**
* this method is only used on initialization
*/
int TriggerShape::findAngleIndex(float target) const {
2018-12-25 09:27:34 -08:00
int engineCycleEventCount = getLength();
efiAssert(CUSTOM_ERR_ASSERT, engineCycleEventCount > 0, "engineCycleEventCount", 0);
uint32_t left = 0;
uint32_t right = engineCycleEventCount - 1;
/**
* Let's find the last trigger angle which is less or equal to the desired angle
* todo: extract binary search as template method?
*/
while (left <= right) {
int middle = (left + right) / 2;
angle_t eventAngle = eventAngles[middle];
if (eventAngle < target) {
left = middle + 1;
} else if (eventAngle > target) {
right = middle - 1;
} else {
// Values are equal
return middle; // Key found
}
}
return left - 1;
}
void TriggerShape::setShapeDefinitionError(bool value) {
shapeDefinitionError = value;
}
void TriggerShape::findTriggerPosition(event_trigger_position_s *position,
2019-10-07 21:30:20 -07:00
angle_t angle DEFINE_CONFIG_PARAM(angle_t, globalTriggerAngleOffset)) {
efiAssertVoid(CUSTOM_ERR_6574, !cisnan(angle), "findAngle#1");
assertAngleRange(angle, "findAngle#a1", CUSTOM_ERR_6545);
2018-12-25 09:27:34 -08:00
efiAssertVoid(CUSTOM_ERR_6575, !cisnan(tdcPosition), "tdcPos#1")
assertAngleRange(tdcPosition, "tdcPos#a1", CUSTOM_UNEXPECTED_TDC_ANGLE);
2018-12-25 09:27:34 -08:00
efiAssertVoid(CUSTOM_ERR_6576, !cisnan(CONFIG_PARAM(globalTriggerAngleOffset)), "tdcPos#2")
assertAngleRange(CONFIG_PARAM(globalTriggerAngleOffset), "tdcPos#a2", CUSTOM_INVALID_GLOBAL_OFFSET);
2018-12-25 09:27:34 -08:00
// convert engine cycle angle into trigger cycle angle
2019-10-07 21:30:20 -07:00
angle += tdcPosition + CONFIG_PARAM(globalTriggerAngleOffset);
efiAssertVoid(CUSTOM_ERR_6577, !cisnan(angle), "findAngle#2");
fixAngle2(angle, "addFuel#2", CUSTOM_ERR_6555, getEngineCycle(operationMode));
2018-12-25 09:27:34 -08:00
2019-10-07 21:42:47 -07:00
int triggerEventIndex = triggerIndexByAngle[(int)angle];
angle_t triggerEventAngle = eventAngles[triggerEventIndex];
if (angle < triggerEventAngle) {
warning(CUSTOM_OBD_ANGLE_CONSTRAINT_VIOLATION, "angle constraint violation in findTriggerPosition(): %.2f/%.2f", angle, triggerEventAngle);
2018-12-25 09:27:34 -08:00
return;
}
2019-10-07 21:54:19 -07:00
position->triggerEventIndex = triggerEventIndex;
2019-10-07 22:03:57 -07:00
position->triggerEventAngle = triggerEventAngle;
position->angleOffsetFromTriggerEvent = angle - triggerEventAngle;
2018-12-25 09:27:34 -08:00
}
void TriggerShape::prepareShape() {
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT
int engineCycleInt = (int) getEngineCycle(operationMode);
for (int angle = 0; angle < engineCycleInt; angle++) {
int triggerShapeIndex = findAngleIndex(angle);
if (useOnlyRisingEdgeForTriggerTemp) {
// we need even index for front_only mode - so if odd indexes are rounded down
triggerShapeIndex = triggerShapeIndex & 0xFFFFFFFE;
}
triggerIndexByAngle[angle] = triggerShapeIndex;
}
2019-01-31 15:30:40 -08:00
#endif
}
2015-12-27 13:02:44 -08:00
void TriggerShape::setTriggerSynchronizationGap(float syncRatio) {
2018-12-25 07:20:13 -08:00
setTriggerSynchronizationGap3(/*gapIndex*/0, syncRatio * 0.75f, syncRatio * 1.25f);
2015-09-12 12:02:40 -07:00
}
void TriggerShape::setSecondTriggerSynchronizationGap2(float syncRatioFrom, float syncRatioTo) {
2018-12-25 07:20:13 -08:00
setTriggerSynchronizationGap3(/*gapIndex*/1, syncRatioFrom, syncRatioTo);
2015-09-12 12:02:40 -07:00
}
2015-12-27 13:02:44 -08:00
void TriggerShape::setThirdTriggerSynchronizationGap(float syncRatio) {
2018-12-25 07:20:13 -08:00
setTriggerSynchronizationGap3(/*gapIndex*/2, syncRatio * 0.75f, syncRatio * 1.25f);
2015-12-27 13:02:44 -08:00
}
void TriggerShape::setSecondTriggerSynchronizationGap(float syncRatio) {
2018-12-25 07:20:13 -08:00
setTriggerSynchronizationGap3(/*gapIndex*/1, syncRatio * 0.75f, syncRatio * 1.25f);
2015-07-10 06:01:56 -07:00
}
/**
* External logger is needed because at this point our logger is not yet initialized
*/
void TriggerShape::initializeTriggerShape(Logging *logger, operation_mode_e ambiguousOperationMode, bool useOnlyRisingEdgeForTrigger, const trigger_config_s *triggerConfig) {
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
2019-02-23 09:33:49 -08:00
efiAssertVoid(CUSTOM_ERR_6641, getCurrentRemainingStack() > 256, "init t");
scheduleMsg(logger, "initializeTriggerShape(%s/%d)", getTrigger_type_e(triggerConfig->type), (int) triggerConfig->type);
#endif
shapeDefinitionError = false;
this->useOnlyRisingEdgeForTriggerTemp = useOnlyRisingEdgeForTrigger;
switch (triggerConfig->type) {
case TT_TOOTHED_WHEEL:
initializeSkippedToothTriggerShapeExt(this, triggerConfig->customTotalToothCount,
triggerConfig->customSkippedToothCount, ambiguousOperationMode);
break;
case TT_MAZDA_MIATA_NA:
2019-02-02 21:40:25 -08:00
initializeMazdaMiataNaShape(this);
break;
case TT_MAZDA_MIATA_NB1:
initializeMazdaMiataNb1Shape(this);
break;
case TT_MAZDA_MIATA_VVT_TEST:
initializeMazdaMiataVVtTestShape(this);
break;
case TT_MAZDA_Z5:
initialize_Mazda_Engine_z5_Shape(this);
break;
case TT_MIATA_VVT:
initializeMazdaMiataNb2Crank(this);
break;
case TT_DODGE_NEON_1995:
configureNeon1995TriggerShape(this);
break;
case TT_DODGE_NEON_1995_ONLY_CRANK:
configureNeon1995TriggerShapeOnlyCrank(this);
break;
case TT_DODGE_STRATUS:
configureDodgeStratusTriggerShape(this);
break;
case TT_DODGE_NEON_2003_CAM:
configureNeon2003TriggerShapeCam(this);
break;
case TT_DODGE_NEON_2003_CRANK:
configureNeon2003TriggerShapeCam(this);
// configureNeon2003TriggerShapeCrank(triggerShape);
break;
case TT_FORD_ASPIRE:
configureFordAspireTriggerShape(this);
break;
case TT_GM_7X:
configureGmTriggerShape(this);
break;
case TT_MAZDA_DOHC_1_4:
configureMazdaProtegeLx(this);
break;
case TT_ONE_PLUS_ONE:
configureOnePlusOne(this, ambiguousOperationMode);
break;
case TT_3_1_CAM:
configure3_1_cam(this, ambiguousOperationMode);
break;
case TT_ONE_PLUS_TOOTHED_WHEEL_60_2:
configureOnePlus60_2(this, ambiguousOperationMode);
break;
case TT_ONE:
setToothedWheelConfiguration(this, 1, 0, ambiguousOperationMode);
break;
case TT_MAZDA_SOHC_4:
configureMazdaProtegeSOHC(this);
break;
case TT_MINI_COOPER_R50:
configureMiniCooperTriggerShape(this);
break;
case TT_TOOTHED_WHEEL_60_2:
setToothedWheelConfiguration(this, 60, 2, ambiguousOperationMode);
break;
case TT_60_2_VW:
setVwConfiguration(this);
break;
case TT_TOOTHED_WHEEL_36_1:
setToothedWheelConfiguration(this, 36, 1, ambiguousOperationMode);
break;
case TT_HONDA_4_24_1:
configureHonda_1_4_24(this, true, true, T_CHANNEL_3, T_PRIMARY, 0);
break;
case TT_HONDA_4_24:
configureHonda_1_4_24(this, false, true, T_NONE, T_PRIMARY, 0);
break;
case TT_HONDA_1_24:
configureHonda_1_4_24(this, true, false, T_PRIMARY, T_NONE, 10);
break;
case TT_HONDA_ACCORD_1_24_SHIFTED:
configureHondaAccordShifted(this);
break;
case TT_HONDA_1_4_24:
configureHondaAccordCDDip(this);
break;
case TT_HONDA_CBR_600:
configureHondaCbr600(this);
break;
case TT_HONDA_CBR_600_CUSTOM:
configureHondaCbr600custom(this);
break;
case TT_MITSUBISHI:
initializeMitsubishi4g18(this);
break;
case TT_DODGE_RAM:
initDodgeRam(this);
break;
case TT_JEEP_4_CYL:
initJeep_XJ_4cyl_2500(this);
break;
case TT_JEEP_18_2_2_2:
initJeep18_2_2_2(this);
break;
case TT_SUBARU_7_6:
initializeSubaru7_6(this);
break;
case TT_36_2_2_2:
initialize36_2_2_2(this);
break;
case TT_2JZ_3_34:
initialize2jzGE3_34(this);
break;
case TT_2JZ_1_12:
initialize2jzGE1_12(this);
break;
case TT_NISSAN_SR20VE:
initializeNissanSR20VE_4(this);
break;
case TT_NISSAN_SR20VE_360:
initializeNissanSR20VE_4_360(this);
break;
case TT_ROVER_K:
initializeRoverK(this);
break;
case TT_FIAT_IAW_P8:
configureFiatIAQ_P8(this);
break;
case TT_GM_LS_24:
initGmLS24(this);
break;
default:
setShapeDefinitionError(true);
warning(CUSTOM_ERR_NO_SHAPE, "initializeTriggerShape() not implemented: %d", triggerConfig->type);
}
2019-02-02 21:40:25 -08:00
/**
* Feb 2019 suggestion: it would be an improvement to remove 'expectedEventCount' logic from 'addEvent'
* and move it here, after all events were added.
*/
calculateExpectedEventCounts(useOnlyRisingEdgeForTrigger);
version++;
if (!shapeDefinitionError) {
wave.checkSwitchTimes(getSize());
}
}