STM32的串口采用DMA方式接收数据测试

本文介绍在STM32单片机中使用DMA方式接收数据的方法,并提供了实现细节和源代码。通过三种策略解决接收可变长度数据的问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

STM32的串口采用DMA方式接收数据测试


本文博客链接:http://blog.csdn.net/jdh99,作者:jdh,转载请注明.


参考链接:http://www.amobbs.com/forum.php?mod=viewthread&tid=5511863&highlight=dma%E6%8E%A5%E6%94%B6


环境:

主机:WINXP

开发环境:MDK4.23

MCU:STM32F103CBT6


说明:

串口可以配置成用DMA的方式接收数据,不过DMA需要定长才能产生接收中断,如何接收可变长度的数据呢?

方法有以下3种:

1.将RX脚与一路时钟外部引脚相连,当串口一帧发完,即可利用此定时器产生超时中断.这个实时性较高,可以做到1个字节实时监测.

2.不改变硬件,开启一个定时器监控DMA接收,如果超时则产生中断.这个实时性不高,因为超时时间必须要大于需要接收帧的时间,精度不好控制.

3.STM32单片机有的串口可以监测总线是否处于空闲,如果空闲则产生中断.可以用它来监测DMA接收是否完毕.这种方式实时性很高.

本文采用第3种方式.在波特率576000下大数据包冲击证明可行.


源代码:

[cpp]  view plain copy
  1. //串口接收DMA缓存  
  2. #define UART_RX_LEN     128  
  3. extern uint8_t Uart_Rx[UART_RX_LEN];  

[cpp]  view plain copy
  1. //串口接收DMA缓存  
  2. uint8_t Uart_Rx[UART_RX_LEN] = {0};  

[cpp]  view plain copy
  1. //---------------------串口功能配置---------------------  
  2.     //打开串口对应的外设时钟    
  3.     RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 , ENABLE);   
  4.     //串口发DMA配置    
  5.     //启动DMA时钟  
  6.     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);  
  7.     //DMA发送中断设置  
  8.     NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_IRQn;  
  9.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3;  
  10.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;  
  11.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  
  12.     NVIC_Init(&NVIC_InitStructure);  
  13.     //DMA1通道4配置  
  14.     DMA_DeInit(DMA1_Channel4);  
  15.     //外设地址  
  16.     DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART1->DR);  
  17.     //内存地址  
  18.     DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)Uart_Send_Buffer;  
  19.     //dma传输方向单向  
  20.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;  
  21.     //设置DMA在传输时缓冲区的长度  
  22.     DMA_InitStructure.DMA_BufferSize = 100;  
  23.     //设置DMA的外设递增模式,一个外设  
  24.     DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  
  25.     //设置DMA的内存递增模式  
  26.     DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  
  27.     //外设数据字长  
  28.     DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  
  29.     //内存数据字长  
  30.     DMA_InitStructure.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;  
  31.     //设置DMA的传输模式  
  32.     DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  
  33.     //设置DMA的优先级别  
  34.     DMA_InitStructure.DMA_Priority = DMA_Priority_High;  
  35.     //设置DMA的2个memory中的变量互相访问  
  36.     DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  
  37.     DMA_Init(DMA1_Channel4,&DMA_InitStructure);  
  38.     DMA_ITConfig(DMA1_Channel4,DMA_IT_TC,ENABLE);  
  39.       
  40.     //使能通道4  
  41.     //DMA_Cmd(DMA1_Channel4, ENABLE);  
  42.   
  43.     //串口收DMA配置    
  44.     //启动DMA时钟  
  45.     RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);  
  46.     //DMA1通道5配置  
  47.     DMA_DeInit(DMA1_Channel5);  
  48.     //外设地址  
  49.     DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART1->DR);  
  50.     //内存地址  
  51.     DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)Uart_Rx;  
  52.     //dma传输方向单向  
  53.     DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;  
  54.     //设置DMA在传输时缓冲区的长度  
  55.     DMA_InitStructure.DMA_BufferSize = UART_RX_LEN;  
  56.     //设置DMA的外设递增模式,一个外设  
  57.     DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  
  58.     //设置DMA的内存递增模式  
  59.     DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  
  60.     //外设数据字长  
  61.     DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  
  62.     //内存数据字长  
  63.     DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;  
  64.     //设置DMA的传输模式  
  65.     DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  
  66.     //设置DMA的优先级别  
  67.     DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;  
  68.     //设置DMA的2个memory中的变量互相访问  
  69.     DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  
  70.     DMA_Init(DMA1_Channel5,&DMA_InitStructure);  
  71.   
  72.     //使能通道5  
  73.     DMA_Cmd(DMA1_Channel5,ENABLE);  
  74.       
  75.         
  76.     //初始化参数    
  77.     //USART_InitStructure.USART_BaudRate = DEFAULT_BAUD;    
  78.     USART_InitStructure.USART_WordLength = USART_WordLength_8b;    
  79.     USART_InitStructure.USART_StopBits = USART_StopBits_1;    
  80.     USART_InitStructure.USART_Parity = USART_Parity_No;    
  81.     USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;    
  82.     USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;      
  83.     USART_InitStructure.USART_BaudRate = DEFAULT_BAUD;   
  84.     //初始化串口   
  85.     USART_Init(USART1,&USART_InitStructure);    
  86.     //TXE发送中断,TC传输完成中断,RXNE接收中断,PE奇偶错误中断,可以是多个     
  87.     //USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);  
  88.       
  89.     //中断配置  
  90.     USART_ITConfig(USART1,USART_IT_TC,DISABLE);  
  91.     USART_ITConfig(USART1,USART_IT_RXNE,DISABLE);  
  92.     USART_ITConfig(USART1,USART_IT_IDLE,ENABLE);    
  93.   
  94.     //配置UART1中断    
  95.     NVIC_PriorityGroupConfig(NVIC_PriorityGroup_3);  
  96.     NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;               //通道设置为串口1中断    
  97.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;       //中断占先等级0    
  98.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;              //中断响应优先级0    
  99.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                 //打开中断    
  100.     NVIC_Init(&NVIC_InitStructure);     
  101.           
  102.     //采用DMA方式发送  
  103.     USART_DMACmd(USART1,USART_DMAReq_Tx,ENABLE);  
  104.     //采用DMA方式接收  
  105.     USART_DMACmd(USART1,USART_DMAReq_Rx,ENABLE);  
  106.     //启动串口    
  107.     USART_Cmd(USART1, ENABLE);   


