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

HAL read/peek timeout with native freeRTOS blocking mechanism #3004

Closed
Closed
Show file tree
Hide file tree
Changes from 1 commit
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
34 changes: 26 additions & 8 deletions cores/esp32/HardwareSerial.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -116,25 +116,43 @@ int HardwareSerial::availableForWrite(void)

int HardwareSerial::peek(void)
{
if (available()) {
return uartPeek(_uart);
}
return -1;
uint8_t data;
if (!uartPeek(_uart, &data, 0))
return -1;

return data;
}

int HardwareSerial::read(void)
{
if(available()) {
return uartRead(_uart);
}
return -1;
uint8_t data;
if (!uartRead(_uart, &data, 0))
return -1;

return data;
}

void HardwareSerial::flush()
{
uartFlush(_uart);
}

int HardwareSerial::timedPeek() {
uint8_t data;
if (!uartPeek(_uart, &data, _timeout))
return -1;

return data;
}

int HardwareSerial::timedRead() {
uint8_t data;
if (!uartRead(_uart, &data, _timeout))
return -1;

return data;
}

size_t HardwareSerial::write(uint8_t c)
{
uartWrite(_uart, c);
Expand Down
2 changes: 2 additions & 0 deletions cores/esp32/HardwareSerial.h
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,8 @@ class HardwareSerial: public Stream
int availableForWrite(void);
int peek(void);
int read(void);
virtual int timedRead(); // private method to read uart with timeout
virtual int timedPeek(); // private method to peek uart with timeout
Copy link
Collaborator

Choose a reason for hiding this comment

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

fix the indent

void flush(void);
size_t write(uint8_t);
size_t write(const uint8_t *buffer, size_t size);
Expand Down
4 changes: 2 additions & 2 deletions cores/esp32/Stream.h
Original file line number Diff line number Diff line change
Expand Up @@ -40,8 +40,8 @@ class Stream: public Print
protected:
unsigned long _timeout; // number of milliseconds to wait for the next char before aborting timed read
unsigned long _startMillis; // used for timeout measurement
int timedRead(); // private method to read stream with timeout
int timedPeek(); // private method to peek stream with timeout
virtual int timedRead(); // private method to read stream with timeout
virtual int timedPeek(); // private method to peek stream with timeout
Copy link
Collaborator

Choose a reason for hiding this comment

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

this may break compatibility with libraries.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

@atanisoft can you be more precise and tell how will it break things? any usage when timedRead should be called directly on Stream.h without virtual inheritance?

Copy link
Collaborator

@atanisoft atanisoft Jul 18, 2019

Choose a reason for hiding this comment

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

Stream.h is defined by the Arduino core specification, you can see the declaration of these methods here. By adding virtual to these methods it will potentially break compatibility with any library which overrides these methods without the declaration of virtual. A similar problem has been seen in #2755

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I cannot imagine any viable scenario about how virtual can break the things. I've got your points about UART HAL methods, and will workaround them. But it is useless if you discard this PR because of virtual methods. Let's decide if this have chance to be merged?

@me-no-dev, what's your opinion about this ?

Copy link
Collaborator

Choose a reason for hiding this comment

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

I'm not against the addition of virtual to the declaration but it may have unintended breakages in libraries which extend this class

int peekNextDigit(); // returns the next numeric digit in the stream or -1 if timeout

public:
Expand Down
19 changes: 11 additions & 8 deletions cores/esp32/esp32-hal-uart.c
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
#include "soc/dport_reg.h"
#include "soc/rtc.h"
#include "esp_intr_alloc.h"
#include "sdkconfig.h"

#define UART_REG_BASE(u) ((u==0)?DR_REG_UART_BASE:( (u==1)?DR_REG_UART1_BASE:( (u==2)?DR_REG_UART2_BASE:0)))
#define UART_RXD_IDX(u) ((u==0)?U0RXD_IN_IDX:( (u==1)?U1RXD_IN_IDX:( (u==2)?U2RXD_IN_IDX:0)))
Expand Down Expand Up @@ -277,26 +278,24 @@ uint32_t uartAvailableForWrite(uart_t* uart)
return 0x7f - uart->dev->status.txfifo_cnt;
}

