STM32菜鸟成长记录---RS485通讯协议的应用

发布者:浅唱梦幻最新更新时间:2016-12-28 来源: eefocus关键字:STM32  RS485  通讯协议 手机看文章 扫描二维码
随时随地手机看文章

RS485缺点:

RS485总线是一种常规的通信总线,它不能够做总线的自动仲裁,也就是不能够同时发送数据以避免总线竞争,所以整个系统的通信效率必然较低,数据冗余量较大,对于速度要求高的应用场所不适应用RS485总线。同时由于RS485总线上通常只有一台主机,所以这种总线方式是典型的集中—分散型控制系统。一旦主机出现故障,会使整个系统的通信限于瘫痪状态,因此做好主机的在线备份是一个重要措施。

**传统光电隔离的典型电路:

VDD与+5V1(VCC485)是两组不共地的电源,一般用隔离型的DC-DC来实现。通过光耦隔离来实现信号的隔离传输,ISL3152EIBZ与MCU系统不共地,完全隔离则有效的抑制了高共模电压的产生,大大降低485的损坏率,提高了系统稳定性。但也存在电路体积过大、电路繁琐、分立器件过多,传输速率受光电器件限制等缺点,对整个系统的稳定性也有一定影响。

***RXD1 :串口接收端

***TXD1 :串口传输端

***TRE1 :为控制位:控制发送还是接收数据;

当TRE1=1(高电平时),光耦电路121截止,/RE=1(无效),DE=1(有效),即发送数据;

当TRE=0   (低电平时),光耦电路导通,/RE=0(有效),即接收数据,DE=0(无效);

/RE: 485接收端

DE:485发送端

第一步,配置好串口发送、接收端引脚和485控制引脚;

因为RXD1引脚相对于STM32芯片来说是接收外来数据,所以设置为输入;

TXD1引脚相对于STM32芯片来说是对外发送数据,所以设置为输出;

TRE1 引脚是对外发送“1”或“0”高低电平命令,所以设置为输出;


  1. /*****************************************************************  

  2. *函数名称:  UART2Init  

  3. *功能描述:  对串口2参数进行设置、485控制端口初始化  

  4. *   

  5. *输入参数:无  

  6. *返 回 值:无  

  7. *其他说明:无  

  8. *当前版本:v1.0  

  9.  


  10. *修改日期      版本号      修改人      修改内容  

  11. *-----------------------------------------------------------------  

  12. *  

  13. ******************************************************************/  

  14. void UART2Init(void)  

  15. {  

  16.         GPIO_InitTypeDef GPIO_InitStructure;  

  17.         USART_InitTypeDef USART_InitStructure;  

  18.   

  19.         RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);  

  20.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);  

  21.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);  

  22.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);//使能外设时钟      

  23.   

  24.         //GPIO结构的成员设置如下:  

  25.          /*--------------485控制端初始化------PA1----------*/  

  26.          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 ;  

  27.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //50M时钟速度  

  28.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出  

  29.         GPIO_Init(GPIOA, &GPIO_InitStructure);  

  30.   

  31.           

  32.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;          //485_TX  

  33.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   

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

  35.         GPIO_Init(GPIOA, &GPIO_InitStructure);  

  36.   

  37.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;           //485_RX  

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

  39.         GPIO_Init(GPIOA, &GPIO_InitStructure);  

  40.   

  41.         //串口的结构成员设置如下:  

  42.         USART_InitStructure.USART_BaudRate = 9600;           

  43.         USART_InitStructure.USART_WordLength = USART_WordLength_8b;  

  44.         USART_InitStructure.USART_StopBits = USART_StopBits_1;  

  45.         USART_InitStructure.USART_Parity = USART_Parity_No;  

  46.         USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;  

  47.         USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;  

  48.         USART_Init(USART2, &USART_InitStructure);  

  49.   

  50.         USART_Cmd(USART2, ENABLE);  

  51.   

  52.         /*方法一: 清发送完成标志*/  

  53.       //  USART_ClearFlag(USART3, USART_FLAG_TC);   

  54.           

  55.        /*方法二:获取串口1状态标志位*/  

  56.        USART_GetITStatus(USART1, USART_FLAG_TC);  

  57.        

  58. }  

第二步:发送数据

这里需要注意的是:

/* CPU的小缺陷:串口配置好,如果直接Send,则第1个字节发送不出去

如下两个方法语句解决第1个字节无法正确发送出去的问题 */

方法一:USART_ClearFlag(USART3, USART_FLAG_TC); /*清发送完成标志,Transmission Complete flag */

方法二:/*获取串口1状态标志位*/

               USART_GetITStatus(USART1, USART_FLAG_TC);

刚上电时出现乱码的原因:

while(USART_GetFlagStatus(USART2, USART_FLAG_TXE) == RESET);  // USART_FLAG_TXE---检测发送数据寄存器空标志位

如果USART_FLAG_TC---发送完成标志位

(1)       当设为USART_FLAG_TXE---检测发送数据寄存器空标志位—为空,但是发送移位寄存器不为空,数据还没有完全的发送出去,又有数据就被写进来了,所以就会容易出现乱码;

