rusefi/firmware/development/logic_analyzer.cpp

281 lines
7.5 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file logic_analyzer.cpp
2019-05-02 14:52:48 -07:00
* @brief Initialization of Input Capture pins used for rusEfi console sniffer
2015-07-10 06:01:56 -07:00
*
* This file is responsible for sniffing of external digital signals and registering
2019-05-02 14:52:48 -07:00
* these digital events in WaveChart used by the Engine Sniffer tab of rusEfi Console.
2015-07-10 06:01:56 -07:00
*
2018-04-01 20:28:04 -07:00
* this is rusEfi build-in logic analyzer
2015-07-10 06:01:56 -07:00
*
* @date Jan 7, 2013
2020-01-07 21:02:40 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
2015-07-10 06:01:56 -07:00
*/
#include "pch.h"
#include "logic_analyzer.h"
#include "os_access.h"
2015-07-10 06:01:56 -07:00
#include "eficonsole.h"
#include "trigger_central.h"
2019-07-06 17:15:49 -07:00
#include "os_util.h"
2015-07-10 06:01:56 -07:00
#include "rpm_calculator.h"
2015-07-15 18:01:45 -07:00
#include "engine_sniffer.h"
2015-07-10 06:01:56 -07:00
#if EFI_LOGIC_ANALYZER
2015-07-10 06:01:56 -07:00
#define CHART_RESET_DELAY 1
#define MAX_ICU_COUNT 5
2019-04-12 19:10:57 -07:00
#if EFI_ENGINE_SNIFFER
2015-07-10 06:01:56 -07:00
extern WaveChart waveChart;
#endif /* EFI_ENGINE_SNIFFER */
2015-07-10 06:01:56 -07:00
/**
* Difference between current 1st trigger event and previous 1st trigger event.
*/
static volatile uint32_t engineCycleDurationUs;
static volatile efitimeus_t previousEngineCycleTimeUs = 0;
2015-07-10 06:01:56 -07:00
static int waveReaderCount = 0;
static WaveReader readers[MAX_ICU_COUNT];
static void ensureInitialized(WaveReader *reader) {
/*may be*/UNUSED(reader);
efiAssertVoid(CUSTOM_ERR_6654, reader->hw != NULL && reader->hw->started, "wave analyzer NOT INITIALIZED");
2015-07-10 06:01:56 -07:00
}
static void waAnaWidthCallback(WaveReader *reader) {
efitick_t nowUs = getTimeNowUs();
2017-01-12 06:02:19 -08:00
reader->riseEventCounter++;
2015-07-10 06:01:56 -07:00
reader->lastActivityTimeUs = nowUs;
2018-09-10 19:24:50 -07:00
assertIsrContext(CUSTOM_ERR_6670);
addEngineSnifferEvent(reader->name, PROTOCOL_ES_UP);
2015-07-10 06:01:56 -07:00
uint32_t width = nowUs - reader->periodEventTimeUs;
reader->last_wave_low_widthUs = width;
reader->signalPeriodUs = nowUs - reader->widthEventTimeUs;
reader->widthEventTimeUs = nowUs;
}
void WaveReader::onFallEvent() {
efitick_t nowUs = getTimeNowUs();
2017-01-12 06:02:19 -08:00
fallEventCounter++;
2015-07-10 06:01:56 -07:00
lastActivityTimeUs = nowUs;
2018-09-10 19:24:50 -07:00
assertIsrContext(CUSTOM_ERR_6670);
addEngineSnifferEvent(name, PROTOCOL_ES_DOWN);
2015-07-10 06:01:56 -07:00
efitick_t width = nowUs - widthEventTimeUs;
last_wave_high_widthUs = width;
#if EFI_SHAFT_POSITION_INPUT
int revolutionCounter = getRevolutionCounter();
#else
int revolutionCounter = 0;
#endif
2015-07-10 06:01:56 -07:00
totalOnTimeAccumulatorUs += width;
if (currentRevolutionCounter != revolutionCounter) {
/**
* We are here in case of a new engine cycle
*/
currentRevolutionCounter = revolutionCounter;
prevTotalOnTimeUs = totalOnTimeAccumulatorUs;
totalOnTimeAccumulatorUs = 0;
waveOffsetUs = nowUs - previousEngineCycleTimeUs;
}
periodEventTimeUs = nowUs;
}
static void waIcuPeriodCallback(WaveReader *reader) {
reader->onFallEvent();
}
static void initWave(const char *name, int index) {
brain_pin_e brainPin = engineConfiguration->logicAnalyzerPins[index];
2015-07-10 06:01:56 -07:00
waveReaderCount++;
2018-07-25 20:03:04 -07:00
efiAssertVoid(CUSTOM_ERR_6655, index < MAX_ICU_COUNT, "too many ICUs");
2015-07-10 06:01:56 -07:00
WaveReader *reader = &readers[index];
if (!isBrainPinValid(brainPin)) {
/**
* in case we are running, and we select none for a channel that was running,
* this way we ensure that we do not get false report from that channel
**/
reader->hw = nullptr;
return;
}
2015-07-10 06:01:56 -07:00
reader->name = name;
reader->hw = startDigitalCapture("wave input", brainPin);
if (reader->hw != NULL) {
reader->hw->setWidthCallback((VoidInt)(void*) waAnaWidthCallback, (void*) reader);
2015-07-10 06:01:56 -07:00
reader->hw->setPeriodCallback((VoidInt)(void*) waIcuPeriodCallback, (void*) reader);
}
2015-07-10 06:01:56 -07:00
efiPrintf("wave%d input on %s", index, hwPortname(brainPin));
2015-07-10 06:01:56 -07:00
}
WaveReader::WaveReader() {
2019-10-07 22:26:35 -07:00
hw = nullptr;
2015-07-10 06:01:56 -07:00
}
void waTriggerEventListener(trigger_event_e ckpSignalType, uint32_t index, efitick_t edgeTimestamp) {
2015-07-10 06:01:56 -07:00
(void)ckpSignalType;
if (index != 0) {
return;
}
efitimeus_t nowUs = NT2US(edgeTimestamp);
2015-07-10 06:01:56 -07:00
engineCycleDurationUs = nowUs - previousEngineCycleTimeUs;
previousEngineCycleTimeUs = nowUs;
}
static float getSignalOnTime(int index) {
WaveReader *reader = &readers[index];
ensureInitialized(reader);
if (getTimeNowUs() - reader->lastActivityTimeUs > 4 * US_PER_SECOND) {
return 0.0f; // dwell time has expired
}
return reader->last_wave_high_widthUs / 1000.0f;
}
static efitime_t getWaveOffset(int index) {
WaveReader *reader = &readers[index];
ensureInitialized(reader);
return reader->waveOffsetUs;
}
static float getSignalPeriodMs(int index) {
WaveReader *reader = &readers[index];
ensureInitialized(reader);
return reader->signalPeriodUs / 1000.0f;
}
static void reportWave(Logging *logging, int index) {
2020-01-20 22:47:58 -08:00
if (readers[index].hw == nullptr) {
return;
}
2015-07-10 06:01:56 -07:00
if (readers[index].hw->started) {
// int counter = getEventCounter(index);
// debugInt2(logging, "ev", index, counter);
float dwellMs = getSignalOnTime(index);
float periodMs = getSignalPeriodMs(index);
logging->appendPrintf("duty%d%s", index, LOG_DELIMITER);
logging->appendFloat(100.0f * dwellMs / periodMs, 2);
logging->appendPrintf("%s", LOG_DELIMITER);
2015-07-10 06:01:56 -07:00
/**
* that's the ON time of the LAST signal
*/
logging->appendPrintf("dwell%d%s", index, LOG_DELIMITER);
logging->appendFloat(dwellMs, 2);
logging->appendPrintf("%s", LOG_DELIMITER);
2015-07-10 06:01:56 -07:00
/**
* that's the total ON time during the previous engine cycle
*/
logging->appendPrintf("total_dwell%d%s", index, LOG_DELIMITER);
logging->appendFloat(readers[index].prevTotalOnTimeUs / 1000.0f, 2);
logging->appendPrintf("%s", LOG_DELIMITER);
2015-07-10 06:01:56 -07:00
logging->appendPrintf("period%d%s", index, LOG_DELIMITER);
logging->appendFloat(periodMs, 2);
logging->appendPrintf("%s", LOG_DELIMITER);
2015-07-10 06:01:56 -07:00
uint32_t offsetUs = getWaveOffset(index);
2022-01-20 19:38:08 -08:00
int rpm = Sensor::getOrZero(SensorType::Rpm);
2017-12-02 15:57:52 -08:00
if (rpm != 0) {
float oneDegreeUs = getOneDegreeTimeUs(rpm);
logging->appendPrintf("advance%d%s", index, LOG_DELIMITER);
2017-12-02 15:57:52 -08:00
float angle = (offsetUs / oneDegreeUs) - tdcPosition();
2018-07-23 18:38:05 -07:00
fixAngle(angle, "waveAn", CUSTOM_ERR_6564);
logging->appendFloat(angle, 3);
logging->appendPrintf("%s", LOG_DELIMITER);
2017-12-02 15:57:52 -08:00
}
2015-07-10 06:01:56 -07:00
}
}
void printWave(Logging *logging) {
reportWave(logging, 0);
reportWave(logging, 1);
}
2017-01-12 06:02:19 -08:00
void showWaveInfo(void) {
efiPrintf("logic input #1: %d/%d", readers[0].fallEventCounter, readers[0].riseEventCounter);
2017-01-12 06:02:19 -08:00
}
void initWaveAnalyzer() {
2016-06-01 17:01:36 -07:00
if (hasFirmwareError()) {
return;
}
addConsoleAction("waveinfo", showWaveInfo);
}
void startLogicAnalyzerPins() {
initWave(PROTOCOL_WA_CHANNEL_1, 0);
initWave(PROTOCOL_WA_CHANNEL_2, 1);
initWave(PROTOCOL_WA_CHANNEL_3, 2);
initWave(PROTOCOL_WA_CHANNEL_4, 3);
2015-07-10 06:01:56 -07:00
}
void stopLogicAnalyzerPins() {
for (int index = 0; index < LOGIC_ANALYZER_CHANNEL_COUNT; index++) {
brain_pin_e brainPin = activeConfiguration.logicAnalyzerPins[index];
if (isBrainPinValid(brainPin)) {
stopDigitalCapture("wave input", brainPin);
}
}
}
static void getChannelFreqAndDuty(int index, scaled_channel<float> *duty, scaled_channel<uint32_t> *freq) {
float high,period;
if ((duty == nullptr) || (freq == nullptr)) {
return;
}
if (readers[index].hw == nullptr) {
*duty = 0.0;
*freq = 0;
} else {
high = getSignalOnTime(index);
period = getSignalPeriodMs(index);
if ((period != 0) && (readers[index].hw->started)) {
*duty = (high * 1000.0f) /(period * 10.0f);
*freq = (int)(1 / (period / 1000.0f));
} else {
*duty = 0.0;
*freq = 0;
}
}
}
void reportLogicAnalyzerToTS() {
#if EFI_TUNER_STUDIO
scaled_channel<uint32_t> tmp;
getChannelFreqAndDuty(0,&engine->outputChannels.debugFloatField1, &engine->outputChannels.debugIntField1);
getChannelFreqAndDuty(1,&engine->outputChannels.debugFloatField2, &engine->outputChannels.debugIntField2);
getChannelFreqAndDuty(2,&engine->outputChannels.debugFloatField3, &engine->outputChannels.debugIntField3);
getChannelFreqAndDuty(3,&engine->outputChannels.debugFloatField4, &tmp);
engine->outputChannels.debugIntField4 = (uint16_t)tmp;
#endif
}
#endif /* EFI_LOGIC_ANALYZER */