custom-board-bundle-sample-.../firmware/controllers/trigger/trigger_simulator.cpp

156 lines
5.3 KiB
C++
Raw Normal View History

2015-09-23 18:02:33 -07:00
/*
* @file trigger_simulator.cpp
*
* @date Sep 23, 2015
2020-01-13 18:57:43 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
2015-09-23 18:02:33 -07:00
*/
2018-09-16 19:26:57 -07:00
#include "global.h"
2015-09-23 18:02:33 -07:00
#include "engine.h"
#include "trigger_simulator.h"
2017-03-03 21:17:53 -08:00
#include "trigger_emulator_algo.h"
2015-09-23 18:02:33 -07:00
#define SIMULATION_CYCLE_PERIOD 720000
EXTERN_ENGINE;
2017-03-03 21:17:53 -08:00
// this is not the only place where we have 'isUpEvent'. todo: reuse
2020-01-27 21:27:30 -08:00
static const bool isRisingEdge[HW_EVENT_TYPES] = { false, true, false, true, false, true };
2017-03-03 21:17:53 -08:00
/**
* todo: should this method be invoked somewhere deeper? at the moment we have too many usages too high
* @return true if front should be decoded further, false if we are not interested
*/
bool isUsefulSignal(trigger_event_e signal, const TriggerConfiguration& triggerConfiguration) {
return !triggerConfiguration.UseOnlyRisingEdgeForTrigger || isRisingEdge[(int) signal];
2017-03-03 21:17:53 -08:00
}
2019-04-12 19:07:03 -07:00
#if EFI_UNIT_TEST
2017-03-04 06:07:10 -08:00
extern bool printTriggerDebug;
#endif /* ! EFI_UNIT_TEST */
void TriggerStimulatorHelper::feedSimulatedEvent(
const TriggerStateCallback triggerCycleCallback,
const TriggerConfiguration& triggerConfiguration,
TriggerState& state,
const TriggerWaveform& shape,
int i
2020-08-23 23:23:13 -07:00
) {
efiAssertVoid(CUSTOM_ERR_6593, shape.getSize() > 0, "size not zero");
int stateIndex = i % shape.getSize();
int size = shape.getSize();
2015-09-23 18:02:33 -07:00
int loopIndex = i / shape.getSize();
2015-09-23 18:02:33 -07:00
int time = (int) (SIMULATION_CYCLE_PERIOD * (loopIndex + shape.wave.getSwitchTime(stateIndex)));
2015-09-23 18:02:33 -07:00
const MultiChannelStateSequence& multiChannelStateSequence = shape.wave;
2019-04-14 10:52:27 -07:00
#if EFI_UNIT_TEST
int prevIndex = getPreviousIndex(stateIndex, shape.getSize());
2019-04-14 10:52:27 -07:00
pin_state_t primaryWheelState = multiChannelStateSequence.getChannelState(0, prevIndex);
pin_state_t newPrimaryWheelState = multiChannelStateSequence.getChannelState(0, stateIndex);
2015-09-23 18:02:33 -07:00
pin_state_t secondaryWheelState = multiChannelStateSequence.getChannelState(1, prevIndex);
pin_state_t newSecondaryWheelState = multiChannelStateSequence.getChannelState(1, stateIndex);
2015-09-23 18:02:33 -07:00
// pin_state_t thirdWheelState = multiChannelStateSequence->getChannelState(2, prevIndex);
// pin_state_t new3rdWheelState = multiChannelStateSequence->getChannelState(2, stateIndex);
2015-09-23 18:02:33 -07:00
2017-03-04 06:07:10 -08:00
if (printTriggerDebug) {
2020-07-19 11:17:15 -07:00
printf("TriggerStimulator: simulatedEvent: %d>%d primary %d>%d secondary %d>%d\r\n", prevIndex, stateIndex, primaryWheelState, newPrimaryWheelState,
2017-03-04 06:07:10 -08:00
secondaryWheelState, newSecondaryWheelState );
}
#endif /* EFI_UNIT_TEST */
2017-03-03 21:08:56 -08:00
// todo: code duplication with TriggerEmulatorHelper::handleEmulatorCallback?
if (needEvent(stateIndex, size, multiChannelStateSequence, 0)) {
pin_state_t currentValue = multiChannelStateSequence.getChannelState(/*phaseIndex*/0, stateIndex);
2020-08-26 14:30:13 -07:00
trigger_event_e event = currentValue ? SHAFT_PRIMARY_RISING : SHAFT_PRIMARY_FALLING;
if (isUsefulSignal(event, triggerConfiguration)) {
state.decodeTriggerEvent(shape,
2020-01-26 00:33:45 -08:00
triggerCycleCallback,
/* override */ nullptr,
2020-08-23 22:21:42 -07:00
triggerConfiguration,
2020-08-26 14:30:13 -07:00
event, time);
}
2015-09-23 18:02:33 -07:00
}
if (needEvent(stateIndex, size, multiChannelStateSequence, 1)) {
pin_state_t currentValue = multiChannelStateSequence.getChannelState(/*phaseIndex*/1, stateIndex);
2020-08-26 14:30:13 -07:00
trigger_event_e event = currentValue ? SHAFT_SECONDARY_RISING : SHAFT_SECONDARY_FALLING;
if (isUsefulSignal(event, triggerConfiguration)) {
state.decodeTriggerEvent(shape,
2020-01-26 00:33:45 -08:00
triggerCycleCallback,
/* override */ nullptr,
2020-08-23 22:21:42 -07:00
triggerConfiguration,
2020-08-26 14:30:13 -07:00
event, time);
}
2015-09-23 18:02:33 -07:00
}
if (needEvent(stateIndex, size, multiChannelStateSequence, 2)) {
pin_state_t currentValue = multiChannelStateSequence.getChannelState(/*phaseIndex*/2, stateIndex);
2020-08-26 14:30:13 -07:00
trigger_event_e event = currentValue ? SHAFT_3RD_RISING : SHAFT_3RD_FALLING;
if (isUsefulSignal(event, triggerConfiguration)) {
state.decodeTriggerEvent(shape,
2020-01-26 00:33:45 -08:00
triggerCycleCallback,
/* override */ nullptr,
2020-08-23 22:21:42 -07:00
triggerConfiguration,
2020-08-26 14:30:13 -07:00
event, time);
}
2015-09-23 18:02:33 -07:00
}
}
void TriggerStimulatorHelper::assertSyncPositionAndSetDutyCycle(
const TriggerStateCallback triggerCycleCallback,
const TriggerConfiguration& triggerConfiguration,
const uint32_t syncIndex,
TriggerState& state,
TriggerWaveform& shape
2020-08-23 23:23:13 -07:00
) {
2015-09-23 18:02:33 -07:00
2018-02-05 14:41:05 -08:00
/**
* let's feed two more cycles to validate shape definition
*/
for (uint32_t i = syncIndex + 1; i <= syncIndex + GAP_TRACKING_LENGTH * shape.getSize(); i++) {
2020-08-23 22:21:42 -07:00
feedSimulatedEvent(triggerCycleCallback,
triggerConfiguration,
2020-08-23 23:23:13 -07:00
state, shape, i);
2015-09-23 18:02:33 -07:00
}
int revolutionCounter = state.getTotalRevolutionCounter();
2018-10-28 14:20:43 -07:00
if (revolutionCounter != GAP_TRACKING_LENGTH + 1) {
warning(CUSTOM_OBD_TRIGGER_WAVEFORM, "sync failed/wrong gap parameters trigger=%s rc=%d", getTrigger_type_e(triggerConfiguration.TriggerType), revolutionCounter);
shape.setShapeDefinitionError(true);
2016-01-24 15:01:56 -08:00
return;
}
shape.shapeDefinitionError = false;
2015-09-23 18:02:33 -07:00
for (int i = 0; i < PWM_PHASE_MAX_WAVE_PER_PWM; i++) {
shape.expectedDutyCycle[i] = 1.0 * state.expectedTotalTime[i] / SIMULATION_CYCLE_PERIOD;
2015-09-23 18:02:33 -07:00
}
}
2017-03-04 06:07:10 -08:00
/**
* @return trigger synchronization point index, or error code if not found
*/
uint32_t TriggerStimulatorHelper::findTriggerSyncPoint(
TriggerWaveform& shape,
const TriggerConfiguration& triggerConfiguration,
TriggerState& state) {
2015-09-23 18:02:33 -07:00
for (int i = 0; i < 4 * PWM_PHASE_MAX_COUNT; i++) {
2020-08-23 22:21:42 -07:00
feedSimulatedEvent(nullptr,
triggerConfiguration,
2020-08-23 23:23:13 -07:00
state, shape, i);
2015-09-23 18:02:33 -07:00
if (state.shaft_is_synchronized) {
2015-09-23 18:02:33 -07:00
return i;
2020-01-26 11:20:55 -08:00
}
2015-09-23 18:02:33 -07:00
}
shape.setShapeDefinitionError(true);
2017-03-06 14:31:20 -08:00
warning(CUSTOM_ERR_TRIGGER_SYNC, "findTriggerZeroEventIndex() failed");
2015-09-23 18:02:33 -07:00
return EFI_ERROR_CODE;
}