(2)       当设为USART_FLAG_TC—检测发送完成标志位—为空,即发送移位寄存器为空,数据才真正的发送出去,因此此时又有数据被写进来也不会发生乱码

STM32的数据发送有两个中断标志,一个是发送数据寄存器空标志,一个是发送完毕标志。两个标志都可以引起中断. 

要以中断的方式发送一个数据包,流程是这样的:

1.设置RS485的方向为发送,使能发送寄存器空中断,使能完毕进入串口中断。

2.串口中断里读取串口状态,并填充一个数据到发送数据寄存器,硬件自动清除发送数据寄存器空标志,串口数据发送开始。

3.串口发送完一个数据,发送数据寄存器变空,再进入中断,继续填充下一个数据,直到最后一个数据填充完,使能串口

 发送完毕中断。

4.最后一个数据发送完毕,再次进入中断,清除发送数据寄存器空标志,清除发送完毕中断标志,清除这两个中断标志

 的使能位,设置RS485的方向为接收.


  1. /*****************************************************************  

  2.   

  3. *                                      宏定义  

  4.   

  5. ******************************************************************/  

  6.   

  7. #define RX_485  GPIO_SetBits(GPIOA,GPIO_Pin_1);  

  8.   

  9. #define TX_485  GPIO_ResetBits(GPIOA,GPIO_Pin_1);  

  10.   

  11. /*****************************************************************  

  12.   

  13. *函数名称:  UART2_TX485_Puts  

  14.   

  15. *功能描述:  发送字符串数据  

  16.   

  17. *   

  18.   

  19. *输入参数:str:要发送的字符串  

  20.   

  21. *返回值:无  

  22.   

  23. *其他说明:无  

  24.   

  25. *当前版本:v1.0  

  26.   


  27.   


  28.   

  29. *修改日期      版本号      修改人      修改内容  

  30.   

  31. *-----------------------------------------------------------------  

  32.   

  33. *  

  34.   

  35. ******************************************************************/  

  36.   

  37.    

  38.   

  39. void UART2_TX485_Puts(char * str)  

  40.   

  41. {     

  42.   

  43.      

  44.   

  45.     while(*str)  

  46.   

  47.     {    

  48.   

  49.        TX_485;     //打开485发送DE端口,关闭接收/RE端口  

  50.   

  51.        DelayNmS(1);  

  52.   

  53.         USART_SendData(USART2, *str++);  

  54.   

  55.         /* Loop until the end of transmission */  

  56.   

  57.         while(USART_GetFlagStatus(USART2, USART_FLAG_TXE) == RESET);  //检测发送数据寄存器空标志位  

  58.   

  59.        DelayNmS(1);  

  60.   

  61.         RX_485;     //关闭发送DE端口,打开接收/RE485端口,  

  62.   

  63.     }  

  64.   

  65. }   

在485芯片的通信中,尤其要注意对485控制端DE的软件编程。为了可靠工作,在485总线状态切换时需要做适当延时,再进行数据收发。具体的做法是在数据发送状态下,   先将控制端置“1”,延时1ms左右的时间,在发送有效的数据,一包数据发送结束后再延时1ms后,将控制端置“0”,这样处理会使总线在状态切换时,有一个稳定的工作过程。


关键字:STM32  RS485  通讯协议 引用地址:STM32菜鸟成长记录---RS485通讯协议的应用

上一篇:STM32菜鸟成长记录---窗口看门狗
下一篇:STM32菜鸟成长记录---系统滴答定时器(systick)应用

推荐阅读最新更新时间:2024-03-16 15:27

