-
Notifications
You must be signed in to change notification settings - Fork 7.6k
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
Changes from 1 commit
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -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 | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this may break compatibility with libraries. There was a problem hiding this comment. Choose a reason for hiding this commentThe 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? There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 ? There was a problem hiding this comment. Choose a reason for hiding this commentThe 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: | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -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))) | ||
|
@@ -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; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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. There was a problem hiding this comment. Choose a reason for hiding this commentThe 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. There was a problem hiding this comment. Choose a reason for hiding this commentThe 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). There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So, what do you recommend? add overloaded functions? There was a problem hiding this comment. Choose a reason for hiding this commentThe 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. There was a problem hiding this comment. Choose a reason for hiding this commentThe 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; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @atanisoft same here |
||
} | ||
return 0; | ||
} | ||
|
@@ -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(); | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Fix the indent for this block |
||
} else { | ||
//todo: | ||
// set baudrate | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
fix the indent