环境:
主机:WIN8
开发环境:MDK5.13
mcu: stm32f407VGT6
说明:
之前用STM32F103实现DMA收发串口数据,现在项目中采用STM32F407,所以将此机制移植到F4上。
STM32F103上用DMA收发串口数据文章:
STM32的串口采用DMA方式发送数据测试
STM32的串口采用DMA方式接收数据测试
源代码:
串口初始化代码:
/*********************************************************************
* 初始化串口
**********************************************************************/
static void init_uart(void)
{
//定义中断结构体
NVIC_InitTypeDef NVIC_InitStructure ;
//定义IO初始化结构体
GPIO_InitTypeDef GPIO_InitStructure;
//定义串口结构体
USART_InitTypeDef USART_InitStructure;
//定义DMA结构体
DMA_InitTypeDef DMA_InitStructure;
//打开串口对应的外设时钟
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
//串口发DMA配置
//启动DMA时钟
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
//DMA发送中断设置
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream6_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
//DMA通道配置
DMA_DeInit(DMA1_Stream6);
DMA_InitStructure.DMA_Channel = DMA_Channel_4;
//外设地址
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&USART2->DR);
//内存地址
DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)Tx_Buf_Gsm;
//dma传输方向
DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral;
//设置DMA在传输时缓冲区的长度
DMA_InitStructure.DMA_BufferSize = TX_LEN_GSM;
//设置DMA的外设递增模式,一个外设
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
//设置DMA的内存递增模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
//外设数据字长
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
//内存数据字长
DMA_InitStructure.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;
//设置DMA的传输模式
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
//设置DMA的优先级别
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
//指定如果FIFO模式或直接模式将用于指定的流 : 不使能FIFO模式
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
//指定了FIFO阈值水平
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
//指定的Burst转移配置内存传输
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
//指定的Burst转移配置外围转移 */
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
//配置DMA1的通道
DMA_Init(DMA1_Stream6, &DMA_InitStructure);
//使能中断
DMA_ITConfig(DMA1_Stream6,DMA_IT_TC,ENABLE);
//串口收DMA配置
//启动DMA时钟
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
//DMA通道配置
DMA_DeInit(DMA1_Stream5);
DMA_InitStructure.DMA_Channel = DMA_Channel_4;
//外设地址
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&USART2->DR);
//内存地址
DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)Rx_Buf_Gsm;
//dma传输方向
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
//设置DMA在传输时缓冲区的长度
DMA_InitStructure.DMA_BufferSize = RX_LEN_GSM;
//设置DMA的外设递增模式,一个外设
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
//设置DMA的内存递增模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
//外设数据字长
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
//内存数据字长
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
//设置DMA的传输模式
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
//设置DMA的优先级别
DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;
//指定如果FIFO模式或直接模式将用于指定的流 : 不使能FIFO模式
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
//指定了FIFO阈值水平
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
//指定的Burst转移配置内存传输
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
//指定的Burst转移配置外围转移 */
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
//配置DMA1的通道
DMA_Init(DMA1_Stream5, &DMA_InitStructure);
//使能通道
DMA_Cmd(DMA1_Stream5,ENABLE);
//初始化串口参数
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
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_InitStructure.USART_BaudRate = DEFAULT_BAUD_GSM;
//初始化串口
USART_Init(USART2,&USART_InitStructure);
//中断配置
USART_ITConfig(USART2,USART_IT_TC,DISABLE);
USART_ITConfig(USART2,USART_IT_RXNE,DISABLE);
USART_ITConfig(USART2,USART_IT_IDLE,ENABLE);
//配置中断
//NVIC_PriorityGroupConfig(NVIC_PriorityGroup_3);
NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn; //通道设置为串口中断
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //中断占先等级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //中断响应优先级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //打开中断
NVIC_Init(&NVIC_InitStructure);
//采用DMA方式发送
USART_DMACmd(USART2,USART_DMAReq_Tx,ENABLE);
//采用DMA方式接收
USART_DMACmd(USART2,USART_DMAReq_Rx,ENABLE);
//中断配置
USART_ITConfig(USART2,USART_IT_TC,DISABLE);
USART_ITConfig(USART2,USART_IT_RXNE,DISABLE);
USART_ITConfig(USART2,USART_IT_TXE,DISABLE);
USART_ITConfig(USART2,USART_IT_IDLE,ENABLE);
//启动串口
USART_Cmd(USART2, ENABLE);
//设置IO口时钟
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_USART2);
GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_USART2);
//管脚模式:输出口
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
//类型:推挽模式
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
//上拉下拉设置
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
//IO口速度
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
//管脚指定
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
//初始化
GPIO_Init(GPIOA, &GPIO_InitStructure);
//管脚模式:输入口
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
//上拉下拉设置
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
//管脚指定
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
//初始化
GPIO_Init(GPIOA, &GPIO_InitStructure);
}
发送数据:
/*********************************************************************
* 接口函数:向gsm模块发送数据
*参数:data:发送数据存放地址
* size:发送数据字节数
**********************************************************************/
void drv_gsm_tx(uint8_t *data,uint16_t size)
{
//等待空闲
while (Flag_Tx_Gsm_Busy);
Flag_Tx_Gsm_Busy = 1;
//复制数据
memcpy(Tx_Buf_Gsm,data,size);
//设置传输数据长度
DMA_SetCurrDataCounter(DMA1_Stream6,size);
//打开DMA,开始发送
DMA_Cmd(DMA1_Stream6,ENABLE);
}
中断处理函数:
/*********************************************************************
* 接口函数:DMA发送中断处理函数
**********************************************************************/
void drv_gsm_deal_irq_dma_tx(void)
{
if(DMA_GetITStatus(DMA1_Stream6,DMA_IT_TCIF6) != RESET)
{
//清除标志位
DMA_ClearFlag(DMA1_Stream6,DMA_FLAG_TCIF6);
//关闭DMA
DMA_Cmd(DMA1_Stream6,DISABLE);
//打开发送完成中断,发送最后两个字节
USART_ITConfig(USART2,USART_IT_TC,ENABLE);
}
}
/*********************************************************************
* 处理发送完成中断
*返回:0:未产生,1:已经产生
**********************************************************************/
uint8_t drv_gsm_deal_irq_tx_end(void)
{
if(USART_GetITStatus(USART2, USART_IT_TC) != RESET)
{
//关闭发送完成中断
USART_ITConfig(USART2,USART_IT_TC,DISABLE);
//发送完成
Flag_Tx_Gsm_Busy = 0;
return 1;
}
return 0;
}
/*********************************************************************
* 处理接收完成中断
*参数:buf:接收的数据
* len:接收的数据长度
*返回:0:未产生,其他:已经产生,此值为接收的数据长度
**********************************************************************/
uint8_t drv_gsm_deal_irq_rx_end(uint8_t *buf)
{
uint16_t len = 0;
//接收完成中断
if(USART_GetITStatus(USART2, USART_IT_IDLE) != RESET)
{
USART2->SR;
USART2->DR; //清USART_IT_IDLE标志
//关闭DMA
DMA_Cmd(DMA1_Stream5,DISABLE);
//清除标志位
DMA_ClearFlag(DMA1_Stream5,DMA_FLAG_TCIF5);
//获得接收帧帧长
len = RX_LEN_GSM - DMA_GetCurrDataCounter(DMA1_Stream5);
memcpy(buf,Rx_Buf_Gsm,len);
//设置传输数据长度
DMA_SetCurrDataCounter(DMA1_Stream5,RX_LEN_GSM);
//打开DMA
DMA_Cmd(DMA1_Stream5,ENABLE);
return len;
}
return 0;
}
中断函数:
/*********************************************************************
* GSM模块:DMA发送中断处理函数
**********************************************************************/
void DMA1_Stream6_IRQHandler(void)
{
gsm_dma_tx_irq_handler();
}
/*********************************************************************
* GSM模块:串口中断处理函数
**********************************************************************/
void USART2_IRQHandler(void)
{
gsm_irq_handler();
}
注意:
接收完成中断中必须要清除标志位
//清除标志位
DMA_ClearFlag(DMA1_Stream5,DMA_FLAG_TCIF5);
并且必须在关闭DMA之后,这个语句在F103中并非必须的
中断处理函数:
/*********************************************************************
* 接口函数:DMA发送中断处理函数
**********************************************************************/
void gsm_dma_tx_irq_handler(void)
{
inf_gsm_deal_irq_dma_tx();
}
/*********************************************************************
* 接口函数:串口中断处理函数
*参数:data:接收数据存放地址
*返回:接收数据长度
**********************************************************************/
上一篇:stm32F4各个库文件的作用分析
下一篇:STM32F407上移植emwin
推荐阅读
史海拾趣
尽管已经取得了显著的成就,但ConnectBlue公司并没有止步不前。相反,公司继续坚持创新驱动的发展战略,不断推出新产品、新技术和新服务。同时,公司还积极关注行业发展趋势和未来技术方向,为公司的长期发展做好充分准备。展望未来,ConnectBlue公司将继续致力于无线通信技术的研发和应用,为推动电子行业的进步和发展做出更大的贡献。
这五个故事展示了ConnectBlue公司在电子行业中的发展历程和取得的成就。从创立初期的艰难起步到产品创新的突破、市场拓展的成功、被u-blox收购后的新跨越以及持续创新的未来展望,ConnectBlue公司始终保持着积极进取的精神和不断创新的态度,在无线通信领域取得了显著的成绩。
自上世纪60年代起,TDK就开始在中国布局。首先,在台湾建立合资公司,随后在香港设立销售网络及生产线。进入80年代后,TDK正式踏足中国大陆,在华东及华南多个地区建立生产基地,并逐渐将业务扩展至全国。TDK在中国的发展不仅推动了当地电子工业的进步,也为公司自身赢得了广阔的市场空间。目前,TDK在中国的员工人数近6万人,占TDK全球员工总数的近55%。
在不断追求技术创新的道路上,Auris公司并未止步。近年来,该公司正在积极研发柔性机器人技术,这种技术有望在治疗喉咙、肺和肠胃疾病方面发挥重要作用。柔性机器人的研发不仅展示了Auris在技术创新方面的持续努力,也为其在未来的医疗市场中抢占先机提供了有力支持。
这五个故事展示了Auris公司在电子行业中的发展历程,从创始人的科技愿景到技术突破获得认可,再到与强生的合作以及柔性机器人的研发,每一步都体现了Auris对于技术创新和市场拓展的不懈追求。这些事实性的故事不仅展现了Auris公司的成长轨迹,也反映了电子行业中技术创新和市场竞争的激烈态势。
Auris公司的创始人Federic Moll博士,被誉为手术机器人之父。他在2007年创立了Auris Health(原名Auris Surgical Robotics),总部位于美国加利福尼亚州圣卡洛斯。Moll博士此前已经成功创办了多家手术机器人企业,其中最为知名的是上市公司Intuitive Surgical,其达芬奇手术机器人技术已在全球范围内广泛应用。Moll博士的深厚技术背景和丰富的创业经验为Auris的起步奠定了坚实的基础。
证券行业普遍面临的问题: 信息交互问题:信息传递的时间段集中、信息交互量大,交互次数频繁,难以实现数据集中以及实时交互管理; 资讯服务问题:股市行情多变,客户不便随时随地及时准确的了解交易情况和信息; 系统及数据安全; 客户管理; ...… 查看全部问答∨ |
|
C51单片机中P0口和P2口作为拓展片外RAM时是如何工作的? 确切的说,我想知道用C语言如何对P0口P2口赋值以达到对片外RAM中数据的访问。但现在的问题是,我不知道P0口和P2口如何工作的?P0口作为“数据/地址”复用端口时,对片外RAM的数据访问是自动切换的,还是要手动的对P0口P2口赋值呢? ============== ...… 查看全部问答∨ |
|
本人stm32还未入门者,图中画红线的这句话令我犯晕,既然该位被置’1’ 。为什么在该位中写入’1’可以清除它??各位大神求解释 … 查看全部问答∨ |
圆点博士微型四轴飞行器第二次裸飞成功圆点博士微型四轴飞行器第二次裸飞成功。所谓裸飞是指还没上陀螺仪的PID控制,加电让它飞。 $(\'flv_H1b\').innerHTML=(mobileplayer() ? \"\" : AC_FL_RunContent(\'width\', \'500\', \'height\', \'375\ ...… 查看全部问答∨ |
小弟是刚开始学的51单片机,想做个科研立项,没有什么好的想法 ,麻烦各位师兄师姐帮帮忙,想下,最好是新颖些,难度不要太难,我想做一次就做好它,先谢谢大家了… 查看全部问答∨ |