/** * See also BinarySensorLog.java * See also mlq_file_format.txt */ #include "pch.h" #include "binary_logging.h" #include "log_field.h" #include "buffered_writer.h" #include "tunerstudio.h" #define TIME_PRECISION 1000 // floating number of seconds with millisecond precision static scaled_channel packedTime; // todo: we are at the edge of sdLogBuffer size and at the moment we have no code to make sure buffer does not overflow // todo: make this logic smarter // The list of logged fields lives in a separate file so it can eventually be tool-generated #include "log_fields_generated.h" static constexpr uint16_t computeFieldsRecordLength() { uint16_t recLength = 0; for (size_t i = 0; i < efi::size(fields); i++) { recLength += fields[i].getSize(); } return recLength; } #if EFI_FILE_LOGGING // this one needs to be in main ram so that SD card SPI DMA works fine static NO_CACHE char sdLogBuffer[250]; static uint64_t binaryLogCount = 0; extern bool main_loop_started; void writeSdLogLine(Writer& bufferedWriter) { if (!main_loop_started) return; if (binaryLogCount == 0) { writeFileHeader(bufferedWriter); } else { updateTunerStudioState(); size_t length = writeBlock(sdLogBuffer); efiAssertVoid(OBD_PCM_Processor_Fault, length <= efi::size(sdLogBuffer), "SD log buffer overflow"); bufferedWriter.write(sdLogBuffer, length); } binaryLogCount++; } #endif /* EFI_FILE_LOGGING */ static constexpr uint16_t recordLength = computeFieldsRecordLength(); void writeFileHeader(Writer& outBuffer) { char buffer[MLQ_HEADER_SIZE]; // File format: MLVLG\0 strncpy(buffer, "MLVLG", 6); // Format version = 01 buffer[6] = 0; buffer[7] = 1; // Timestamp buffer[8] = 0; buffer[9] = 0; buffer[10] = 0; buffer[11] = 0; // Info data start buffer[12] = 0; buffer[13] = 0; size_t headerSize = MLQ_HEADER_SIZE + efi::size(fields) * 55; // Data begin index: begins immediately after the header buffer[14] = 0; buffer[15] = 0; buffer[16] = (headerSize >> 8) & 0xFF; buffer[17] = headerSize & 0xFF; // Record length - length of a single data record: sum size of all fields buffer[18] = recordLength >> 8; buffer[19] = recordLength & 0xFF; // Number of logger fields buffer[20] = 0; buffer[21] = efi::size(fields); outBuffer.write(buffer, MLQ_HEADER_SIZE); // Write the actual logger fields, offset 22 for (size_t i = 0; i < efi::size(fields); i++) { fields[i].writeHeader(outBuffer); } } static uint8_t blockRollCounter = 0; //static efitimeus_t prevSdCardLineTime = 0; size_t writeBlock(char* buffer) { // Offset 0 = Block type, standard data block in this case buffer[0] = 0; // Offset 1 = rolling counter sequence number buffer[1] = blockRollCounter++; // Offset 2, size 2 = Timestamp at 10us resolution efitimeus_t nowUs = getTimeNowUs(); uint16_t timestamp = nowUs / 10; buffer[2] = timestamp >> 8; buffer[3] = timestamp & 0xFF; // todo: add a log field for SD card period // prevSdCardLineTime = nowUs; packedTime = getTimeNowMs() * 1.0 / TIME_PRECISION; // Offset 4 = field data const char* dataBlockStart = buffer + 4; char* dataBlock = buffer + 4; uint8_t sum = 0; for (size_t fieldIndex = 0; fieldIndex < efi::size(fields); fieldIndex++) { size_t entrySize = fields[fieldIndex].writeData(dataBlock); for (size_t byteIndex = 0; byteIndex < entrySize; byteIndex++) { // "CRC" at the end is just the sum of all bytes sum += dataBlock[byteIndex]; } // Increment pointer to next entry dataBlock += entrySize; } size_t dataBlockSize = dataBlock - dataBlockStart; *dataBlock = sum; // Total size has 4 byte header + 1 byte checksum return dataBlockSize + 5; }