mbed library sources. Supersedes mbed-src.

Fork of mbed-dev by mbed official

Files at this revision

API Documentation at this revision

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

targets/hal/TARGET_STM/TARGET_STM32F4/TARGET_B96B_F446VE/device.h Show annotated file Show diff for this revision Revisions of this file
targets/hal/TARGET_STM/TARGET_STM32F4/TARGET_B96B_F446VE/objects.h Show annotated file Show diff for this revision Revisions of this file
targets/hal/TARGET_STM/TARGET_STM32F4/serial_api.c Show annotated file Show diff for this revision Revisions of this file
--- 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