HAL_ChibiOS: re-implement half-duplex using HDSEL switching

this makes half-duplex more reliable on UARTs with pullups or level
shifters
This commit is contained in:
Andrew Tridgell 2019-12-27 18:27:11 +11:00
parent faf34970e1
commit b1ca7380be
2 changed files with 75 additions and 43 deletions

View File

@ -48,10 +48,13 @@ UARTDriver *UARTDriver::uart_drivers[UART_MAX_DRIVERS];
// event used to wake up waiting thread. This event number is for // event used to wake up waiting thread. This event number is for
// caller threads // caller threads
#define EVT_DATA EVENT_MASK(0) #define EVT_DATA EVENT_MASK(10)
// event for parity error // event for parity error
#define EVT_PARITY EVENT_MASK(1) #define EVT_PARITY EVENT_MASK(11)
// event for transmit end for half-duplex
#define EVT_TRANSMIT_END EVENT_MASK(12)
#ifndef HAL_UART_MIN_TX_SIZE #ifndef HAL_UART_MIN_TX_SIZE
#define HAL_UART_MIN_TX_SIZE 1024 #define HAL_UART_MIN_TX_SIZE 1024
@ -98,8 +101,9 @@ void UARTDriver::uart_thread(void* arg)
if (uart_drivers[i] == nullptr) { if (uart_drivers[i] == nullptr) {
continue; continue;
} }
if ((mask & EVENT_MASK(i)) && if (uart_drivers[i]->_initialised &&
uart_drivers[i]->_initialised) { (mask & EVENT_MASK(i) ||
(uart_drivers[i]->hd_tx_active && (mask & EVT_TRANSMIT_END)))) {
uart_drivers[i]->_timer_tick(); uart_drivers[i]->_timer_tick();
} }
} }
@ -209,8 +213,12 @@ void UARTDriver::begin(uint32_t b, uint16_t rxS, uint16_t txS)
chVTObjectInit(&tx_timeout); chVTObjectInit(&tx_timeout);
tx_bounce_buf_ready = true; tx_bounce_buf_ready = true;
} }
if (half_duplex) {
rx_dma_enabled = tx_dma_enabled = false;
} else {
rx_dma_enabled = rx_bounce_buf[0] != nullptr && rx_bounce_buf[1] != nullptr; rx_dma_enabled = rx_bounce_buf[0] != nullptr && rx_bounce_buf[1] != nullptr;
tx_dma_enabled = tx_bounce_buf != nullptr; tx_dma_enabled = tx_bounce_buf != nullptr;
}
#endif #endif
/* /*
@ -294,7 +302,7 @@ void UARTDriver::begin(uint32_t b, uint16_t rxS, uint16_t txS)
sercfg.speed = _baudrate; sercfg.speed = _baudrate;
// start with options from set_options() // start with options from set_options()
sercfg.cr1 = 0; sercfg.cr1 = _cr1_options;
sercfg.cr2 = _cr2_options; sercfg.cr2 = _cr2_options;
sercfg.cr3 = _cr3_options; sercfg.cr3 = _cr3_options;
@ -468,13 +476,6 @@ void UARTDriver::rxbuff_full_irq(void* self, uint32_t flags)
/* /*
we have data to copy out we have data to copy out
*/ */
if (uart_drv->half_duplex) {
uint32_t now = AP_HAL::micros();
if (now - uart_drv->hd_write_us < uart_drv->hd_read_delay_us) {
len = 0;
}
}
stm32_cacheBufferInvalidate(uart_drv->rx_bounce_buf[bounce_idx], len); stm32_cacheBufferInvalidate(uart_drv->rx_bounce_buf[bounce_idx], len);
uart_drv->_readbuf.write(uart_drv->rx_bounce_buf[bounce_idx], len); uart_drv->_readbuf.write(uart_drv->rx_bounce_buf[bounce_idx], len);
uart_drv->receive_timestamp_update(); uart_drv->receive_timestamp_update();
@ -798,10 +799,6 @@ void UARTDriver::write_pending_bytes_DMA(uint32_t n)
} }
} }
if (half_duplex) {
half_duplex_setup_delay(tx_len);
}
dmaStreamDisable(txdma); dmaStreamDisable(txdma);
tx_bounce_buf_ready = false; tx_bounce_buf_ready = false;
osalDbgAssert(txdma != nullptr, "UART TX DMA allocation failed"); osalDbgAssert(txdma != nullptr, "UART TX DMA allocation failed");
@ -827,6 +824,10 @@ void UARTDriver::write_pending_bytes_NODMA(uint32_t n)
ByteBuffer::IoVec vec[2]; ByteBuffer::IoVec vec[2];
uint16_t nwritten = 0; uint16_t nwritten = 0;
if (half_duplex) {
half_duplex_setup_tx();
}
const auto n_vec = _writebuf.peekiovec(vec, n); const auto n_vec = _writebuf.peekiovec(vec, n);
for (int i = 0; i < n_vec; i++) { for (int i = 0; i < n_vec; i++) {
int ret = -1; int ret = -1;
@ -856,10 +857,6 @@ void UARTDriver::write_pending_bytes_NODMA(uint32_t n)
} }
_total_written += nwritten; _total_written += nwritten;
if (half_duplex) {
half_duplex_setup_delay(nwritten);
}
} }
/* /*
@ -927,15 +924,24 @@ void UARTDriver::write_pending_bytes(void)
} }
/* /*
setup a delay after writing bytes to a half duplex UART to prevent setup for half duplex tramsmit. To cope with uarts that have level
read-back of the same bytes that we wrote. half-duplex protocols shifters and pullups we need to play a trick where we temporarily
tend to have quite loose timing, which makes this possible disable half-duplex while transmitting. That disables the receive
part of the uart on the pin which allows the transmit side to
correctly setup the idle voltage before the transmit starts.
*/ */
void UARTDriver::half_duplex_setup_delay(uint16_t len) void UARTDriver::half_duplex_setup_tx(void)
{ {
const uint16_t pad_us = 1000; #ifdef HAVE_USB_SERIAL
hd_write_us = AP_HAL::micros(); if (!hd_tx_active) {
hd_read_delay_us = ((1000000UL * len * 10) / _baudrate) + pad_us; chEvtGetAndClearFlags(&hd_listener);
SerialDriver *sd = (SerialDriver*)(sdef.serial);
hd_tx_active = true;
sdStop(sd);
sercfg.cr3 &= ~USART_CR3_HDSEL;
sdStart(sd, &sercfg);
}
#endif
} }
@ -948,6 +954,20 @@ void UARTDriver::_timer_tick(void)
{ {
if (!_initialised) return; if (!_initialised) return;
#ifdef HAVE_USB_SERIAL
if (hd_tx_active && (chEvtGetAndClearFlags(&hd_listener) & CHN_OUTPUT_EMPTY) != 0) {
/*
half-duplex transmit has finished. We now re-enable the
HDSEL bit for receive
*/
SerialDriver *sd = (SerialDriver*)(sdef.serial);
sdStop(sd);
sercfg.cr3 |= USART_CR3_HDSEL;
sdStart(sd, &sercfg);
hd_tx_active = false;
}
#endif
#ifndef HAL_UART_NODMA #ifndef HAL_UART_NODMA
if (rx_dma_enabled && rxdma) { if (rx_dma_enabled && rxdma) {
chSysLock(); chSysLock();
@ -1023,15 +1043,10 @@ void UARTDriver::_timer_tick(void)
ret = -1; ret = -1;
} }
#endif #endif
if (half_duplex) { if (!hd_tx_active) {
uint32_t now = AP_HAL::micros();
if (now - hd_write_us < hd_read_delay_us) {
break;
}
}
_readbuf.commit((unsigned)ret); _readbuf.commit((unsigned)ret);
receive_timestamp_update(); receive_timestamp_update();
}
/* stop reading as we read less than we asked for */ /* stop reading as we read less than we asked for */
if ((unsigned)ret < vec[i].len) { if ((unsigned)ret < vec[i].len) {
@ -1361,7 +1376,21 @@ bool UARTDriver::set_options(uint8_t options)
if (options & OPTION_HDPLEX) { if (options & OPTION_HDPLEX) {
cr3 |= USART_CR3_HDSEL; cr3 |= USART_CR3_HDSEL;
_cr3_options |= USART_CR3_HDSEL; _cr3_options |= USART_CR3_HDSEL;
if (!half_duplex) {
chEvtRegisterMaskWithFlags(chnGetEventSource((SerialDriver*)sdef.serial),
&hd_listener,
EVT_TRANSMIT_END,
CHN_OUTPUT_EMPTY);
half_duplex = true; half_duplex = true;
}
#ifndef HAL_UART_NODMA
if (rx_dma_enabled && rxdma) {
dmaStreamDisable(rxdma);
}
#endif
// force DMA off when using half-duplex as the timing may affect other devices
// sharing the DMA channel
rx_dma_enabled = tx_dma_enabled = false;
} else { } else {
cr3 &= ~USART_CR3_HDSEL; cr3 &= ~USART_CR3_HDSEL;
} }

View File

@ -185,17 +185,20 @@ private:
uint32_t _first_write_started_us; uint32_t _first_write_started_us;
uint32_t _total_written; uint32_t _total_written;
// we remember cr2 and cr2 options from set_options to apply on sdStart() // we remember config options from set_options to apply on sdStart()
uint32_t _cr3_options; uint32_t _cr1_options;
uint32_t _cr2_options; uint32_t _cr2_options;
uint32_t _cr3_options;
uint8_t _last_options; uint8_t _last_options;
// half duplex control. After writing we throw away bytes for 4 byte widths to // half duplex control. After writing we throw away bytes for 4 byte widths to
// prevent reading our own bytes back // prevent reading our own bytes back
#if CH_CFG_USE_EVENTS == TRUE
bool half_duplex; bool half_duplex;
uint32_t hd_read_delay_us; event_listener_t hd_listener;
uint32_t hd_write_us; bool hd_tx_active;
void half_duplex_setup_delay(uint16_t len); void half_duplex_setup_tx(void);
#endif
// set to true for unbuffered writes (low latency writes) // set to true for unbuffered writes (low latency writes)
bool unbuffered_writes; bool unbuffered_writes;