fome-fw/firmware/development/perf_trace.cpp

102 lines
2.1 KiB
C++
Raw Normal View History

2019-10-11 17:43:21 -07:00
2019-10-14 23:34:12 -07:00
#include "efifeatures.h"
2019-10-11 17:43:21 -07:00
#include "perf_trace.h"
#include "efitime.h"
#include "os_util.h"
#ifndef TRACE_BUFFER_LENGTH
2019-10-14 23:34:12 -07:00
#define TRACE_BUFFER_LENGTH 2048
#endif /* TRACE_BUFFER_LENGTH */
2019-10-11 17:43:21 -07:00
2019-10-14 23:34:12 -07:00
enum class EPhase : char
2019-10-11 17:43:21 -07:00
{
Start,
End,
InstantThread,
InstantGlobal,
};
struct TraceEntry
{
PE Event;
2019-10-14 23:34:12 -07:00
EPhase Phase;
2019-10-11 17:43:21 -07:00
uint8_t Data;
uint8_t ThreadId;
uint32_t Timestamp;
};
// Ensure that the struct is the size we think it is - the binary layout is important
2019-10-11 17:43:21 -07:00
static_assert(sizeof(TraceEntry) == 8);
// This buffer stores a trace - we write the full buffer once, then disable tracing
2019-10-11 17:43:21 -07:00
static TraceEntry s_traceBuffer[TRACE_BUFFER_LENGTH];
static size_t s_nextIdx = 0;
static bool s_isTracing = false;
2019-10-14 23:34:12 -07:00
void perfEventImpl(PE event, EPhase phase, uint8_t data)
2019-10-11 17:43:21 -07:00
{
// Bail if we aren't allowed to trace
2019-10-14 23:34:12 -07:00
if constexpr (!ENABLE_PERF_TRACE) {
return;
}
// Bail if we aren't tracing
2019-10-14 23:34:12 -07:00
if (!s_isTracing) {
2019-10-11 17:43:21 -07:00
return;
}
uint32_t timestamp = getTimeNowLowerNt();
size_t idx;
// Critical section: reserve index under lock
{
bool wasLocked = lockAnyContext();
idx = s_nextIdx++;
2019-10-14 23:34:12 -07:00
if (s_nextIdx >= TRACE_BUFFER_LENGTH) {
2019-10-11 17:43:21 -07:00
s_nextIdx = 0;
2019-10-14 23:34:12 -07:00
s_isTracing = false;
2019-10-11 17:43:21 -07:00
}
if (!wasLocked) {
unlockAnyContext();
}
}
// We can safely write data out of the lock, our spot is reserved
2019-10-14 23:34:12 -07:00
volatile TraceEntry& entry = s_traceBuffer[idx];
2019-10-11 17:43:21 -07:00
entry.Event = event;
2019-10-14 23:34:12 -07:00
entry.Phase = phase;
// Get the current active interrupt - this is the "thread ID"
2019-10-14 23:34:12 -07:00
entry.ThreadId = static_cast<uint8_t>(SCB->ICSR & SCB_ICSR_VECTACTIVE_Msk);
2019-10-11 17:43:21 -07:00
entry.Timestamp = timestamp;
entry.Data = data;
}
2019-10-14 23:34:12 -07:00
void perfEventBegin(PE event, uint8_t data) {
perfEventImpl(event, EPhase::Start, data);
2019-10-11 17:43:21 -07:00
}
2019-10-14 23:34:12 -07:00
void perfEventEnd(PE event, uint8_t data) {
perfEventImpl(event, EPhase::End, data);
2019-10-11 17:43:21 -07:00
}
2019-10-14 23:34:12 -07:00
void perfEventInstantGlobal(PE event, uint8_t data) {
perfEventImpl(event, EPhase::InstantGlobal, data);
2019-10-11 17:43:21 -07:00
}
2019-10-14 23:40:51 -07:00
void perfTraceEnable() {
2019-10-14 23:34:12 -07:00
s_isTracing = true;
}
2019-10-14 23:40:51 -07:00
const TraceBufferResult perfTraceGetBuffer() {
// stop tracing if you try to get the buffer early
s_isTracing = false;
return {reinterpret_cast<const uint8_t*>(s_traceBuffer), sizeof(s_traceBuffer)};
2019-10-11 17:43:21 -07:00
}