STM32串口DMA发送中断配置

发布者:Xiaoxue666最新更新时间:2019-08-10 来源: eefocus关键字:STM32  串口  DMA发送  中断配置 手机看文章 扫描二维码
随时随地手机看文章

本文以STM32F1xx的串口1为例,简单呈现下使用DMA中断连续发送的代码

串口DMA配置通常可以分为2个部分:

1.串口配置

2.DMA配置


串口配置

void UART1_Init(u32 bound){

GPIO_InitTypeDef GPIO_InitStructure;

USART_InitTypeDef USART_InitStructure;

NVIC_InitTypeDef NVIC_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO, ENABLE); //使能USART1,GPIOA时钟

  

//USART1_TX   GPIOA.9

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出

GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9


NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器  

  //USART1_RX   GPIOA.10初始化

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA10

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入

GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10  


NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子优先级3

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能 

//USART 初始化设置

USART_InitStructure.USART_BaudRate = bound;//串口波特率

USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式

USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位

USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位

USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制

USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //收发模式


USART_Init(USART1, &USART_InitStructure); //初始化串口1

USART_ITConfig(USART1,USART_IT_IDLE,ENABLE);  //使能空闲中断

USART_Cmd(USART1, ENABLE);                    //使能串口1 

}


2.DMA配置

本代码封装了下DMA配置


void DMA_Config(DMA_Channel_TypeDef*DMA_CHx,u32 cpar,u32 cmar,u32 dma_dir,u32 priority,u16 bufsize)

{

DMA_InitTypeDef DMA_InitStructure;

  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输

  DMA_DeInit(DMA_CHx);   //将DMA的通道1寄存器重设为缺省值

DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;  //DMA外设ADC基地址

DMA_InitStructure.DMA_MemoryBaseAddr = cmar;  //DMA内存基地址

DMA_InitStructure.DMA_DIR = dma_dir;  //数据传输方向,从内存读取发送到外设

DMA_InitStructure.DMA_BufferSize = bufsize;  //DMA通道的DMA缓存的大小

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //内存地址寄存器递增

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //数据宽度为8位

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; //数据宽度为8位

DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //工作在正常缓存模式

DMA_InitStructure.DMA_Priority = priority; //DMA通道 x拥有中优先级 

DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输

DMA_Init(DMA_CHx, &DMA_InitStructure);  


void UART_DMA_Config{

NVIC_InitTypeDef NVIC_InitStructure;


 //配置串口1发送DMA

DMA_Config(DMA1_Channel4,(u32)&USART1->DR,(u32)USART1_TX_BUF,DMA_DIR_PeripheralDST,DMA_Priority_Medium,0);//DMA1通道4,外设为串口1Tx,存储器为USART_TX_BUF 

//配置串口1接收DMA

DMA_Config(DMA1_Channel5,(u32)&USART1->DR,(u32)USART1_RX_BUF,DMA_DIR_PeripheralSRC,DMA_Priority_Medium,0);//DMA1通道5,外设为串口1Rx,存储器为USART_RX_BUF


NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0 ;//抢占优先级3

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //子优先级3

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能

NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器


DMA_ITConfig(DMA1_Channel4,DMA_IT_TC,ENABLE); //开USART1 Tx DMA中断

USART_DMACmd(USART1,USART_DMAReq_Tx,ENABLE);  //使能串口3的DMA接收

USART_DMACmd(USART1,USART_DMAReq_Rx,ENABLE);  //使能串口3的DMA接收

}


最后是中断函数


void USART1_IRQHandler(void)            //串口1中断服务程序

{

if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET)  

{  

USART1_RX_CNT = USART1->SR;  

USART1_RX_CNT = USART1->DR; //清USART_IT_IDLE标志

DMA_Cmd(DMA1_Channel5,DISABLE);  

USART_ITConfig(USART1, USART_IT_IDLE, DISABLE); //静止中断,防覆盖

USART1_RX_CNT = USART1_REC_LEN - DMA_GetCurrDataCounter(DMA1_Channel5);  

DMA_SetCurrDataCounter(DMA1_Channel5,USART1_REC_LEN);  

DMA_Cmd(DMA1_Channel5,ENABLE);  

USART_ITConfig(USART1, USART_IT_IDLE, ENABLE); 

}   

}


 //串口1 TX DMA终端

