OpenEdv-开源电子网

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

串口DMA-空闲中断接收-判断数据长度正确-但是DMA缓存数据错误

[复制链接]

22

主题

69

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
303
金钱
303
注册时间
2019-11-10
在线时间
61 小时
发表于 2021-7-31 12:47:27 | 显示全部楼层 |阅读模式
2金钱
本帖最后由 woshi520laoxie 于 2021-8-1 16:48 编辑

串口DMA接收异常
问题1:空闲中断正常、读取DMA接收数据正常,但是里面的数据不对,一直都是第一次的数据
          如果在空闲中断中打一个断点,那么数据也正常,全速运行的时候,数据就不对了。
问题1的尝试:空闲中断的时候加一个延时操作之后再去读取DMA里面的数据,也是不对的。
                        为何打个断点就对了

1、硬件平台:  STM32H743
2、使用串口 DMA收发模式
使用场景:DMA的接收使用空闲中断的方式,当时进入IDLE中断的时候,判断串口接收DMA里面的数据量,如果有数据就把数据提取出来。
空闲中断的代码:
    if((__HAL_UART_GET_FLAG(phuart,UART_FLAG_IDLE) != RESET))
    {   
                __IS_NULL(pUsart);
        __HAL_UART_CLEAR_IDLEFLAG(phuart);
                pUsart->RxLen =  pUsart->buf_len - __HAL_DMA_GET_COUNTER(&gRxDMA_Handler[ch]);
                if (pUsart->RxLen > 0)
                {
                        HAL_UART_DMAStop(phuart);
                        if (HAL_UART_Receive_DMA(phuart,(uint8_t *)pUsart->RxDmaBuffer,pUsart->buf_len) != HAL_OK)   /* 重新启用接收 */
                        {
                                Error_Handler();
                        }
                        pUsart->RxFlag = 1;
                }
    }
这个问题一开始的时候就遇到过,之后网上找一些例子,例子看着都大同小异,然后改着改着不知道为什么就正常了,现在加多一个串口又出现这个问题了。
但是不清楚问题的原因。不知道什么因素导致的。

实际测试情况如下:
1、如果我直接把接收到数据的缓存的地址给到发送,那么接下来接收到数据都是正确的(第一个箭头所指)。
2、如果我先把DMA接收到的数据复制一份出来,然后换一个地址进行发送,那么接下来DMA接收到数据就不对了(第二个箭头所指)。
2021-08-01_162226.png
我在论坛找了一个例子,试了一下,一样的效果。我把那个代码贴出来,测试的时候把宏改一下就可以了。
2021-08-01_164617.png








H743-DMA串口收发 空闲中断.rar

1.48 MB, 下载次数: 23

最佳答案

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

H7芯片带有cache,不能直接去读DMA的数据。这个cache很重要,上手H7芯片 一定要看这个。 我当初就被这个坑死了。 或者你可以试试在读取数据之前加上 SCB_InvalidateDCache_by_Addr((uint32_t *)data, data.length); 这个函数
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

3

主题

117

帖子

0

精华

高级会员

Rank: 4

积分
559
金钱
559
注册时间
2018-10-3
在线时间
107 小时
发表于 2021-7-31 12:47:28 | 显示全部楼层
H7芯片带有cache,不能直接去读DMA的数据。这个cache很重要,上手H7芯片 一定要看这个。 我当初就被这个坑死了。 或者你可以试试在读取数据之前加上
SCB_InvalidateDCache_by_Addr((uint32_t *)data, data.length);  这个函数
回复

使用道具 举报

22

主题

69

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
303
金钱
303
注册时间
2019-11-10
在线时间
61 小时
 楼主| 发表于 2021-7-31 17:37:24 | 显示全部楼层
