rusefi/firmware/console/binary/tooth_logger.cpp

253 lines
6.6 KiB
C++
Raw Normal View History

2019-07-07 12:22:46 -07:00
/*
* @file tooth_logger.cpp
*
* @date Jul 7, 2019
* @author Matthew Kennedy
*/
#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
#include <cstddef>
#include "efitime.h"
#include "efilib.h"
2020-05-25 10:02:05 -07:00
#include "tunerstudio_outputs.h"
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;
bool trigger : 1;
bool sync : 1;
2020-05-25 21:07:18 -07:00
bool coil : 1;
bool injector : 1;
} 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.
*/
static composite_logger_s buffer[COMPOSITE_PACKET_COUNT] CCM_OPTIONAL;
static composite_logger_s *ptr_buffer_first = &buffer[0];
static composite_logger_s *ptr_buffer_second = &buffer[(COMPOSITE_PACKET_COUNT/2)-1];
static size_t NextIdx = 0;
static volatile bool ToothLoggerEnabled = false;
static volatile bool firstBuffer = true;
static uint32_t lastEdgeTimestamp = 0;
2021-07-02 16:28:09 -07:00
static bool currentTrigger1 = false;
static bool currentTrigger2 = false;
static bool currentTdc = false;
2020-05-25 21:07:18 -07:00
// any coil, all coils thrown together
2021-07-02 16:28:09 -07:00
static bool currentCoilState = false;
2020-05-25 21:07:18 -07:00
// same about injectors
2021-07-02 16:28:09 -07:00
static bool currentInjectorState = false;
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) {
2020-12-08 10:59:40 -08:00
for (size_t i = 0; i < NextIdx; i++) {
2020-07-19 21:36:10 -07:00
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;
event->isTDC = buffer[i].trigger;
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
2021-07-02 16:28:09 -07:00
static void SetNextCompositeEntry(efitick_t timestamp DECLARE_ENGINE_PARAMETER_SUFFIX) {
uint32_t nowUs = NT2US(timestamp);
// TS uses big endian, grumble
buffer[NextIdx].timestamp = SWAP_UINT32(nowUs);
2021-07-02 16:28:09 -07:00
buffer[NextIdx].priLevel = currentTrigger1;
buffer[NextIdx].secLevel = currentTrigger2;
buffer[NextIdx].trigger = currentTdc;
2021-07-03 07:37:03 -07:00
buffer[NextIdx].sync = engine->triggerCentral.triggerState.getShaftSynchronized();
2021-07-02 16:28:09 -07:00
buffer[NextIdx].coil = currentCoilState;
buffer[NextIdx].injector = currentInjectorState;
NextIdx++;
static_assert(sizeof(composite_logger_s) == COMPOSITE_PACKET_SIZE, "composite packet size");
//If we hit the end, loop
if ((firstBuffer) && (NextIdx >= (COMPOSITE_PACKET_COUNT/2))) {
/* first half is full */
#if EFI_TUNER_STUDIO
tsOutputChannels.toothLogReady = true;
#endif
firstBuffer = false;
}
if ((!firstBuffer) && (NextIdx >= sizeof(buffer) / sizeof(buffer[0]))) {
#if EFI_TUNER_STUDIO
tsOutputChannels.toothLogReady = true;
#endif
NextIdx = 0;
firstBuffer = true;
}
/////tsOutputChannels.toothLogReady = true;
}
void LogTriggerTooth(trigger_event_e tooth, efitick_t timestamp DECLARE_ENGINE_PARAMETER_SUFFIX) {
// bail if we aren't enabled
2019-09-03 17:35:52 -07:00
if (!ToothLoggerEnabled) {
return;
}
// Don't log at significant engine speed
if (engine->rpmCalculator.getRpm() > 4000) {
return;
}
ScopePerf perf(PE::LogTriggerTooth);
/*
// 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:
2021-07-02 16:28:09 -07:00
currentTrigger1 = false;
break;
case SHAFT_PRIMARY_RISING:
2021-07-02 16:28:09 -07:00
currentTrigger1 = true;
break;
case SHAFT_SECONDARY_FALLING:
2021-07-02 16:28:09 -07:00
currentTrigger2 = false;
break;
case SHAFT_SECONDARY_RISING:
2021-07-02 16:28:09 -07:00
currentTrigger2 = true;
break;
2021-07-02 17:12:31 -07:00
// major hack to get most value of limited logic data write
#if EFI_UNIT_TEST
case SHAFT_3RD_FALLING:
currentCoilState = false;
break;
case SHAFT_3RD_RISING:
currentCoilState = true;
break;
#endif
default:
break;
2019-09-03 17:35:52 -07:00
}
2021-07-02 16:28:09 -07:00
SetNextCompositeEntry(timestamp PASS_ENGINE_PARAMETER_SUFFIX);
}
void LogTriggerTopDeadCenter(efitick_t timestamp DECLARE_ENGINE_PARAMETER_SUFFIX) {
// bail if we aren't enabled
if (!ToothLoggerEnabled) {
return;
}
2021-07-02 16:28:09 -07:00
currentTdc = true;
SetNextCompositeEntry(timestamp PASS_ENGINE_PARAMETER_SUFFIX);
currentTdc = false;
SetNextCompositeEntry(timestamp + 10 PASS_ENGINE_PARAMETER_SUFFIX);
}
2020-05-25 21:07:18 -07:00
void LogTriggerCoilState(efitick_t timestamp, bool state DECLARE_ENGINE_PARAMETER_SUFFIX) {
if (!ToothLoggerEnabled) {
return;
}
2021-07-02 16:28:09 -07:00
currentCoilState = state;
UNUSED(timestamp);
//SetNextCompositeEntry(timestamp, trigger1, trigger2, trigger PASS_ENGINE_PARAMETER_SUFFIX);
2020-05-25 21:07:18 -07:00
}
void LogTriggerInjectorState(efitick_t timestamp, bool state DECLARE_ENGINE_PARAMETER_SUFFIX) {
if (!ToothLoggerEnabled) {
return;
}
2021-07-02 16:28:09 -07:00
currentInjectorState = state;
UNUSED(timestamp);
//SetNextCompositeEntry(timestamp, trigger1, trigger2, trigger PASS_ENGINE_PARAMETER_SUFFIX);
2020-05-25 21:07:18 -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
lastEdgeTimestamp = getTimeNowUs();
// 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
// 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 = false;
2020-07-19 19:41:34 -07:00
#endif // EFI_TUNER_STUDIO
}
2020-05-26 19:30:53 -07:00
void EnableToothLoggerIfNotEnabled() {
if (!ToothLoggerEnabled) {
EnableToothLogger();
}
}
void DisableToothLogger() {
ToothLoggerEnabled = false;
2020-07-19 19:41:34 -07:00
#if EFI_TUNER_STUDIO
tsOutputChannels.toothLogReady = false;
2020-07-19 19:41:34 -07:00
#endif // EFI_TUNER_STUDIO
}
ToothLoggerBuffer GetToothLoggerBuffer() {
if (firstBuffer) {
#if EFI_TUNER_STUDIO
tsOutputChannels.toothLogReady = false;
#endif
return { reinterpret_cast<uint8_t*>(ptr_buffer_second), (sizeof(buffer)/2) };
} else {
#if EFI_TUNER_STUDIO
tsOutputChannels.toothLogReady = false;
#endif
return { reinterpret_cast<uint8_t*>(ptr_buffer_first), (sizeof(buffer)/2) };
}
}
2019-07-07 12:22:46 -07:00
2019-07-07 12:22:46 -07:00
#endif /* EFI_TOOTH_LOGGER */