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

  • As Ole Morten wrote: »… Blocking will be from ~1 to ~6 ms …«, 4.800 Baud won't be slow enough. At 4.800 Baud you could have ~2,1 ms for each byte (assuming 8n1). Therefore I think, you have to slow down to 2.400 Baud (~4,8 ms per byte) to prevent data loss.

Reply Children
No Data
Related