2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file loggingcentral.cpp
|
|
|
|
*
|
2019-05-04 07:18:49 -07:00
|
|
|
*
|
|
|
|
* As of May 2019 we have given up on text-based 'push' terminal mode. At the moment binary protocol
|
|
|
|
* is the consumen of this logging buffer.
|
|
|
|
*
|
2015-07-10 06:01:56 -07:00
|
|
|
* @date Mar 8, 2015
|
2020-01-13 18:57:43 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
2018-09-16 19:26:57 -07:00
|
|
|
#include "global.h"
|
2020-11-19 03:56:02 -08:00
|
|
|
#include "os_access.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "efilib.h"
|
|
|
|
|
2020-08-06 21:59:00 -07:00
|
|
|
#if EFI_UNIT_TEST || EFI_SIMULATOR
|
2020-07-03 21:31:29 -07:00
|
|
|
extern bool verboseMode;
|
|
|
|
#endif /* EFI_UNIT_TEST */
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
typedef char log_buf_t[DL_OUTPUT_BUFFER];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* we need to leave a byte for zero terminator, also two bytes for the \r\n in
|
|
|
|
* printWithLength, also couple of bytes just in case
|
|
|
|
*/
|
|
|
|
#define MAX_DL_CAPACITY (DL_OUTPUT_BUFFER - 5)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This is the buffer into which all the data providers write
|
|
|
|
*/
|
|
|
|
static char *accumulationBuffer;
|
|
|
|
|
2017-05-17 17:36:07 -07:00
|
|
|
static log_buf_t pendingBuffers0;
|
2015-07-10 06:01:56 -07:00
|
|
|
static log_buf_t pendingBuffers1;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* We copy all the pending data into this buffer once we are ready to push it out
|
|
|
|
*/
|
|
|
|
static char * outputBuffer;
|
|
|
|
|
2020-05-03 08:49:06 -07:00
|
|
|
|
|
|
|
class LoggingCentral {
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Class constructors are a great way to have simple initialization sequence
|
|
|
|
*/
|
|
|
|
LoggingCentral() {
|
|
|
|
pendingBuffers0[0] = 0;
|
|
|
|
pendingBuffers1[0] = 0;
|
|
|
|
accumulationBuffer = pendingBuffers0;
|
|
|
|
outputBuffer = pendingBuffers1;
|
|
|
|
accumulatedSize = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* amount of data accumulated so far
|
|
|
|
*/
|
|
|
|
uint32_t accumulatedSize;
|
|
|
|
};
|
|
|
|
|
|
|
|
static LoggingCentral loggingCentral;
|
|
|
|
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
2019-05-04 07:18:49 -07:00
|
|
|
* This method appends the content of specified thread-local logger into the global buffer
|
|
|
|
* of logging content.
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2020-11-19 03:56:02 -08:00
|
|
|
static void scheduleLoggingInternal(Logging *logging) {
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_TEXT_LOGGING
|
2019-05-05 08:06:27 -07:00
|
|
|
#ifdef EFI_PRINT_MESSAGES_TO_TERMINAL
|
|
|
|
print(logging->buffer);
|
|
|
|
print("\r\n");
|
|
|
|
#endif /* EFI_PRINT_MESSAGES_TO_TERMINAL */
|
2015-07-10 06:01:56 -07:00
|
|
|
// this could be done without locking
|
|
|
|
int newLength = efiStrlen(logging->buffer);
|
|
|
|
|
2020-11-19 03:56:02 -08:00
|
|
|
chibios_rt::CriticalSectionLocker csl;
|
2020-05-03 08:49:06 -07:00
|
|
|
if (loggingCentral.accumulatedSize + newLength >= MAX_DL_CAPACITY) {
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* if no one is consuming the data we have to drop it
|
|
|
|
* this happens in case of serial-over-USB, todo: find a better solution?
|
|
|
|
*/
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// memcpy is faster then strcpy because it is not looking for line terminator
|
2020-05-03 08:49:06 -07:00
|
|
|
memcpy(accumulationBuffer + loggingCentral.accumulatedSize, logging->buffer, newLength + 1);
|
|
|
|
loggingCentral.accumulatedSize += newLength;
|
2018-09-16 20:54:14 -07:00
|
|
|
#endif /* EFI_TEXT_LOGGING */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2020-11-19 03:56:02 -08:00
|
|
|
void scheduleLogging(Logging* logging) {
|
|
|
|
scheduleLoggingInternal(logging);
|
|
|
|
logging->reset();
|
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
2019-05-04 07:18:49 -07:00
|
|
|
* Actual communication layer invokes this method when it's ready to send some data out
|
|
|
|
*
|
2015-07-10 06:01:56 -07:00
|
|
|
* this method should always be invoked from the same thread!
|
2019-05-04 07:18:49 -07:00
|
|
|
* @return pointer to the buffer which should be print to console
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2021-03-15 03:42:58 -07:00
|
|
|
const char* swapOutputBuffers(size_t* actualOutputBufferSize) {
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_ENABLE_ASSERTS
|
2015-07-10 06:01:56 -07:00
|
|
|
int expectedOutputSize;
|
2016-12-30 09:03:15 -08:00
|
|
|
#endif /* EFI_ENABLE_ASSERTS */
|
2015-07-10 06:01:56 -07:00
|
|
|
{ // start of critical section
|
2020-11-19 03:56:02 -08:00
|
|
|
chibios_rt::CriticalSectionLocker csl;
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* we cannot output under syslock, we simply rotate which buffer is which
|
|
|
|
*/
|
|
|
|
char *temp = outputBuffer;
|
|
|
|
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_ENABLE_ASSERTS
|
2020-05-03 08:49:06 -07:00
|
|
|
expectedOutputSize = loggingCentral.accumulatedSize;
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif /* EFI_ENABLE_ASSERTS */
|
|
|
|
outputBuffer = accumulationBuffer;
|
|
|
|
|
|
|
|
accumulationBuffer = temp;
|
2020-05-03 08:49:06 -07:00
|
|
|
loggingCentral.accumulatedSize = 0;
|
2015-07-10 06:01:56 -07:00
|
|
|
accumulationBuffer[0] = 0;
|
|
|
|
} // end of critical section
|
|
|
|
|
|
|
|
*actualOutputBufferSize = efiStrlen(outputBuffer);
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_ENABLE_ASSERTS
|
2016-12-30 09:03:15 -08:00
|
|
|
if (*actualOutputBufferSize != expectedOutputSize) {
|
2017-04-22 06:50:09 -07:00
|
|
|
int sizeToShow = minI(10, *actualOutputBufferSize);
|
|
|
|
int offsetToShow = *actualOutputBufferSize - sizeToShow;
|
2017-05-29 16:23:15 -07:00
|
|
|
firmwareError(ERROR_LOGGING_SIZE_CALC, "lsize mismatch %d/%d [%s]", *actualOutputBufferSize, expectedOutputSize,
|
2017-04-22 06:50:09 -07:00
|
|
|
&outputBuffer[offsetToShow]);
|
2016-12-30 09:03:15 -08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif /* EFI_ENABLE_ASSERTS */
|
2015-07-10 06:01:56 -07:00
|
|
|
return outputBuffer;
|
|
|
|
}
|
|
|
|
|
2018-08-31 19:30:03 -07:00
|
|
|
/**
|
2019-05-04 07:18:49 -07:00
|
|
|
* rusEfi business logic invokes this method in order to eventually print stuff to rusEfi console
|
|
|
|
*
|
2018-08-31 19:30:03 -07:00
|
|
|
* this whole method is executed under syslock so that we can have multiple threads use the same shared buffer
|
|
|
|
* in order to reduce memory usage
|
2019-05-04 07:18:49 -07:00
|
|
|
*
|
|
|
|
* this is really 'global lock + printf + scheduleLogging + unlock' a bit more clear
|
2018-08-31 19:30:03 -07:00
|
|
|
*/
|
2020-07-03 21:31:29 -07:00
|
|
|
void scheduleMsg(Logging *logging, const char *format, ...) {
|
2020-08-06 21:59:00 -07:00
|
|
|
#if EFI_UNIT_TEST || EFI_SIMULATOR
|
2020-07-03 21:31:29 -07:00
|
|
|
if (verboseMode) {
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
|
|
vprintf(format, ap);
|
|
|
|
va_end(ap);
|
|
|
|
printf("\r\n");
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for (unsigned int i = 0;i<strlen(format);i++) {
|
2019-05-04 07:18:49 -07:00
|
|
|
// todo: open question which layer would not handle CR/LF properly?
|
2020-07-03 21:31:29 -07:00
|
|
|
efiAssertVoid(OBD_PCM_Processor_Fault, format[i] != '\n', "No CRLF please");
|
2019-04-13 07:59:29 -07:00
|
|
|
}
|
2019-04-12 19:10:57 -07:00
|
|
|
#if EFI_TEXT_LOGGING
|
2018-08-31 19:30:03 -07:00
|
|
|
if (logging == NULL) {
|
|
|
|
warning(CUSTOM_ERR_LOGGING_NULL, "logging NULL");
|
|
|
|
return;
|
|
|
|
}
|
2020-11-19 03:56:02 -08:00
|
|
|
|
|
|
|
chibios_rt::CriticalSectionLocker csl;
|
2020-09-13 01:49:25 -07:00
|
|
|
logging->reset(); // todo: is 'reset' really needed here?
|
2018-08-31 19:30:03 -07:00
|
|
|
appendMsgPrefix(logging);
|
|
|
|
|
|
|
|
va_list ap;
|
2020-07-03 21:31:29 -07:00
|
|
|
va_start(ap, format);
|
|
|
|
logging->vappendPrintf(format, ap);
|
2018-08-31 19:30:03 -07:00
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
appendMsgPostfix(logging);
|
|
|
|
scheduleLogging(logging);
|
2018-09-16 20:54:14 -07:00
|
|
|
#endif /* EFI_TEXT_LOGGING */
|
2020-07-03 21:31:29 -07:00
|
|
|
#endif /* EFI_UNIT_TEST */
|
2018-08-31 19:30:03 -07:00
|
|
|
}
|
|
|
|
|