OpenEdv-开源电子网

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

STM32F4,串口DMA中断方式接收定长数据失败,论坛里面全是不定长数据空闲中断接收的,救救孩子吧

[复制链接]

2

主题

5

帖子

0

精华

新手上路

积分
22
金钱
22
注册时间
2019-8-29
在线时间
5 小时
发表于 2020-2-13 02:44:47 | 显示全部楼层 |阅读模式
4金钱
u8 readbuf[44];
void uart2_init(u32 bound){
        //GPIO¶Ë¿úéèÖÃ
        GPIO_InitTypeDef GPIO_InitStructure;
        USART_InitTypeDef USART_InitStructure;
        NVIC_InitTypeDef NVIC_InitStructure;
        DMA_InitTypeDef DMA_InitStructure;       
       
        RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA|RCC_AHB1Periph_DMA1,ENABLE); //ê1ÄüGPIOAê±Öó
        RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);//ê1ÄüUSART2ê±Öó
       
        //′®¿ú1¶Ôó|òy½Å¸′óÃó3éä
        GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_USART2); //GPIOA2¸′óÃÎaUSART2
        GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_USART2); //GPIOA3¸′óÃÎaUSART2

        //USART2¶Ë¿úÅäÖÃ
        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3; //GPIOA2óëGPIOA3
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//¸′óÃ1|Äü
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;        //Ëù¶è50MHz
        GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //íÆíì¸′óÃêä3ö
        GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; //éÏà-
        GPIO_Init(GPIOA,&GPIO_InitStructure); //3õê¼»ˉPA9£&#172A10

        //USART2 3õê¼»ˉéèÖÃ
        USART_InitStructure.USART_BaudRate = bound;//2¨ìØÂêéèÖÃ
        USART_InitStructure.USART_WordLength = USART_WordLength_8b;//×Ö3¤Îa8λêy¾Y¸ñê½
        USART_InitStructure.USART_StopBits = USART_StopBits_1;//ò»¸öí£Ö1λ
        USART_InitStructure.USART_Parity = USART_Parity_No;//ÎTÆæÅ¼D£Ñéλ
        USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//ÎTó2¼têy¾Yá÷¿ØÖÆ
        USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;        //êÕ·¢Ä£ê½
        USART_Init(USART2, &USART_InitStructure); //3õê¼»ˉ′®¿ú2
       
       
        //DMAÅäÖÃ
        DMA_DeInit(DMA1_Stream5);
        while(DMA_GetCmdStatus(DMA1_Stream5)!=DISABLE);
        DMA_InitStructure.DMA_BufferSize = 44;
        DMA_InitStructure.DMA_Channel = DMA_Channel_4;
        DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
        DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Enable;
        DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
        DMA_InitStructure.DMA_Memory0BaseAddr = (u32)readbuf;
        DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
        DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
        DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
        DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
        DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&(USART2->DR);
        DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
        DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
        DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
        DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
        DMA_Init(DMA1_Stream5,&DMA_InitStructure);
        //USART_ClearFlag(USART1, USART_FLAG_TC);
        DMA_ClearFlag(DMA1_Stream5,DMA_FLAG_TCIF4);
        NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream5_IRQn;//′®¿ú2 DMAÖD¶Ï
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;//ÇàÕ¼óÅÏè¼¶1
        NVIC_InitStructure.NVIC_IRQChannelSubPriority =1;                //×óóÅÏè¼¶1
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                        //IRQí¨μàê1Äü
        NVIC_Init(&NVIC_InitStructure);        //¸ù¾YÖ¸¶¨μÄ2Îêy3õê¼»ˉVIC¼Ä′æÆ÷¡¢
        DMA_ITConfig(DMA1_Stream5,DMA_IT_TC, ENABLE);//¿aÆôÏà1ØÖD¶Ï
       
        DMA_Cmd(DMA1_Stream5,ENABLE);
        USART_DMACmd(USART2,USART_DMAReq_Rx,ENABLE);
        USART_Cmd(USART2, ENABLE);  //ê1Äü′®¿ú2
}