void DMA1_Channel4_IRQHandler(void)   

{

if(DMA_GetITStatus(DMA1_FLAG_TC4))

{

DMA_ClearFlag(DMA1_FLAG_TC4);

if(Txed_Num > Max_Tx_Cnt)

{

Txed_Num = 0;

return;

}

TX_DATA(DATA_Len);

Txed_Num = Txed_Num + 1;

}

}


//DMA发送代码

void TX_DATA(u8 len)

{

while(DMA_GetCurrDataCounter(DMA1_Channel4)!=0); //等待通道7传输完成   

DMA_Cmd(DMA1_Channel4, DISABLE );  //关闭 指示的通道        

DMA_SetCurrDataCounter(DMA1_Channel4,len);//DMA通道的DMA缓存的大小

DMA_Cmd(DMA1_Channel4, ENABLE);           //开启DMA传输

}


总结

串口DMA发送中断配置时,易遗漏


DMA_ITConfig(DMA1_Channel4,DMA_IT_TC,ENABLE);


本例是DMA发送中断在 连续、分次发送 场景下的应用。



关键字:STM32  串口  DMA发送  中断配置 引用地址:STM32串口DMA发送中断配置

上一篇:正点原子stm32模拟IIC
下一篇:STM32三线检测八个按键(74HC165)

推荐阅读最新更新时间:2024-11-08 19:52

