stm32f429(HAL库)串口1 DMA双缓冲接收数据模仿录音实验,失败!!!

2019-07-20 08:55发布

相关的配置函数

//串口1接收DMA初始化配置
//DMA2 数据流5  通道2或5
void UART1RxDMA_Config(u8* buf0,u8 *buf1,u16 num,u8 width)
{
    u32 memwidth=0,perwidth=0;     
    switch(width)
    {
        case 0:      
            memwidth=DMA_MDATAALIGN_BYTE;
            perwidth=DMA_PDATAALIGN_BYTE;
            break;
        case 1:     
            memwidth=DMA_MDATAALIGN_HALFWORD;
            perwidth=DMA_PDATAALIGN_HALFWORD;
            break;
        case 2:      
            memwidth=DMA_MDATAALIGN_WORD;
            perwidth=DMA_PDATAALIGN_WORD;
            break;
            
    }
        //impinjbuf1=mymalloc(SRAMEX,IMPINJ_BUF_SIZE);
        //impinjbuf2=mymalloc(SRAMEX,IMPINJ_BUF_SIZE);
    __HAL_RCC_DMA2_CLK_ENABLE();                                
    __HAL_LINKDMA(&UART1_Handler,hdmarx,UART1RxDMA_Handler);      
    UART1RxDMA_Handler.Instance=DMA2_Stream5;                                          
    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=perwidth;           
    UART1RxDMA_Handler.Init.MemDataAlignment=memwidth;            
    UART1RxDMA_Handler.Init.Mode=DMA_CIRCULAR;                       
    UART1RxDMA_Handler.Init.Priority=DMA_PRIORITY_MEDIUM;           
    UART1RxDMA_Handler.Init.FIFOMode=DMA_FIFOMODE_DISABLE;         
    UART1RxDMA_Handler.Init.MemBurst=DMA_MBURST_SINGLE;            
    UART1RxDMA_Handler.Init.PeriphBurst=DMA_PBURST_SINGLE;           
    HAL_DMA_DeInit(&UART1RxDMA_Handler);                           
    HAL_DMA_Init(&UART1RxDMA_Handler);                              
   
    HAL_DMAEx_MultiBufferStart(&UART1RxDMA_Handler,(u32)&USART1->DR,(u32)buf0,(u32)buf1,num);
    __HAL_DMA_DISABLE(&UART1RxDMA_Handler);                        
    delay_us(10);                                                
    __HAL_DMA_CLEAR_FLAG(&UART1RxDMA_Handler,DMA_FLAG_TCIF1_5);//清除传输完成中断标志位     
    __HAL_DMA_ENABLE_IT(&UART1RxDMA_Handler,DMA_IT_TC);    //开启传输完成中断         
   
    HAL_NVIC_SetPriority(DMA2_Stream5_IRQn,0,1);                 
    HAL_NVIC_EnableIRQ(DMA2_Stream5_IRQn);
}

void ImpinjRxDMA_Start(void)
{   
    __HAL_DMA_ENABLE(&UART1RxDMA_Handler);
}

void ImpinjRxDMA_Stop(void)
{
    __HAL_DMA_DISABLE(&UART1RxDMA_Handler);
}

void DMA2_Stream5_IRQHandler(void)
{  
    if(__HAL_DMA_GET_FLAG(&UART1RxDMA_Handler,DMA_FLAG_TCIF1_5)!=RESET) //DMA传输完成
    {
        __HAL_DMA_CLEAR_FLAG(&UART1RxDMA_Handler,DMA_FLAG_TCIF1_5);     //清除DMA传输完成中断标志位
        UART1RxDMA_callback();    //执行回调函数,读取数据等操作在这里面处理  
    }        
}

//这个回调函数里是切换传输完成内存块的地址
//我所用的双缓冲是在一个大缓存里取相邻的两个小缓存
void UART1RxDMA_callback(void)
{      
        sw_cnt=sw_cnt+1;
      if(sw_cnt/2==1)
        {
            sw_0=sw_0+1;
        }
        else
        {
            sw_1=sw_1+1;
        }            
        if(DMA2_Stream5->CR&(1<<19))
        {
            DMA2_Stream1->M0AR = (u32)&IPJ_BUF[sw_0*2*IPJ_miniBUF_SIZE];
            LED0=!LED0;
        }
        else
        {
            DMA2_Stream1->M1AR = (u32)&IPJ_BUF[IPJ_miniBUF_SIZE+sw_1*2*IPJ_miniBUF_SIZE];
            LED1=!LED1;
        }            
}

//初始化
IPJ_BUF_MALLOC();   //大缓存内存申请
UART1RxDMA_Config(IPJ_BUF,&IPJ_BUF[IPJ_miniBUF_SIZE],IPJ_miniBUF_SIZE,0);//开始采用大缓存的首地址和一个偏移地址,传输项目量为偏移地址大小即小缓存大小
ImpinjRxDMA_Start();//开启传输


问题:无法进入传输完成中断

友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
15条回答
mowenhui
2019-07-22 00:57
mowenhui 发表于 2016-11-24 22:44
现在能分别进入M0AR和M1AR切换了,但是第一包数据(很大的话)能两个MxAR多次切换成功,而第二包数据就不 ...

说错了 不是CR寄存器的问题  我跟踪到应该是 DMA_SxCR_DBM这个寄存器的值没有被置位
导致

if(((hdma->Instance->CR) & (uint32_t)(DMA_SxCR_DBM)) != RESET)
      {
        /* Current memory buffer used is Memory 0 */
        if((hdma->Instance->CR & DMA_SxCR_CT) == RESET)
        {
            //自定义1号缓存函数
                        Memory1CpltCallback(hdma);
        }
        /* Current memory buffer used is Memory 1 */
        else
        {
                        //自定义0号缓存函数
                        Memory0CpltCallback(hdma);
        }
      }
      /* Disable the transfer complete interrupt if the DMA mode is not CIRCULAR */
      else if((hdma->Instance->CR) != RESET)
      {
        if((hdma->Instance->CR & DMA_SxCR_CIRC) == RESET)
        {
          /* Disable the transfer complete interrupt */
          hdma->Instance->CR  &= ~(DMA_IT_TC);

          /* Process Unlocked */
          __HAL_UNLOCK(hdma);

          /* Change the DMA state */
          hdma->State = HAL_DMA_STATE_READY;
        }

                RxDMA_callback(hdma);  //自定义DMA接收完成处理函数
      }

一周热门 更多>