相关的配置函数
//串口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();//开启传输
问题:无法进入传输完成中断
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
说错了 不是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接收完成处理函数
}
一周热门 更多>