2015-07-10 06:01:56 -07:00
|
|
|
/**
|
2019-12-03 22:11:10 -08:00
|
|
|
* @file state_sequence.h
|
2015-07-10 06:01:56 -07:00
|
|
|
*
|
|
|
|
* @date May 18, 2014
|
2020-01-07 21:02:40 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2019-12-03 22:11:10 -08:00
|
|
|
|
|
|
|
#pragma once
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2021-11-10 16:47:27 -08:00
|
|
|
#include <stdint.h>
|
2022-09-05 00:07:03 -07:00
|
|
|
#include "rusefi_enums.h"
|
|
|
|
#include "expected.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2022-09-10 23:57:35 -07:00
|
|
|
enum class TriggerValue : uint8_t {
|
|
|
|
FALL = 0,
|
|
|
|
RISE = 1
|
|
|
|
};
|
2022-09-10 23:16:47 -07:00
|
|
|
|
|
|
|
// see also 'HW_EVENT_TYPES'
|
|
|
|
typedef enum {
|
|
|
|
SHAFT_PRIMARY_FALLING = 0,
|
|
|
|
SHAFT_PRIMARY_RISING = 1,
|
|
|
|
SHAFT_SECONDARY_FALLING = 2,
|
|
|
|
SHAFT_SECONDARY_RISING = 3,
|
|
|
|
} trigger_event_e;
|
|
|
|
|
2019-03-29 07:37:33 -07:00
|
|
|
/**
|
|
|
|
* This layer has two primary usages:
|
|
|
|
* 1) 'simple' PWM generation is used to produce actuator square control wave
|
|
|
|
* 2) 'complex' PWM generation is used for trigger simulator.
|
|
|
|
* Some triggers like Nissan 360 slot optical wheel need a lot of points to describe the shape of the wave.
|
2019-03-29 07:49:56 -07:00
|
|
|
* Looks like 252 is explained by 60 tooth * 2 (number of fronts) * 2 (number of crank rotations within engine cycle)
|
2019-03-29 07:37:33 -07:00
|
|
|
*/
|
2019-03-29 07:29:01 -07:00
|
|
|
#ifndef PWM_PHASE_MAX_COUNT
|
2020-04-19 15:58:22 -07:00
|
|
|
// as of April 2020, trigger which requires most array length is REMIX_66_2_2_2
|
|
|
|
// we can probably reduce RAM usage if we have more custom logic of triggers with large number of tooth while
|
|
|
|
// pretty easy logic. like we do not need to REALLY have an array to remember the shape of evenly spaces 360 or 60/2 trigger :)
|
2021-07-20 18:10:53 -07:00
|
|
|
// todo https://github.com/rusefi/rusefi/issues/3003
|
2020-04-19 11:54:02 -07:00
|
|
|
#define PWM_PHASE_MAX_COUNT 280
|
2019-03-29 07:29:01 -07:00
|
|
|
#endif /* PWM_PHASE_MAX_COUNT */
|
2022-06-01 18:24:20 -07:00
|
|
|
#define PWM_PHASE_MAX_WAVE_PER_PWM 2
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2022-09-10 23:57:35 -07:00
|
|
|
typedef TriggerValue pin_state_t;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
/**
|
2021-11-21 01:56:07 -08:00
|
|
|
* This class represents multi-channel logical signals with shared time axis
|
2018-12-24 19:40:48 -08:00
|
|
|
*
|
2021-11-21 01:56:07 -08:00
|
|
|
* This is a semi-abstract interface so that implementations can exist for either regularized
|
|
|
|
* patterns (60-2, etc) or completely arbitrary patterns stored in arrays.
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2021-11-21 01:56:07 -08:00
|
|
|
class MultiChannelStateSequence {
|
2015-07-10 06:01:56 -07:00
|
|
|
public:
|
2018-12-24 19:40:48 -08:00
|
|
|
/**
|
2021-11-21 01:56:07 -08:00
|
|
|
* values in the (0..1] range which refer to points within the period at at which pin state
|
|
|
|
* should be changed So, in the simplest case we turn pin off at 0.3 and turn it on at 1 -
|
|
|
|
* that would give us a 70% duty cycle PWM
|
2018-12-24 19:40:48 -08:00
|
|
|
*/
|
2021-11-21 01:56:07 -08:00
|
|
|
virtual float getSwitchTime(int phaseIndex) const = 0;
|
|
|
|
virtual pin_state_t getChannelState(int channelIndex, int phaseIndex) const = 0;
|
2018-12-24 19:40:48 -08:00
|
|
|
|
2021-11-21 01:56:07 -08:00
|
|
|
// Make sure the switch times are in order and end at the very end.
|
|
|
|
void checkSwitchTimes(float scale) const;
|
|
|
|
|
2022-08-28 06:43:21 -07:00
|
|
|
// Find the exact angle, or unexpected if it doesn't exist
|
|
|
|
expected<int> findAngleMatch(float angle) const;
|
2021-11-21 01:56:07 -08:00
|
|
|
|
|
|
|
// returns the index at which given value would need to be inserted into sorted array
|
|
|
|
int findInsertionAngle(float angle) const;
|
|
|
|
|
|
|
|
uint16_t phaseCount = 0; // Number of timestamps
|
|
|
|
uint16_t waveCount = 0; // Number of waveforms
|
2015-07-10 06:01:56 -07:00
|
|
|
};
|
|
|
|
|
2021-11-21 01:56:07 -08:00
|
|
|
template<unsigned max_phase>
|
|
|
|
class MultiChannelStateSequenceWithData : public MultiChannelStateSequence {
|
2015-07-10 06:01:56 -07:00
|
|
|
public:
|
2021-11-21 01:56:07 -08:00
|
|
|
float getSwitchTime(int phaseIndex) const override {
|
|
|
|
return switchTimes[phaseIndex];
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2021-11-21 01:56:07 -08:00
|
|
|
pin_state_t getChannelState(int channelIndex, int phaseIndex) const override {
|
|
|
|
if (channelIndex >= waveCount) {
|
|
|
|
// todo: would be nice to get this asserting working
|
|
|
|
//firmwareError(OBD_PCM_Processor_Fault, "channel index %d/%d", channelIndex, waveCount);
|
|
|
|
}
|
2022-09-10 23:57:35 -07:00
|
|
|
return ((waveForm[phaseIndex] >> channelIndex) & 1) ? TriggerValue::RISE : TriggerValue::FALL;
|
2021-11-21 01:56:07 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void reset() {
|
|
|
|
waveCount = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setSwitchTime(const int phaseIndex, const float value) {
|
|
|
|
switchTimes[phaseIndex] = value;
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2021-11-21 01:56:07 -08:00
|
|
|
void setChannelState(const int channelIndex, const int phaseIndex, pin_state_t state) {
|
|
|
|
if (channelIndex >= waveCount) {
|
|
|
|
// todo: would be nice to get this asserting working
|
|
|
|
//firmwareError(OBD_PCM_Processor_Fault, "channel index %d/%d", channelIndex, waveCount);
|
|
|
|
}
|
|
|
|
uint8_t & ref = waveForm[phaseIndex];
|
2022-09-10 23:57:35 -07:00
|
|
|
ref = (ref & ~(1U << channelIndex)) | ((state == TriggerValue::RISE ? 1 : 0) << channelIndex);
|
2021-11-21 01:56:07 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
float switchTimes[max_phase];
|
|
|
|
uint8_t waveForm[max_phase];
|
|
|
|
};
|
2019-12-03 22:11:10 -08:00
|
|
|
|