chuangyi2000 发表于 2018-3-8 10:36:00

关于HAL库的STM32F767的DMA通过IDLE中断接收数据

本帖最后由 chuangyi2000 于 2018-3-9 08:01 编辑

问题描述;开启了USART1的DMA接收和发送,开启了USART1的IDLE中断,IDLE中断正常,能成功接收到数据。通过仿真能看到接收缓冲区数组中的数据,但不能访问,如果不访问,再开启DMA接收能正常接收,并且在仿真状态下也能查看。只要访问一次这个数组,下一次开启DMA接收后就不能接收数据了。

原代码如下:

                if(rx_end == 1){
                        rx_end = 0;
                        
                        HAL_UART_DMAStop(&UART1_Handler);

                        LCD_ShowString(300, 400, 200, 16, 16, "success!");
                        LCD_ShowString(230, 400, 200, 16, 16, aRxBuffer);            //这条不能执行,执行就不能再接收数据
                                                                                                                     //屏蔽掉这条在仿真状态下能观察aRxBuffer接收正常
                        
                        HAL_UART_Receive_DMA(&UART1_Handler, aRxBuffer, RXBUFFERSIZE);
                }      


配置原代码如下:

{
      __HAL_RCC_DMA2_CLK_ENABLE();                        //DMA2ʱÖÓʹÄÜ      
      __HAL_RCC_GPIOA_CLK_ENABLE();                        //ʹÄÜGPIOAʱÖÓ
      __HAL_RCC_USART1_CLK_ENABLE();                        //ʹÄÜUSART1ʱÖÓ

      GPIO_Initure.Pin=GPIO_PIN_9;                        //PA9
      GPIO_Initure.Mode=GPIO_MODE_AF_PP;                //¸´ÓÃÍÆÍìÊä³ö
      GPIO_Initure.Pull=GPIO_PULLUP;                        //ÉÏÀ­
      GPIO_Initure.Speed=GPIO_SPEED_FAST;                //¸ßËÙ
      GPIO_Initure.Alternate=GPIO_AF7_USART1;      //¸´ÓÃΪUSART1
      HAL_GPIO_Init(GPIOA,&GPIO_Initure);                   //³õʼ»¯PA9

      GPIO_Initure.Pin=GPIO_PIN_10;                        //PA10
      HAL_GPIO_Init(GPIOA,&GPIO_Initure);                   //³õʼ»¯PA10
      
      UART1_Handler.Instance=USART1;                                          //USART1
      UART1_Handler.Init.BaudRate=9600;                                    //²¨ÌØÂÊ
      UART1_Handler.Init.WordLength=UART_WORDLENGTH_8B;   //×Ö³¤Îª8λÊý¾Ý¸ñʽ
      UART1_Handler.Init.StopBits=UART_STOPBITS_1;            //Ò»¸öֹͣλ
      UART1_Handler.Init.Parity=UART_PARITY_NONE;                  //ÎÞÆæÅ¼Ð£Ñéλ
      UART1_Handler.Init.HwFlowCtl=UART_HWCONTROL_NONE;   //ÎÞÓ²¼þÁ÷¿Ø
      UART1_Handler.Init.Mode=UART_MODE_TX_RX;                  //ÊÕ·¢Ä£Ê½
      HAL_UART_Init(&UART1_Handler);                                          //HAL_UART_Init()»áʹÄÜUART1

    //Tx DMAÅäÖÃ
      UART1TxDMA_Handler.Instance = DMA2_Stream7;                           //Êý¾ÝÁ÷Ñ¡Ôñ
      UART1TxDMA_Handler.Init.Channel = DMA_CHANNEL_4;                      //ͨµÀÑ¡Ôñ
      UART1TxDMA_Handler.Init.Direction = DMA_MEMORY_TO_PERIPH;             //´æ´¢Æ÷µ½ÍâÉè
      UART1TxDMA_Handler.Init.PeriphInc = DMA_PINC_DISABLE;               //ÍâÉè·ÇÔöÁ¿Ä£Ê½
      UART1TxDMA_Handler.Init.MemInc = DMA_MINC_ENABLE;                     //´æ´¢Æ÷ÔöÁ¿Ä£Ê½
      UART1TxDMA_Handler.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;    //ÍâÉèÊý¾Ý³¤¶È:8λ
      UART1TxDMA_Handler.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;       //´æ´¢Æ÷Êý¾Ý³¤¶È:8λ
      UART1TxDMA_Handler.Init.Mode = DMA_NORMAL;                            //ÍâÉèÁ÷¿ØÄ£Ê½
      UART1TxDMA_Handler.Init.Priority = DMA_PRIORITY_MEDIUM;               //ÖеÈÓÅÏȼ¶
      UART1TxDMA_Handler.Init.FIFOMode = DMA_FIFOMODE_DISABLE;            
      UART1TxDMA_Handler.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;      
      UART1TxDMA_Handler.Init.MemBurst = DMA_MBURST_SINGLE;               //´æ´¢Æ÷Í»·¢µ¥´Î´«Êä
      UART1TxDMA_Handler.Init.PeriphBurst = DMA_PBURST_SINGLE;            //ÍâÉèÍ»·¢µ¥´Î´«Êä

//    HAL_DMA_DeInit(&UART1TxDMA_Handler);   
      HAL_DMA_Init(&UART1TxDMA_Handler);
      __HAL_LINKDMA(&UART1_Handler, hdmatx, UART1TxDMA_Handler);

    //Rx DMAÅäÖÃ
      UART1RxDMA_Handler.Instance = DMA2_Stream2;                           //Êý¾ÝÁ÷Ñ¡Ôñ
      UART1RxDMA_Handler.Init.Channel = DMA_CHANNEL_4;                      //ͨµÀÑ¡Ôñ
      UART1RxDMA_Handler.Init.Direction = DMA_PERIPH_TO_MEMORY;             //´æ´¢Æ÷µ½ÍâÉè
      UART1RxDMA_Handler.Init.PeriphInc = DMA_PINC_DISABLE;               //ÍâÉè·ÇÔöÁ¿Ä£Ê½
      UART1RxDMA_Handler.Init.MemInc = DMA_MINC_ENABLE;                     //´æ´¢Æ÷ÔöÁ¿Ä£Ê½
      UART1RxDMA_Handler.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;    //ÍâÉèÊý¾Ý³¤¶È:8λ
      UART1RxDMA_Handler.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;       //´æ´¢Æ÷Êý¾Ý³¤¶È:8λ
      UART1RxDMA_Handler.Init.Mode = DMA_NORMAL;                            //ÍâÉèÁ÷¿ØÄ£Ê½
      UART1RxDMA_Handler.Init.Priority = DMA_PRIORITY_MEDIUM;               //ÖеÈÓÅÏȼ¶
      UART1RxDMA_Handler.Init.FIFOMode = DMA_FIFOMODE_DISABLE;            
      UART1RxDMA_Handler.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;      
      UART1RxDMA_Handler.Init.MemBurst = DMA_MBURST_SINGLE;               //´æ´¢Æ÷Í»·¢µ¥´Î´«Êä
      UART1RxDMA_Handler.Init.PeriphBurst = DMA_PBURST_SINGLE;            //ÍâÉèÍ»·¢µ¥´Î´«Êä

      HAL_DMA_Init(&UART1RxDMA_Handler);
      __HAL_LINKDMA(&UART1_Handler, hdmarx, UART1RxDMA_Handler);

      __HAL_UART_CLEAR_IDLEFLAG(&UART1_Handler);
      __HAL_UART_ENABLE_IT(&UART1_Handler, UART_IT_RXNE);                //¿ªÆô½ÓÊÕÖжÏ
      HAL_NVIC_EnableIRQ(USART1_IRQn);                                                      //ʹÄÜUSART1ÖжÏͨµÀ
      HAL_NVIC_SetPriority(USART1_IRQn, 3, 3);                                        //ÇÀÕ¼ÓÅÏȼ¶3£¬×ÓÓÅÏȼ¶3

      HAL_UART_Receive_DMA(&UART1_Handler, aRxBuffer, RXBUFFERSIZE);            //开启了一次

}

接收中断函数

void USART1_IRQHandler(void)                        
{
unsigned short y;
      
      if((__HAL_UART_GET_FLAG(&UART1_Handler, UART_IT_RXNE) != RESET))
      {
                y = 400;
                LCD_ShowString(30, y, 200, 16, 16, "UART_IT_RXNE");
                LCD_ShowNum(180, y, RX_ADDR, 3, 16);
               
                rx_end = 1;
               
                if(RX_ADDR < 900){
                        RX_ADDR++;
                }
                else{
                        RX_ADDR = 0;
                }
      }

      HAL_UART_IRQHandler(&UART1_Handler);      
}



jjbboox 发表于 2018-3-8 11:19:08

本帖最后由 jjbboox 于 2018-3-8 11:23 编辑

你没有把DMA关闭。先把DMA关闭以后才可以访问DMA内存区域。另外一般读取串口数据都应该是在主循环里面,判断读取标志为1,不要在中断函数中处理读取到的内容。

__HAL_UART_CLEAR_IDLEFLAG(&UART1_Handler);

y= UART1_Handler.Instance->SR;
y= UART1_Handler.Instance->DR;

HAL_UART_DMAStop(&UART1_Handler);






chuangyi2000 发表于 2018-3-8 11:54:14

jjbboox 发表于 2018-3-8 11:19
你没有把DMA关闭。先把DMA关闭以后才可以访问DMA内存区域。另外一般读取串口数据都应该是在主循环里面,判 ...

