程序是这样的
设置usart1的DMA发送功能,当串口接收完数据后,触发dma传输完成中断,但是进入中断时,程序跑飞了
不开dma中断的话,程序是可以正常运行的
下面是是实现的代码
//DMA1_Channel4 usart1发送
void MYDMA_USART_Send(DMA_Channel_TypeDef*DMA_CHx, u32 cpar, u32 cmar, u16 cndtr)
{
RCC->AHBENR|=1<<0; //开启DMA1时钟
delay_ms(1); //等待DMA时钟稳定
DMA_CHx->CPAR=cpar; //DMA1 外设地址
DMA_CHx->CMAR=(u32)cmar; //DMA1,存储器地址
DMA1_MEM_LEN1=cndtr; //保存DMA传输数据量
DMA_CHx->CNDTR=cndtr; //DMA1,传输数据量
DMA_CHx->CCR=0X00000000; //复位
DMA_CHx->CCR|=1<<4; //从存储器读
DMA_CHx->CCR|=0<<5; //普通模式
DMA_CHx->CCR|=0<<6; //外设地址非增量模式
DMA_CHx->CCR|=1<<7; //存储器增量模式
DMA_CHx->CCR|=0<<8; //外设数据宽度为8位
DMA_CHx->CCR|=0<<10; //存储器数据宽度8位
DMA_CHx->CCR|=1<<12; //中等优先级
DMA_CHx->CCR|=0<<14; //非存储器到存储器模式
DMA_CHx->CCR|=1<<1; //允许传输完成中断
MY_NVIC_Init(3,1,DMA1_Channel4_IRQChannel,2);
}
//开启一次DMA传输
void MYDMA_Enable_Send(DMA_Channel_TypeDef*DMA_CHx,u16 cndtr)
{
DMA_CHx->CCR&=~(1<<0); //关闭DMA传输
DMA_CHx->CNDTR=cndtr; //DMA1,传输数据量
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
void DMA1_Channel4_IRQHandler(void)
{
if((DMA1->ISR & DMA1_FLAG_TC4)==(u32)1)//判断传输完成中断位是否置位
{
DMA1->IFCR |= DMA1_FLAG_TC4; //清除完成传输中断
LED0=!LED0;
//printf("111");
}
}
void uart_init1(u32 pclk2,u32 bound)
{
float temp;
u16 mantissa;
u16 fraction;
temp=(float)(pclk2*1000000)/(bound*16);//得到USARTDIV
mantissa=temp; //得到整数部分
fraction=(temp-mantissa)*16; //得到小数部分
mantissa<<=4;
mantissa+=fraction;
RCC->APB2ENR|=1<<2; //使能PORTA口时钟
RCC->APB2ENR|=1<<14; //使能串口时钟
GPIOA->CRH&=0XFFFFF00F;
GPIOA->CRH|=0X000008B0; //IO状态设置
RCC->APB2RSTR|=1<<14; //复位串口1
RCC->APB2RSTR&=~(1<<14); //停止复位
//波特率设置
USART1->BRR=mantissa; // 波特率设置
USART1->CR1|=0X200C ; //1位停止,无校验位.
USART1->CR3|=1<<7; //使能DMA发送
//使能接收中断
// USART1->CR1|=1<<8 ; //PE中断使能
// USART1->CR1|=1<<5 ; //接收缓冲区非空中断使能
// MY_NVIC_Init(3,3,USART1_IRQChannel,2);//组2,最低优先级
MYDMA_USART_Send (DMA1_Channel4, (u32)&USART1->DR, (u32)txdata, 0);
}
|