基于MSP430的无线传输协议
该协议基于在同一块MSP430上用串口1发射,串口0接收,使用两块无线收发的NRF401模块。 初始化串口0,用于接收 void init_UART0(void) { UCTL0 &= ~SWRST; UCTL0 |=CHAR;//0X10; UBR00 = 0x03; UBR10 = 0x00; UMCTL0 = 0x4A; UTCTL0 = SSEL0;//0X10; ME1 |= UTXE0+URXE0 ; //enable the UART0 IE1 |=URXIE0; P3SEL|=BIT4+BIT5; P3DIR|=BIT4; Clinet_Receive; } 初始化串口1,用于
[单片机]
STM32 USB NAND Flash模拟U盘无法格式化问题的解决
前几天,一直在寻找NAND Flash模拟U盘程序无法格式化的问题。在中秋月圆之夜,还苦逼地在实验室调代码,也许是杭州大圆月的原因,今晚感觉整人特别亢奋,效率也特别高,灵感也多。终于,在不懈的努力下,找到代码中的害群之马,把无法格式的问题解决掉了。下面就来说说。 这几天一直在想问题出在哪里,不知道自己的代码跟官方的例程对照了多少次,把不一样的地方全都改了一遍,最终未果。今晚思路特别清晰,于是在想到格式化实际上就是向存储器写数据而已,而设计到写数据部分的代码就只在mass_mal.c、memory.c以及存储器的驱动文件。于是反复检查这几个文件,终于发现一点端倪了,问题出在memroy.c这个文件里。 memory.c这个文件只有两
[单片机]
STM32值系统时钟初始化程序设计
正如之前文章所述,在使用外设功能时,都必须提前对其精心初始化才能使用其功能。 以下是系统定时器初始化的程序清单。 #include stm32f10x_lib.h unsigned char sys_nub;//系统定时器中断计数变量 //SysTick 设置 void Systick_Config(void) { //失能SysTick定时器 SysTick_CounterCmd(SysTick_Counter_Disable); //失能SysTick中断 SysTick_ITConfig(DISABLE); //设置SysTick时钟源 SysTick_ClockSourceConfig(SysTick_Source_HC
[单片机]
STM32 USB CAN 学习笔记 - 共享RAM的用法
USB 时钟可以一直使能。 如果CAN时钟没有使能,RAM 能被软件读写。(CANBus 不能发送和接受Message) 如果CAN时钟使能,RAM不能软件被写。 CANBus Core 控制此RAM区域。(CANBus 可以发送和接受Message) CANBus 占用了 256 字节(地址范围 512 字节)。 开始区域是Filters = (F0R0、F0R1), (F1R0、F1R1), ... (F13R0、F13R1). 紧接着是 Rx0 FiFo , Tx0, Tx1, Tx2, Rx0 FiFo , Rx0 FiFo , Rx1 FiFo , Rx1 FiFo , Rx1 FiFo 。 USB 可以使用
[单片机]
STM32通过IIC读写EEPROM(24C02)
STM32作为主机I2C,读写24C02 EEPROM 1、 时钟和数据的传输:开始和停止条件,数据在SCL的高电平期间有效,在SCL的低电平期间改变。 2、 开始条件:在SCL高电平期间,SDA产生一个下降沿 3、 停止条件:在SCL高电平期间,SDA产生一个上升沿 4、 应答:成功接收到数据(地址和数据),产生一个应答位(在第9个时钟周期,将SDA拉低) 下面是源程序:原理上说,下面程序再移植时,只要将数据类型变化,可以应用到任何处理器 AT24c02.h #ifndef __24CXX_H #define __24CXX_H #include i2c.h /*********************************
[单片机]
stm32快速学习1——环境的建立
安装keil for arm 下载stm32固件库3.4 解压,勿乱移动解压文件内的文件,因为项目内使用的都是相对路径。 进入 STM32F10x_StdPeriph_Lib_V3.4.0\Project\STM32F10x_StdPeriph_Template\MDK-ARM 打开keil文件,修改一下这个项目模板 删除左侧的stm32_eval文件夹、doc文件夹、使用的是stm32f103RBT6(中等容量的芯片)然后MDK-ARM下的文件除了 startup_stm32f10x_md.s(根据芯片类型),其他都删除 startup_stm32f10x_md.s右键,选中include in target build 左侧最
[单片机]
STM32学习——矩阵键盘
在之前的学习中就学过了IO口的操作,矩阵键盘就是对IO口的操作,还有一个就是扫描模式。 方法一: 逐行扫描:通过高四位轮流输出低电平来对矩阵键盘进行逐行扫描,当低四位接收到的数据不全为1的时候,说明有按键按下,然后通过接收到的数据是哪一位为0来判断是哪一个按键被按下。 方法二: 行列扫描:通过高四位全部输出低电平,低四位输出高电平。当接收到的数据,低四位不全为高电平时,说明有按键按下,然后通过接收的数据值,判断是哪一列有按键按下,然后再反过来,高四位输出高电平,低四位输出低电平,然后根据接收到的高四位的值判断是那一行有按键按下,这样就能够确定是哪一个按键按下了。 在这里用到了中断的方法,行输出高电平,列做模拟输入口。当有按键按
[单片机]
<font color='red'>STM32</font>学习——矩阵键盘
STM32 FSMC 总结
背景 做DP开发过程中,需要将DP芯片和ARM芯片进行通信,一般有两种方式: 1)IO端口(总线)方式来进行通信; 2)FSMC-存储器映射方式来进行通信; 之前的开发是采用第一种方式在F103芯片上开发,现在换F4平台,采用第二种方式来进行交互数据。 FSMC的基础 FSMC功能是类似 51 单片机的存储器映射功能,能管理多个外部不同种类的存储设备,具有方便,简单,快捷的优点。 目前支持的类型有: SRAM 、 ROM 、 PSRAM 、 NOR Flash 和NANDFlash 存储器 。 更多的关于FSMC的资料需要阅读STM32 手册对于FSMC的详细介绍。 VPC3+C + ARM +
[单片机]
<font color='red'>STM32</font> FSMC 总结
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved