OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 2982|回复: 1

串口1使用DMA发送接收(使用寄存器),搞定。

[复制链接]

15

主题

68

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
290
金钱
290
注册时间
2014-1-2
在线时间
62 小时
发表于 2017-2-2 14:00:36 | 显示全部楼层 |阅读模式
本帖最后由 llqzx 于 2017-2-2 14:01 编辑

搜了好久,都没有在论坛和其他地方找到这个,关于串口1使用DMA发送和接收的程序,论坛有一个是使用的库函数,但本着研究寄存器的精神,我还是花了点时间自己搞定了这个。
其间,由于一个小小的失误,怎么都调不通,看来基本功还是要练习。不多说上代码。
[mw_shl_code=applescript,true]void uart_init(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;//IO状态设置
        GPIOA->CRH|=0X000008B0;//IO状态设置
        RCC->APB2RSTR|=1<<14;   //复位串口1
        RCC->APB2RSTR&=~(1<<14);//停止复位                     
         USART1->BRR=mantissa; // 波特率设置

        USART1->CR1 |= 0X200C;  //无校验位.
        USART1->CR2 &= 0<<12;//1位停止
        USART1->CR3 |= 1<<6;   //使能DMA发送接收
        USART1->CR3 |= 1<<7;   //使能DMA发送接收               

#if EN_USART1_RX                  //如果使能了接收
        USART1->CR1 |= 1<<4;    //空闲中断使能                    
        MY_NVIC_Init(3,2,USART1_IRQn,2);//组2,最低优先级
#endif        
        
        RCC->AHBENR |= 1<<0;                        //开启DMA1时钟
        delay_ms(10);                                //等待DMA时钟稳定
//        //=============DMA发送配置=========================================
        DMA1_Channel4->CPAR = (u32)(&USART1->DR);                  //DMA1 外设地址
        DMA1_Channel4->CMAR = (u32)DMA_Tran_Buf;         //DMA1,存储器地址
        DMA1_Channel4->CNDTR = DMA_Tran_Len;  //DMA1,传输数据量
        DMA1_Channel4->CCR = 0x00000000;            //复位
        DMA1_Channel4->CCR |= 1<<4;                        //传输方向,从内存读取发送到外设
        DMA1_Channel4->CCR |= 0<<5;                        //普通模式,不执行循环操作
        DMA1_Channel4->CCR |= 0<<6;                       //外设地址非增量模式
        DMA1_Channel4->CCR |= 1<<7;                        //存储器增量模式
        DMA1_Channel4->CCR |= 0<<8;                        //外设数据宽度为8位
        DMA1_Channel4->CCR |= 0<<10;                       //存储器数据宽度8位
        DMA1_Channel4->CCR |= 1<<12;                       //中等优先级
        DMA1_Channel4->CCR |= 0<<14;                       //非存储器到存储器模式        
        
        DMA1->IFCR |= 1<<12;                                                                        //清除Channel4所有标志
  DMA1_Channel4->CCR |= 1<<1;                                                  //允许传输完成中断
  MY_NVIC_Init(3,3,DMA1_Channel4_IRQn,2);
        DMA1_Channel4->CCR &= ~(1<<0);        //关闭DMA的Channel4        
        //=============DMA接收配置=========================================
        DMA1_Channel5->CPAR = (u32)(&USART1->DR);                  //DMA1 外设地址
        DMA1_Channel5->CMAR = (u32)DMA_Rece_Buf;         //DMA1,存储器地址
        DMA1_Channel5->CNDTR = DMA_Rec_Len;            //DMA1,传输数据量
        DMA1_Channel5->CCR = 0x00000000;                                //复位
        DMA1_Channel5->CCR |= 0<<4;                                                  //传输方向,从外设读取发送到内存
        DMA1_Channel5->CCR |= 0<<5;                                                  //普通模式
        DMA1_Channel5->CCR |= 0<<6;                                                 //外设地址非增量模式
        DMA1_Channel5->CCR |= 1<<7;                                                  //存储器增量模式
        DMA1_Channel5->CCR |= 0<<8;                                                  //外设数据宽度为8位
        DMA1_Channel5->CCR |= 0<<10;                                                 //存储器数据宽度8位
        DMA1_Channel5->CCR |= 1<<12;                                                 //中等优先级
        DMA1_Channel5->CCR |= 0<<14;                                                 //非存储器到存储器模式        
        
        DMA1->IFCR |= 1<<16 ;                                                                         //清除Channel5全局中断标志
        DMA1_Channel5->CCR |= 1<<0;                          //开启DMA传输  //使能USART1 RX DMA1 所指示的通道
}
[/mw_shl_code]

由于开始的时候把 外设地址那一行写成了,DMA1_Channel4->CPAR = USART1->DR;,怎么调都是错误,这句话在语法上是没问题的,检查的时候都忽略了这一行,最后还是这一样出了问题,艾西吧。

下面上传源工程,希望对后来者有帮助。
工程目的:串口发送数据,DMA接收,在空闲中断中,DMA发送接收到的数据,DMA发送中断,关DMA发送,上位机收到刚才发送的内容。
UART DMA发送接收 OK.rar (565.86 KB, 下载次数: 157)
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

5

主题

179

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
8195
金钱
8195
注册时间
2016-9-7
在线时间
1113 小时
发表于 2017-2-2 23:38:31 | 显示全部楼层
感谢分享。开工后试试看
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-8 00:07

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表