rusefi/firmware/development/wave_analyzer.cpp

259 lines
6.8 KiB
C++
Raw Normal View History

2014-08-29 07:52:33 -07:00
/**
* @file wave_analyzer.cpp
* @brief Initialization of Input Capture pins used for dev console sniffer
*
* This file is responsible for sniffing of external digital signals and registering
* these digital events in WaveChart used by the Sniffer tab of Dev Console.
*
*
* @date Jan 7, 2013
2015-01-12 15:04:10 -08:00
* @author Andrey Belomutskiy, (c) 2012-2015
2014-08-29 07:52:33 -07:00
*/
#include "main.h"
#include "wave_analyzer.h"
#include "eficonsole.h"
#include "data_buffer.h"
#include "pin_repository.h"
#include "engine_state.h"
#include "signal_executor.h"
#include "engine_configuration.h"
#include "trigger_central.h"
#include "rfiutil.h"
#include "engine_math.h"
#include "engine.h"
2014-09-01 06:02:46 -07:00
#include "rpm_calculator.h"
2014-11-23 19:03:07 -08:00
#include "wave_chart.h"
2014-08-29 07:52:33 -07:00
2014-12-23 20:03:31 -08:00
#if EFI_WAVE_ANALYZER || defined(__DOXYGEN__)
2014-11-07 19:04:45 -08:00
EXTERN_ENGINE;
2014-08-29 07:52:33 -07:00
#define CHART_RESET_DELAY 1
2014-11-21 22:03:21 -08:00
#define MAX_ICU_COUNT 5
2014-08-29 07:52:33 -07:00
2014-11-21 22:03:21 -08:00
extern WaveChart waveChart;
2014-11-24 09:03:09 -08:00
extern bool hasFirmwareErrorFlag;
2014-08-29 07:52:33 -07:00
2014-09-01 17:02:55 -07:00
/**
* Difference between current 1st trigger event and previous 1st trigger event.
*/
static volatile uint32_t engineCycleDurationUs;
static volatile uint64_t previousEngineCycleTimeUs = 0;
2014-08-29 07:52:33 -07:00
static int waveReaderCount = 0;
static WaveReader readers[MAX_ICU_COUNT];
2014-11-21 22:03:21 -08:00
static THD_WORKING_AREA(waThreadStack, UTILITY_THREAD_STACK_SIZE);
2015-01-14 15:04:00 -08:00
static Logging * logger;
2014-08-29 07:52:33 -07:00
static void ensureInitialized(WaveReader *reader) {
efiAssertVoid(reader->hw.started, "wave analyzer NOT INITIALIZED");
}
#if EFI_WAVE_ANALYZER || defined(__DOXYGEN__)
static void waAnaWidthCallback(WaveReader *reader) {
uint64_t nowUs = getTimeNowUs();
reader->eventCounter++;
reader->lastActivityTimeUs = nowUs;
2014-11-18 06:03:12 -08:00
addWaveChartEvent(reader->name, WC_UP);
2014-08-29 07:52:33 -07:00
uint32_t width = nowUs - reader->periodEventTimeUs;
reader->last_wave_low_widthUs = width;
reader->signalPeriodUs = nowUs - reader->widthEventTimeUs;
reader->widthEventTimeUs = nowUs;
}
2014-09-01 06:02:46 -07:00
void WaveReader::onFallEvent() {
2014-08-29 07:52:33 -07:00
uint64_t nowUs = getTimeNowUs();
2014-09-01 06:02:46 -07:00
eventCounter++;
lastActivityTimeUs = nowUs;
2014-11-18 06:03:12 -08:00
addWaveChartEvent(name, WC_DOWN);
2014-08-29 07:52:33 -07:00
2014-09-01 06:02:46 -07:00
uint64_t width = nowUs - widthEventTimeUs;
last_wave_high_widthUs = width;
2014-08-29 07:52:33 -07:00
2014-09-01 06:02:46 -07:00
int revolutionCounter = getRevolutionCounter();
totalOnTimeAccumulatorUs += width;
if (currentRevolutionCounter != revolutionCounter) {
/**
* We are here in case of a new engine cycle
*/
currentRevolutionCounter = revolutionCounter;
prevTotalOnTimeUs = totalOnTimeAccumulatorUs;
totalOnTimeAccumulatorUs = 0;
2014-08-29 07:52:33 -07:00
2014-09-01 17:02:55 -07:00
waveOffsetUs = nowUs - previousEngineCycleTimeUs;
2014-08-29 07:52:33 -07:00
}
2014-09-01 17:02:55 -07:00
periodEventTimeUs = nowUs;
2014-09-01 06:02:46 -07:00
2014-09-01 17:02:55 -07:00
// uint32_t period = engineCycleDurationUs; // local copy of volatile variable
2014-09-01 06:02:46 -07:00
}
static void waIcuPeriodCallback(WaveReader *reader) {
reader->onFallEvent();
2014-08-29 07:52:33 -07:00
}
static void setWaveModeSilent(int index, int mode) {
WaveReader *reader = &readers[index];
2014-12-27 20:03:27 -08:00
startInputDriver(&reader->hw, mode);
2014-08-29 07:52:33 -07:00
}
2014-09-11 18:02:50 -07:00
//static int getEventCounter(int index) {
// WaveReader *reader = &readers[index];
// ensureInitialized(reader);
// return reader->eventCounter;
//}
2014-08-29 07:52:33 -07:00
2014-08-31 15:02:50 -07:00
static void initWave(const char *name, int index) {
brain_pin_e brainPin = boardConfiguration->logicAnalyzerPins[index];
bool mode = boardConfiguration->logicAnalyzerMode[index];
2014-08-29 07:52:33 -07:00
waveReaderCount++;
efiAssertVoid(index < MAX_ICU_COUNT, "too many ICUs");
WaveReader *reader = &readers[index];
WaveReaderHw *hw = &reader->hw;
reader->name = name;
2015-01-14 06:05:59 -08:00
hw->widthListeners.registerCallback((IntListener) waAnaWidthCallback, (void*) reader);
2014-08-29 07:52:33 -07:00
2015-01-14 06:05:59 -08:00
hw->periodListeners.registerCallback((IntListener) waIcuPeriodCallback, (void*) reader);
2014-08-29 07:52:33 -07:00
2014-12-27 19:05:10 -08:00
initWaveAnalyzerDriver(hw, brainPin);
2014-08-29 07:52:33 -07:00
print("wave%d input on %s%d\r\n", index, portname(reader->hw.port), reader->hw.pin);
2014-12-27 20:03:27 -08:00
startInputDriver(hw, mode);
2014-08-29 07:52:33 -07:00
}
#endif
2014-11-24 18:03:34 -08:00
static void waTriggerEventListener(trigger_event_e ckpSignalType, uint32_t index DECLARE_ENGINE_PARAMETER_S) {
2014-09-24 12:03:03 -07:00
(void)ckpSignalType;
2014-08-31 17:02:54 -07:00
if (index != 0) {
2014-08-29 07:52:33 -07:00
return;
2014-08-31 17:02:54 -07:00
}
2014-08-29 07:52:33 -07:00
uint64_t nowUs = getTimeNowUs();
2014-09-01 17:02:55 -07:00
engineCycleDurationUs = nowUs - previousEngineCycleTimeUs;
previousEngineCycleTimeUs = nowUs;
2014-08-29 07:52:33 -07:00
}
static msg_t waThread(void *arg) {
2014-09-24 12:03:03 -07:00
(void)arg;
2014-08-29 07:52:33 -07:00
chRegSetThreadName("Wave Analyzer");
#if EFI_WAVE_CHART
while (TRUE) {
chThdSleepSeconds(CHART_RESET_DELAY);
2014-09-11 08:03:38 -07:00
waveChart.publishChartIfFull();
2014-08-29 07:52:33 -07:00
}
#endif /* EFI_WAVE_CHART */
#if defined __GNUC__
return -1;
#endif
}
2014-09-01 06:02:46 -07:00
static uint32_t getWaveLowWidth(int index) {
2014-08-29 07:52:33 -07:00
WaveReader *reader = &readers[index];
ensureInitialized(reader);
return reader->last_wave_low_widthUs;
}
2014-09-01 06:02:46 -07:00
static float getSignalOnTime(int index) {
2014-08-29 07:52:33 -07:00
WaveReader *reader = &readers[index];
ensureInitialized(reader);
if (getTimeNowUs() - reader->lastActivityTimeUs > 4 * US_PER_SECOND) {
return 0.0f; // dwell time has expired
2014-08-31 17:02:54 -07:00
}
2014-08-29 07:52:33 -07:00
return reader->last_wave_high_widthUs / 1000.0f;
}
2014-09-01 06:02:46 -07:00
static uint64_t getWaveOffset(int index) {
2014-08-29 07:52:33 -07:00
WaveReader *reader = &readers[index];
ensureInitialized(reader);
return reader->waveOffsetUs;
}
2014-09-01 06:02:46 -07:00
static float getSignalPeriodMs(int index) {
2014-08-29 07:52:33 -07:00
WaveReader *reader = &readers[index];
ensureInitialized(reader);
return reader->signalPeriodUs / 1000.0f;
}
2014-09-11 18:02:50 -07:00
//static uint64_t getWidthEventTime(int index) {
// WaveReader *reader = &readers[index];
// ensureInitialized(reader);
// return reader->widthEventTimeUs;
//}
2014-08-29 07:52:33 -07:00
static void reportWave(Logging *logging, int index) {
2014-09-11 18:02:50 -07:00
if (readers[index].hw.started) {
2014-08-29 07:52:33 -07:00
// int counter = getEventCounter(index);
// debugInt2(logging, "ev", index, counter);
2014-09-11 18:02:50 -07:00
float dwellMs = getSignalOnTime(index);
float periodMs = getSignalPeriodMs(index);
2014-08-29 07:52:33 -07:00
2014-09-11 18:02:50 -07:00
appendPrintf(logging, "duty%d%s", index, DELIMETER);
appendFloat(logging, 100.0f * dwellMs / periodMs, 2);
appendPrintf(logging, "%s", DELIMETER);
2014-08-29 07:52:33 -07:00
2014-09-11 18:02:50 -07:00
/**
* that's the ON time of the LAST signal
*/
appendPrintf(logging, "dwell%d%s", index, DELIMETER);
appendFloat(logging, dwellMs, 2);
appendPrintf(logging, "%s", DELIMETER);
2014-09-01 06:02:46 -07:00
2014-09-11 18:02:50 -07:00
/**
* that's the total ON time during the previous engine cycle
*/
appendPrintf(logging, "total_dwell%d%s", index, DELIMETER);
appendFloat(logging, readers[index].prevTotalOnTimeUs / 1000.0f, 2);
appendPrintf(logging, "%s", DELIMETER);
2014-09-01 06:02:46 -07:00
2014-09-11 18:02:50 -07:00
appendPrintf(logging, "period%d%s", index, DELIMETER);
appendFloat(logging, periodMs, 2);
appendPrintf(logging, "%s", DELIMETER);
2014-08-29 07:52:33 -07:00
2014-09-11 18:02:50 -07:00
uint32_t offsetUs = getWaveOffset(index);
float oneDegreeUs = getOneDegreeTimeUs(getRpm());
2014-08-29 07:52:33 -07:00
2014-09-11 18:02:50 -07:00
appendPrintf(logging, "advance%d%s", index, DELIMETER);
2014-11-25 07:03:25 -08:00
float angle = (offsetUs / oneDegreeUs) - engineConfiguration->globalTriggerAngleOffset;
2014-11-25 08:04:15 -08:00
fixAngle(angle);
2014-11-25 07:03:25 -08:00
appendFloat(logging, angle, 3);
2014-09-11 18:02:50 -07:00
appendPrintf(logging, "%s", DELIMETER);
}
2014-08-29 07:52:33 -07:00
}
void printWave(Logging *logging) {
reportWave(logging, 0);
reportWave(logging, 1);
}
2015-01-14 15:04:00 -08:00
void initWaveAnalyzer(Logging *sharedLogger) {
logger = sharedLogger;
2014-08-29 07:52:33 -07:00
#if EFI_WAVE_ANALYZER || defined(__DOXYGEN__)
2014-08-31 15:02:50 -07:00
initWave(WA_CHANNEL_1, 0);
initWave(WA_CHANNEL_2, 1);
2014-08-29 07:52:33 -07:00
2014-11-26 07:03:13 -08:00
// addTriggerEventListener(waTriggerEventListener, "wave analyzer", NULL);
2014-08-29 07:52:33 -07:00
2014-08-31 15:02:50 -07:00
addConsoleActionII("set_logic_input_mode", setWaveModeSilent);
2014-08-29 07:52:33 -07:00
2014-08-31 17:02:54 -07:00
chThdCreateStatic(waThreadStack, sizeof(waThreadStack), NORMALPRIO, waThread, (void*) NULL);
2014-08-29 07:52:33 -07:00
#else
print("wave disabled\r\n");
#endif
}
2014-12-23 20:03:31 -08:00
#endif