别用keil了,CLion同样可以用于STM32开发
0.前言 最近有用到 STM32 这款MCU,好久没用 Keil 感觉各种不适应,遂尝试能不能把 STM32 的开发环境部署到其他更强大的IDE,结果很圆满,以下是配置 Clion 用于 STM32 开发的过程记录,供大家参考~ 做嵌入式开发的人对 STM32 这个平台应该都是非常熟悉的,在国内尤其流行,很多产品里面都是基于这个平台做的方案。多数人在开发 STM32 的时候用的都是 Keil 这个老牌IDE,很大一部分原因是因为大多数人最初是从51单片机学习过来的,51就是基于 Keil 去开发的,然后迁移到 STM32 的时候也就沿用下来了。 Keil 操作简单,容易上手,而且可以很方便地进行调试。但是对于以前不是做嵌入式开
[单片机]
STM32自己总结如何正确创建一个工程
1、建立好文件夹,文件夹中有相应的库函数和头文件等等 以上通过我的G盘里面的建立工程模板解压即可得到 2、通过keil创建test_creat.uvproj到以目录 选择完芯片以后,弹出窗口一定要选择“否” 3、添加文件,只用添加C文件进去 添加完成以后会看到: 4、头文件路径的添加 stm32f10x_it.h ,stm32f10x.h, stm32f10x_conf.h ,core_cm3.h, system_stm32f10x.h 主要添加以上的头文件位置还有各种ppp外设头文件的位置也就是src文件夹 5.编译会出现的问题及解决方法 1、去掉STM32F10
[单片机]
<font color='red'>STM32</font>自己总结如何正确创建一个工程
stm32 - keil中启动文件的选择和固件库宏定义
● 启动文件的选择 1. Low-density devices (STM32F10nx4 = 16k, STM32F10nx6 = 32k) - startup_stm32f10x_ld_vl.s: 小容量超值型。STM32F100xx系列,闪存16k~32k字节。 - startup_stm32f10x_ld.s: 小容量型。STM32F101xx, STM32F102xx, STM32F103xx系列,闪存16k~32k字节。 2. Medium-density devices (STM32F10nx8 = 64k, STM32F10nxB = 125k) - startup_stm32f10x_md_vl.s: 中
[单片机]
<font color='red'>stm32</font> - keil中启动文件的选择和固件库宏定义
基于stm32的keil开发环境搭建
1 只要按照下面的一步步来,绝对能从0开始建立一个STM32工程。不仅包括工程建立过程,还有Jlink设置方法。本文使用芯片为STM32F103CB。 1 下载stm32F10x的官方库 2 新建工程 工程名设为stm32_demo,选择芯片型号为STM32F103B,如图: 因为下载的stm32库中有启动代码,所以这里选择 否 ,不拷贝启动代码。 在工程文件下,新建Startup 、Headers、User、Libraries、CMSIS、Lists、Output文件夹。 至此,stm32的工程文件结构如下 3 库文件拷贝 把下载stm32库中文件拷贝到新建工程中 文件
[单片机]
基于<font color='red'>stm32</font>的keil开发环境搭建
STM32 | STM32的复用时钟何时开启?
STM32的AFIO时钟真的是在开启引脚复用功能的时候开启吗?其实并不是~ 什么是复用? 我们知道,STM32有很多外设,这些外设的外部引脚都是与GPIO共用的。我们可以通过软件来配置引脚作为GPIO引脚还是作为外设引脚。当引脚配置为外设引脚时就叫做复用。如串口默认复用的引脚为: PA9、PA10引脚可配置为普通IO,也可配置为串口引脚(PA9为TX,PA10为RX)。 例子:串口1为例 关于外设的配置可查阅:《STM32参考手册》关于通用和复用功能I/O(GPIO和AFIO) 的章节,如串口引脚配置如下: 最近看到一份代码(基于STM32F103ZET6)的串口配置如下: 看到这里开启了AFIO时钟,让我疑惑
[单片机]
<font color='red'>STM32</font> | <font color='red'>STM32</font>的复用时钟何时开启?
STM32】中断和中断优先级理解
  在嵌入式系统开发过程中,不可避免的会和“中断”打交道。很多初学者对中断的理解不深,往往是看到中断绕着走。其实不然,费点时间搞懂中断能解决不少问题。   中断英文“Interrupt”,字面很好理解,就是打断的意思。在程序中“Interrupt”打断的自然也是程序。举个例子,有一个遥控的自动避障小车,小车在向前跑的过程中程序一直监测周围环境,实现避障功能,这个时候你想遥控小车停个三分钟,就要用到中断了,也就是打断原有的“监测环境——避障”程序流程。打断的目的就是执行“别跑啦,停下来歇歇”这个程序,我们称中断之后执行的程序为“中断服务函数”。一般情况下,中断服务函数执行完后(小车休息三分钟,或者干其他的事情完事后),会清理一下中断
[单片机]
STM32 断言的使用
由于进入岁末,放假还有几天,无事可做便开始玩玩STM32芯片。 以前虽然也使用过ST8系列的8位单片机,但是相对于32位的STM32系列单片机来讲,还是稍微复杂一点的。但ST半导体出品的CPU都提供固件库给大家使用,这使得开发周期缩短了很多。在仔细阅读官方下载的固件库时,发现ST的技术人员编写的固件库很值得我们学习。 1、首先它的封装很独立,便于移植,功能模块独立; 2、通过 include.h 对包含的固件库头文件进行剪裁,方便加载需要的头文件; 3、在功能函数中都进行断言判断,以便设计人员在开发调试阶段进行稳定性测试; 。。。。。。 /* Exported ty
[单片机]
STM32外部中断执行过程
一、MCU中断程序执行过程 以MCU裸机程序框架为例,MCU 的主函数是个死循环: 主函数: main(void) { While(1) { //主程序 } } 如果没有中断或异常产生会一直在while(1)里执行主程序代码。当中断产生后,当前执行的任务会被打断,程序跳转到中断处理函数执行,执行完会返回之前的主程序断点处继续执行。 中断处理函数: void IRQ_handler(void) { //中断处理程序 //清除中断标志,否则会不停的进入中断处理程序。 } 在中断处理函数中不要做复杂费时的事情,中断内做的事情尽可能少。 二、中断使用实例 STM32的IO都可以配置成外部中断,但不是同时都可以配成外部中断。需要遵循如下
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

换一换 更多 相关热搜器件
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved