OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 3986|回复: 6

DMA串口接收到数据,不能立刻发送数据(HAL库)

[复制链接]

41

主题

278

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2357
金钱
2357
注册时间
2019-10-29
在线时间
380 小时
发表于 2020-11-19 17:22:23 | 显示全部楼层 |阅读模式
1金钱
本帖最后由 chenyuan 于 2020-11-19 17:27 编辑

今天用到串口DMA+空闲中断来进行收发数据,但是在收到数据后,要延时 2ms 才能发送数据,一旦去掉延时,发送的数据就全是0xFF;请问各位大佬,接收完后,要进行什么操作?? 才能紧接着就可以发送数据!!  谢谢啦

  1. void USART3_4_IRQHandler(void)
  2. {
  3.   if((__HAL_UART_GET_FLAG(&huart3, UART_FLAG_IDLE) == true))
  4.         {
  5.                 /* Clear IDLE Flag */
  6.         __HAL_UART_CLEAR_IDLEFLAG(&huart3);
  7.                
  8.                 /* Stop Receive */
  9.                 HAL_UART_DMAStop(&huart3);                        //如果发生了空闲中断,则先关闭DMA,不再接受数据

  10.                 /* len */
  11.                 relayLen = 70 - huart3.hdmarx->Instance->CNDTR;

  12.                 if(relayLen == 6)
  13.                 {
  14.                   relayFlyBuf[0]=0;
  15.                  <b> relay_Done=1;</b>
  16.                 }
  17.                 else
  18.                 {
  19.                         RelayFly_Init();//重新开始DMA接收
  20.                         return;
  21.                 }
  22.         }
  23.   HAL_UART_IRQHandler(&huart3);
  24. }

  25. void main(void)
  26. {
  27.         while(1)
  28.         {
  29.                 if(<b>relay_Done</b>)
  30.                 {
  31.                         HAL_Delay(2);
  32.                         //while(HAL_DMA_GetState(&hdma_usart3_tx) != HAL_DMA_STATE_READY);
  33.                         //while(HAL_UART_GetState(&huart3) != HAL_UART_STATE_READY);
  34.                         //while(huart3.hdmarx->Instance->CNDTR!=0)
  35.                         //while(__HAL_UART_GET_FLAG(&huart3, UART_FLAG_RXNE)!=1)

  36.                         HAL_GPIO_WritePin(WBUS2_CTL_GPIO_Port,WBUS2_CTL_Pin, GPIO_PIN_RESET);//设置发射
  37.                         HAL_UART_Transmit_DMA(&huart3, buff_temp, 31);
  38.                         while(HAL_UART_GetState(&huart3) != HAL_UART_STATE_READY);//等待发送完成
  39.                         
  40.                         HAL_GPIO_WritePin(WBUS2_CTL_GPIO_Port, WBUS2_CTL_Pin, GPIO_PIN_SET);//设置接收        
  41.                         HAL_UART_Receive_DMA(&huart3, relayFlyBuf, 70);//开始接收

  42.                         relay_Done=0;
  43.                 }
  44.         }

复制代码


最佳答案

查看完整内容[请看2#楼]

搞半天是发送端问题,我发送端之前是这样的:然后我将延时1ms改成了 while(HAL_UART_GetState(&huart3) != HAL_UART_STATE_READY); 最后接收端收到数据不用在延时2ms,也可以正常发送数据 吸取了一个教训,逻辑分析仪采集到数据的结束,但并不代表着DMA通道已经复位好,准备好了下一次数据的传输 另外,我发现在接收端,将重新设置接收放在if(收到数据){}里面,他有时候会出现收不到数据的现象,而把重新设置接收放在while中的话 ...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

41

主题

278

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2357
金钱
2357
注册时间
2019-10-29
在线时间
380 小时
 楼主| 发表于 2020-11-19 17:22:24 | 显示全部楼层
搞半天是发送端问题,我发送端之前是这样的:
  1. HAL_GPIO_WritePin(WBUS2_CTL_GPIO_Port,WBUS2_CTL_Pin, GPIO_PIN_RESET);//设置发射使能
  2. HAL_UART_Transmit_DMA(&huart3, buff_temp, 6);
  3. HAL_Delay(1);
  4. HAL_GPIO_WritePin(WBUS2_CTL_GPIO_Port,WBUS2_CTL_Pin, GPIO_PIN_SET);<span style="background-color: rgb(255, 255, 255); white-space: pre;">        </span>
复制代码
然后我将延时1ms改成了 while(HAL_UART_GetState(&huart3) != HAL_UART_STATE_READY);
  1. HAL_GPIO_WritePin(WBUS2_CTL_GPIO_Port,WBUS2_CTL_Pin, GPIO_PIN_RESET);//设置发射使能
  2. HAL_UART_Transmit_DMA(&huart3, buff_temp, 6);
  3. <span style="background-color: rgb(255, 255, 255);">while(HAL_UART_GetState(&huart3) != HAL_UART_STATE_READY);</span>
  4. HAL_GPIO_WritePin(WBUS2_CTL_GPIO_Port,WBUS2_CTL_Pin, GPIO_PIN_SET);        
复制代码
最后接收端收到数据不用在延时2ms,也可以正常发送数据
吸取了一个教训,逻辑分析仪采集到数据的结束,但并不代表着DMA通道已经复位好,准备好了下一次数据的传输
另外,我发现在接收端,将重新设置接收放在if(收到数据){}里面,他有时候会出现收不到数据的现象,而把重新设置接收放在while中的话,就没碰到偶尔收不到数据的现象
回复

使用道具 举报

10

主题

778

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
6745
金钱
6745
注册时间
2017-4-12
在线时间
1258 小时
发表于 2020-11-19 19:41:07 | 显示全部楼层
DMA发送前,先读取寄存器,看看是否发送完成了,发送完成,再去发送
回复

使用道具 举报

2

主题

685

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3448
金钱
3448
注册时间
2017-7-4
在线时间
869 小时
发表于 2020-11-19 22:45:21 | 显示全部楼层
开启DMA发送完成中断,根据中断状态判断DMA是否处于发送状态;空闲则启动发送,否则数据发送失败。
参考标准库实现:https://acuity.blog.csdn.net/article/details/108367512
源码:https://github.com/Prry/stm32-uart-dma
回复

使用道具 举报

41

主题

278

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2357
金钱
2357
注册时间
2019-10-29
在线时间
380 小时
 楼主| 发表于 2020-11-20 09:19:04 | 显示全部楼层
245925587 发表于 2020-11-19 19:41
DMA发送前,先读取寄存器,看看是否发送完成了,发送完成,再去发送

读取哪个寄存器??  我试了那些代码上35行的那些,都没有用
回复

使用道具 举报

41

主题

278

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2357
金钱
2357
注册时间
2019-10-29
在线时间
380 小时
 楼主| 发表于 2020-11-20 11:35:19 | 显示全部楼层
Acuity 发表于 2020-11-19 22:45
开启DMA发送完成中断,根据中断状态判断DMA是否处于发送状态;空闲则启动发送,否则数据发送失败。
参考标 ...

我在HAL_UART_Transmit_DMA(&huart3, buff_temp, 31);前加上这句
while(HAL_UART_GetState(&huart3) != HAL_UART_STATE_READY);
等待串口准备好,因为我看了HAL_UART_Transmit_DMA()进去后会判断串口是否准备好了,但是还是没用,如果不加延时2ms,加上while(HAL_UART_GetState(&huart3) != HAL_UART_STATE_READY);这句的话,采集的时序是正确收到数据,然后40us左右就开始发送数据,数据全是0xFF,数据的长度倒是没变,发送了31字节。
大佬你的代码和CSDN上的也看了,太过愚钝,没太看懂具体的操作
我这现在就是收到数据后立马发送的话就会发送全是0xFF的数据,串口的TX/RX公用一个引脚(硬件上的修改),相当于一个开关,打下来就是TX,打上去就是接收。
回复

使用道具 举报

41

主题

278

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2357
金钱
2357
注册时间
2019-10-29
在线时间
380 小时
 楼主| 发表于 2020-11-20 11:39:40 | 显示全部楼层
245925587 发表于 2020-11-19 19:41
DMA发送前,先读取寄存器,看看是否发送完成了,发送完成,再去发送

大佬  可以告诉具体是哪个寄存器吗?我debug调试,在延时前后加断点,都没有发现usart3的寄存器有什么变化;
奇怪我在网上找到的HAL的空闲中断+DMA收发数据,也没看到收到数据,再发送有加什么延时或者等待什么标志??
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-22 18:37

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表