From 1f16228bdb8b87770d930e04c7d4b1bbbef714b7 Mon Sep 17 00:00:00 2001 From: Matthew Kennedy Date: Wed, 29 Apr 2020 07:53:35 -0700 Subject: [PATCH] Perf Trace Improvements (#1383) * add tid field to thread * change perf trace format * trace consumers * Update java parsing * guard for test + kinetis * stubs * ram Co-authored-by: Matthew Kennedy --- firmware/bootloader/src/rusefi_stubs.cpp | 1 + firmware/config/boards/kinetis/chconf.h | 20 ---------- firmware/config/chconf_common.h | 20 ++++++++++ firmware/config/stm32f4ems/chconf.h | 20 ---------- firmware/config/stm32f7ems/chconf.h | 20 ---------- firmware/controllers/engine_controller.cpp | 2 +- .../controllers/engine_controller_misc.cpp | 7 ++++ .../controllers/trigger/trigger_central.cpp | 2 +- .../controllers/trigger/trigger_decoder.cpp | 2 +- firmware/development/perf_trace.cpp | 31 ++++++++++------ firmware/development/perf_trace.h | 19 ++++------ .../models/src/com/rusefi/tracing/Entry.java | 15 +++++--- .../src/com/rusefi/tracing/JsonOutput.java | 37 ++++++++++--------- 13 files changed, 86 insertions(+), 110 deletions(-) diff --git a/firmware/bootloader/src/rusefi_stubs.cpp b/firmware/bootloader/src/rusefi_stubs.cpp index fa3e79bfe1..04c07dce70 100644 --- a/firmware/bootloader/src/rusefi_stubs.cpp +++ b/firmware/bootloader/src/rusefi_stubs.cpp @@ -33,4 +33,5 @@ LoggingWithStorage::LoggingWithStorage(const char *name) : Logging(name, DEFAULT void irqEnterHook() {} void irqExitHook() {} void contextSwitchHook() {} +void threadInitHook(void*) {} //#endif /* ENABLE_PERF_TRACE */ \ No newline at end of file diff --git a/firmware/config/boards/kinetis/chconf.h b/firmware/config/boards/kinetis/chconf.h index 199a7e52ae..fd52d41f4c 100644 --- a/firmware/config/boards/kinetis/chconf.h +++ b/firmware/config/boards/kinetis/chconf.h @@ -641,26 +641,6 @@ /* Add threads initialization code here.*/ \ } -/** - * @brief Threads descriptor structure extension. - * @details User fields added to the end of the @p thread_t structure. - */ -#define CH_CFG_THREAD_EXTRA_FIELDS \ - void *activeStack; \ - int remainingStack; \ - /* Add threads custom fields here.*/ - -/** - * @brief Threads initialization hook. - * @details User initialization code added to the @p chThdInit() API. - * - * @note It is invoked from within @p chThdInit() and implicitly from all - * the threads creation APIs. - */ -#define CH_CFG_THREAD_INIT_HOOK(tp) { \ - /* Add threads initialization code here.*/ \ -} - /** * @brief Threads finalization hook. * @details User finalization code added to the @p chThdExit() API. diff --git a/firmware/config/chconf_common.h b/firmware/config/chconf_common.h index 83c028e1b8..76f5555829 100644 --- a/firmware/config/chconf_common.h +++ b/firmware/config/chconf_common.h @@ -21,6 +21,7 @@ extern "C" void irqEnterHook(void); void irqExitHook(void); void contextSwitchHook(void); + void threadInitHook(void* tp); #endif /* __ASSEMBLER__ */ #ifdef __cplusplus } @@ -85,4 +86,23 @@ extern "C" { irqExitHook(); \ } +/** + * @brief Threads descriptor structure extension. + * @details User fields added to the end of the @p thread_t structure. + */ +#define CH_CFG_THREAD_EXTRA_FIELDS \ + void *activeStack; \ + int remainingStack; \ + unsigned char threadId; \ + /* Add threads custom fields here.*/ + +/** + * @brief Threads initialization hook. + * @details User initialization code added to the @p chThdInit() API. + * + * @note It is invoked from within @p chThdInit() and implicitly from all + * the threads creation APIs. + */ +#define CH_CFG_THREAD_INIT_HOOK(tp) { threadInitHook(tp); } + #endif /* CONFIG_CHCONF_COMMON_H_ */ diff --git a/firmware/config/stm32f4ems/chconf.h b/firmware/config/stm32f4ems/chconf.h index 90ac197973..a05a9ff776 100644 --- a/firmware/config/stm32f4ems/chconf.h +++ b/firmware/config/stm32f4ems/chconf.h @@ -632,26 +632,6 @@ /* Add threads initialization code here.*/ \ } -/** - * @brief Threads descriptor structure extension. - * @details User fields added to the end of the @p thread_t structure. - */ -#define CH_CFG_THREAD_EXTRA_FIELDS \ - void *activeStack; \ - int remainingStack; \ - /* Add threads custom fields here.*/ - -/** - * @brief Threads initialization hook. - * @details User initialization code added to the @p chThdInit() API. - * - * @note It is invoked from within @p chThdInit() and implicitly from all - * the threads creation APIs. - */ -#define CH_CFG_THREAD_INIT_HOOK(tp) { \ - /* Add threads initialization code here.*/ \ -} - /** * @brief Threads finalization hook. * @details User finalization code added to the @p chThdExit() API. diff --git a/firmware/config/stm32f7ems/chconf.h b/firmware/config/stm32f7ems/chconf.h index f28eaa8e2f..e0e638900c 100644 --- a/firmware/config/stm32f7ems/chconf.h +++ b/firmware/config/stm32f7ems/chconf.h @@ -632,26 +632,6 @@ /* Add threads initialization code here.*/ \ } -/** - * @brief Threads descriptor structure extension. - * @details User fields added to the end of the @p thread_t structure. - */ -#define CH_CFG_THREAD_EXTRA_FIELDS \ - void *activeStack; \ - int remainingStack; \ - /* Add threads custom fields here.*/ - -/** - * @brief Threads initialization hook. - * @details User initialization code added to the @p chThdInit() API. - * - * @note It is invoked from within @p chThdInit() and implicitly from all - * the threads creation APIs. - */ -#define CH_CFG_THREAD_INIT_HOOK(tp) { \ - /* Add threads initialization code here.*/ \ -} - /** * @brief Threads finalization hook. * @details User finalization code added to the @p chThdExit() API. diff --git a/firmware/controllers/engine_controller.cpp b/firmware/controllers/engine_controller.cpp index df9c3eea5f..1704599af8 100644 --- a/firmware/controllers/engine_controller.cpp +++ b/firmware/controllers/engine_controller.cpp @@ -732,7 +732,7 @@ void initEngineContoller(Logging *sharedLogger DECLARE_ENGINE_PARAMETER_SUFFIX) // help to notice when RAM usage goes up - if a code change adds to RAM usage these variables would fail // linking process which is the way to raise the alarm #ifndef RAM_UNUSED_SIZE -#define RAM_UNUSED_SIZE 14500 +#define RAM_UNUSED_SIZE 14300 #endif #ifndef CCM_UNUSED_SIZE #define CCM_UNUSED_SIZE 2900 diff --git a/firmware/controllers/engine_controller_misc.cpp b/firmware/controllers/engine_controller_misc.cpp index a9b4fdcb8e..be51bbe8c7 100644 --- a/firmware/controllers/engine_controller_misc.cpp +++ b/firmware/controllers/engine_controller_misc.cpp @@ -15,6 +15,12 @@ EXTERN_ENGINE; extern LoggingWithStorage sharedLogger; #if ENABLE_PERF_TRACE +static uint8_t nextThreadId = 0; +void threadInitHook(void* vtp) { + // No lock required, this is already under lock + auto tp = reinterpret_cast(vtp); + tp->threadId = ++nextThreadId; +} void irqEnterHook() { perfEventBegin(PE::ISR); @@ -29,6 +35,7 @@ void contextSwitchHook() { } #else +void threadInitHook(void*) {} void irqEnterHook() {} void irqExitHook() {} void contextSwitchHook() {} diff --git a/firmware/controllers/trigger/trigger_central.cpp b/firmware/controllers/trigger/trigger_central.cpp index 902a87aa2a..8aed68df0c 100644 --- a/firmware/controllers/trigger/trigger_central.cpp +++ b/firmware/controllers/trigger/trigger_central.cpp @@ -227,7 +227,7 @@ uint32_t triggerDuration; uint32_t triggerMaxDuration = 0; void hwHandleShaftSignal(trigger_event_e signal, efitick_t timestamp) { - ScopePerf perf(PE::HandleShaftSignal, static_cast(signal)); + ScopePerf perf(PE::HandleShaftSignal); #if EFI_TOOTH_LOGGER // Log to the Tunerstudio tooth logger diff --git a/firmware/controllers/trigger/trigger_decoder.cpp b/firmware/controllers/trigger/trigger_decoder.cpp index f9acab3996..31cb817ca4 100644 --- a/firmware/controllers/trigger/trigger_decoder.cpp +++ b/firmware/controllers/trigger/trigger_decoder.cpp @@ -383,7 +383,7 @@ void TriggerState::onShaftSynchronization(const TriggerStateCallback triggerCycl void TriggerState::decodeTriggerEvent(TriggerWaveform *triggerShape, const TriggerStateCallback triggerCycleCallback, TriggerStateListener * triggerStateListener, trigger_event_e const signal, efitick_t nowNt DECLARE_CONFIG_PARAMETER_SUFFIX) { - ScopePerf perf(PE::DecodeTriggerEvent, static_cast(signal)); + ScopePerf perf(PE::DecodeTriggerEvent); if (nowNt - previousShaftEventTimeNt > NT_PER_SECOND) { /** diff --git a/firmware/development/perf_trace.cpp b/firmware/development/perf_trace.cpp index 66c570e590..3e13b54a59 100644 --- a/firmware/development/perf_trace.cpp +++ b/firmware/development/perf_trace.cpp @@ -27,7 +27,7 @@ struct TraceEntry { PE Event; EPhase Phase; - uint8_t Data; + int8_t IsrId; uint8_t ThreadId; uint32_t Timestamp; }; @@ -41,7 +41,7 @@ static size_t s_nextIdx = 0; static bool s_isTracing = false; -void perfEventImpl(PE event, EPhase phase, uint8_t data) +static void perfEventImpl(PE event, EPhase phase) { // Bail if we aren't allowed to trace if constexpr (!ENABLE_PERF_TRACE) { @@ -81,22 +81,31 @@ void perfEventImpl(PE event, EPhase phase, uint8_t data) entry.Event = event; entry.Phase = phase; - // Get the current active interrupt - this is the "thread ID" - entry.ThreadId = static_cast(SCB->ICSR & SCB_ICSR_VECTACTIVE_Msk); + // Get the current active interrupt - this is the "process ID" + auto isr = static_cast(SCB->ICSR & SCB_ICSR_VECTACTIVE_Msk); + entry.IsrId = isr - 16; + + // Get the current thread (if not interrupt) and use as the thread ID + if (isr == 0) { + entry.ThreadId = chThdGetSelfX()->threadId; + } else { + // Interrupts have no thread - all are T0 + entry.ThreadId = 0; + } + entry.Timestamp = timestamp; - entry.Data = data; } -void perfEventBegin(PE event, uint8_t data) { - perfEventImpl(event, EPhase::Start, data); +void perfEventBegin(PE event) { + perfEventImpl(event, EPhase::Start); } -void perfEventEnd(PE event, uint8_t data) { - perfEventImpl(event, EPhase::End, data); +void perfEventEnd(PE event) { + perfEventImpl(event, EPhase::End); } -void perfEventInstantGlobal(PE event, uint8_t data) { - perfEventImpl(event, EPhase::InstantGlobal, data); +void perfEventInstantGlobal(PE event) { + perfEventImpl(event, EPhase::InstantGlobal); } void perfTraceEnable() { diff --git a/firmware/development/perf_trace.h b/firmware/development/perf_trace.h index 431e7bc886..5fb68948c3 100644 --- a/firmware/development/perf_trace.h +++ b/firmware/development/perf_trace.h @@ -65,9 +65,9 @@ enum class PE : uint8_t { // please note that the tool requires a comma at the end of last value }; -void perfEventBegin(PE event, uint8_t data = 0); -void perfEventEnd(PE event, uint8_t data = 0); -void perfEventInstantGlobal(PE event, uint8_t data = 0); +void perfEventBegin(PE event); +void perfEventEnd(PE event); +void perfEventInstantGlobal(PE event); // Enable one buffer's worth of perf tracing, and retrieve the buffer size in bytes void perfTraceEnable(); @@ -85,28 +85,23 @@ const TraceBufferResult perfTraceGetBuffer(); class ScopePerf { public: - ScopePerf(PE event) : ScopePerf(event, 0) {} - - ScopePerf(PE event, uint8_t data) : m_event(event), m_data(data) - { - perfEventBegin(event, data); + ScopePerf(PE event) : m_event(event) { + perfEventBegin(event); } ~ScopePerf() { - perfEventEnd(m_event, m_data); + perfEventEnd(m_event); } private: const PE m_event; - const uint8_t m_data; }; #else /* if ENABLE_PERF_TRACE */ struct ScopePerf { - ScopePerf(PE event) { (void)event; } - ScopePerf(PE event, uint8_t data) { (void)event; (void)data; } + ScopePerf(PE) {} }; #endif /* ENABLE_PERF_TRACE */ diff --git a/java_console/models/src/com/rusefi/tracing/Entry.java b/java_console/models/src/com/rusefi/tracing/Entry.java index 535f71a211..73f87bd4ba 100644 --- a/java_console/models/src/com/rusefi/tracing/Entry.java +++ b/java_console/models/src/com/rusefi/tracing/Entry.java @@ -14,11 +14,15 @@ public class Entry { private static final double MAGIC_NT = 168.0; private final String name; private final Phase phase; + private final int isr; + private final int thread; private double timestampSeconds; - public Entry(String name, Phase phase, double timestampSeconds) { + public Entry(String name, Phase phase, double timestampSeconds, int isr, int thread) { this.name = name; this.phase = phase; + this.isr = isr; + this.thread = thread; this.timestampSeconds = timestampSeconds; } @@ -65,7 +69,7 @@ public class Entry { for (int i = 0; i < packet.length - 1; i += 8) { byte type = is.readByte(); byte phase = is.readByte(); - byte data = is.readByte(); + byte isr = is.readByte(); byte thread = is.readByte(); int timestampNt = readInt(is); @@ -91,7 +95,7 @@ public class Entry { name = TypeNames[type]; } - Entry e = new Entry(name, Phase.decode(phase), timestampSeconds); + Entry e = new Entry(name, Phase.decode(phase), timestampSeconds, isr, thread); result.add(e); } @@ -119,10 +123,9 @@ public class Entry { sb.append(","); AppendKeyValuePair(sb, "ph", phase.toString()); sb.append(","); - AppendKeyValuePair(sb, "tid", 0); + AppendKeyValuePair(sb, "tid", thread); sb.append(","); - - AppendKeyValuePair(sb, "pid", 0); + AppendKeyValuePair(sb, "pid", isr); sb.append(","); AppendKeyValuePair(sb, "ts", timestampSeconds); sb.append("}"); diff --git a/java_console/models/src/com/rusefi/tracing/JsonOutput.java b/java_console/models/src/com/rusefi/tracing/JsonOutput.java index 1858a9b9ad..45e4244962 100644 --- a/java_console/models/src/com/rusefi/tracing/JsonOutput.java +++ b/java_console/models/src/com/rusefi/tracing/JsonOutput.java @@ -15,25 +15,26 @@ public class JsonOutput { /** * those are special entries that change display settings * those set thread names and sort index based on those thread IDs - * (those thread IDs are interrupt numbers, and the name is the name of the interrupt handler) + * (those process IDs are interrupt numbers, and the name is the name of the interrupt handler) */ - private static final String FORMATTING_SETTINGS - = "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":18,\"pid\":0,\"args\":{\"name\":\"ADC\"}}," + - "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":29,\"pid\":0,\"args\":{\"name\":\"TIM3\"}}," + - "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":30,\"pid\":0,\"args\":{\"name\":\"TIM4\"}}," + - "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":44,\"pid\":0,\"args\":{\"name\":\"TIM8/13\"}}," + - "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":50,\"pid\":0,\"args\":{\"name\":\"TIM5\"}}," + - "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":58,\"pid\":0,\"args\":{\"name\":\"DMA2s2\"}}," + - "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":60,\"pid\":0,\"args\":{\"name\":\"DMA2s4\"}}," + - "{\"name\":\"thread_name\",\"ph\":\"M\",\"tid\":67,\"pid\":0,\"args\":{\"name\":\"USB\"}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":18,\"pid\":0,\"args\":{\"sort_index\":4}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":29,\"pid\":0,\"args\":{\"sort_index\":2}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":30,\"pid\":0,\"args\":{\"sort_index\":5}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":44,\"pid\":0,\"args\":{\"sort_index\":7}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":50,\"pid\":0,\"args\":{\"sort_index\":3}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":58,\"pid\":0,\"args\":{\"sort_index\":6}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":60,\"pid\":0,\"args\":{\"sort_index\":8}}," + - "{\"name\":\"thread_sort_index\",\"ph\":\"M\",\"tid\":67,\"pid\":0,\"args\":{\"sort_index\":9}}" + private static final String FORMATTING_SETTINGS = + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":-16,\"tid\":0,\"args\":{\"name\":\"Main\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":-1,\"tid\":0,\"args\":{\"name\":\"SysTick\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":18,\"tid\":0,\"args\":{\"name\":\"ADC\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":19,\"tid\":0,\"args\":{\"name\":\"CAN1 TX\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":20,\"tid\":0,\"args\":{\"name\":\"CAN1 RX 0\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":21,\"tid\":0,\"args\":{\"name\":\"CAN1 RX 1\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":22,\"tid\":0,\"args\":{\"name\":\"CAN1 SCE\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":29,\"tid\":0,\"args\":{\"name\":\"TIM3\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":30,\"tid\":0,\"args\":{\"name\":\"TIM4\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":44,\"tid\":0,\"args\":{\"name\":\"TIM8/13\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":50,\"tid\":0,\"args\":{\"name\":\"TIM5\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":54,\"tid\":0,\"args\":{\"name\":\"TIM6\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":57,\"tid\":0,\"args\":{\"name\":\"DMA2s1\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":58,\"tid\":0,\"args\":{\"name\":\"DMA2s2\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":59,\"tid\":0,\"args\":{\"name\":\"DMA2s3\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":60,\"tid\":0,\"args\":{\"name\":\"DMA2s4\"}}," + + "{\"name\":\"process_name\",\"ph\":\"M\",\"pid\":67,\"tid\":0,\"args\":{\"name\":\"USB\"}}," ; private static final String EOL = "\r\n";