[STM32F4] F4 串口DMA接收,只进入一次DMA中断!

[复制链接]
 楼主| 紫宸 发表于 2014-7-28 13:21 | 显示全部楼层 |阅读模式
本帖最后由 紫宸 于 2014-7-28 13:25 编辑

我的问题,我想使用F4的DMA进行串口的DMA接收,配置过程:GPIO配置、UART配置、DMA优先级配置、DMA参数配置,就是这几个函数。
测试中,我用电脑串口发送数据给F4,并使用LED的翻转来表示进入中断。可问题出现了,只有我发送成功第一次的数据(设定的是6个字节)的时候,LED翻转,说明进入了DMA发送完成中断,但是以后不会再进入。更奇怪的是,我把存放DMA接收数据打印出来的 时候,发现里面的数据是实时更新的,也就是说DMA依然在正常工作,想不明白,那为什么不进入中断了呢?
请高手帮忙看看,谢谢!
谁有更好的串口DMA例程,能拿出来交流交流吗?

  1. /*---------------------- COM 端口及引脚选择 --------------------------*/
  2. /*COM*/
  3. #define COM_USART                                                USART2
  4. #define COM_GPIO_AF                                                GPIO_AF_USART2
  5. #define COM_RCC_USART_CLK                                RCC_APB1Periph_USART2
  6. #define COM_IRQ_Channel                                        USART2_IRQn
  7. #define COM_IRQHandler                                        USART2_IRQHandler
  8. /*TX*/
  9. #define COM_TX_PIN                              GPIO_Pin_2
  10. #define COM_TX_GPIO_PORT                               GPIOA
  11. #define COM_TX_GPIO_CLK                                RCC_AHB1Periph_GPIOA
  12. #define COM_TX_GPIO_PinSource                        GPIO_PinSource2
  13. /*RX*/
  14. #define COM_RX_PIN                                          GPIO_Pin_3
  15. #define COM_RX_GPIO_PORT                                GPIOA
  16. #define COM_RX_GPIO_CLK                                        RCC_AHB1Periph_GPIOA
  17. #define COM_RX_GPIO_PinSource                        GPIO_PinSource3

  18. /*--------------------------------------------------------------------*/

  19. /**************************实现函数********************************************
  20. *函数原型:                void COM_Init(u32 baudrate)
  21. *功  能:                初始化串口1
  22. 输入参数:                u32 baudrate        串口的波特率
  23. 输出参数:没有       
  24. *******************************************************************************/
  25. void COM_Init(u32 baudrate)
  26. {
  27.         USART_InitTypeDef USART_InitStructure;        //定义USART初始化结构体
  28.        
  29.         COM_GPIO_Config();                                                //配置UART1的相关引脚
  30.        
  31.        
  32.         RCC_APB1PeriphClockCmd(COM_RCC_USART_CLK,ENABLE);//使能USART1外设时钟
  33.        
  34.         /* Enable the USART OverSampling by 8 */
  35.         USART_OverSampling8Cmd(COM_USART, ENABLE);
  36.        
  37.         /* USARTx configured as follow:
  38.         - BaudRate = 115200 baud  
  39.         - Word Length = 8 Bits
  40.         - One Stop Bit
  41.         - No parity
  42.         - Hardware flow control disabled (RTS and CTS signals)
  43.         - Receive and transmit enabled
  44.         */
  45.         USART_InitStructure.USART_BaudRate = baudrate;
  46.         USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  47.         USART_InitStructure.USART_StopBits = USART_StopBits_1;
  48.         USART_InitStructure.USART_Parity = USART_Parity_No;
  49.         USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  50.         USART_InitStructure.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;
  51.         USART_Init(COM_USART , &USART_InitStructure);

  52.         COM_DMA_NVIC_Config();                                                        //配置串口DMA中断的优先级
  53.         COM_DAM_Config();                                                                //配置串口DMA

  54.         USART_ClearFlag(COM_USART,USART_FLAG_TC);                //清除标志位,防止第一个数据发送失败
  55.         USART_Cmd(COM_USART,ENABLE);                                        //使能串口
  56. }

  57. void COM_GPIO_Config(void)
  58. {
  59.         GPIO_InitTypeDef GPIO_InitStructure;                                        //定义GPIO初始化结构体
  60.         RCC_AHB1PeriphClockCmd(COM_TX_GPIO_CLK|COM_RX_GPIO_CLK,ENABLE);        //使能USART1对应的引脚端口TX RX
  61.        
  62.         /* Configure USART Tx as alternate function  */
  63.         GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;                //推挽输出
  64.         GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;                //上拉
  65.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;                //使用复用功能
  66.         GPIO_InitStructure.GPIO_Pin = COM_TX_PIN;                                //TX pin
  67.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;        //输出频率 50M
  68.         GPIO_Init(COM_TX_GPIO_PORT, &GPIO_InitStructure);                //TX PIN 初始化

  69.         /* Configure USART Rx as alternate function  */
  70.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;                //使用复用功能
  71.         GPIO_InitStructure.GPIO_Pin = COM_RX_PIN        ;                        //RX pin
  72.         GPIO_Init(COM_RX_GPIO_PORT, &GPIO_InitStructure);                //RX PIN初始化
  73.        
  74.          /* Connect PXx to USARTx_Tx*/
  75.         GPIO_PinAFConfig(COM_TX_GPIO_PORT , COM_TX_GPIO_PinSource , COM_GPIO_AF);
  76.        
  77.         /* Connect PXx to USARTx_Rx*/
  78.         GPIO_PinAFConfig(COM_RX_GPIO_PORT , COM_RX_GPIO_PinSource , COM_GPIO_AF);
  79.        
  80. }
  81. /**********************************************************************************************
  82. *函数名:void COM_DMA_NVIC_Config(void)
  83. * 参 数
  84. * 返回值:
  85. * 功能:配置串口DMA的中断优先级
  86. **********************************************************************************************/
  87. void COM_DMA_NVIC_Config(void)
  88. {
  89.         NVIC_InitTypeDef NVIC_InitStructure;
  90.        
  91.         NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream5_IRQn;
  92.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02;
  93.         NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02;
  94.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  95.         NVIC_Init(&NVIC_InitStructure);
  96. }
  97. /**********************************************************************************************
  98. *函数名:void COM_DAM_Config(void)
  99. * 参 数
  100. * 返回值:
  101. * 功能:        配置串口的DMA
  102. **********************************************************************************************/
  103. void COM_DAM_Config(void)
  104. {
  105.         DMA_InitTypeDef  DMA_InitStructure;
  106.         /* Enable the DMA clock */
  107.         RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
  108.         /* Configure DMA controller to manage USART TX and RX DMA request ----------*/
  109.        
  110.         DMA_DeInit(DMA1_Stream5);       
  111.        
  112.         /* Configure DMA Initialization Structure */
  113.         DMA_InitStructure.DMA_BufferSize = 6 ;
  114.         DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable ;
  115.         DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_1QuarterFull ;
  116.         DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single ;
  117.         DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
  118.         DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  119.         DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
  120.         DMA_InitStructure.DMA_PeripheralBaseAddr =(uint32_t) (&(COM_USART->DR)) ;
  121.         DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
  122.         DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
  123.         DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  124.         DMA_InitStructure.DMA_Priority = DMA_Priority_High;
  125.         DMA_InitStructure.DMA_Channel = DMA_Channel_4 ;
  126.         DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory ;
  127.         DMA_InitStructure.DMA_Memory0BaseAddr =(uint32_t)COM_Rx_Buf;
  128.         DMA_Init(DMA1_Stream5,&DMA_InitStructure);

  129.         /* Enable DMA Stream Transfer Complete interrupt */
  130.         DMA_ITConfig(DMA1_Stream5, DMA_IT_TC, ENABLE);                        //开启DMA数据发送完成中断

  131.     DMA_ClearITPendingBit(DMA1_Stream5, DMA_IT_TCIF5);      //清标志
  132.        
  133.         /* Enable USART RX DMA */
  134.         USART_DMACmd(COM_USART, USART_DMAReq_Rx, ENABLE);                //开启USART接收DMA请求
  135.         DMA_Cmd(DMA1_Stream5,ENABLE);
  136. }

  137. /**
  138.   * [url=home.php?mod=space&uid=247401]@brief[/url]  handles DMA Stream interrupt request.
  139.   * @param  None
  140.   * @retval None
  141.   */
  142. void DMA1_Stream5_IRQHandler(void)
  143. {
  144.         if(DMA_GetITStatus(DMA1_Stream5, DMA_IT_TCIF5))
  145.         {
  146.                 /* Clear DMA Stream Transfer Complete interrupt pending bit */
  147.                 DMA_ClearITPendingBit(DMA1_Stream5, DMA_IT_TCIF5);
  148.                 STM_EVAL_LEDOn(LED2);
  149.         }
  150. }

 楼主| 紫宸 发表于 2014-7-29 01:32 | 显示全部楼层
小弟出丑了,原来每次进入中断执行的都是STM_EVAL_LEDOn(LED2);led灯点亮函数,我以为是IO口翻转函数,怪不得没有现象!这一个小小的错误浪费了我一下午加一晚上的时间!,真不该!
电子无聊大神 发表于 2014-7-29 08:37 | 显示全部楼层
ahh 偶尔会这样的,上次我把定时器中断函数写成TIM3_IRQHanlder() 也是折腾了一天..
您需要登录后才可以回帖 登录 | 注册

本版积分规则

3

主题

15

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部

3

主题

15

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部