补充说明
1、DMA的发送正常
2、接口空闲中断也正常
3、判断DMA里面的数据长度正常
4、唯独接收到的数据,不正常
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165475
金钱
165475
注册时间
2010-12-1
在线时间
2115 小时
发表于 2021-8-1 02:02:26 | 显示全部楼层
可以论坛搜索参考下其他网友的代码
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

22

主题

69

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
303
金钱
303
注册时间
2019-11-10
在线时间
61 小时
 楼主| 发表于 2021-8-1 16:50:00 | 显示全部楼层
正点原子 发表于 2021-8-1 02:02
可以论坛搜索参考下其他网友的代码

我更新了一下描述,论坛上面找了一个代码,我试了一下,也是这样的。就是发送的时候如果发送的不是DMA接收到的数据(的地址),那么接下来接收到的数据就不对了
回复

使用道具 举报

22

主题

69

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
303
金钱
303
注册时间
2019-11-10
在线时间
61 小时
 楼主| 发表于 2021-8-1 16:53:27 | 显示全部楼层
因为我测试的时候就是直接做数据回显,但是实际使用的时候,都是把数据复制出来,然后再做处理的。
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165475
金钱
165475
注册时间
2010-12-1
在线时间
2115 小时
发表于 2021-8-2 01:39:13 | 显示全部楼层
建议把cache关了试试,尤其是d cache
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

4

主题

83

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
205
金钱
205
注册时间
2019-11-12
在线时间
77 小时
发表于 2021-8-2 08:26:32 | 显示全部楼层
你用哪个版本的ide, 申请变量不在函数体头部?另外你是怎么判断接收的数据不对,注意你的static的buff没有清空操作哦。
回复

使用道具 举报

13

主题

40

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
395
金钱
395
注册时间
2021-3-6
在线时间
116 小时
发表于 2021-8-2 08:37:39 | 显示全部楼层
试试把DMA配置成循环模式,我用寄存器写过空闲中断,现象和你描述的很像,后来把DMA配置成循环模式接可以了,
回复

使用道具 举报

22

主题

69

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
303
金钱
303
注册时间
2019-11-10
在线时间
61 小时
 楼主| 发表于 2021-8-2 10:14:33 | 显示全部楼层
正点原子 发表于 2021-8-2 01:39
建议把cache关了试试,尤其是d cache

我测试了,的确是如你们所说的那样,是由于cache的缘故,关闭了就正常了,或者和7#的坛友描述的那样,在读取数据之前处理
执行:SCB_InvalidateDCache 函数,用于丢弃 D Cache 当前数据,重新从 SRAM 获取数据。
回复

使用道具 举报

22

主题

69

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
303
金钱
303
注册时间
2019-11-10
在线时间
61 小时
 楼主| 发表于 2021-8-2 10:17:16 | 显示全部楼层
wklhwkl 发表于 2021-8-2 09:08
H7芯片带有cache,不能直接去读DMA的数据。这个cache很重要,上手H7芯片 一定要看这个。 我当初就被这个坑 ...

谢谢啊,这个问题终于是知道了原因,的确是如你所说的那样,是由于cache的缘故,是得好好看看cache了,否则很多莫名其妙的问题,搞得怀疑人生。

我测试如下:
在读取数据之前做一个处理,目前可以正常获取到DMA的数据了。
执行:SCB_InvalidateDCache 函数,用于丢弃 D Cache 当前数据,重新从 SRAM 获取数据。
回复

使用道具 举报

22

主题

69

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
303
金钱
303
注册时间
2019-11-10
在线时间
61 小时
 楼主| 发表于 2021-8-2 10:47:24 | 显示全部楼层
正点原子 发表于 2021-8-2 01:39
建议把cache关了试试,尤其是d cache

似乎这个透写并非完全作用到了,因为在读取DMA数据的时候主动运行SCB_InvalidateDCache就正常了,表示数据应该还在D-cache里面。还不在ram里面。

SCB->CACR|=1<<2;   //强制D-Cache透写,如不开启,实际使用中可能遇到各种问题
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-5-6 12:56

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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