void DMA1_Stream5_IRQHandler(void){
        printf("HAHA\r\n");
        if(DMA_GetITStatus(DMA1_Stream5,DMA_IT_TCIF4)!=RESET){
                for(int i=0;i<44;i++){
                        USART1->DR = readbuf[i];
                }
                DMA_ClearFlag(DMA1_Stream5,DMA_IT_TC);
        }

}
配置DMA和串口的代码如上,测试可以进入DMA中断处理函数,但是速度极快,我的输入数据帧频率才10HZ,理论上不可能这么快进入中断处理函数
导致程序一直卡在中断处理函数这里,无法执行主函数的程序
而且无法进入if(DMA_GetITStatus(DMA1_Stream5,DMA_IT_TCIF4)!=RESET),这个if判断
救救孩子吧,论坛和网上都是用空闲中断接收不定长数据的,原子哥历程只有DMA发送,救救孩子吧,搞了一天,要自闭了

最佳答案

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

已经解决了,问题出在应该判断标志位TCIF5,然后要手动清除中断标志位
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

2

主题

5

帖子

0

精华

新手上路

积分
22
金钱
22
注册时间
2019-8-29
在线时间
5 小时
 楼主| 发表于 2020-2-13 02:44:48 | 显示全部楼层
已经解决了,问题出在应该判断标志位TCIF5,然后要手动清除中断标志位
回复

使用道具 举报

35

主题

560

帖子

2

精华

资深版主

Rank: 8Rank: 8

积分
17747
金钱
17747
注册时间
2018-3-3
在线时间
523 小时
发表于 2020-2-13 11:02:51 | 显示全部楼层
不定长 不就是兼容定长数据的吗,,直接用就可以的。太快进入估计就是标志位没有清理好。可以仿真看看即将出中断处理函数时 SR寄存其是否还有值 再分析
/*
*
*
*
*
*
*/
回复

使用道具 举报

2

主题

5

帖子

0

精华

新手上路

积分
22
金钱
22
注册时间
2019-8-29
在线时间
5 小时
 楼主| 发表于 2020-2-13 12:26:19 | 显示全部楼层
Psheng 发表于 2020-2-13 11:02
不定长 不就是兼容定长数据的吗,,直接用就可以的。太快进入估计就是标志位没有清理好。可以仿真看看即将 ...

我在想,空闲中断的中断处理函数,处理的时间是上一个串口数据完成到下一个串口数据开始,而用DMA中断,中断函数处理的时间段是,上一个串口数据完成+串口数据间隔+下一个串口数据完成。可以看到如果用DMA发生中断,可以更高效的利用CPU资源。
回复

使用道具 举报

3

主题

1155

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
7462
金钱
7462
注册时间
2015-1-15
在线时间
1367 小时
发表于 2020-2-13 15:28:00 | 显示全部楼层
bonoy0328 发表于 2020-2-13 12:26
我在想,空闲中断的中断处理函数,处理的时间是上一个串口数据完成到下一个串口数据开始,而用DMA中断, ...

你协议帧都是定义等长的?
一分耕耘一分收获。
回复

使用道具 举报

2

主题

5

帖子

0

精华

新手上路

积分
22
金钱
22
注册时间
2019-8-29
在线时间
5 小时
 楼主| 发表于 2020-2-13 16:16:13 | 显示全部楼层
yklstudent 发表于 2020-2-13 15:28
你协议帧都是定义等长的?

嗯,每帧44个字节
回复

使用道具 举报

2

主题

474

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
6411
金钱
6411
注册时间
2018-6-27
在线时间
543 小时
发表于 2020-10-30 12:44:39 | 显示全部楼层
学习学习,学习学习。
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-4-29 23:30

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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