Skip to content

Commit

Permalink
Flush TCP socket on exception
Browse files Browse the repository at this point in the history
Not all bytes may have been read from the TCP socket read buffer when
the request processing is aborted by an exception.

- remember the frame length from the message header
- derease remaining length on every successful read
- after sending the exception flush any remaining bytes from the socket

Fixes #67
  • Loading branch information
stefanb2 committed Sep 27, 2024
1 parent 96126e5 commit 2a5342a
Show file tree
Hide file tree
Showing 2 changed files with 17 additions and 2 deletions.
18 changes: 16 additions & 2 deletions nanomodbus.c
Original file line number Diff line number Diff line change
Expand Up @@ -254,8 +254,12 @@ static nmbs_error recv(nmbs_t* nmbs, uint16_t count) {
int32_t ret =
nmbs->platform.read(nmbs->msg.buf + nmbs->msg.buf_idx, count, nmbs->byte_timeout_ms, nmbs->platform.arg);

if (ret == count)
if (ret == count) {
if (nmbs->platform.transport == NMBS_TRANSPORT_TCP)
nmbs->msg.length -= count;

return NMBS_ERROR_NONE;
}

if (ret < count) {
if (ret < 0)
Expand Down Expand Up @@ -355,6 +359,7 @@ static nmbs_error recv_msg_header(nmbs_t* nmbs, bool* first_byte_received) {
nmbs->msg.transaction_id = get_2(nmbs);
uint16_t protocol_id = get_2(nmbs);
uint16_t length = get_2(nmbs); // We should actually check the length of the request against this value
// length includes all message bytes from here
nmbs->msg.unit_id = get_1(nmbs);
nmbs->msg.fc = get_1(nmbs);

Expand All @@ -363,6 +368,9 @@ static nmbs_error recv_msg_header(nmbs_t* nmbs, bool* first_byte_received) {

if (length > 255)
return NMBS_ERROR_INVALID_TCP_MBAP;

// remaining bytes in socket buffer: unit_id & fc already processed
nmbs->msg.length = length - 1 - 1;
}

return NMBS_ERROR_NONE;
Expand Down Expand Up @@ -445,7 +453,13 @@ static nmbs_error send_exception_msg(nmbs_t* nmbs, uint8_t exception) {

NMBS_DEBUG_PRINT("%d NMBS res -> address_rtu %d\texception %d", nmbs->address_rtu, nmbs->address_rtu, exception);

return send_msg(nmbs);
nmbs_error err = send_msg(nmbs);

if (err == NMBS_ERROR_NONE && nmbs->platform.transport == NMBS_TRANSPORT_TCP && nmbs->msg.length > 0)
// Flush the remaining data in the socket buffer
nmbs->platform.read(nmbs->msg.buf, nmbs->msg.length, nmbs->read_timeout_ms, nmbs->platform.arg);

return err;
}
#endif

Expand Down
1 change: 1 addition & 0 deletions nanomodbus.h
Original file line number Diff line number Diff line change
Expand Up @@ -239,6 +239,7 @@ typedef struct nmbs_t {
uint8_t unit_id;
uint8_t fc;
uint16_t transaction_id;
uint16_t length; // only valid for NMBS_TRANSPORT_TCP
bool broadcast;
bool ignored;
} msg;
Expand Down

0 comments on commit 2a5342a

Please sign in to comment.