Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Made verifying the checksum of a message optional #18

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
48 changes: 32 additions & 16 deletions src/dsmr/parser.h
Original file line number Diff line number Diff line change
Expand Up @@ -306,7 +306,7 @@ struct P1Parser {
* pointer in the result will indicate the next unprocessed byte.
*/
template <typename... Ts>
static ParseResult<void> parse(ParsedData<Ts...> *data, const char *str, size_t n, bool unknown_error = false) {
static ParseResult<void> parse(ParsedData<Ts...> *data, const char *str, size_t n, bool unknown_error = false, bool checksum = true) {
ParseResult<void> res;
if (!n || str[0] != '/')
return res.fail(F("Data should start with /"), str);
Expand All @@ -316,27 +316,43 @@ struct P1Parser {

// Look for ! that terminates the data
const char *data_end = data_start;
uint16_t crc = _crc16_update(0, *str); // Include the / in CRC
while (data_end < str + n && *data_end != '!') {
crc = _crc16_update(crc, *data_end);
++data_end;
}
const char *next = NULL;
if(checksum) {
uint16_t crc = _crc16_update(0, *str); // Include the / in CRC
while (data_end < str + n && *data_end != '!') {
crc = _crc16_update(crc, *data_end);
++data_end;
}

if (data_end >= str + n)
return res.fail(F("No checksum found"), data_end);
if (data_end >= str + n)
return res.fail(F("No checksum found"), data_end);

crc = _crc16_update(crc, *data_end); // Include the ! in CRC
crc = _crc16_update(crc, *data_end); // Include the ! in CRC

ParseResult<uint16_t> check_res = CrcParser::parse(data_end + 1, str + n);
if (check_res.err)
return check_res;
ParseResult<uint16_t> check_res = CrcParser::parse(data_end + 1, str + n);
if (check_res.err)
return check_res;

// Check CRC
if (check_res.result != crc)
return res.fail(F("Checksum mismatch"), data_end + 1);
// Check CRC
if (check_res.result != crc)
return res.fail(F("Checksum mismatch"), data_end + 1);

next = check_res.next;
} else {
// There's no CRC check, still need to know where the message ends (!)
while (data_end < str + n && *data_end != '!') {
++data_end;
}

// Skip to end-of-line, everything afther that is not yet processed.
next = data_end;
while (next < str + n && *next != '\r' && *next != '\n') {
++next;
}
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this needed? Is there a CRLF after the !? If I read the v4 spec correctly, there should be no CRLF after the last line (which I think is the checksum line in v4, and the ! line in v3)? Are you seeing a CRLF in your messages?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, you could be right, I've checked the v3 spec and there's no CRLF. Need to check if that's somehow added to my recording.

}

res = parse_data(data, data_start, data_end, unknown_error);
res.next = check_res.next;
res.next = next;
return res;
}

Expand Down