-
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
Adds HardwareSerial::setRxTimeout() #6397
Adds HardwareSerial::setRxTimeout() #6397
Conversation
More information about this. I've tested this timeout on the oscilloscope and it's working as expected! 🎉🎉🎉 One consideration. If an overflow occurs, and then you don't process the data on the buffer with onReceiveError callback, then onReceive will never get called again. Maybe I should add a warning on the comments. |
Thanks @gonzabrusco ! |
Hello I use UART2 with the pins (GPIO) 12 and 13 Torsten |
Hello @TS6000 . This is a Pull Request. Please create a new issue if you have one. But before creating one, take note that the UART code have changed a lot in the latest master. So please try again using master or wait for 2.0.3 version. |
cores/esp32/HardwareSerial.cpp
Outdated
break; | ||
case UART_FIFO_OVF: | ||
log_w("UART%d FIFO Overflow. Consider adding Hardware Flow Control to your Application.", uart->_uart_nr); | ||
if(uart->_onReceiveErrorCB) uart->_onReceiveErrorCB(UART_FIFO_OVF_ERROR); | ||
xQueueReset(uartEventQueue); |
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.
why lose all previous data?
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.
I did this because it was like this on the previous UART task (before latest SuGlider commit). But maybe this could be ommited. The reason is that when you have a big overflow (overflowing multiple times the FIFO) the event queue gets filled with one UART_BUFFER_FULL and multiple UART_FIFO_OVF. I though it was redundant to be called multiple times for the same event.
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.
This is what I was talking about:
arduino-esp32/cores/esp32/esp32-hal-uart.c
Line 117 in c014eaf
xQueueReset(uart->uart_event_queue); |
arduino-esp32/cores/esp32/esp32-hal-uart.c
Line 123 in c014eaf
xQueueReset(uart->uart_event_queue); |
If you think this should be deleted, I have no opposition.
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.
I would rather discuss it, than just delete it :) @SuGlider want to pitch in an idea too?
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.
Well, I guess this is a bit relative. It could happen that you have a very fast overflow and you will be called multiple times with the UART_FIFO_OVF event. But thinking about this more thoroughly, maybe that is not a "normal" or "frequent" situation. I added it back because I saw it on the original esp32-hal-uart code and when I was testing the overflow with the serial monitor, I noticed that behaviour. But as I said, probably is not a frequent situation and you could eventually lose something important on the event queue? (maybe an UART_DATA event after an overflow event?). Probably this should be deleted. But I will let @SuGlider give this opinion about this. I would love to have this on 2.0.3 because I will be using that version as the new core for my project (2.0.2 has many bugs with UART).
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.
@gonzabrusco - about your consideration:
One consideration. If an overflow occurs, and then you don't process the data on the buffer with onReceiveError callback, then onReceive will never get called again.
It is a real problem for most Arduino users.
onReceive Callback should be called also when RX FIFO is full and UART driver copies that data to the RX ringbuffer.
I think it should pass a parameter to the call back function informing if there was a Timeout instead of forcing only calling it when timeout occurs. This way the user can decide what to do in the callback and it is more flexible.
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.
Regarding xQueueReset(uart->uart_event_queue);
, it shall be removed from the task.
I suggest that it may become a new function such as eventQueueReset()
in HardwareSerial API.
This way users can decide when to reset the queue instead of forcing it in the task loop.
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.
Last suggestion is about changing the name of the function void HardwareSerial::onReceiveTimeout(uint8_t symbols_timeout)
to void HardwareSerial::setOnReceiveTimeout(uint8_t symbols_timeout)
or some name like setRxTimeout()
.
Timeout mode shall not be forced in the code... it shall be an option. check
// from void HardwareSerial::onReceiveTimeout(uint8_t symbols_timeout) :::
if(symbols_timeout == 0) {
_onReceiveTimeout = 1; // Never disable timeout
}
// from void HardwareSerial::begin(...) :::
// Set UART RX timeout
if (_uart != NULL && _onReceiveCB != NULL) {
uart_set_rx_timeout(_uart_nr, _onReceiveTimeout);
}
// from void HardwareSerial::_uartEventTask(void *args)) ::: "&& event.timeout_flag"
case UART_DATA:
if(uart->_onReceiveCB && uart->available() > 0 && event.timeout_flag) uart->_onReceiveCB();
break;
cores/esp32/HardwareSerial.cpp
Outdated
break; | ||
case UART_BUFFER_FULL: | ||
log_w("UART%d Buffer Full. Consider encreasing your buffer size of your Application.", uart->_uart_nr); | ||
if(uart->_onReceiveErrorCB) uart->_onReceiveErrorCB(UART_BUFFER_FULL_ERROR); | ||
xQueueReset(uartEventQueue); |
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.
same question as above.
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.
Same answer.
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.
xQueueReset(uartEventQueue);
could go on a API function just for that, such as void HardwareSerial::eventQueueReset()
.
This way the sketch can decide when to flush the events.
Please remove it.
Yes, this is correct, onReceive Callback will be called right after 120 bytes or less. I got confused with the Function Name I will review the code and reseach about the implications, given that it may affect basic functions such as |
Hi @SuGlider. About this, Now the onReceive Callback gets called on 10 symbol time UART silence by default (because it's IDF default) or 120 bytes received. This callback just allows you to set the callback timeout (because for example modbus has an interframe delay of 3.5 symbols) and disables the redundant call every 120 bytes (is that really needed?). That way you can be called only when all the data was already received (and not in the middle of the reception for messages longer than 120 bytes). A Modbus frame could be as long as 255 bytes for example. The other alternative is to inform the onReceive callback if the callback was produced because of a timeout or because 120 bytes were received. That way the callback can be able to discern between the two and take different actions.
No problem about the name change. I just gave it that name because it was the timeout on which onReceive callback will be called. Take you time if you need to (although I would love this to be available on 2.0.3 which I will use for my project). About the implications on other functions, I'm not so sure if there is any. This just sets the timeout for the event UART_DATA. Not sure how this could affect other functions like read(). |
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.
Please review all the suggestions. Thanks!
cores/esp32/HardwareSerial.cpp
Outdated
HSERIAL_MUTEX_UNLOCK(); | ||
} | ||
|
||
void HardwareSerial::onReceiveTimeout(uint8_t symbols_timeout) |
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.
Suggestion: change name to something like setOnReceiveTimeout()
or setRxTimeout()
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.
Ok.
cores/esp32/HardwareSerial.cpp
Outdated
HSERIAL_MUTEX_LOCK(); | ||
|
||
if(symbols_timeout == 0) { | ||
_onReceiveTimeout = 1; // Never disable timeout |
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.
I think timeout shall not be enforced. Let the user decide to disable it by setting it to 0.
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.
Ok.
cores/esp32/HardwareSerial.cpp
Outdated
@@ -128,7 +128,8 @@ HardwareSerial::HardwareSerial(int uart_nr) : | |||
_uart_nr(uart_nr), | |||
_uart(NULL), | |||
_rxBufferSize(256), | |||
_onReceiveCB(NULL), | |||
_onReceiveCB(NULL), | |||
_onReceiveTimeout(10), |
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.
I think it shall have as default value ZERO. Disabled. This way it may be enabled using the API.
This will force other changes in the code, mainly because of issues with RX Overflow.
When RX overflow happens, IDF driver disables RX Timeout interrupt flag!
Thus, onReceive will never be called again and the queue will continue full.
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.
This is the only thing I disagree. By default now (as before my pull request), onReceive callback is being called by timeout (10 symbos by default) or by rx fifo full (120 bytes). That is the standard behaviour by IDF. I don't think that Arduino core should use another default. I would prefer to let the timeout on because if you receive 119 bytes, you will never get called (unless you receive on more byte that will trigger UART_INTR_RXFIFO_FULL interrupt)
cores/esp32/HardwareSerial.cpp
Outdated
@@ -210,15 +228,17 @@ void HardwareSerial::_uartEventTask(void *args) | |||
if(xQueueReceive(uartEventQueue, (void * )&event, (portTickType)portMAX_DELAY)) { | |||
switch(event.type) { | |||
case UART_DATA: | |||
if(uart->_onReceiveCB && uart->available() > 0) uart->_onReceiveCB(); | |||
if(uart->_onReceiveCB && uart->available() > 0 && event.timeout_flag) uart->_onReceiveCB(); |
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.
I think it shall not force testing timeout_flag
in order to activate the callback.
Set it as a parameter to the callback function and let the application decide what to do with this informtation.
It also create issues when the IDF ringbuffer is full!
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.
Ok. I will pass the timeout flag as a parameter to the callback.
cores/esp32/HardwareSerial.cpp
Outdated
if (_uart != NULL && _onReceiveCB != NULL) { | ||
uart_set_rx_timeout(_uart_nr, _onReceiveTimeout); | ||
} | ||
|
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.
This shall not be here. It shall be only in the setRxTimeout(...)
API function.
It is enforcing timeout usage to the sketch. That shall be optional.
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.
As I said before. The timeout behaviour is the default as per IDF (10 symbols defined in UART_TOUT_THRESH_DEFAULT). I think this is needed because the user could call setRxTimeout before or after the begin() . This is the same principle as you took with onReceive(). This is because uart_set_rx_timeout should be called once the uart is initialized.
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.
OK. I agree. Let's set the default always.
cores/esp32/HardwareSerial.h
Outdated
@@ -138,6 +150,7 @@ class HardwareSerial: public Stream | |||
uart_t* _uart; | |||
size_t _rxBufferSize; | |||
OnReceiveCb _onReceiveCB; | |||
uint8_t _onReceiveTimeout; |
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.
this may not be necessary to exist anymore as per the suggestions already made here.
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.
I will stress my previous comments here. Because you need to call uart_set_rx_timeout after the uart initialization, then I need a local variable to store my timeout config in case setRxTimeout was called before begin().
I commited the requested changes but I'm still testing it. Please wait before reviewing again. @SuGlider |
This is ready for review again. I'm not sure if I should mark "Resolved" all previous conversations. @SuGlider Here´s a simple sketch to test this.
|
check if _uart is not NULL before using it.
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.
Thanks @gonzabrusco !
I just added a NULL test to eventQueueReset()
, in case it is called before Serial.begin()
, which can reset the board.
(No need for it - reverted)
Nice catch! What is the etiquete here in Espressif repositories about "conversations"? Should I mark them resolved or wait for @me-no-dev or you to resolve them? Don't want to make a mess... |
reverting last change made - it is not necessary.
Actually, that change was not necessary because the function to get the queue already tests it. So I reverted the commit.
I'll run a final testing sequence here and then I'll post a message to me-no-dev asking for his final review. Thanks for the PR! |
In order to allow the user to choose if onReceive() call back will be called only when UART Rx timeout happens or also when UART FIFO gets 120 bytes, a new parameter has been added to onReceive() with the default behavior based on timeout. void onReceive(OnReceiveCb function, bool onlyOnTimeout = true); onReceive will setup a callback that will be called whenever an UART interruption occurs (UART_INTR_RXFIFO_FULL or UART_INTR_RXFIFO_TOUT) UART_INTR_RXFIFO_FULL interrupt triggers at UART_FULL_THRESH_DEFAULT bytes received (defined as 120 bytes by default in IDF) UART_INTR_RXFIFO_TOUT interrupt triggers at UART_TOUT_THRESH_DEFAULT symbols passed without any reception (defined as 10 symbos by default in IDF) onlyOnTimeout parameter will define how onReceive will behave: Default: true -- The callback will only be called when RX Timeout happens. Whole stream of bytes will be ready for being read on the callback function at once. This option may lead to Rx Overflow depending on the Rx Buffer Size and number of bytes received in the streaming false -- The callback will be called when FIFO reaches 120 bytes and also on RX Timeout. The stream of incommig bytes will be "split" into blocks of 120 bytes on each callback. This option avoid any sort of Rx Overflow, but leaves the UART packet reassembling work to the Application.
I have tested this PR and also the last PR #6364 Running the example sketch, but removing the I think that your intention was to only get called back when timeout happens, right? I have changed this PR to reflect this intention, which in fact is better for the user, as you have suggested. New way of doing it: #include <Arduino.h>
void receiveFnc() {
Serial.printf("Received UART Data. Available For Reading = %u\n", Serial.available());
while(Serial.available()) Serial.read(); // Discard received data.
}
void receiveErrorFnc(hardwareSerial_error_t error){
Serial.print("UART Reception Error: ");
switch(error) {
case UART_BREAK_ERROR:
Serial.println("UART_BREAK_ERROR");
break;
case UART_BUFFER_FULL_ERROR:
Serial.println("UART_BUFFER_FULL_ERROR");
break;
case UART_FIFO_OVF_ERROR:
Serial.println("UART_FIFO_OVF_ERROR");
break;
case UART_FRAME_ERROR:
Serial.println("UART_FRAME_ERROR");
break;
case UART_PARITY_ERROR:
Serial.println("UART_PARITY_ERROR");
break;
}
}
void setup() {
// Setting UART Ring Buffer to 512 bytes
Serial.setRxBufferSize(512);
// Define UART Reception Callback
Serial.onReceive(receiveFnc); // default is being called on UART RX timeout
// in order to be called when RX FIFO reaches 120 bytes or when RX timeout(with less than 120 bytes),
// set the callback in this way:
// Serial.onReceive(receiveFnc, false);
// Define UART Error Callback
Serial.onReceiveError(receiveErrorFnc);
// Serial speed 9600 symbos per second
Serial.begin(9600);
}
void loop() {
// Nothing to do
} Please let me know if you agree to these changes I have commited to your PR. |
@gonzabrusco No need to setRxTimeout because it already works as intended. This may be confusing for the regular Arduino user. eventQueueReset() may be useful, so let's keep it. Testing the example I have added here above, the user can decide in an easy way if the callback shall work only on timeout or when blocks of 120 bytes or timeout happens. It is flexible and simple. Please let me know your thoughts. |
Hi @SuGlider . Thanks for the changes in the PR but unfortunately I have to disagree on the decision to remove setRXTimeout which was the primary goal of this PR. The reason I needed this method is to be able to set correctly the timeout (in symbols) in which my callback was going to be called. This is CRITICAL on modbus over RS485 because you HAVE TO process any message in the bus in exactly 3.5 symbols time. If you process the buffer in 10 symbols time (like IDF default), you are arriving too late and you could end up seeing one master request and one slave response joined as a single message. You can see that LIBMODBUS on IDF sets this timeout to 3 for example (using the same function I was using on the first place uart_set_rx_timeout). I agree with your changes but please revert the removal of setRXTimeout. This is needed for Arduino power users! PD: This is whay I was talking about in libmodbus: https://github.com/espressif/esp-idf/blob/a5227db2a75102ca1a17860188c3c352a529a01b/components/freemodbus/port/portserial_m.c#L244 |
@gonzabrusco Please let me know if you also agree to this changes in order to start merging process. |
I loved the changes. Thanks! |
@me-no-dev - This PR is ready for merging, just waiting for your final review. |
@gonzabrusco @SuGlider here seems to be a conflict now. Please fix |
Conflict solved. Thanks. |
Summary
Adds onReceiveTimeout() to HardwareSerial. Default value of the timeout is 10 symbols. This method allows the user to change the timeout on which the onReceive callback is called.
Before this change, the callback was called on 120 bytes always and then on timeout (when receiving more than 120 bytes). Now the callback gets called ONLY on timeout.
Impact
No impact. Everything should work as always.
Related links
No issue related. I did this modifications myself because on my modbus library I was seeing that on long messages (more than 120 bytes), the callbacks was getting called twice. Now it only calls for timeouts (of no receiving data).