mbed library sources. Supersedes mbed-src. GR-PEACH runs on RAM.
Fork of mbed-dev by
Revision 55:814265bf5462, committed 2016-01-29
- Comitter:
- mbed_official
- Date:
- Fri Jan 29 13:30:11 2016 +0000
- Parent:
- 54:2a2065e67ef6
- Child:
- 56:05912f50f004
- Commit message:
- Synchronized with git revision bad9c120124289d7176748df4ebcc6b2e1cbfd1e
Full URL: https://github.com/mbedmicro/mbed/commit/bad9c120124289d7176748df4ebcc6b2e1cbfd1e/
B96B async serial
Changed in this revision
--- a/targets/hal/TARGET_STM/TARGET_STM32F4/TARGET_B96B_F446VE/device.h Thu Jan 28 18:15:10 2016 +0000 +++ b/targets/hal/TARGET_STM/TARGET_STM32F4/TARGET_B96B_F446VE/device.h Fri Jan 29 13:30:11 2016 +0000 @@ -40,8 +40,8 @@ #define DEVICE_ANALOGOUT 1 #define DEVICE_SERIAL 1 -#define DEVICE_SERIAL_ASYNCH 0 -#define DEVICE_SERIAL_ASYNCH_DMA 0 +#define DEVICE_SERIAL_ASYNCH 1 +#define DEVICE_SERIAL_ASYNCH_DMA 1 #define DEVICE_SERIAL_FC 0 #define DEVICE_I2C 1
--- a/targets/hal/TARGET_STM/TARGET_STM32F4/TARGET_B96B_F446VE/objects.h Thu Jan 28 18:15:10 2016 +0000
+++ b/targets/hal/TARGET_STM/TARGET_STM32F4/TARGET_B96B_F446VE/objects.h Fri Jan 29 13:30:11 2016 +0000
@@ -74,6 +74,9 @@
uint32_t parity;
PinName pin_tx;
PinName pin_rx;
+#if DEVICE_SERIAL_ASYNCH
+ uint32_t events;
+#endif
};
struct spi_s {
--- a/targets/hal/TARGET_STM/TARGET_STM32F4/serial_api.c Thu Jan 28 18:15:10 2016 +0000
+++ b/targets/hal/TARGET_STM/TARGET_STM32F4/serial_api.c Fri Jan 29 13:30:11 2016 +0000
@@ -39,36 +39,104 @@
#include "mbed_error.h"
#define UART_NUM (8)
+#define UART_STATE_RX_ACTIVE 0x20
+#define UART_STATE_TX_ACTIVE 0x10
static uint32_t serial_irq_ids[UART_NUM] = {0, 0, 0, 0, 0, 0, 0, 0};
+#if DEVICE_SERIAL_ASYNCH_DMA
+static const uint32_t DMA_UartRx_Channel[UART_NUM] = {DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_5, DMA_CHANNEL_5, DMA_CHANNEL_5};
+static const uint32_t DMA_UartRx_Stream[UART_NUM] = {(uint32_t)DMA2_Stream5, (uint32_t) DMA1_Stream5, (uint32_t) DMA1_Stream1, (uint32_t) DMA1_Stream2, (uint32_t) DMA1_Stream0, (uint32_t) DMA2_Stream5, (uint32_t) DMA1_Stream3, (uint32_t) DMA1_Stream6};
+static const uint32_t DMA_UartTx_Channel[UART_NUM] = {DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_4, DMA_CHANNEL_5, DMA_CHANNEL_5, DMA_CHANNEL_5};
+static const uint32_t DMA_UartTx_Stream[UART_NUM] = {(uint32_t)DMA2_Stream7, (uint32_t) DMA1_Stream6, (uint32_t) DMA1_Stream3, (uint32_t) DMA1_Stream4, (uint32_t) DMA1_Stream7, (uint32_t) DMA2_Stream6, (uint32_t) DMA1_Stream1, (uint32_t) DMA1_Stream0};
+
+#endif
static uart_irq_handler irq_handler;
+DMA_HandleTypeDef DmaHandle;
UART_HandleTypeDef UartHandle;
int stdio_uart_inited = 0;
serial_t stdio_uart;
+#if DEVICE_SERIAL_ASYNCH
+#define SERIAL_OBJ(X) (obj->serial.X)
+#else
+#define SERIAL_OBJ(X) (obj->X)
+#endif
+
static void init_uart(serial_t *obj)
{
- UartHandle.Instance = (USART_TypeDef *)(obj->uart);
+#if DEVICE_SERIAL_ASYNCH_DMA
+ static DMA_HandleTypeDef hdma_tx;
+ static DMA_HandleTypeDef hdma_rx;
+#endif
+
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
- UartHandle.Init.BaudRate = obj->baudrate;
- UartHandle.Init.WordLength = obj->databits;
- UartHandle.Init.StopBits = obj->stopbits;
- UartHandle.Init.Parity = obj->parity;
+ UartHandle.Init.BaudRate = SERIAL_OBJ(baudrate);
+ UartHandle.Init.WordLength = SERIAL_OBJ(databits);
+ UartHandle.Init.StopBits = SERIAL_OBJ(stopbits);
+ UartHandle.Init.Parity = SERIAL_OBJ(parity);
UartHandle.Init.HwFlowCtl = UART_HWCONTROL_NONE;
+ UartHandle.Init.OverSampling = UART_OVERSAMPLING_16;
- if (obj->pin_rx == NC) {
+ if (SERIAL_OBJ(pin_rx) == NC) {
UartHandle.Init.Mode = UART_MODE_TX;
- } else if (obj->pin_tx == NC) {
+ } else if (SERIAL_OBJ(pin_tx) == NC) {
UartHandle.Init.Mode = UART_MODE_RX;
} else {
UartHandle.Init.Mode = UART_MODE_TX_RX;
}
+#if DEVICE_SERIAL_ASYNCH_DMA
+ if (SERIAL_OBJ(pin_tx) != NC) {
+ // set DMA in the UartHandle
+ /* Configure the DMA handler for Transmission process */
+ hdma_tx.Instance = (DMA_Stream_TypeDef *)DMA_UartTx_Stream[SERIAL_OBJ(index)];
+ hdma_tx.Init.Channel = DMA_UartTx_Channel[SERIAL_OBJ(index)];
+ hdma_tx.Init.Direction = DMA_MEMORY_TO_PERIPH;
+ hdma_tx.Init.PeriphInc = DMA_PINC_DISABLE;
+ hdma_tx.Init.MemInc = DMA_MINC_ENABLE;
+ hdma_tx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;
+ hdma_tx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;
+ hdma_tx.Init.Mode = DMA_NORMAL;
+ hdma_tx.Init.Priority = DMA_PRIORITY_LOW;
+ hdma_tx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
+ hdma_tx.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
+ hdma_tx.Init.MemBurst = DMA_MBURST_INC4;
+ hdma_tx.Init.PeriphBurst = DMA_PBURST_INC4;
+
+ HAL_DMA_Init(&hdma_tx);
+
+ /* Associate the initialized DMA handle to the UART handle */
+ __HAL_LINKDMA(&UartHandle, hdmatx, hdma_tx);
+ }
+
+ if (SERIAL_OBJ(pin_rx) != NC) {
+ /* Configure the DMA handler for reception process */
+ hdma_rx.Instance = (DMA_Stream_TypeDef *)DMA_UartRx_Stream[SERIAL_OBJ(index)];
+ hdma_rx.Init.Channel = DMA_UartRx_Channel[SERIAL_OBJ(index)];
+ hdma_rx.Init.Direction = DMA_PERIPH_TO_MEMORY;
+ hdma_rx.Init.PeriphInc = DMA_PINC_DISABLE;
+ hdma_rx.Init.MemInc = DMA_MINC_ENABLE;
+ hdma_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;
+ hdma_rx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;
+ hdma_rx.Init.Mode = DMA_NORMAL;
+ hdma_rx.Init.Priority = DMA_PRIORITY_HIGH;
+ hdma_rx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
+ hdma_rx.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
+ hdma_rx.Init.MemBurst = DMA_MBURST_INC4;
+ hdma_rx.Init.PeriphBurst = DMA_PBURST_INC4;
+
+ HAL_DMA_Init(&hdma_rx);
+
+ /* Associate the initialized DMA handle to the UART handle */
+ __HAL_LINKDMA(&UartHandle, hdmarx, hdma_rx);
+ }
+#endif
if (HAL_UART_Init(&UartHandle) != HAL_OK) {
- error("Cannot initialize UART");
+ error("Cannot initialize UART\n");
}
}
@@ -79,53 +147,78 @@
UARTName uart_rx = (UARTName)pinmap_peripheral(rx, PinMap_UART_RX);
// Get the peripheral name (UART_1, UART_2, ...) from the pin and assign it to the object
- obj->uart = (UARTName)pinmap_merge(uart_tx, uart_rx);
- MBED_ASSERT(obj->uart != (UARTName)NC);
+ SERIAL_OBJ(uart) = (UARTName)pinmap_merge(uart_tx, uart_rx);
+
+ MBED_ASSERT(SERIAL_OBJ(uart) != (UARTName)NC);
// Enable USART clock
- switch (obj->uart) {
+ switch (SERIAL_OBJ(uart)) {
case UART_1:
__HAL_RCC_USART1_CLK_ENABLE();
- obj->index = 0;
+ SERIAL_OBJ(index) = 0;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA2_CLK_ENABLE();
+#endif
break;
case UART_2:
__HAL_RCC_USART2_CLK_ENABLE();
- obj->index = 1;
+ SERIAL_OBJ(index) = 1;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA1_CLK_ENABLE();
+#endif
break;
#if defined(USART3_BASE)
case UART_3:
__HAL_RCC_USART3_CLK_ENABLE();
- obj->index = 2;
+ SERIAL_OBJ(index) = 2;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA1_CLK_ENABLE();
+#endif
break;
#endif
#if defined(UART4_BASE)
case UART_4:
__HAL_RCC_UART4_CLK_ENABLE();
- obj->index = 3;
+ SERIAL_OBJ(index) = 3;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA1_CLK_ENABLE();
+#endif
break;
#endif
#if defined(UART5_BASE)
case UART_5:
__HAL_RCC_UART5_CLK_ENABLE();
- obj->index = 4;
+ SERIAL_OBJ(index) = 4;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA1_CLK_ENABLE();
+#endif
break;
#endif
#if defined(USART6_BASE)
case UART_6:
__HAL_RCC_USART6_CLK_ENABLE();
- obj->index = 5;
+ SERIAL_OBJ(index) = 5;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA2_CLK_ENABLE();
+#endif
break;
#endif
#if defined(UART7_BASE)
case UART_7:
__HAL_RCC_UART7_CLK_ENABLE();
- obj->index = 6;
+ SERIAL_OBJ(index) = 6;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA1_CLK_ENABLE();
+#endif
break;
#endif
#if defined(UART8_BASE)
case UART_8:
__HAL_RCC_UART8_CLK_ENABLE();
- obj->index = 7;
+ SERIAL_OBJ(index) = 7;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA1_CLK_ENABLE();
+#endif
break;
#endif
}
@@ -133,6 +226,7 @@
// Configure the UART pins
pinmap_pinout(tx, PinMap_UART_TX);
pinmap_pinout(rx, PinMap_UART_RX);
+
if (tx != NC) {
pin_mode(tx, PullUp);
}
@@ -141,18 +235,18 @@
}
// Configure UART
- obj->baudrate = 9600;
- obj->databits = UART_WORDLENGTH_8B;
- obj->stopbits = UART_STOPBITS_1;
- obj->parity = UART_PARITY_NONE;
+ SERIAL_OBJ(baudrate) = 9600;
+ SERIAL_OBJ(databits) = UART_WORDLENGTH_8B;
+ SERIAL_OBJ(stopbits) = UART_STOPBITS_1;
+ SERIAL_OBJ(parity) = UART_PARITY_NONE;
- obj->pin_tx = tx;
- obj->pin_rx = rx;
+ SERIAL_OBJ(pin_tx) = tx;
+ SERIAL_OBJ(pin_rx) = rx;
init_uart(obj);
// For stdio management
- if (obj->uart == STDIO_UART) {
+ if (SERIAL_OBJ(uart) == STDIO_UART) {
stdio_uart_inited = 1;
memcpy(&stdio_uart, obj, sizeof(serial_t));
}
@@ -161,7 +255,7 @@
void serial_free(serial_t *obj)
{
// Reset UART and disable clock
- switch (obj->uart) {
+ switch (SERIAL_OBJ(uart)) {
case UART_1:
__USART1_FORCE_RESET();
__USART1_RELEASE_RESET();
@@ -184,6 +278,9 @@
__UART4_FORCE_RESET();
__UART4_RELEASE_RESET();
__UART4_CLK_DISABLE();
+#if DEVICE_SERIAL_ASYNCH_DMA
+ __HAL_RCC_DMA1_CLK_DISABLE();
+#endif
break;
#endif
#if defined(UART5_BASE)
@@ -216,44 +313,44 @@
#endif
}
// Configure GPIOs
- pin_function(obj->pin_tx, STM_PIN_DATA(STM_MODE_INPUT, GPIO_NOPULL, 0));
- pin_function(obj->pin_rx, STM_PIN_DATA(STM_MODE_INPUT, GPIO_NOPULL, 0));
+ pin_function(SERIAL_OBJ(pin_tx), STM_PIN_DATA(STM_MODE_INPUT, GPIO_NOPULL, 0));
+ pin_function(SERIAL_OBJ(pin_rx), STM_PIN_DATA(STM_MODE_INPUT, GPIO_NOPULL, 0));
- serial_irq_ids[obj->index] = 0;
+ serial_irq_ids[SERIAL_OBJ(index)] = 0;
}
void serial_baud(serial_t *obj, int baudrate)
{
- obj->baudrate = baudrate;
+ SERIAL_OBJ(baudrate) = baudrate;
init_uart(obj);
}
void serial_format(serial_t *obj, int data_bits, SerialParity parity, int stop_bits)
{
if (data_bits == 9) {
- obj->databits = UART_WORDLENGTH_9B;
+ SERIAL_OBJ(databits) = UART_WORDLENGTH_9B;
} else {
- obj->databits = UART_WORDLENGTH_8B;
+ SERIAL_OBJ(databits) = UART_WORDLENGTH_8B;
}
switch (parity) {
case ParityOdd:
case ParityForced0:
- obj->parity = UART_PARITY_ODD;
+ SERIAL_OBJ(parity) = UART_PARITY_ODD;
break;
case ParityEven:
case ParityForced1:
- obj->parity = UART_PARITY_EVEN;
+ SERIAL_OBJ(parity) = UART_PARITY_EVEN;
break;
default: // ParityNone
- obj->parity = UART_PARITY_NONE;
+ SERIAL_OBJ(parity) = UART_PARITY_NONE;
break;
}
if (stop_bits == 2) {
- obj->stopbits = UART_STOPBITS_2;
+ SERIAL_OBJ(stopbits) = UART_STOPBITS_2;
} else {
- obj->stopbits = UART_STOPBITS_1;
+ SERIAL_OBJ(stopbits) = UART_STOPBITS_1;
}
init_uart(obj);
@@ -277,6 +374,23 @@
}
}
}
+#if DEVICE_SERIAL_ASYNCH_DMA
+static void dma_irq(DMAName name, int id)
+{
+ // TO DO
+ DmaHandle.Instance = (DMA_Stream_TypeDef *)name;
+ if (serial_irq_ids[id] != 0) {
+ if (__HAL_DMA_GET_TC_FLAG_INDEX(&DmaHandle) != RESET) {
+ irq_handler(serial_irq_ids[id], TxIrq);
+ __HAL_DMA_CLEAR_FLAG(&DmaHandle, DMA_FLAG_TCIF0_4);
+ }
+ if (__HAL_DMA_GET_TC_FLAG_INDEX(&DmaHandle) != RESET) {
+ irq_handler(serial_irq_ids[id], RxIrq);
+ __HAL_DMA_CLEAR_FLAG(&DmaHandle, DMA_FLAG_TCIF2_6);
+ }
+ }
+}
+#endif
static void uart1_irq(void)
{
@@ -300,6 +414,19 @@
{
uart_irq(UART_4, 3);
}
+#if DEVICE_SERIAL_ASYNCH_DMA
+
+static void dma1_stream2_irq(void)
+{
+ dma_irq(DMA_1, 3 /* TO DO : ??? WHAT IS THIS 3 ??? */);
+}
+
+
+static void dma1_stream4_irq(void)
+{
+ dma_irq(DMA_1, 3 /* TO DO : ??? WHAT IS THIS 3 ??? */);
+}
+#endif
#endif
#if defined(UART5_BASE)
@@ -333,17 +460,21 @@
void serial_irq_handler(serial_t *obj, uart_irq_handler handler, uint32_t id)
{
irq_handler = handler;
- serial_irq_ids[obj->index] = id;
+ serial_irq_ids[SERIAL_OBJ(index)] = id;
}
void serial_irq_set(serial_t *obj, SerialIrq irq, uint32_t enable)
{
IRQn_Type irq_n = (IRQn_Type)0;
uint32_t vector = 0;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ IRQn_Type irqn_dma = (IRQn_Type)0;
+ uint32_t vector_dma = 0;
+#endif
- UartHandle.Instance = (USART_TypeDef *)(obj->uart);
+ UartHandle.Instance = (USART_TypeDef *)SERIAL_OBJ(uart);
- switch (obj->uart) {
+ switch (SERIAL_OBJ(uart)) {
case UART_1:
irq_n = USART1_IRQn;
vector = (uint32_t)&uart1_irq;
@@ -363,6 +494,15 @@
case UART_4:
irq_n = UART4_IRQn;
vector = (uint32_t)&uart4_irq;
+#if DEVICE_SERIAL_ASYNCH_DMA
+ if (irq == RxIrq) {
+ irqn_dma = DMA1_Stream2_IRQn;
+ vector_dma = (uint32_t)&dma1_stream2_irq;
+ } else {
+ irqn_dma = DMA1_Stream4_IRQn;
+ vector_dma = (uint32_t)&dma1_stream4_irq;
+ }
+#endif
break;
#endif
#if defined(UART5_BASE)
@@ -395,13 +535,24 @@
if (irq == RxIrq) {
__HAL_UART_ENABLE_IT(&UartHandle, UART_IT_RXNE);
+#if DEVICE_SERIAL_ASYNCH_DMA
+ NVIC_SetVector(irq_n, vector_dma);
+ NVIC_EnableIRQ(irq_n);
+ NVIC_SetVector(irqn_dma, vector_dma);
+ NVIC_EnableIRQ(irqn_dma);
+#else
+ NVIC_SetVector(irq_n, vector);
+ NVIC_EnableIRQ(irq_n);
+#endif
} else { // TxIrq
__HAL_UART_ENABLE_IT(&UartHandle, UART_IT_TC);
+ NVIC_SetVector(irq_n, vector);
+ NVIC_EnableIRQ(irq_n);
+#if DEVICE_SERIAL_ASYNCH_DMA
+ NVIC_SetVector(irqn_dma, vector_dma);
+ NVIC_EnableIRQ(irqn_dma);
+#endif
}
-
- NVIC_SetVector(irq_n, vector);
- NVIC_EnableIRQ(irq_n);
-
} else { // disable
int all_disabled = 0;
@@ -416,7 +567,12 @@
if ((UartHandle.Instance->CR1 & USART_CR1_RXNEIE) == 0) all_disabled = 1;
}
- if (all_disabled) NVIC_DisableIRQ(irq_n);
+ if (all_disabled) {
+ NVIC_DisableIRQ(irq_n);
+#if DEVICE_SERIAL_ASYNCH_DMA
+ NVIC_DisableIRQ(irqn_dma);
+#endif
+ }
}
}
@@ -427,14 +583,14 @@
int serial_getc(serial_t *obj)
{
- USART_TypeDef *uart = (USART_TypeDef *)(obj->uart);
+ USART_TypeDef *uart = (USART_TypeDef *)(SERIAL_OBJ(uart));
while (!serial_readable(obj));
return (int)(uart->DR & 0x1FF);
}
void serial_putc(serial_t *obj, int c)
{
- USART_TypeDef *uart = (USART_TypeDef *)(obj->uart);
+ USART_TypeDef *uart = (USART_TypeDef *)(SERIAL_OBJ(uart));
while (!serial_writable(obj));
uart->DR = (uint32_t)(c & 0x1FF);
}
@@ -442,7 +598,7 @@
int serial_readable(serial_t *obj)
{
int status;
- UartHandle.Instance = (USART_TypeDef *)(obj->uart);
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
// Check if data is received
status = ((__HAL_UART_GET_FLAG(&UartHandle, UART_FLAG_RXNE) != RESET) ? 1 : 0);
return status;
@@ -451,7 +607,7 @@
int serial_writable(serial_t *obj)
{
int status;
- UartHandle.Instance = (USART_TypeDef *)(obj->uart);
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
// Check if data is transmitted
status = ((__HAL_UART_GET_FLAG(&UartHandle, UART_FLAG_TXE) != RESET) ? 1 : 0);
return status;
@@ -459,7 +615,7 @@
void serial_clear(serial_t *obj)
{
- UartHandle.Instance = (USART_TypeDef *)(obj->uart);
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
__HAL_UART_CLEAR_FLAG(&UartHandle, UART_FLAG_TXE);
__HAL_UART_CLEAR_FLAG(&UartHandle, UART_FLAG_RXNE);
}
@@ -471,7 +627,7 @@
void serial_break_set(serial_t *obj)
{
- UartHandle.Instance = (USART_TypeDef *)(obj->uart);
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
HAL_LIN_SendBreak(&UartHandle);
}
@@ -479,4 +635,546 @@
{
}
+//########################################################################################
+
+#if DEVICE_SERIAL_ASYNCH
+
+//----------------------------------------------------------------------------------------
+// LOCAL HELPER FUNCTIONS
+//----------------------------------------------------------------------------------------
+
+/** Configure the TX buffer for an asynchronous write serial transaction
+ *
+ * @param obj The serial object.
+ * @param tx The buffer for sending.
+ * @param tx_length The number of words to transmit.
+ */
+static void h_serial_tx_buffer_set(serial_t *obj, void *tx, int tx_length, uint8_t width)
+{
+ // We only support byte buffers for now
+ MBED_ASSERT(width == 8);
+ UartHandle.Instance = (USART_TypeDef *)SERIAL_OBJ(uart);
+
+ // Exit if a transmit is already on-going
+ if (serial_tx_active(obj)) return;
+
+ obj->tx_buff.buffer = tx;
+ obj->tx_buff.length = tx_length;
+ obj->tx_buff.pos = 0;
+
+ return;
+}
+
+/** Configure the RX buffer for an asynchronous write serial transaction
+ *
+ * @param obj The serial object.
+ * @param tx The buffer for sending.
+ * @param tx_length The number of words to transmit.
+ */
+static void h_serial_rx_buffer_set(serial_t *obj, void *rx, int rx_length, uint8_t width)
+{
+ /* Sanity check arguments */
+ MBED_ASSERT(obj);
+ MBED_ASSERT(rx != (void*)0);
+ // We only support byte buffers for now
+ MBED_ASSERT(width == 8);
+
+ // Exit if a reception is already on-going
+ if (serial_rx_active(obj)) return;
+
+ obj->rx_buff.buffer = rx;
+ obj->rx_buff.length = rx_length;
+ obj->rx_buff.pos = 0;
+
+ return;
+}
+
+/** Configure TX events
+ *
+ * @param obj The serial object
+ * @param event The logical OR of the TX events to configure
+ * @param enable Set to non-zero to enable events, or zero to disable them
+ */
+static void h_serial_tx_enable_event(serial_t *obj, int event, uint8_t enable)
+{
+ // Shouldn't have to enable TX interrupt here, just need to keep track of the requested events.
+ if (enable) SERIAL_OBJ(events) |= event;
+ else SERIAL_OBJ(events) &= ~event;
+}
+
+/** Configure RX events
+ *
+ * @param obj The serial object
+ * @param event The logical OR of the RX events to configure
+ * @param enable Set to non-zero to enable events, or zero to disable them
+ */
+static void h_serial_rx_enable_event(serial_t *obj, int event, uint8_t enable)
+{
+ // Shouldn't have to enable RX interrupt here, just need to keep track of the requested events.
+ if (enable) SERIAL_OBJ(events) |= event;
+ else SERIAL_OBJ(events) &= ~event;
+}
+
+/**
+* Get index of serial object TX IRQ, relating it to the physical peripheral.
+*
+* @param obj pointer to serial object
+* @return internal NVIC TX IRQ index of U(S)ART peripheral
+*/
+static IRQn_Type h_serial_get_irq_index(serial_t *obj)
+{
+ IRQn_Type irq_n = (IRQn_Type)0;
+
+ UartHandle.Instance = (USART_TypeDef *)SERIAL_OBJ(uart);
+
+ switch (SERIAL_OBJ(uart)) {
+#if defined(USART1_BASE)
+ case UART_1:
+ irq_n = USART1_IRQn;
+ break;
#endif
+#if defined(USART2_BASE)
+ case UART_2:
+ irq_n = USART2_IRQn;
+ break;
+#endif
+#if defined(USART3_BASE)
+ case UART_3:
+ irq_n = USART3_IRQn;
+ break;
+#endif
+#if defined(UART4_BASE)
+ case UART_4:
+ irq_n = UART4_IRQn;
+ break;
+#endif
+#if defined(UART5_BASE)
+ case UART_5:
+ irq_n = UART5_IRQn;
+ break;
+#endif
+#if defined(USART6_BASE)
+ case UART_6:
+ irq_n = USART6_IRQn;
+ break;
+#endif
+#if defined(UART7_BASE)
+ case UART_7:
+ irq_n = UART7_IRQn;
+ break;
+#endif
+#if defined(UART8_BASE)
+ case UART_8:
+ irq_n = UART8_IRQn;
+ break;
+#endif
+ default:
+ irq_n = (IRQn_Type)0;
+ }
+
+ return irq_n;
+}
+
+#if DEVICE_SERIAL_ASYNCH_DMA
+
+/** The asynchronous TX and RX handler.
+ *
+ * @param obj The serial object
+ * @return Returns event flags if a TX/RX transfer termination condition was met or 0 otherwise
+ */
+static void h_serial_txdma_irq_handler_asynch()
+{
+ HAL_DMA_IRQHandler(UartHandle.hdmatx);
+}
+/** The asynchronous TX and RX handler.
+ *
+ * @param obj The serial object
+ * @return Returns event flags if a TX/RX transfer termination condition was met or 0 otherwise
+ */
+void h_serial_rxdma_irq_handler_asynch(serial_t *obj)
+{
+// UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
+ HAL_DMA_IRQHandler(UartHandle.hdmarx);
+}
+
+/**
+* Get index of serial object TX DMA IRQ, relating it to the physical peripheral.
+*
+* @param obj pointer to serial object
+* @return internal NVIC TX DMA IRQ index of U(S)ART peripheral
+*/
+static IRQn_Type h_serial_tx_get_irqdma_index(serial_t *obj)
+{
+ IRQn_Type irq_n = (IRQn_Type)0;
+
+ UartHandle.Instance = (USART_TypeDef *)SERIAL_OBJ(uart);
+
+ switch (SERIAL_OBJ(uart)) {
+#if defined(USART1_BASE)
+ case UART_1:
+ irq_n = DMA2_Stream7_IRQn;
+ break;
+#endif
+#if defined(USART2_BASE)
+ case UART_2:
+ irq_n = DMA1_Stream6_IRQn;
+ break;
+#endif
+#if defined(USART3_BASE)
+ case UART_3:
+ irq_n = DMA1_Stream3_IRQn;
+ break;
+#endif
+#if defined(UART4_BASE)
+ case UART_4:
+ irq_n = DMA1_Stream4_IRQn;
+ break;
+#endif
+#if defined(UART5_BASE)
+ case UART_5:
+ irq_n = DMA1_Stream7_IRQn;
+ break;
+#endif
+#if defined(USART6_BASE)
+ case UART_6:
+ irq_n = DMA2_Stream6_IRQn;
+ break;
+#endif
+#if defined(UART7_BASE)
+ case UART_7:
+ irq_n = DMA1_Stream1_IRQn;
+ break;
+#endif
+#if defined(UART8_BASE)
+ case UART_8:
+ irq_n = DMA1_Stream0_IRQn;
+ break;
+#endif
+ default:
+ irq_n = (IRQn_Type)0;
+ }
+
+ return irq_n;
+}
+/**
+* Get index of serial object RX DMA IRQ, relating it to the physical peripheral.
+*
+* @param obj pointer to serial object
+* @return internal NVIC RX DMA IRQ index of U(S)ART peripheral
+*/
+static IRQn_Type h_serial_rx_get_irqdma_index(serial_t *obj)
+{
+ IRQn_Type irq_n = (IRQn_Type)0;
+
+ UartHandle.Instance = (USART_TypeDef *)SERIAL_OBJ(uart);
+
+ switch (SERIAL_OBJ(uart)) {
+#if defined(USART1_BASE)
+ case UART_1:
+ irq_n = DMA2_Stream5_IRQn;
+ break;
+#endif
+#if defined(USART2_BASE)
+ case UART_2:
+ irq_n = DMA1_Stream5_IRQn;
+ break;
+#endif
+#if defined(USART3_BASE)
+ case UART_3:
+ irq_n = DMA1_Stream1_IRQn;
+ break;
+#endif
+#if defined(UART4_BASE)
+ case UART_4:
+ irq_n = DMA1_Stream2_IRQn;
+ break;
+#endif
+#if defined(UART5_BASE)
+ case UART_5:
+ irq_n = DMA1_Stream0_IRQn;
+ break;
+#endif
+#if defined(USART6_BASE)
+ case UART_6:
+ irq_n = DMA2_Stream2_IRQn;
+ break;
+#endif
+#if defined(UART7_BASE)
+ case UART_7:
+ irq_n = DMA1_Stream3_IRQn;
+ break;
+#endif
+#if defined(UART8_BASE)
+ case UART_8:
+ irq_n = DMA1_Stream6_IRQn;
+ break;
+#endif
+ default:
+ irq_n = (IRQn_Type)0;
+ }
+
+ return irq_n;
+}
+#endif
+//----------------------------------------------------------------------------------------
+// MBED API FUNCTIONS
+//----------------------------------------------------------------------------------------
+
+/** Begin asynchronous TX transfer. The used buffer is specified in the serial object,
+ * tx_buff
+ *
+ * @param obj The serial object
+ * @param tx The buffer for sending
+ * @param tx_length The number of words to transmit
+ * @param tx_width The bit width of buffer word
+ * @param handler The serial handler
+ * @param event The logical OR of events to be registered
+ * @param hint A suggestion for how to use DMA with this transfer
+ * @return Returns number of data transfered, or 0 otherwise
+ */
+int serial_tx_asynch(serial_t *obj, const void *tx, size_t tx_length, uint8_t tx_width, uint32_t handler, uint32_t event, DMAUsage hint)
+{
+
+ // Check buffer is ok
+ MBED_ASSERT(tx != (void*)0);
+ MBED_ASSERT(tx_width == 8); // support only 8b width
+
+ if (tx_length == 0) return 0;
+
+ // Set up buffer
+ h_serial_tx_buffer_set(obj, (void *)tx, tx_length, tx_width);
+
+ // Set up events
+ h_serial_tx_enable_event(obj, SERIAL_EVENT_TX_ALL, 0); // Clear all events
+ h_serial_tx_enable_event(obj, event, 1); // Set only the wanted events
+
+ // Enable interrupt
+ IRQn_Type irqn = h_serial_get_irq_index(obj);
+ NVIC_ClearPendingIRQ(irqn);
+ NVIC_DisableIRQ(irqn);
+ NVIC_SetPriority(irqn, 1);
+ NVIC_SetVector(irqn, (uint32_t)handler);
+ UartHandle.Instance = (USART_TypeDef *)SERIAL_OBJ(uart);
+ NVIC_EnableIRQ(irqn);
+
+#if DEVICE_SERIAL_ASYNCH_DMA
+ // Enable DMA interrupt
+ irqn = h_serial_tx_get_irqdma_index(obj);
+ NVIC_ClearPendingIRQ(irqn);
+ NVIC_DisableIRQ(irqn);
+ NVIC_SetPriority(irqn, 1);
+// NVIC_SetVector(irqn, (uint32_t)&h_serial_txdma_irq_handler_asynch);
+ NVIC_SetVector(irqn, (uint32_t)handler);
+ NVIC_EnableIRQ(irqn);
+
+ // the following function will enable program and enable the DMA transfer
+ if (HAL_UART_Transmit_DMA(&UartHandle, (uint8_t*)tx, tx_length) != HAL_OK)
+ {
+ /* Transfer error in transmission process */
+ return 0;
+ }
+#else
+ // the following function will enable UART_IT_TXE and error interrupts
+ if (HAL_UART_Transmit_IT(&UartHandle, (uint8_t*)tx, tx_length) != HAL_OK)
+ {
+ /* Transfer error in transmission process */
+ return 0;
+ }
+#endif
+
+ return tx_length;
+}
+
+/** Begin asynchronous RX transfer (enable interrupt for data collecting)
+ * The used buffer is specified in the serial object - rx_buff
+ *
+ * @param obj The serial object
+ * @param rx The buffer for sending
+ * @param rx_length The number of words to transmit
+ * @param rx_width The bit width of buffer word
+ * @param handler The serial handler
+ * @param event The logical OR of events to be registered
+ * @param handler The serial handler
+ * @param char_match A character in range 0-254 to be matched
+ * @param hint A suggestion for how to use DMA with this transfer
+ */
+void serial_rx_asynch(serial_t *obj, void *rx, size_t rx_length, uint8_t rx_width, uint32_t handler, uint32_t event, uint8_t char_match, DMAUsage hint)
+{
+ /* Sanity check arguments */
+ MBED_ASSERT(obj);
+ MBED_ASSERT(rx != (void*)0);
+ MBED_ASSERT(rx_width == 8); // support only 8b width
+
+ h_serial_rx_enable_event(obj, SERIAL_EVENT_RX_ALL, 0);
+ h_serial_rx_enable_event(obj, event, 1);
+ // set CharMatch
+ if (char_match != SERIAL_RESERVED_CHAR_MATCH) {
+ obj->char_match = char_match;
+ }
+ h_serial_rx_buffer_set(obj, rx, rx_length, rx_width);
+
+ IRQn_Type irqn = h_serial_get_irq_index(obj);
+ NVIC_ClearPendingIRQ(irqn);
+ NVIC_DisableIRQ(irqn);
+ NVIC_SetPriority(irqn, 0);
+ NVIC_SetVector(irqn, (uint32_t)handler);
+ NVIC_EnableIRQ(irqn);
+
+ UartHandle.Instance = (USART_TypeDef *)SERIAL_OBJ(uart);
+ // flush current data + error flags
+ __HAL_UART_CLEAR_PEFLAG(&UartHandle);
+#if DEVICE_SERIAL_ASYNCH_DMA
+ // Enable DMA interrupt
+ irqn = h_serial_rx_get_irqdma_index(obj);
+ NVIC_ClearPendingIRQ(irqn);
+ NVIC_DisableIRQ(irqn);
+ NVIC_SetPriority(irqn, 1);
+ NVIC_SetVector(irqn, (uint32_t)handler);
+
+ NVIC_EnableIRQ(irqn);
+ // following HAL function will program and enable the DMA transfer
+ HAL_UART_Receive_DMA(&UartHandle, (uint8_t*)rx, rx_length);
+#else
+ // following HAL function will enable the RXNE interrupt + error interrupts
+ HAL_UART_Receive_IT(&UartHandle, (uint8_t*)rx, rx_length);
+#endif
+ /* Enable the UART Error Interrupt: (Frame error, noise error, overrun error) */
+ __HAL_UART_ENABLE_IT(&UartHandle, UART_IT_ERR);
+
+ return;
+}
+
+/** Attempts to determine if the serial peripheral is already in use for TX
+ *
+ * @param obj The serial object
+ * @return Non-zero if the TX transaction is ongoing, 0 otherwise
+ */
+uint8_t serial_tx_active(serial_t *obj)
+{
+ MBED_ASSERT(obj);
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
+ return ((HAL_UART_GetState(&UartHandle) & UART_STATE_TX_ACTIVE) ? 1 : 0);
+}
+
+/** Attempts to determine if the serial peripheral is already in use for RX
+ *
+ * @param obj The serial object
+ * @return Non-zero if the RX transaction is ongoing, 0 otherwise
+ */
+uint8_t serial_rx_active(serial_t *obj)
+{
+ MBED_ASSERT(obj);
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
+ return ((HAL_UART_GetState(&UartHandle) & UART_STATE_RX_ACTIVE) ? 1 : 0);
+
+}
+
+/** The asynchronous TX and RX handler.
+ *
+ * @param obj The serial object
+ * @return Returns event flags if a TX/RX transfer termination condition was met or 0 otherwise
+ */
+int serial_irq_handler_asynch(serial_t *obj)
+{
+ volatile int return_event = 0;
+ uint8_t *buf = (uint8_t*)obj->rx_buff.buffer;
+ uint8_t i = 0;
+
+ // Irq handler is common to Tx and Rx
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
+#if DEVICE_SERIAL_ASYNCH_DMA
+ if ((UartHandle.Instance->CR3 & USART_CR3_DMAT) !=0) {
+ // call dma tx interrupt
+ HAL_DMA_IRQHandler(UartHandle.hdmatx);
+ }
+ if ((UartHandle.Instance->CR3 & USART_CR3_DMAR) !=0) {
+ // call dma rx interrupt
+ HAL_DMA_IRQHandler(UartHandle.hdmarx);
+ }
+#endif
+ HAL_UART_IRQHandler(&UartHandle);
+ // TX PART:
+ if (__HAL_UART_GET_FLAG(&UartHandle, UART_FLAG_TC) != RESET) {
+ __HAL_UART_CLEAR_FLAG(&UartHandle, UART_FLAG_TC);
+ // return event SERIAL_EVENT_TX_COMPLETE if requested
+ if ((SERIAL_OBJ(events) & SERIAL_EVENT_TX_COMPLETE ) != 0){
+ return_event |= SERIAL_EVENT_TX_COMPLETE & obj->serial.events;
+ }
+ }
+ // handle error events:
+ if (__HAL_UART_GET_FLAG(&UartHandle, HAL_UART_ERROR_PE)) {
+ __HAL_UART_CLEAR_FLAG(&UartHandle, HAL_UART_ERROR_PE);
+ return_event |= SERIAL_EVENT_RX_PARITY_ERROR & obj->serial.events;
+ }
+ if (__HAL_UART_GET_FLAG(&UartHandle, HAL_UART_ERROR_NE)||(UartHandle.ErrorCode & HAL_UART_ERROR_NE)!=0) {
+ __HAL_UART_CLEAR_FLAG(&UartHandle, HAL_UART_ERROR_NE);
+ // not supported by mbed
+ }
+ if (__HAL_UART_GET_FLAG(&UartHandle, HAL_UART_ERROR_FE)||(UartHandle.ErrorCode & HAL_UART_ERROR_FE)!=0) {
+ __HAL_UART_CLEAR_FLAG(&UartHandle, HAL_UART_ERROR_FE);
+ return_event |= SERIAL_EVENT_RX_FRAMING_ERROR & obj->serial.events;
+ }
+ if (__HAL_UART_GET_FLAG(&UartHandle, HAL_UART_ERROR_ORE)||(UartHandle.ErrorCode & HAL_UART_ERROR_ORE)!=0) {
+ __HAL_UART_CLEAR_FLAG(&UartHandle, HAL_UART_ERROR_ORE);
+ return_event |= SERIAL_EVENT_RX_OVERRUN_ERROR & obj->serial.events;
+ }
+
+ //RX PART
+ // increment rx_buff.pos
+ if (UartHandle.RxXferSize !=0) {
+ obj->rx_buff.pos = UartHandle.RxXferSize - UartHandle.RxXferCount;
+ }
+ if ((UartHandle.RxXferCount==0)&&(obj->rx_buff.pos >= (obj->rx_buff.length - 1))) {
+ return_event |= SERIAL_EVENT_RX_COMPLETE & obj->serial.events;
+ }
+ // Chek if Char_match is present
+ if (SERIAL_OBJ(events) & SERIAL_EVENT_RX_CHARACTER_MATCH) {
+ if (buf != NULL){
+ while((buf[i] != obj->char_match)&&(i<UartHandle.RxXferSize)){//for (i=0;i<UartHandle.RxXferSize;i++){
+ i++;//if (buf[i] == obj->char_match{
+ //}
+ }
+ if (i<UartHandle.RxXferSize){
+ obj->rx_buff.pos = i;
+ return_event |= SERIAL_EVENT_RX_CHARACTER_MATCH & obj->serial.events;
+ }
+ }
+ }
+ return return_event;
+}
+
+/** Abort the ongoing TX transaction. It disables the enabled interupt for TX and
+ * flush TX hardware buffer if TX FIFO is used
+ *
+ * @param obj The serial object
+ */
+void serial_tx_abort_asynch(serial_t *obj)
+{
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
+ __HAL_UART_DISABLE_IT(&UartHandle, UART_IT_TC|UART_IT_TXE);
+ UartHandle.Instance = 0;
+
+ obj->tx_buff.buffer = 0;
+ obj->tx_buff.length = 0;
+
+}
+
+/** Abort the ongoing RX transaction It disables the enabled interrupt for RX and
+ * flush RX hardware buffer if RX FIFO is used
+ *
+ * @param obj The serial object
+ */
+void serial_rx_abort_asynch(serial_t *obj)
+{
+ UartHandle.Instance = (USART_TypeDef *)(SERIAL_OBJ(uart));
+ __HAL_UART_DISABLE_IT(&UartHandle, UART_IT_RXNE);
+ UartHandle.Instance = 0;
+
+ obj->rx_buff.buffer = 0;
+ obj->rx_buff.length = 0;
+
+}
+
+#endif
+
+#endif
