rusefi-full/firmware/util/loggingcentral.cpp

242 lines
6.8 KiB
C++

/**
* @file loggingcentral.cpp
*
* This file implements text logging.
*
* Uses a queue of buffers so that the expensive printf operation doesn't require exclusive access
* (ie, global system lock) to log. In the past there have been serious performance problems caused
* by heavy logging on a low prioriy thread that blocks the rest of the system running (trigger errors, etc).
*
* Uses ChibiOS message queues to maintain one queue of free buffers, and one queue of used buffers.
* When a thread wants to write, it acquires a free buffer, prints to it, and pushes it in to the
* used queue. A dedicated thread then dequeues and writes lines from the used buffer in to the
* large output buffer.
*
* Later, the binary TS thread will request access to the output log buffer for reading, so a lock is taken,
* buffers, swapped, and the back buffer returned. This blocks neither output nor logging in any case, as
* each operation operates on a different buffer.
*
* @date Mar 8, 2015, heavily revised April 2021
* @author Andrey Belomutskiy, (c) 2012-2021
* @author Matthew Kennedy
*/
#include "pch.h"
#include "thread_controller.h"
template <size_t TBufferSize>
void LogBuffer<TBufferSize>::writeLine(LogLineBuffer* line) {
writeInternal(line->buffer);
}
template <size_t TBufferSize>
void LogBuffer<TBufferSize>:: writeLogger(Logging* logging) {
writeInternal(logging->buffer);
}
template <size_t TBufferSize>
size_t LogBuffer<TBufferSize>::length() const {
return m_writePtr - m_buffer;
}
template <size_t TBufferSize>
void LogBuffer<TBufferSize>::reset() {
m_writePtr = m_buffer;
memset(m_buffer, 0, TBufferSize);
}
template <size_t TBufferSize>
const char* LogBuffer<TBufferSize>::get() const {
return m_buffer;
}
template <size_t TBufferSize>
void LogBuffer<TBufferSize>::writeInternal(const char* buffer) {
size_t len = efiStrlen(buffer);
// leave one byte extra at the end to guarantee room for a null terminator
size_t available = TBufferSize - length() - 1;
// If we can't fit the whole thing, write as much as we can
len = minI(available, len);
memcpy(m_writePtr, buffer, len);
m_writePtr += len;
// Ensure the output buffer is always null terminated (in case we did a partial write)
*m_writePtr = '\0';
}
// for unit tests
template class LogBuffer<10>;
#if (EFI_PROD_CODE || EFI_SIMULATOR) && EFI_TEXT_LOGGING
// This mutex protects the LogBuffer instances below
chibios_rt::Mutex logBufferMutex;
// Two buffers:
// - we copy line buffers to writeBuffer in LoggingBufferFlusher
// - and read from readBuffer via TunerStudio protocol commands
using LB = LogBuffer<DL_OUTPUT_BUFFER>;
LB buffers[2];
LB* writeBuffer = &buffers[0];
LB* readBuffer = &buffers[1];
/**
* Actual communication layer invokes this method when it's ready to send some data out
*
* @return pointer to the buffer which should be print to console
*/
const char* swapOutputBuffers(size_t* actualOutputBufferSize) {
{
chibios_rt::MutexLocker lock(logBufferMutex);
// Swap buffers under lock
auto temp = writeBuffer;
writeBuffer = readBuffer;
readBuffer = temp;
// Reset the front buffer - it's now empty
writeBuffer->reset();
}
*actualOutputBufferSize = readBuffer->length();
#if EFI_ENABLE_ASSERTS
size_t expectedOutputSize = efiStrlen(readBuffer->get());
// Check that the actual length of the buffer matches the expected length of how much we thought we wrote
if (*actualOutputBufferSize != expectedOutputSize) {
firmwareError(ERROR_LOGGING_SIZE_CALC, "lsize mismatch %d vs strlen %d", *actualOutputBufferSize, expectedOutputSize);
return nullptr;
}
#endif /* EFI_ENABLE_ASSERTS */
return readBuffer->get();
}
// These buffers store lines queued to be written to the writeBuffer
constexpr size_t lineBufferCount = 24;
static LogLineBuffer lineBuffers[lineBufferCount];
// freeBuffers contains a queue of buffers that are not in use
static chibios_rt::Mailbox<LogLineBuffer*, lineBufferCount> freeBuffers;
// filledBuffers contains a queue of buffers currently waiting to be written to the output buffer
static chibios_rt::Mailbox<LogLineBuffer*, lineBufferCount> filledBuffers;
class LoggingBufferFlusher : public ThreadController<256> {
public:
LoggingBufferFlusher() : ThreadController("log flush", PRIO_TEXT_LOG) { }
void ThreadTask() override {
while (true) {
// Fetch a queued message
LogLineBuffer* line;
msg_t msg = filledBuffers.fetch(&line, TIME_INFINITE);
if (msg == MSG_RESET) {
// todo?
// what happens if MSG_RESET?
} else {
// Lock the buffer mutex - inhibit buffer swaps while writing
chibios_rt::MutexLocker lock(logBufferMutex);
// Write the line out to the output buffer
writeBuffer->writeLine(line);
// Return this line buffer to the free list
freeBuffers.post(line, TIME_INFINITE);
}
}
}
};
static LoggingBufferFlusher lbf;
void startLoggingProcessor() {
// Push all buffers in to the free queue
for (size_t i = 0; i < lineBufferCount; i++) {
freeBuffers.post(&lineBuffers[i], TIME_INFINITE);
}
// Start processing used buffers
lbf.start();
}
#endif // EFI_PROD_CODE
#if EFI_UNIT_TEST || EFI_SIMULATOR
extern bool verboseMode;
#endif
namespace priv
{
void efiPrintfInternal(const char *format, ...) {
#if EFI_UNIT_TEST || EFI_SIMULATOR
if (verboseMode) {
va_list ap;
va_start(ap, format);
vprintf(format, ap);
va_end(ap);
printf("\r\n");
}
#endif
#if (EFI_PROD_CODE || EFI_SIMULATOR) && EFI_TEXT_LOGGING
for (unsigned int i = 0; i < strlen(format); i++) {
// todo: open question which layer would not handle CR/LF properly?
efiAssertVoid(OBD_PCM_Processor_Fault, format[i] != '\n', "No CRLF please");
}
LogLineBuffer* lineBuffer;
msg_t msg;
{
// Acquire a buffer we can write to
chibios_rt::CriticalSectionLocker csl;
msg = freeBuffers.fetchI(&lineBuffer);
}
// No free buffers available, so we can't log
if (msg != MSG_OK) {
return;
}
// Write the formatted string to the output buffer
va_list ap;
va_start(ap, format);
chvsnprintf(lineBuffer->buffer, sizeof(lineBuffer->buffer), format, ap);
va_end(ap);
// Ensure that the string is comma-terminated in case it overflowed
lineBuffer->buffer[sizeof(lineBuffer->buffer) - 1] = LOG_DELIMITER[0];
{
// Push the buffer in to the written list so it can be written back
chibios_rt::CriticalSectionLocker csl;
filledBuffers.postI(lineBuffer);
}
#endif
}
} // namespace priv
/**
* This method appends the content of specified thread-local logger into the global buffer
* of logging content.
*
* This is a legacy function, most normal logging should use efiPrintf
*/
void scheduleLogging(Logging *logging) {
#if (EFI_PROD_CODE || EFI_SIMULATOR) && EFI_TEXT_LOGGING
// Lock the buffer mutex - inhibit buffer swaps while writing
{
chibios_rt::MutexLocker lock(logBufferMutex);
writeBuffer->writeLogger(logging);
}
// Reset the logging now that it's been written out
logging->reset();
#endif
}