2019-07-07 12:22:46 -07:00
|
|
|
/*
|
|
|
|
* @file tooth_logger.cpp
|
|
|
|
*
|
|
|
|
* @date Jul 7, 2019
|
|
|
|
* @author Matthew Kennedy
|
|
|
|
*/
|
|
|
|
|
2019-07-06 17:48:58 -07:00
|
|
|
#include "tooth_logger.h"
|
|
|
|
|
2019-07-07 12:22:46 -07:00
|
|
|
#include "global.h"
|
2020-10-03 21:15:40 -07:00
|
|
|
#include "perf_trace.h"
|
2019-07-07 12:22:46 -07:00
|
|
|
|
|
|
|
#if EFI_TOOTH_LOGGER
|
|
|
|
|
2020-04-14 12:17:15 -07:00
|
|
|
EXTERN_ENGINE;
|
|
|
|
|
2019-07-06 17:48:58 -07:00
|
|
|
#include <cstddef>
|
|
|
|
#include "efitime.h"
|
|
|
|
#include "efilib.h"
|
2020-05-25 10:02:05 -07:00
|
|
|
#include "tunerstudio_outputs.h"
|
2019-07-06 17:48:58 -07:00
|
|
|
|
2020-04-13 17:47:54 -07:00
|
|
|
typedef struct __attribute__ ((packed)) {
|
|
|
|
uint16_t timestamp;
|
|
|
|
} tooth_logger_s;
|
|
|
|
|
|
|
|
typedef struct __attribute__ ((packed)) {
|
|
|
|
// the whole order of all packet bytes is reversed, not just the 'endian-swap' integers
|
|
|
|
uint32_t timestamp;
|
|
|
|
// unfortunately all these fields are required by TS...
|
|
|
|
bool priLevel : 1;
|
|
|
|
bool secLevel : 1;
|
2020-07-20 09:45:26 -07:00
|
|
|
bool isTDC : 1;
|
2020-04-13 17:47:54 -07:00
|
|
|
bool sync : 1;
|
2020-05-25 21:07:18 -07:00
|
|
|
bool coil : 1;
|
|
|
|
bool injector : 1;
|
2020-04-13 17:47:54 -07:00
|
|
|
} composite_logger_s;
|
|
|
|
|
2020-05-25 19:38:57 -07:00
|
|
|
/**
|
|
|
|
* Engine idles around 20Hz and revs up to 140Hz, at 60/2 and 8 cylinders we have about 20Khz events
|
|
|
|
* If we can read buffer at 50Hz we want buffer to be about 400 elements.
|
|
|
|
*/
|
2020-04-13 17:47:54 -07:00
|
|
|
static composite_logger_s buffer[COMPOSITE_PACKET_COUNT] CCM_OPTIONAL;
|
2019-07-06 17:48:58 -07:00
|
|
|
static size_t NextIdx = 0;
|
|
|
|
static volatile bool ToothLoggerEnabled = false;
|
|
|
|
|
|
|
|
static uint32_t lastEdgeTimestamp = 0;
|
|
|
|
|
2020-04-13 17:47:54 -07:00
|
|
|
static bool trigger1 = false;
|
|
|
|
static bool trigger2 = false;
|
2020-05-25 21:07:18 -07:00
|
|
|
// any coil, all coils thrown together
|
|
|
|
static bool coil = false;
|
|
|
|
// same about injectors
|
|
|
|
static bool injector = false;
|
2020-04-13 17:47:54 -07:00
|
|
|
|
2020-05-26 19:30:53 -07:00
|
|
|
int getCompositeRecordCount() {
|
|
|
|
return NextIdx;
|
|
|
|
}
|
|
|
|
|
2020-07-19 21:36:10 -07:00
|
|
|
|
|
|
|
#if EFI_UNIT_TEST
|
|
|
|
#include "logicdata.h"
|
|
|
|
int copyCompositeEvents(CompositeEvent *events) {
|
|
|
|
for (int i = 0;i < NextIdx;i++) {
|
|
|
|
CompositeEvent *event = &events[i];
|
2020-07-20 06:04:29 -07:00
|
|
|
event->timestamp = SWAP_UINT32(buffer[i].timestamp);
|
2020-07-19 21:36:10 -07:00
|
|
|
event->primaryTrigger = buffer[i].priLevel;
|
|
|
|
event->secondaryTrigger = buffer[i].secLevel;
|
2020-07-20 09:45:26 -07:00
|
|
|
event->isTDC = buffer[i].isTDC;
|
2020-07-19 21:36:10 -07:00
|
|
|
event->sync = buffer[i].sync;
|
|
|
|
event->coil = buffer[i].coil;
|
|
|
|
event->injector = buffer[i].injector;
|
|
|
|
}
|
|
|
|
return NextIdx;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // EFI_UNIT_TEST
|
|
|
|
|
2020-04-19 17:46:29 -07:00
|
|
|
static void SetNextCompositeEntry(efitick_t timestamp, bool trigger1, bool trigger2,
|
|
|
|
bool isTDC DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
uint32_t nowUs = NT2US(timestamp);
|
2019-07-06 17:48:58 -07:00
|
|
|
// TS uses big endian, grumble
|
2020-04-13 17:47:54 -07:00
|
|
|
buffer[NextIdx].timestamp = SWAP_UINT32(nowUs);
|
|
|
|
buffer[NextIdx].priLevel = trigger1;
|
|
|
|
buffer[NextIdx].secLevel = trigger2;
|
2020-07-20 09:45:26 -07:00
|
|
|
buffer[NextIdx].isTDC = isTDC;
|
2020-04-19 17:46:29 -07:00
|
|
|
buffer[NextIdx].sync = engine->triggerCentral.triggerState.shaft_is_synchronized;
|
2020-05-25 21:07:18 -07:00
|
|
|
buffer[NextIdx].coil = coil;
|
|
|
|
buffer[NextIdx].injector = injector;
|
2020-04-13 17:47:54 -07:00
|
|
|
|
2019-07-06 17:48:58 -07:00
|
|
|
NextIdx++;
|
|
|
|
|
2020-04-13 17:47:54 -07:00
|
|
|
static_assert(sizeof(composite_logger_s) == COMPOSITE_PACKET_SIZE, "composite packet size");
|
|
|
|
|
2019-07-06 17:48:58 -07:00
|
|
|
// If we hit the end, loop
|
|
|
|
if (NextIdx >= sizeof(buffer) / sizeof(buffer[0])) {
|
|
|
|
NextIdx = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-14 12:17:15 -07:00
|
|
|
void LogTriggerTooth(trigger_event_e tooth, efitick_t timestamp DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2020-10-03 21:15:40 -07:00
|
|
|
ScopePerf perf(PE::LogTriggerTooth);
|
2019-07-06 17:48:58 -07:00
|
|
|
// bail if we aren't enabled
|
2019-09-03 17:35:52 -07:00
|
|
|
if (!ToothLoggerEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
2019-07-06 17:48:58 -07:00
|
|
|
|
2020-04-13 17:47:54 -07:00
|
|
|
/*
|
|
|
|
// We currently only support the primary trigger falling edge
|
|
|
|
// (this is the edge that VR sensors are accurate on)
|
|
|
|
// Since VR sensors are the most useful case here, this is okay for now.
|
|
|
|
if (tooth != SHAFT_PRIMARY_FALLING) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t nowUs = NT2US(timestamp);
|
|
|
|
// 10us per LSB - this gives plenty of accuracy, yet fits 655.35 ms in to a uint16
|
|
|
|
uint16_t delta = static_cast<uint16_t>((nowUs - lastEdgeTimestamp) / 10);
|
|
|
|
lastEdgeTimestamp = nowUs;
|
|
|
|
|
|
|
|
SetNextEntry(delta);
|
|
|
|
*/
|
|
|
|
|
|
|
|
switch (tooth) {
|
|
|
|
case SHAFT_PRIMARY_FALLING:
|
|
|
|
trigger1 = false;
|
|
|
|
break;
|
|
|
|
case SHAFT_PRIMARY_RISING:
|
|
|
|
trigger1 = true;
|
|
|
|
break;
|
|
|
|
case SHAFT_SECONDARY_FALLING:
|
|
|
|
trigger2 = false;
|
|
|
|
break;
|
|
|
|
case SHAFT_SECONDARY_RISING:
|
|
|
|
trigger2 = true;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2019-09-03 17:35:52 -07:00
|
|
|
}
|
2019-07-06 17:48:58 -07:00
|
|
|
|
2020-04-19 17:46:29 -07:00
|
|
|
SetNextCompositeEntry(timestamp, trigger1, trigger2, false PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogTriggerTopDeadCenter(efitick_t timestamp DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
// bail if we aren't enabled
|
|
|
|
if (!ToothLoggerEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
SetNextCompositeEntry(timestamp, trigger1, trigger2, true PASS_ENGINE_PARAMETER_SUFFIX);
|
2020-07-20 09:45:26 -07:00
|
|
|
SetNextCompositeEntry(timestamp + 10, trigger1, trigger2, false PASS_ENGINE_PARAMETER_SUFFIX);
|
2019-07-06 17:48:58 -07:00
|
|
|
}
|
|
|
|
|
2020-05-25 21:07:18 -07:00
|
|
|
void LogTriggerCoilState(efitick_t timestamp, bool state DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
if (!ToothLoggerEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
coil = state;
|
|
|
|
SetNextCompositeEntry(timestamp, trigger1, trigger2, false PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogTriggerInjectorState(efitick_t timestamp, bool state DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
if (!ToothLoggerEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
injector = state;
|
|
|
|
SetNextCompositeEntry(timestamp, trigger1, trigger2, false PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
|
2019-07-06 17:48:58 -07:00
|
|
|
void EnableToothLogger() {
|
|
|
|
// Clear the buffer
|
|
|
|
memset(buffer, 0, sizeof(buffer));
|
|
|
|
|
|
|
|
// Reset the last edge to now - this prevents the first edge logged from being bogus
|
2020-01-19 19:23:41 -08:00
|
|
|
lastEdgeTimestamp = getTimeNowUs();
|
2019-07-06 17:48:58 -07:00
|
|
|
|
|
|
|
// Reset write index
|
|
|
|
NextIdx = 0;
|
|
|
|
|
|
|
|
// Enable logging of edges as they come
|
|
|
|
ToothLoggerEnabled = true;
|
|
|
|
|
2020-07-19 19:41:34 -07:00
|
|
|
#if EFI_TUNER_STUDIO
|
2019-07-06 17:48:58 -07:00
|
|
|
// Tell TS that we're ready for it to read out the log
|
|
|
|
// nb: this is a lie, as we may not have written anything
|
|
|
|
// yet. However, we can let it continuously read out the buffer
|
|
|
|
// as we update it, which looks pretty nice.
|
|
|
|
tsOutputChannels.toothLogReady = true;
|
2020-07-19 19:41:34 -07:00
|
|
|
#endif // EFI_TUNER_STUDIO
|
2019-07-06 17:48:58 -07:00
|
|
|
}
|
|
|
|
|
2020-05-26 19:30:53 -07:00
|
|
|
void EnableToothLoggerIfNotEnabled() {
|
|
|
|
if (!ToothLoggerEnabled) {
|
|
|
|
EnableToothLogger();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-06 17:48:58 -07:00
|
|
|
void DisableToothLogger() {
|
|
|
|
ToothLoggerEnabled = false;
|
2020-07-19 19:41:34 -07:00
|
|
|
#if EFI_TUNER_STUDIO
|
2019-07-06 17:48:58 -07:00
|
|
|
tsOutputChannels.toothLogReady = false;
|
2020-07-19 19:41:34 -07:00
|
|
|
#endif // EFI_TUNER_STUDIO
|
2019-07-06 17:48:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ToothLoggerBuffer GetToothLoggerBuffer() {
|
|
|
|
return { reinterpret_cast<uint8_t*>(buffer), sizeof(buffer) };
|
|
|
|
}
|
2019-07-07 12:22:46 -07:00
|
|
|
|
|
|
|
#endif /* EFI_TOOTH_LOGGER */
|