[cpp]  view plain copy
  1. //串口1接收中断     
  2. void USART1_IRQHandler(void)                                 
  3. {     
  4.     uint32_t temp = 0;  
  5.     uint16_t i = 0;  
  6.       
  7.     if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET)  
  8.     {  
  9.         //USART_ClearFlag(USART1,USART_IT_IDLE);  
  10.         temp = USART1->SR;  
  11.         temp = USART1->DR; //清USART_IT_IDLE标志  
  12.         DMA_Cmd(DMA1_Channel5,DISABLE);  
  13.   
  14.         temp = UART_RX_LEN - DMA_GetCurrDataCounter(DMA1_Channel5);  
  15.         for (i = 0;i < temp;i++)  
  16.         {  
  17.             Data_Receive_Usart = Uart_Rx[i];  
  18.             //启动串口状态机  
  19.             usart_state_run();   
  20.         }  
  21.   
  22.         //设置传输数据长度  
  23.         DMA_SetCurrDataCounter(DMA1_Channel5,UART_RX_LEN);  
  24.         //打开DMA  
  25.         DMA_Cmd(DMA1_Channel5,ENABLE);  
  26.     }   
  27.       
  28.     __nop();   
  29. }   

测试结果:

条件:单片机运行于72M,与PC通信速率为460800.PC每隔100ms发送一个9个字节的包:c5 5c 6 0 6F 10 5 4e f7.

测试:单片机每次收到此包,一个IO作电平跳转,然后处理返回一包.

示波器显示:


放大显示:

<think>嗯,用户想了解STM32DMA接收功能配置教程和常见问题解决方案。首先,我需要回忆一下STM32DMA相关知识。DMA(直接内存访问)对于减少CPU负担很重要,尤其是在处理串口数据接收这类任务时。记得STM32F1系列中,不同型号的DMA通道数量不同,比如用户提到的STM32F103C8T6只有DMA1,7个通道。这点在配置时需要注意,避免通道冲突。 接下来,配置步骤应该包括初始化DMA串口,使能相应的中断。比如,空闲中断结合DMA可以处理不定长数据,这可能是一个常见需求。需要提到如何设置空闲中断,配置DMA传输方向为外设到内存,设置数据长度等。可能还需要提及中断服务函数的处理,比如在空闲中断触发时,计算接收到的数据长度,然后进行后续处理。 然后,常见问题。比如DMA传输不触发,这可能是因为没有使能DMA请求或者中断配置错误。数据覆盖问题可能由于DMA没有循环模式导致,或者应用程序处理速度跟不上接收速度。内存对齐错误也是常见问题,特别是不同数据宽度时,需要确保外设和内存的数据宽度一致。此外,DMA优先级设置不当可能导致其他外设无法及时响应,需要检查优先级配置。 还要考虑用户可能的实际应用场景,比如串口接收不定长数据,使用空闲中断和DMA的组合方法。这时候需要配置USART的DMA请求,并使能空闲中断,在中断处理函数中处理数据。代码示例可能会有帮助,但用户可能需要更详细的步骤,比如HAL库中的配置流程,或者标准外设库的函数调用顺序。 另外,用户可能遇到的问题包括如何判断DMA传输完成,如何处理接收数据的中断,以及如何避免数据溢出。需要提醒用户启用传输完成中断或半传输中断,如果应用需要的话。同时,内存缓冲区的管理也很重要,比如使用双缓冲或者循环缓冲来避免数据丢失。 可能还需要提到DMA的初始化结构体配置,比如DMA_InitTypeDef中的各个参数:外设地址、内存地址、传输方向、数据大小、是否自动递增等。这些参数设置错误会导致DMA无法正常工作。 最后,引用用户提供的参考资料,比如引用[1]和[2],其中提到了DMA通道的数量和使能DMA接收的代码片段。需要将这些引用自然地融入回答中,确保用户能够参考官方文档或相关资源进行更深入的了解。</think>### STM32 DMA接收功能配置教程 #### 一、硬件基础 1. STM32F1系列DMA控制器包含7个通道(仅DMA1),F4/F7系列包含双DMA控制器 2. 外设与DMA通道的映射关系需参考《参考手册》,例如USART1_RX对应DMA1通道5[^1] #### 二、配置流程(以USART1为例) 1. **DMA初始化** ```c DMA_InitTypeDef DMA_InitStructure; DMA_DeInit(DMA1_Channel5); DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&USART1->DR; DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)RxBuffer; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; // 外设为数据DMA_InitStructure.DMA_BufferSize = BufferSize; DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; // 循环模式防溢出 DMA_InitStructure.DMA_Priority = DMA_Priority_High; DMA_Init(DMA1_Channel5, &DMA_InitStructure); ``` 2. **串口DMA使能** ```c USART_DMACmd(USART1, USART_DMAReq_Rx, ENABLE); // 使能接收DMA[^2] ``` 3. **中断配置** ```c NVIC_InitTypeDef NVIC_InitStructure; // 使能空闲中断 USART_ITConfig(USART1, USART_IT_IDLE, ENABLE); // 配置DMA中断 NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel5_IRQn; NVIC_Init(&NVIC_InitStructure); DMA_ITConfig(DMA1_Channel5, DMA_IT_TC, ENABLE); // 使能传输完成中断 ``` #### 三、典型应用场景 1. **不定长数据接收**(空闲中断+DMA) $$ 接收数据长度 = 总缓冲区大小 - DMA_GetCurrDataCounter(DMA1_Channel5) $$ 2. **高速数据采集**(ADC+DMA) 3. **大容量数据传输**(SPI Flash读写) §§常见问题解决方案§§ 1. **DMA传输不触发** - 检查外设时钟使能(`RCC_AHBPeriphClockCmd`) - 确认DMA请求映射正确(参考《参考手册》通道分配表) - 验证NVIC优先级配置是否冲突 2. **数据覆盖问题** - 使用双缓冲机制:配置两个DMA内存地址交替使用 - 启用半传输中断:`DMA_IT_HT`可实现"乒乓缓冲" - 调整DMA模式为循环缓冲:`DMA_Mode_Circular` 3. **内存对齐错误** - 确保外设/内存数据宽度匹配(32位地址需4字节对齐) - 使用`__align(4)`定义缓冲区 ```c __align(4) uint8_t RxBuffer[256]; ``` 4. **中断响应延迟** - 设置合理的中断优先级分组:`NVIC_PriorityGroupConfig` - DMA中断优先级应高于外设中断(推荐DMA0级,外设1级) [^1]: STM32F103参考手册第10.3节DMA通道映射表 [^2]: STM32标准外设库开发指南第15章DMA配置
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值