本文主要記錄UART DMA操作方式,同時對STM32F103 UART驅動抽象出來實現幀數據接收
1、MDK工程目錄(創建工程方式略)

main.c內容如下
運行后的結果是UART收到數據立即通過TX發送出去,同時LED狀態反轉一次
#include <stdio.h> #include "stm32f10x.h" #include "platform.h" unsigned char led_count = 0; void LED_Init(void){ GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure); } void uart_recv_str(UART_TypeDef *p, uint8_t *pbuf, uint16_t len){ UART_ApiStructure.send_str(p, pbuf, len); GPIO_WriteBit(GPIOB, GPIO_Pin_9, (++led_count & 0x01) ? Bit_SET : Bit_RESET); } int main(void){ LED_Init(); UART_ApiStructure.config(&UART_TypeDef_param, uart_recv_str); UART_ApiStructure.send_str(&UART_TypeDef_param, (uint8_t *)"uart init ok\r\n", 14); for(;;){} }
uart.h
#ifndef __UART_H #define __UART_H typedef void (*pfunc_uart_gpio_init)(void); /** * @addtogroup 串口參數定義 * @{ */ /** UART 數據處理設置 */ typedef struct{ uint8_t *tx_buf; /*!< 發送數據緩存 */ uint8_t *rx_buf; /*!< 接收數據緩存 */ uint16_t tx_buf_len_max; /*!< 發送數據最大緩存空間大小 */ uint16_t rx_buf_len_max; /*!< 接收數據最大緩存空間大小 */ uint16_t rx_tail; /*!< 接收數據緩存尾標識 */ uint16_t rx_head; /*!< 接收數據緩存頭標識 */ }UART_UserParam; /** DMA 參數設置 */ typedef struct{ uint8_t irq_tx; /*!< 發送通道中斷號 */ uint8_t irq_rx; /*!< 接收通道中斷號 */ DMA_Channel_TypeDef *channel_tx; /*!< 發送通道 */ DMA_Channel_TypeDef *channel_rx; /*!< 接收通道 */ uint32_t TX_IT_FLAG_TCx; /*!< 發送完成中斷標志 */ uint32_t RX_IT_FLAG_TCx; /*!< 接收完成中斷標志 */ }DMA_Param; /** UART 參數設置 */ typedef struct{ uint32_t baud; /*!< 波特率參數 */ pfunc_uart_gpio_init pfunc_gpio_init; /*!< 指向串口GPIO初始化函數指針 */ USART_TypeDef* USARTx; /*!< 使用串口號 */ uint8_t USARTx_IRQn; /*!< 串口接收中斷號 */ DMA_Param *DMAx; /*!< DMA參數指針 */ UART_UserParam *user; /*!< UART數據處理指針 */ }UART_TypeDef; /** * @} */ /** * @addtogroup API 接口定義 */ typedef void (*pfunc_uart_rsend_str)(UART_TypeDef *p, uint8_t *pbuf, uint16_t len); typedef uint8_t (*pfunc_uart_config)(UART_TypeDef *p, pfunc_uart_rsend_str cb); typedef struct{ pfunc_uart_config config; /*!< 初始化串口 */ pfunc_uart_rsend_str send_str; /*!< 發送數據 */ pfunc_uart_rsend_str recv_cb; /*!< 接收數據 */ }UART_ApiDef; extern UART_ApiDef UART_ApiStructure; /** * @} */ #ifdef USE_PRINTF extern int __printf(const char *fmt, ...); #else #define __printf(...) #endif #endif
uart.c
/** * @file uart.c USART 驅動程序 * * @author T0213-ZH * @date 2018.06.13 */ #include "platform.h" /** * @brief 配置USART參數 * @param p: 指向串口參數指針 * @param cb: 用於接收數據回調函數指針 * * @retval 0-成功,1-已初始過了,2-未設置GPIO初始函數, 3-串口組參數不匹配 */ static uint8_t uart_init(UART_TypeDef *p, pfunc_uart_rsend_str cb){ USART_InitTypeDef USART_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; DMA_InitTypeDef DMA_InitStructure; //if(g_uart_param) return 1; if(!p->pfunc_gpio_init) return 2; if(cb) UART_ApiStructure.recv_cb = cb; //g_uart_param = p; p->pfunc_gpio_init(); if(p->USARTx == USART1){ RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); }else if(p->USARTx == USART2){ RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE); }else if(p->USARTx == USART3){ RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE); }else{ return 3; } USART_InitStructure.USART_BaudRate = p->baud; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(p->USARTx, &USART_InitStructure); USART_Cmd(p->USARTx, ENABLE); USART_ClearFlag(p->USARTx, USART_FLAG_TC); //注:避免首字符丟掉現象 USART_ITConfig(p->USARTx, USART_IT_IDLE, ENABLE); NVIC_InitStructure.NVIC_IRQChannel = p->USARTx_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); USART_DMACmd(p->USARTx, USART_DMAReq_Tx, ENABLE); USART_DMACmd(p->USARTx, USART_DMAReq_Rx, ENABLE); RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //DMA TX Config DMA_DeInit(p->DMAx->channel_tx); DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&p->USARTx->DR); DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)p->user->tx_buf; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST; DMA_InitStructure.DMA_BufferSize = p->user->tx_buf_len_max; DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; DMA_Init(p->DMAx->channel_tx, &DMA_InitStructure); //DMA_Cmd(p->DMAx->channel_tx, ENABLE) NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); NVIC_InitStructure.NVIC_IRQChannel = p->DMAx->irq_tx; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); //DMA RX Config DMA_DeInit(p->DMAx->channel_rx); DMA_InitStructure.DMA_BufferSize = p->user->rx_buf_len_max;; DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)p->user->rx_buf; DMA_Init(p->DMAx->channel_rx, &DMA_InitStructure); DMA_Cmd(p->DMAx->channel_rx, ENABLE); NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); NVIC_InitStructure.NVIC_IRQChannel = p->DMAx->irq_rx; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); DMA_ITConfig(p->DMAx->channel_tx, DMA_IT_TC, ENABLE); DMA_ITConfig(p->DMAx->channel_rx, DMA_IT_TC, ENABLE); return 0; } /** * @brief 發送數據接口 * @param p: 指向發送數據緩存指針 * @param len: 數據長度 * * @retval none */ static void uart_send_str(UART_TypeDef *p, uint8_t *pbuf, uint16_t len){
if(!len) return;
uint16_t i; for(i=0; i<len; i++){ p->user->tx_buf[i] = *pbuf++; } while(DMA_GetFlagStatus(p->DMAx->TX_IT_FLAG_TCx) == SET); DMA_SetCurrDataCounter(p->DMAx->channel_tx, len); DMA_Cmd(p->DMAx->channel_tx, ENABLE); } /** * @brief 接收數據接口,中斷觸發 * * @retval none */ static void uart_recv_len(UART_TypeDef *p){ p->user->rx_head = p->user->rx_buf_len_max - DMA_GetCurrDataCounter(p->DMAx->channel_rx); if(p->user->rx_tail != p->user->rx_head){ uint16_t i = 0; uint8_t buf[p->user->rx_buf_len_max]; uint16_t head = p->user->rx_head; do{ buf[i++] = p->user->rx_buf[p->user->rx_tail++]; if(p->user->rx_tail == p->user->rx_buf_len_max) p->user->rx_tail = 0; }while(p->user->rx_tail != head); if(UART_ApiStructure.recv_cb){ UART_ApiStructure.recv_cb(p, buf, i); } //uart_send_str(buf, i); } } /** * @brief UART 接口函數 */ UART_ApiDef UART_ApiStructure = { .config = uart_init, /*!< 指向串口參數配置接口函數指針 */ .send_str = uart_send_str, /*!< 指向串口發送字符串接口函數指針 */ .recv_cb = 0, }; void uart_tx_irq(UART_TypeDef *p){ if(DMA_GetITStatus(p->DMAx->TX_IT_FLAG_TCx) == SET){ DMA_Cmd(p->DMAx->channel_tx, DISABLE); DMA_ClearITPendingBit(p->DMAx->TX_IT_FLAG_TCx); } } void uart_rx_irq(UART_TypeDef *p){ if(DMA_GetITStatus(p->DMAx->RX_IT_FLAG_TCx) == SET){ DMA_ClearITPendingBit(p->DMAx->RX_IT_FLAG_TCx); } } void uart_irq(UART_TypeDef *p){ if(USART_GetITStatus(p->USARTx, USART_IT_IDLE) != RESET){ USART_ReceiveData(p->USARTx); USART_ClearITPendingBit(p->USARTx, USART_IT_IDLE); uart_recv_len(p); } } #ifdef USE_PRINTF int __fputc(unsigned char ch, unsigned char *f){ /* Place your implementation of fputc here */ /* e.g. write a character to the USART */ USART_SendData(USART1, (uint8_t) ch); /* Loop until the end of transmission */ while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET) {} return ch; } #endif
platform.c
#include "platform.h" extern void uart_tx_irq(UART_TypeDef *p); extern void uart_rx_irq(UART_TypeDef *p); extern void uart_irq(UART_TypeDef *p); void USART_GpioInit(void){ GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); } /** * @addtogroup * @{ */ #define UART_TX_BUF_MAX (256) #define UART_RX_BUF_MAX (256) uint8_t uart_tx_buf[UART_TX_BUF_MAX]; uint8_t uart_rx_buf[UART_RX_BUF_MAX]; DMA_Param dma_param = { .irq_tx = DMA1_Channel4_IRQn, .irq_rx = DMA1_Channel5_IRQn, .channel_tx = DMA1_Channel4, .channel_rx = DMA1_Channel5, .TX_IT_FLAG_TCx = DMA1_IT_TC4, .RX_IT_FLAG_TCx = DMA1_IT_TC5, }; /** UART */ UART_UserParam uart_param = { .tx_buf = uart_tx_buf, .rx_buf = uart_rx_buf, .tx_buf_len_max = UART_TX_BUF_MAX, .rx_buf_len_max = UART_RX_BUF_MAX, .rx_tail = 0, .rx_head = 0, }; /** UART */ UART_TypeDef UART_TypeDef_param = { .baud = 115200, .pfunc_gpio_init = USART_GpioInit, .USARTx = USART1, .USARTx_IRQn = USART1_IRQn, .DMAx = &dma_param, .user = &uart_param, }; void DMA1_Channel4_IRQHandler(void){ uart_tx_irq(&UART_TypeDef_param); } void DMA1_Channel5_IRQHandler(void){ uart_rx_irq(&UART_TypeDef_param); } void USART1_IRQHandler(void){ uart_irq(&UART_TypeDef_param); } /** * @} */