你好, 谢谢回复。

我的访问是在主程序中, IDLE中断函数中rx_end = 1;主函数中判断rx_end==1 后
关闭DMA   HAL_UART_DMAStop(&UART1_Handler);
访问数据    LCD_ShowString(230, 400, 200, 16, 16, aRxBuffer);
重启DMA    HAL_UART_Receive_DMA(&UART1_Handler, aRxBuffer, RXBUFFERSIZE);


                        y = UART1_Handler.Instance->ISR;
                        y = UART1_Handler.Instance->RDR;
                        HAL_UART_DMAStop(&UART1_Handler);
关闭DMA前加上那两条后现象还是一样的   不能访问接收缓冲数组。

chuangyi2000 发表于 2018-3-8 12:00:56

总是只能显示第一次的数据,访问后重启DMA,后面的数据就收不到了,仿真状态也是看不到缓冲数组中的数据

Dandjinh 发表于 2018-3-8 14:35:41

本帖最后由 Dandjinh 于 2018-3-8 14:42 编辑

有可能是Cache的问题

chuangyi2000 发表于 2018-3-8 17:19:37

Dandjinh 发表于 2018-3-8 14:35
有可能是Cache的问题

谢谢,非常像,因为当我不能查看到接收缓冲数据的时候,DMA接收数据的数量能够正确更新。

chuangyi2000 发表于 2018-3-9 07:54:46

Dandjinh 发表于 2018-3-8 14:35
有可能是Cache的问题

:hug:

jjbboox 发表于 2018-3-9 08:46:01

chuangyi2000 发表于 2018-3-8 11:54
你好, 谢谢回复。

我的访问是在主程序中, IDLE中断函数中rx_end = 1;主函数中判断rx_end==1 后 ...

你可以参考下面这个链接
我就是照着这个方法做的,效果很好。
http://blog.csdn.net/youmeichifan/article/details/51750435

chuangyi2000 发表于 2018-3-12 08:08:07

jjbboox 发表于 2018-3-9 08:46
你可以参考下面这个链接
我就是照着这个方法做的,效果很好。
http://blog.csdn.net/youmeichifan/articl ...

谢谢回复。
我程序功能基本实现了,实际上是访问数据访问的Cache,DMA更新的是缓冲区,导致存储器内容不一致,读出数据给人感觉就是没收到数据一样。

dsjsjf 发表于 2018-3-15 16:22:39

/* USART6 串口空闲中断函数 */
void HeaterComm_Callback(void)
{
uint32_t TmpFlag = 0;
uint32_t TmpLength;

TmpFlag = __HAL_UART_GET_FLAG(&huart6, UART_FLAG_IDLE);      
if((TmpFlag != RESET))
{
    __HAL_UART_CLEAR_IDLEFLAG(&huart6);
    //HAL_UART_AbortReceive_IT(&huart6);                           
    HAL_UART_DMAStop(&huart6);      
    TmpLength = __HAL_DMA_GET_COUNTER((DMA_HandleTypeDef *)&hdma_usart6_rx);         
    //TmpLength = hdma_usart6_rx.Instance->NDTR;
    HeaterComm.DataLength = 32 - TmpLength;
    osSignalSet (HeaterComm_TaskHandle, 1);                     //HeaterComm.ReceiveFlag = 1;
}
}
页: [1]
查看完整版本: 关于HAL库的STM32F767的DMA通过IDLE中断接收数据