OpenEdv-开源电子网

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

F103串口空闲中断+DMA接收碰到0x00直接进入空闲中断

[复制链接]

1

主题

1

帖子

0

精华

初级会员

Rank: 2

积分
66
金钱
66
注册时间
2017-9-15
在线时间
12 小时
发表于 2019-4-24 12:51:00 | 显示全部楼层 |阅读模式
1金钱
最近在尝试使用串口空闲中断+DMA的时候,碰到一个奇怪的情况。当串口接收到的数据时0x00的时候,即使后面仍然有数据没有发送完成,程序也会立刻进入空闲中断。导致的结果就是从0x00开始到后面的数据都接收不到。其他的数据包括0xFF都不会碰到这样的情况,唯独0x00这样子。示波器看了下串口的波形也是对的,调试半天找不到原因。无奈之下寻求各位大佬帮助,有没有碰到过同样问题的?附上代码
[mw_shl_code=c,true]void uart_init(void){
  //GPIO端口设置
  GPIO_InitTypeDef GPIO_InitStructure;
        USART_InitTypeDef USART_InitStructure;
        NVIC_InitTypeDef NVIC_InitStructure;
         
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//使能GPIOD,GPIOA时钟
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
  
        //USART1_TX   GPIOA.9
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;        //复用推挽输出
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9
   
  //USART1_RX          GPIOA.10初始化
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA.10
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//上拉输入
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10


  //Usart1 NVIC 配置
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);        //设置优先级分组:先占优先级和从优先级。具体见固件库使用手册13.2.3
  NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;                //子优先级3
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                        //IRQ通道使能
        NVIC_Init(&NVIC_InitStructure);        //根据指定的参数初始化VIC寄存器
  
   //USART 初始化设置

        USART_InitStructure.USART_BaudRate = Baudrate;//串口波特率
        USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
        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(USART1, &USART_InitStructure); //初始化串口1
       
  USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);//开启串口空闲中断
  USART_Cmd(USART1, ENABLE);                    //使能串口1
        USART_DMACmd(USART1,USART_DMAReq_Tx|USART_DMAReq_Rx,ENABLE);//使能USART1的DMA发送和接收请求

}[/mw_shl_code]


[mw_shl_code=c,true]void DMA_Rx_Config(void)
{
        RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);        //使能DMA时钟
        //DMA_InitTypeDef DMA_InitStructure;
       
  DMA_DeInit(DMA1_Channel5);   //将DMA1的通道5寄存器重设为缺省值

        DMA_RxInitStructure.DMA_PeripheralBaseAddr = (u32)&USART1->DR;  //DMA外设基地址
        DMA_RxInitStructure.DMA_MemoryBaseAddr = (u32)USART_Rx_BUF;  //DMA内存基地址
        DMA_RxInitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;//数据传输方向,从外设读取发送到内存
        DMA_RxInitStructure.DMA_BufferSize = DMA1_Rx_LEN;  //DMA通道的DMA缓存的大小
        DMA_RxInitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变
        DMA_RxInitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //内存地址寄存器递增
        DMA_RxInitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //数据宽度为8位
        DMA_RxInitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //数据宽度为8位
        DMA_RxInitStructure.DMA_Mode = DMA_Mode_Normal;  //工作在正常缓存模式
        DMA_RxInitStructure.DMA_Priority = DMA_Priority_Medium; //DMA通道 x拥有中优先级
        DMA_RxInitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输
        DMA_Init(DMA1_Channel5, &DMA_RxInitStructure);  //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Rx_DMA_Channel所标识的寄存器
}[/mw_shl_code]

[mw_shl_code=c,true]void USART1_IRQHandler(void)
{
        u16 num=0;
        USART_Tx_BUF[0]= 'A';
        if(USART_GetITStatus(USART1,USART_IT_IDLE) == SET)//中断为空闲中断
        {
                num = USART1->SR;
                num = USART1->DR;//清USART_IT_IDLE标志
                DMA_Cmd(DMA1_Channel5,DISABLE);//关闭DMA
                num = DMA1_Rx_LEN -  DMA_GetCurrDataCounter(DMA1_Channel5);//得到真正接收数据个数
                USART_Rx_BUF[num] = '\0';
                DMA1_Channel5->CNDTR = DMA1_Rx_LEN;//重新设置接收数据个数
                DMA_Cmd(DMA1_Channel5,ENABLE);//开启DMA
          receive_flag = 1;//接收数据标志位置1
  }
}[/mw_shl_code]

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

5

主题

106

帖子

0

精华

高级会员

Rank: 4

积分
757
金钱
757
注册时间
2015-10-27
在线时间
180 小时
发表于 2019-4-24 17:26:31 | 显示全部楼层
[mw_shl_code=c,true]void USART1_IRQHandler(void)
{
        uint8_t num=num;
        static uint16_t RxCount=0;//接收数据计数
        if (USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
        //判断读寄存器是否非空
        {
                USART_ClearITPendingBit(USART1, USART_IT_RXNE);//清除标志位       
                Debug_Buf[RxCount++]=USART1->DR;
                if(RxCount>300)
                        RxCount=0;
        }
        if (USART_GetITStatus(USART1, USART_IT_IDLE) != RESET)//1帧数据
        {
                num = USART1->SR;        
                num = USART1->DR; //清USART_IT_IDLE标志       
                Debug_num=RxCount;       
                RxCount=0;               
                Debug_flag = 1; //接收数据标志位置1
        }
}[/mw_shl_code]
很久以前用的串口空闲中断接收,但是没有用DMA接收,直接用中断了。
回复

使用道具 举报

0

主题

4

帖子

0

精华

新手上路

积分
23
金钱
23
注册时间
2019-12-1
在线时间
5 小时
发表于 2020-7-23 09:44:48 | 显示全部楼层
楼主这个问题解决了吗
回复

使用道具 举报

0

主题

5

帖子

0

精华

初级会员

Rank: 2

积分
52
金钱
52
注册时间
2022-11-8
在线时间
13 小时
发表于 2022-11-8 18:48:29 | 显示全部楼层
楼主最后解决了吗
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-2-25 22:02

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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