STM32串口发送中断踩坑
今天想测试下Modbus设备,手上暂时没有串口转485的模块,就打算用手上的stm32f042的开发板做个串口转485模块。如下所示
但是软件实际开发过程中,遇到了麻烦。
现象: 在打开串口接收中断时,串口会一直产生除接收中断外的其它中断,非常奇怪。
USART_ITConfig(InitPort, USART_IT_RXNE, ENABLE); //使能接收中断
通过查手册发现,在打开接收中断时,默认会打开溢出中断
下面的方式是不能清除溢出中断标记。
if (USART_GetFlagStatus(USART2, USART_FLAG_ORE) != RESET){USART_ClearFlag(USART2, USART_FLAG_ORE);}
可使用如下方式清除溢出中断,但是要使能溢出中断
USART_ITConfig(USART1, USART_IT_ORE, ENABLE);
使用下面的方式清除溢出中断。
if (USART_GetITStatus(USART2, USART_IT_ORE) == SET) {USART_ClearITPendingBit(USART2,USART_IT_ORE);}
网上这篇文章也是不错的:http://news.eeworld.com.cn/mcu/article_2018060839675.html
虽然这种方式发送数据,暂时不会一直卡在中断里面,当发送长数据时,还是有时候会出现卡在中断里面的情况。如下我打印出了产生中断种类。
为此,在进入中断函数时,直接清理了中断标志,注意不要清理接收中断。
void USART2_IRQHandler()
{USART_ClearITPendingBit(USART2,USART_IT_ORE);USART_ClearITPendingBit(USART2,USART_IT_IDLE);USART_ClearITPendingBit(USART2,USART_IT_TXE);USART_ClearITPendingBit(USART2,USART_IT_EOB);if( USART_GetITStatus(USART2,USART_IT_RXNE) != RESET ){ USART_ClearITPendingBit(USART2,USART_IT_RXNE);while((USART1->ISR&0x40) == 0);USART1->TDR = USART2->RDR;}
}
这样的话,串口2接收的数据,直接可以通过串口1转发。
完整的串口转485代码
void usart_config(uint8_t port, uint32_t BaudRate)
{ USART_InitTypeDef USART_InitStructure;GPIO_InitTypeDef GPIO_InitStructure;NVIC_InitTypeDef NVIC_InitStructure;USART_TypeDef *InitPort = USART1; //默认是debug口RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE); //使能GPIOA的系统时钟if (port == 1) {RCC_APB2PeriphClockCmd (RCC_APB2Periph_USART1, ENABLE); //打开串口1串口时钟//GPIO MAPGPIO_PinAFConfig (GPIOA, GPIO_PinSource9, GPIO_AF_1); //初始化GPIOA的PIN9为串口功能GPIO_PinAFConfig (GPIOA, GPIO_PinSource10, GPIO_AF_1); //初始化GPIOA的PIN10为串口功能GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10; //初始化GPIOA PIN9,PIN10 口InitPort = USART1;NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;} else if (port == 2) {RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE); //打开串口2串口时钟//GPIO MAPGPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_1); //初始化GPIOA的PIN2为串口功能;GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_1); //初始化GPIOA的PIN2为串口功能;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3; //初始化GPIOA PIN2,PIN3 口InitPort = USART2;NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;}USART_DeInit (InitPort);/* Configure pins as AF pushpull */GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; //串口GPIO复用功能GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽输出GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;GPIO_Init (GPIOA, &GPIO_InitStructure);USART_InitStructure.USART_BaudRate = BaudRate; //模特率设置USART_InitStructure.USART_WordLength = USART_WordLength_8b; //数据长度为8bitUSART_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 (InitPort, &USART_InitStructure); //串口使能USART_ITConfig(InitPort, USART_IT_RXNE, ENABLE); USART_ITConfig(USART1, USART_IT_ORE, ENABLE); USART_Cmd (InitPort, ENABLE);/* USART1 IRQ Channel configuration */NVIC_InitStructure.NVIC_IRQChannelPriority = 0x00;NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;NVIC_Init(&NVIC_InitStructure);
}void USART1_IRQHandler()
{USART_ClearITPendingBit(USART1,USART_IT_ORE);USART_ClearITPendingBit(USART1,USART_IT_IDLE);USART_ClearITPendingBit(USART1,USART_IT_TXE);USART_ClearITPendingBit(USART1,USART_IT_EOB);if( USART_GetITStatus(USART1,USART_IT_RXNE) != RESET ){ USART_ClearITPendingBit(USART1,USART_IT_RXNE);while((USART1->ISR&0x40) == 0);USART2->TDR = USART1->RDR;}
}void USART2_IRQHandler()
{USART_ClearITPendingBit(USART2,USART_IT_ORE);USART_ClearITPendingBit(USART2,USART_IT_IDLE);USART_ClearITPendingBit(USART2,USART_IT_TXE);USART_ClearITPendingBit(USART2,USART_IT_EOB);if( USART_GetITStatus(USART2,USART_IT_RXNE) != RESET ){ USART_ClearITPendingBit(USART2,USART_IT_RXNE);while((USART1->ISR&0x40) == 0);USART1->TDR = USART2->RDR;}
}
发送数据测试结果:
左边发送数据到右边。总共发送了28080个字符,但是接收了27841个字符,少接收了239个字符。
波特率为57600,每次发送184个字节,可以看到发送29256字符后,丢失2个字节。注意这里我在测试时一直电机发送按钮,没有等待。这个误码率还是可以接收。
STM32串口发送中断踩坑相关推荐
- STM32串口发送中断
SECTION 2 先说TC.即Transmission Complete.发送一个字节后才进入中断,这里称为"发送后中断".和原来8051的TI方式一样,都是发送后才进中断,需要 ...
- 向STM32串口发送数据的标准函数
向STM32串口发送数据的标准函数 例子:1 void UART_Send_Message(u8 *Data,u8 lenth) {while(lenth--){USART_SendData(USAR ...
- STM32 串口发送乱码问题
STM32 串口发送乱码问题 一.问题状况: 显示为一堆乱码,
- stm32: 串口空闲中断的实现(HAL库)
STM32利用串口空闲中断来分包(HAL库) 文章目录 STM32利用串口空闲中断来分包(HAL库) 1. 开发环境 2. 串口中断接收的问题和解决办法 3. 串口空闲中断分包的原理 4. STM32 ...
- STM32串口发送数据和接收数据方式总结
文章目录 串口发送数据 串口接受数据 串口发送数据 1.串口发送数据最直接的方式就是标准调用库函数 . void USART_SendData(USART_TypeDef* USARTx, uint1 ...
- STM32串口发送接收数据
目录 1.串口通信 2.串口的结构体 3.如何配置串口的发送 4.通过串口向电脑发送ok字符 5.封装发送字符串函数 6.重定向printf串口发送 7.串口输入控制LED灯开关 遇到的问题 1.串口 ...
- STM32串口用中断还是用轮询
1.从轮询到中断 很多同学都不喜欢用中断,而偏爱用轮询的操作方式. 这是不是和我们的天性有关呢?每个人都喜欢一切尽在掌握中,肯定都不喜欢被打断.我们常常都有这样的经验:正在跟别人说一件事,然后突然有个 ...
- STM32串口IDLE中断
stm32串口的收发似乎不太一样,发只要你把数据送出去就行了,电脑会自动读取(电脑不知道你啥时候发,总不能一直等你的数据吧),而你的stm32要接受一个数据呢?一直在USART_ReceiveData ...
- STM32串口发送Ctrl+C Ctrl+Z的方法
Ctrl+C.Ctrl+Z对应的0x03和0x1A,在串口发送的时候,需要发送十六进制的数据,而不是字符 奋斗STM32中的USART_OUT函数,参数默认是发送字符串 但是进入到USART_OUT的 ...
最新文章
- 许昌科技学校工业机器人_【调研】省人大常委会副主任徐济超到许昌科技学校进行专题调研...
- 安装linux 系统报错:No DEFAULT or UI configuration directive found 解决方法
- Java 并发编程笔记(一)
- TikTok时代细分需求 牛逼亚马逊运营团队打造新爆款
- 3、leetcode 697 数组的度
- 电脑软件:推荐5款实用的效率软件,每一款都爱不释手!
- 画图讲解SQL join 语句
- 系统架构设计师考试999999999999
- 签名证书无效”-在vCenter Server Appliance 6.5 / 6.7上使用Shell脚本重新生成和替换已过期的STS证书(76719)
- iOS 通讯录编程【总结】
- Codejock Suite Pro 19.3.0 Carck版本
- 微信小游戏 资源下载解压
- 共享单车数据集_共享单车数据的数据可视化
- mysqld.exe 无法找到入口
- 201871010133-赵永军《面向对象程序设计(java)》第二周学习总结
- 北京第二外国语学院本科毕业论文答辩PPT模板
- excel不同电脑上显示不一样解决方案
- excel提取奇数行
- java indexeddb_IndexedDB使用与出坑指南
- 画论78 王概《芥子园画传》
热门文章
- java计算机毕业设计培训中心管理系统源码+系统+数据库+lw文档+mybatis+运行部署
- Scope Script Example (Scope)
- VNC 桌面没有图标解决办法
- 【王道数据结构编程题】- 二叉树算法题
- linux库的知识(概)
- 好玩的pywebio,搭建简单的web页面,超简单
- [USACO09FEB]庙会班车Fair Shuttle
- signature=e260e08d0d5973d18c37cc596c51cae8,女性不妊症領域におけるレーザー手術の現況...
- [转] 基本RS触发器
- (二)分布式存储原理与设计