This post is older than 2 years and might not be relevant anymore
More Info: Consider searching for newer posts

uart in fifo mode has long delays between receiving chars and some got lost

Hi, I made simple command shell via UART. It works fine when I type commands in terminal by hand. But I run into troubles when BLE chip is fed from MCU that send characters quickly. I can see several miliseconds delays between echoing characters and even some got lost and received command was corrupted. I hoped that received chars are stored in FIFO so there shouldn't be any data loss. Maybe I do it wrong in my code, any idea?

volatile int uart_cmd_ready; // complette command flag volatile int uart_cmd_index; // cmd buffer rec. char index char uart_cmd[UART_CMD_SIZE]; // cmd buffer that received chars are stored in

int uart_init(void) { // FIFO size must be 2^n app_uart_comm_params_t params; uint32_t err_code;

params.baud_rate=UART_BAUDR; params.flow_control=APP_UART_FLOW_CONTROL_DISABLED; // HW flow disabled - we have only 2-wire UART connection params.use_parity=false; params.tx_pin_no=UART_TX_PIN; params.rx_pin_no=UART_RX_PIN; APP_UART_FIFO_INIT(&params, UART_RX_BUFFER_SIZE, UART_TX_BUFFER_SIZE, uart_evt_handler, APP_IRQ_PRIORITY_HIGH, err_code); uart_cmd_ready=0;
uart_cmd_index=0;
memset(uart_cmd, 0, UART_CMD_SIZE); return(err_code); }

void uart_evt_handler(app_uart_evt_t *p_app_uart_event) { uint8_t c; if (p_app_uart_event->evt_type==APP_UART_DATA_READY) { app_uart_get(&c); // read char from FIFO app_uart_put(c); // and echo it immediately if ((c==(uint8_t)'\r') || (c==(uint8_t)'\n') || (uart_cmd_index>=UART_CMD_SIZE-2)) // if end of line received CR or LF or buffer full { uart_cmd[uart_cmd_index++]='\0'; // zero string termination, inc. index uart_cmd_ready=1; // set cmd complette flag } else if (c==8) // BackSpace { if (uart_cmd_index>0) // if index not at start position uart_cmd_index--; // decrement index } else
uart_cmd[uart_cmd_index++]=c; // insert char to buffer in ic. index }
}

after command is parsed I call

void clear_cmd(void) { sd_nvic_DisableIRQ(UART0_IRQn); // disable IRQ from UART uart_cmd[0]='\0'; // zero string in cmd buffer uart_cmd_ready=0; // reset cmd complette flag uart_cmd_index=0; // reset index sd_nvic_EnableIRQ(UART0_IRQn); // enable IRQ from UART }

  • Seems that we are talking about nRF51822, isn't it? 1st (2 byte rx fifo) or 2nd (6 byte fifo) Revision? Mine is a QFAAG0 which ist 2nd revision with 6 byte fifo). What baud rate are you running? Uart transmission without HW flowcontrol can result in lost bytes when SoftDevice is active and baudrate > 9.600 → https://devzone.nordicsemi.com/index.php/what-s-the-maximum-of-baud-rate-supported-of-uart#reply-889

  • Joe, sorry I missed post that UART_BAUDR = UART_BAUDRATE_BAUDRATE_Baud38400. I belived that FIFO is here to prevent data loss. I would expect that UART Rx interrupt priority is set high because its handling is very fast - just copy 1 reg to buffer so it shouldn't disturb radio communication a lot. What is the longest delay that can fully softdevice occupy the CPU?

  • If you have a first revision chip, and don't use flow control, you may see lost bytes with higher baud rates, since the softdevice will block the CPU during the connection events. Details on this blocking is given in the S110 SoftDevice specification, which I'd recommend you to read thoroughly. Blocking will be from ~1 to ~6 ms depending on the amount of data currently going over the link. Keeping a BLE link gives hard real-time requirements, and at the wrong time, any application level interrupt can cause disruption. With the S110 it is therefore not legal to have interrupts with priority 0; these are exclusive to the softdevice.

    In general, I'd strongly recommend to both make sure to use second revision chips, which have a 6-byte instead of a 2-byte buffer (identified by markings QFAAFA or later), and independently also use flow-control if at all possible. Doing so should avoid any lost bytes independent of baud rate, as long as the other device doesn't misbehave (take a look at Joe's very nice experiement here!).

  • Hi, unfortunately we have old chip version bought probably in late 2013 (marking is QFAAC0) that is already soldered on hundreds of our PCBs and cannot be changed. Also we cannot use HW flow control because our main MCU has only one full-featured UART that is reserved for GSM modem where it is necessary. Maybe RTS&CTS could be emulated by MCU's GPIO but there are no wires from MCU anyway... So only way is decrease baudrate, I hope 4800 would be safe enough and acceptable four our app... BTW I saw that dropped characters even during advertising only, not connection established yet.

    But I think that you could add UART receiver ISR into softdevice code allowing it to use high priority interrupt (copy one char shouldn't disrupt radio timing) and then share/copy buffers with application. Latency wouldn't be problem but it prevent data loss.

  • We're aware that this is not optimal, but I'm afraid adding UART into the softdevice won't happen anytime soon.

    There are several workarounds possible for this, and I'll try to list them all, independent on whether they have been mentioned before here or not:

    • Use flow control.
    • Reduce baud rate.
    • Make sure your serial protocol handles lost bytes gracefully, and retransmits.
    • Use high baud rate and time your transfer to in between radio events, by using the radio notification feature of the softdevice. Depending on advertising/connection interval, you'll have potentially long periods of non-interrupted time in between.
Related