uint8_t uartRead(uart_t* uart)
uint8_t uartRead(uart_t* uart, uint8_t *data, size_t to)
{
if(uart == NULL || uart->queue == NULL) {
return 0;
}
uint8_t c;
if(xQueueReceive(uart->queue, &c, 0)) {
return c;
if(xQueueReceive(uart->queue, data, to)) {
return 1;
Copy link
Collaborator

Choose a reason for hiding this comment

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

this alters the expected behavior of uartRead() which returns the character that was read from the uart device.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

@atanisoft mention 0 timeout in original code. It will return 0 even if there's nothing to read in uart. That's weird behavior and breaks POLA per se and should be fixed.

Copy link
Collaborator

Choose a reason for hiding this comment

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

while it may be weird and break POLA, it is likely going to break a number of existing libraries which rely on the current method signatures and behaviors of the methods. I know I have a library which uses the UART HAL (I'll be moving it over to IDF API instead soon).

Copy link
Contributor Author

Choose a reason for hiding this comment

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

how do you determine case if UART empty or not in your library?

Moving to idf API was the first solution we've considered. But it's not perfect too. If you carefully look into uart api implementation you'll notice that it's impossible to set RX buffer size, according to example given in documentation. So, we've discarded that solution in favor to fix HAL

Copy link
Collaborator

Choose a reason for hiding this comment

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

uartAvailable() is used to check for presence of data.

As for RX buffer size, that is allocated at time of initialization of the uart via IDF API

Copy link
Contributor Author

Choose a reason for hiding this comment

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

@atanisoft

So, what do you recommend? add overloaded functions?

Copy link
Collaborator

Choose a reason for hiding this comment

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

You could add a new method with the updated signature with the existing one calling into the new method for compatibility. Since this is C code and not C++ I'm not sure if an overloaded method with the same name is possible.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Oh, yes. It's C. I'll think how to implement this.

}
return 0;
}

uint8_t uartPeek(uart_t* uart)
uint8_t uartPeek(uart_t* uart, uint8_t *data, size_t to)
{
if(uart == NULL || uart->queue == NULL) {
return 0;
}
uint8_t c;
if(xQueuePeek(uart->queue, &c, 0)) {
return c;
if(xQueuePeek(uart->queue, data, to)) {
return 1;
Copy link
Collaborator

Choose a reason for hiding this comment

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

same for this method, the uartPeek() method returns the next character available to read

Copy link
Contributor Author

Choose a reason for hiding this comment

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

@atanisoft same here

}
return 0;
}
Expand Down Expand Up @@ -377,6 +376,10 @@ static void uart_on_apb_change(void * arg, apb_change_ev_t ev_type, uint32_t old
}
// wait TX empty
while(uart->dev->status.txfifo_cnt || uart->dev->status.st_utx_out);

if (xHigherPriorityTaskWoken) {
portYIELD_FROM_ISR();
}
Copy link
Collaborator

Choose a reason for hiding this comment

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

Fix the indent for this block

} else {
//todo:
// set baudrate
Expand Down
4 changes: 2 additions & 2 deletions cores/esp32/esp32-hal-uart.h
Original file line number Diff line number Diff line change
Expand Up @@ -56,8 +56,8 @@ void uartEnd(uart_t* uart);

uint32_t uartAvailable(uart_t* uart);
uint32_t uartAvailableForWrite(uart_t* uart);
uint8_t uartRead(uart_t* uart);
uint8_t uartPeek(uart_t* uart);
uint8_t uartRead(uart_t* uart, uint8_t *data, size_t to);
uint8_t uartPeek(uart_t* uart, uint8_t *data, size_t to);

void uartWrite(uart_t* uart, uint8_t c);
void uartWriteBuf(uart_t* uart, const uint8_t * data, size_t len);
Expand Down