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
|
|
|
|
2018-12-24 19:40:48 -08:00
|
|
|
#include "global.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
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 :)
|
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 */
|
2015-07-10 06:01:56 -07:00
|
|
|
#define PWM_PHASE_MAX_WAVE_PER_PWM 3
|
|
|
|
|
|
|
|
/**
|
|
|
|
* int8_t is probably less efficient then int32_t but we need
|
|
|
|
* to reduce memory footprint
|
|
|
|
*
|
2017-01-02 11:03:17 -08:00
|
|
|
* todo: migrate to bit-array to save memory?
|
|
|
|
* this would cost some CPU cycles. see std::vector<bool>
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2019-02-02 22:27:47 -08:00
|
|
|
typedef trigger_value_e pin_state_t;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
/**
|
2018-12-24 19:40:48 -08:00
|
|
|
* This class represents one channel of a digital signal state sequence
|
|
|
|
* Each element represents either a HIGH or LOW state - while at the moment this
|
|
|
|
* is not implemented using a bit array, it could absolutely be a bit array
|
|
|
|
*
|
|
|
|
* This sequence does not know anything about signal lengths - only signal state at a given index
|
2019-12-03 22:11:10 -08:00
|
|
|
* This sequence can have consecutive zeros and ones since these sequences work as a group within MultiChannelStateSequence
|
2018-12-24 19:40:48 -08:00
|
|
|
*
|
2015-07-10 06:01:56 -07:00
|
|
|
* @brief PWM configuration for the specific output pin
|
|
|
|
*/
|
2019-12-03 22:11:10 -08:00
|
|
|
class SingleChannelStateSequence {
|
2015-07-10 06:01:56 -07:00
|
|
|
public:
|
2019-12-03 22:11:10 -08:00
|
|
|
SingleChannelStateSequence();
|
|
|
|
explicit SingleChannelStateSequence(pin_state_t *pinStates);
|
2015-07-10 06:01:56 -07:00
|
|
|
void init(pin_state_t *pinStates);
|
2018-12-24 19:40:48 -08:00
|
|
|
/**
|
|
|
|
* todo: confirm that we only deal with two states here, no magic '-1'?
|
|
|
|
* @return HIGH or LOW state at given index
|
|
|
|
*/
|
2019-06-08 06:51:36 -07:00
|
|
|
pin_state_t getState(int switchIndex) const;
|
2019-02-02 22:19:16 -08:00
|
|
|
void setState(int switchIndex, pin_state_t state);
|
2018-12-24 19:40:48 -08:00
|
|
|
|
2018-12-24 19:57:36 -08:00
|
|
|
// todo: make this private by using 'getState' and 'setState' methods
|
2015-07-10 06:01:56 -07:00
|
|
|
pin_state_t *pinStates;
|
|
|
|
};
|
|
|
|
|
2018-12-24 19:57:36 -08:00
|
|
|
/**
|
|
|
|
* This class represents multi-channel logical signals with shared time axis
|
|
|
|
*
|
|
|
|
*/
|
2019-12-03 22:11:10 -08:00
|
|
|
class MultiChannelStateSequence {
|
2015-07-10 06:01:56 -07:00
|
|
|
public:
|
2019-12-03 22:11:10 -08:00
|
|
|
MultiChannelStateSequence();
|
|
|
|
MultiChannelStateSequence(float *switchTimes, SingleChannelStateSequence *waves);
|
|
|
|
void init(float *switchTimes, SingleChannelStateSequence *waves);
|
2015-07-10 06:01:56 -07:00
|
|
|
void reset(void);
|
2019-03-02 19:54:28 -08:00
|
|
|
float getSwitchTime(const int phaseIndex) const;
|
|
|
|
void setSwitchTime(const int phaseIndex, const float value);
|
|
|
|
void checkSwitchTimes(const int size);
|
|
|
|
pin_state_t getChannelState(const int channelIndex, const int phaseIndex) const;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-03-02 19:54:28 -08:00
|
|
|
int findAngleMatch(const float angle, const int size) const;
|
|
|
|
int findInsertionAngle(const float angle, const int size) const;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
/**
|
2018-12-25 07:13:00 -08:00
|
|
|
* Number of signal channels
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
int waveCount;
|
2019-12-03 22:11:10 -08:00
|
|
|
SingleChannelStateSequence *channels = nullptr;
|
2015-07-10 06:01:56 -07:00
|
|
|
//private:
|
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
2019-10-07 22:26:35 -07:00
|
|
|
float *switchTimes = nullptr;
|
2015-07-10 06:01:56 -07:00
|
|
|
};
|
|
|
|
|
2019-12-03 22:11:10 -08:00
|
|
|
|