Skip to content

Commit

Permalink
Supporting historical TIC (partial impl)
Browse files Browse the repository at this point in the history
  • Loading branch information
lains committed Apr 3, 2024
1 parent 0a3349c commit a9c795b
Show file tree
Hide file tree
Showing 4 changed files with 39 additions and 10 deletions.
11 changes: 9 additions & 2 deletions src/TIC/DatasetExtractor.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ unsigned int TIC::DatasetExtractor::pushBytes(const uint8_t* buffer, unsigned in
*/
if (len == 0)
return 0;
//std::cout << "de got " << len << " bytes starting with: " << std::hex << static_cast<unsigned int>(buffer[0]) << " while " << std::string(this->sync?"inside":"outside") << " of a dataset\n";
unsigned int usedBytes = 0;
if (!this->sync) { /* We don't record bytes, we'll just look for a start of dataset */
uint8_t* firstStartOfDataset = (uint8_t*)(memchr(buffer, TIC::DatasetExtractor::START_MARKER, len));
Expand All @@ -41,7 +42,10 @@ unsigned int TIC::DatasetExtractor::pushBytes(const uint8_t* buffer, unsigned in
}
else {
/* We are inside a TIC dataset, search for the end of dataset marker */
uint8_t* endOfDataset = (uint8_t*)(memchr(buffer, TIC::DatasetExtractor::END_MARKER, len)); /* Search for end of dataset */
//FIXME: historical TIC uses LF, standard TIC uses CR
//If we allow LD below, and we have transmission errors, we may become out of sync if we catch a wrong extraneous LF... we will be out of sync and get only empty datasets because we swap start and end markers
//We should have a way to recover from this, for example by detecting 0 size datasets and thus understand we need to invert sync state
uint8_t* endOfDataset = (uint8_t*)(memchr(buffer, TIC::DatasetExtractor::LF, len)); /* Search for end of dataset */
if (endOfDataset) { /* We have an end of dataset marker in the buffer, we can extract the full dataset */
unsigned int leadingBytesInPreviousDataset = endOfDataset - buffer;
usedBytes = this->processIncomingDatasetBytes(buffer, leadingBytesInPreviousDataset, true); /* Copy the buffer up to (but exclusing the end of dataset marker), the dataset is complete */
Expand Down Expand Up @@ -76,7 +80,10 @@ unsigned int TIC::DatasetExtractor::processIncomingDatasetBytes(const uint8_t* b
}

void TIC::DatasetExtractor::processCurrentDataset() {
this->onDatasetExtracted(this->currentDataset, this->nextWriteInCurrentDataset, this->onDatasetExtractedContext);
//std::vector<uint8_t> datasetContent(this->currentDataset, this->currentDataset+this->nextWriteInCurrentDataset);
//std::cout << "New dataset extracted: " << vectorToHexString(datasetContent) << " (as string: \"" << std::string(datasetContent.begin(), datasetContent.end()) << "\")\n";
if (this->onDatasetExtracted)
this->onDatasetExtracted(this->currentDataset, this->nextWriteInCurrentDataset, this->onDatasetExtractedContext);
}

bool TIC::DatasetExtractor::isInSync() const {
Expand Down
28 changes: 21 additions & 7 deletions src/TIC/Unframer.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -46,18 +46,31 @@ unsigned int TIC::Unframer::pushBytes(const uint8_t* buffer, unsigned int len) {
else {
/* We are inside a TIC frame, search for the end of frame (ETX) marker */
uint8_t* etx = (uint8_t*)(memchr(buffer, TIC::Unframer::END_MARKER, len)); /* Search for end of frame */
if (etx) { /* We have an ETX in the buffer, we can extract the full frame */
unsigned int leadingBytesInPreviousFrame = etx - buffer;
usedBytes = this->processIncomingFrameBytes(buffer, leadingBytesInPreviousFrame); /* Copy the buffer up to (but exclusing the ETX marker) */
uint8_t* stx = nullptr;
if (!etx) { /* Historical TIC may not contain any ETX, but a STX would also signify that the current frame is over */
stx = (uint8_t*)(memchr(buffer, TIC::Unframer::START_MARKER, len)); /* Search for a start if not end of frame is found */
}
if (etx || stx) { /* We have detected the end of the current frame in the buffer, we can extract the full frame */
//std::cout << std::string("Got ") << std::string(etx?"end marker":"unexpected start") << "\n";
unsigned int leadingBytesInPreviousFrame;
if (etx) {
leadingBytesInPreviousFrame = etx - buffer;
}
else if (stx) {
leadingBytesInPreviousFrame = stx - buffer;
}
usedBytes = this->processIncomingFrameBytes(buffer, leadingBytesInPreviousFrame); /* Copy the buffer up to (but exclusing the end of frame marker) */
this->processCurrentFrame(); /* The frame is complete */
leadingBytesInPreviousFrame++; /* Skip the ETX marker */
usedBytes++;
if (etx) {
leadingBytesInPreviousFrame++; /* Skip the ETX marker */
usedBytes++;
}
this->sync = false; /* Consider we are outside of a frame now */
if (leadingBytesInPreviousFrame < len) { /* We have at least one byte after the frame ETX */
usedBytes += this->pushBytes(buffer + leadingBytesInPreviousFrame, len - leadingBytesInPreviousFrame); /* Process the trailing bytes (probably the next frame, starting with STX) */
}
}
else { /* No ETX, copy the whole chunk */
else { /* No end of frame was found, copy the whole chunk */
/* Incoming bytes */
usedBytes = this->processIncomingFrameBytes(buffer, len); /* Process these bytes as valid */
}
Expand Down Expand Up @@ -90,8 +103,9 @@ void TIC::Unframer::processCurrentFrame() {
this->onNewFrameBytes(this->currentFrame, this->nextWriteInCurrentFrame, this->parserFuncContext);
this->nextWriteInCurrentFrame = 0; /* Wipe any data in the current frame, start over */
#endif
if (this->onFrameComplete != nullptr)
if (this->onFrameComplete != nullptr) {
this->onFrameComplete(this->parserFuncContext);
}
}

bool TIC::Unframer::isInSync() const {
Expand Down
9 changes: 8 additions & 1 deletion test/src/TicDatasetExtractor_tests.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
#include <string>
#include <iterator>
#include <cstring>
#include <iomanip>

#include "Tools.h"
#include "TIC/DatasetExtractor.h"
Expand Down Expand Up @@ -60,6 +61,11 @@ static void datasetExtractorUnwrapForwardFrameBytes(const uint8_t* buf, unsigned
if (context == NULL)
return; /* Failsafe, discard if no context */
TIC::DatasetExtractor* de = static_cast<TIC::DatasetExtractor*>(context);
// std::cout << "Got " << cnt << " bytes: ";
// for (unsigned int i = 0; i < cnt; i++) {
// std::cout << std::hex << std::setfill('0') << std::setw(2) << static_cast<unsigned int>(buf[i]) << " ";
// }
// std::cout << "\n";
de->pushBytes(buf, cnt);
}

Expand All @@ -72,6 +78,7 @@ static void datasetExtractorUnWrapFrameFinished(void *context) {
if (context == NULL)
return; /* Failsafe, discard if no context */
TIC::DatasetExtractor* de = static_cast<TIC::DatasetExtractor*>(context);
// std::cout << "Frame finished\n";
/* We have finished parsing a frame, if there is an open dataset, we should discard it and start over at the following frame */
de->reset();
}
Expand Down Expand Up @@ -187,7 +194,7 @@ static void TicUnframer_test_file_sent_by_chunks(const std::vector<uint8_t>& tic
}

TEST(TicDatasetExtractor_tests, Chunked_sample_unframe_dsextract_historical_TIC) {
std::vector<uint8_t> ticData = readVectorFromDisk("./samples/continuous_linky_3P_historical_TIC_sample.bin");
std::vector<uint8_t> ticData = readVectorFromDisk("./samples/continuous_linky_3P_historical_TIC_2024_sample.bin");

for (unsigned int chunkSize = 1; chunkSize <= TIC::DatasetExtractor::MAX_DATASET_SIZE; chunkSize++) {
DatasetDecoderStub stub;
Expand Down
1 change: 1 addition & 0 deletions test/src/Tools.h
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
#include <vector>
#include <string>
#include <fstream>
#include "stdint.h"

std::string vectorToHexString(const std::vector<uint8_t> &vec);

Expand Down

0 comments on commit a9c795b

Please sign in to comment.