
个人感觉,做嵌入式,底层就是datasheet,顶层就是数理逻辑。: { T# h( B0 b7 t8 y3 q 不管什么芯片,当我们遇到问题时,通过查阅datasheet或上官网基本上都能找到解决方法。然而,这些基本都是英文。所以,英文好对做研发是有很大益处的。不过好在有翻译工具,如:有道( ![]() C语言虽然没有class,但有struct。我们可以多用struct。ST库和ucos中就包含许多struct。/ u1 G }% H- a5 F 程序结构多用状态机或switch。个人感觉,状态机看起来很有条理,很清晰,后期修改维护也方便。 鄙人浅见,请大家不吝赐教。3 I6 V h6 _: L 进入正题:UART以DMA方式接收和发送的函数调用顺序: 循环模式接收:HAL_UART_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)==》DMA1_Channelx_IRQHandler(void)==》HAL_DMA_IRQHandler(&hdma_usartx_rx)==》UART_DMAReceiveCplt(DMA_HandleTypeDef *hdma)==》HAL_UART_RxCpltCallback(huart) 当然这当中还会调用传输 Half 中断,这里就不讨论了。 8 E7 q5 ?4 B: J' w1 V) q- A 正常模式发送:HAL_UART_Transmit_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)==》DMA1_Channelx_IRQHandler(void)==》HAL_DMA_IRQHandler(&hdma_usartx_tx)==》UART_DMATransmitCplt(DMA_HandleTypeDef *hdma)==》USART3_IRQHandler(void)==》HAL_UART_IRQHandler(&huart3)==》UART_EndTransmit_IT(huart)==》HAL_UART_RxCpltCallback(huart) 8 E) W U, G' v, A0 W% E 这里只分析 HAL_UART_Receive_DMA 的源码,其它的大家自己分析: HAL_StatusTypeDef HAL_UART_Receive_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size). c) h/ u( O1 B( t2 g UART以DMA方式接收指定长度数据到指定缓冲区 { uint32_t *tmp; /* Check that a Rx process is not already ongoing */ if(huart->RxState == HAL_UART_STATE_READY) 检查huart->RxState是否处于 接收空闲 状态。当这一状态标志非READY时,会跳过DMA接收参数设置,直接返回HAL_BUSY。其它的UART接收函数也会检查这个状态,所以,哪个先调用就执行哪个。* X& D: Q1 y8 e6 M { if((pData == NULL) || (Size == 0U)) { return HAL_ERROR;( h. x2 a+ a) h, A% D6 K }& W$ I9 ]3 |9 h /* Process Locked *// k6 S6 }& |; r __HAL_LOCK(huart); 8 K( Z6 ?7 G% R9 R huart->pRxBuffPtr = pData; huart->RxXferSize = Size; 4 ]7 e: G$ M2 G) n* @ huart->ErrorCode = HAL_UART_ERROR_NONE;: P* m1 |3 }% N% {- n huart->RxState = HAL_UART_STATE_BUSY_RX; . T; n6 V7 j& w! @+ E& v /* Set the UART DMA transfer complete callback */ \0 _! O+ z& l9 G! O huart->hdmarx->XferCpltCallback = UART_DMAReceiveCplt; 设置DMA传输完成的回调函数。当DMA以循环方式传输时会调用UART接收完成中断的回调函数;以Normal方式传输时会关闭UART的DMA通道,并使能UART传输完成中断,触发UART传输完成中断,设置huart->RxState为READY,并调用/ i; V( `; @( B& a' _ UART接收完成中断的回调函数。所以,不管DMA按循环或正常模式传输,到最后都会调用UART接收完成中断的回调函数 /* Set the UART DMA Half transfer complete callback */' z( Y0 j- a& o huart->hdmarx->XferHalfCpltCallback = UART_DMARxHalfCplt; /* Set the DMA error callback */5 G; A: y) `3 U/ T6 N) D2 Y- s9 k huart->hdmarx->XferErrorCallback = UART_DMAError;- w( e$ Y9 e5 j7 g- ]2 z% F5 c /* Set the DMA abort callback */- F6 c$ y, S+ y! U$ x huart->hdmarx->XferAbortCallback = NULL;1 f f0 W+ X/ s& p3 w7 ` /* Enable the DMA channel */6 W% j7 d! H* p) ?& M& r& y tmp = (uint32_t*)&pData;) R0 R4 w4 y8 G, X4 W5 h HAL_DMA_Start_IT(huart->hdmarx, (uint32_t)&huart->Instance->DR, *(uint32_t*)tmp, Size);以中断方式打开DMA传输。所以CubeMx中DMA中断默认是打开的切不可更改3 X6 s4 T+ a! ^/ k V- F5 C; h /* Clear the Overrun flag just before enabling the DMA Rx request: can be mandatory for the second transfer */ __HAL_UART_CLEAR_OREFLAG(huart); 0 a3 B/ D& u5 c/ s7 Y } /* Process Unlocked *// t. e0 E6 m0 ?" p) p! ?$ | __HAL_UNLOCK(huart); 4 V$ i* u+ p8 m/ } /* Enable the UART Parity Error Interrupt */ SET_BIT(huart->Instance->CR1, USART_CR1_PEIE); /* Enable the UART Error Interrupt: (Frame error, noise error, overrun error) */ SET_BIT(huart->Instance->CR3, USART_CR3_EIE);* \' R0 U# K7 t+ K1 ]2 W$ u% w' L + Q: f0 W+ J2 M /* Enable the DMA transfer for the receiver request by setting the DMAR bit in the UART CR3 register */ SET_BIT(huart->Instance->CR3, USART_CR3_DMAR);: `7 V, D& \6 b$ a/ B return HAL_OK; } else {$ k s8 l/ F O/ K- x return HAL_BUSY;2 R* b3 u) o* h# ~6 k$ C }) f2 @: H5 x8 [8 d$ J2 i } 下面上我做的测试:一、Cube配置 1、DMA接收要用 循环模式,只需调用一次接收函数即可,重复调用也只有第一次调用有效。若是 正常模式,需要在每次接收完成后重复调用。 2、DMA发送要用 正常模式,需在每次发送时重复调用。还要打开UART中断,否则即使重复调用DMA发送函数也只有第一次发送有效,其余不会执行(UART发送状态一直处于 BUSY)。若是 循环模式,则会连续不断地发送,不会停止。+ N; v9 V) }8 o+ s, {3 K; Q 以上两点大家可以自己改改测试。 ![]() ![]() ![]() 二、Keil例程编写, x/ L3 f7 I+ z: L2 q ![]() ![]() ![]() 三、测试" O% Y8 \: z/ r1 A 可以看到,发送数 = 接收数。大家可以做更大数据的短间隔长时间测试,看看会不会出错。% |3 m+ k+ V2 u9 S: n1 [, a: Y9 W ![]() 最后,奉上测试工程: ![]() |
6 ^ ^0 D3 c, q( u a: B
1 B0 g- \) ^) X, J- m
"no source": Error: #5: cannot open source input file "C:/Users/Administrator/STM32Cube/Repository/STM32Cube_FW_F1_V1.6.0/Drivers/STM32F1xx_HAL_Driver/Src/stm32f1xx_hal_rcc.c": No such file or directory
https://www.stmcu.org.cn/module/forum/thread-614550-1-1.html