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 }

Parents
  • 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!).

  • 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.
Reply
  • 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.
Children
No Data
Related