STM32 485通信 自我学习总结

发布者:AngelicWhisper最新更新时间:2017-11-28 来源: eefocus关键字:STM32  485通信 手机看文章 扫描二维码
随时随地手机看文章

准备总结一下学习过程中的485通信知识!----------------------------------------------------

先描述一下学习STM32与485通信的时候想实现的功能--------


首先是完成双机通信中的双向通信----这里定义A为主机发送指令给从机B,从机B在接收到主机A的指令后,判断有效位的正确性,如果正确将这个指令通过RS232串口显示到串口助手里,观察整个指令是否接收正确-----同时在从机B接收到主机A指令后,向主机A发送指定数据----主机A在接收到从机B数据后判断有效位的正确性,如果正确则通过RS232打印到串口助手,然后观察完整数据。--------这里的判断比较简单------只是判断接收数据的最后一位是否是规定的字符----为了数据的准确性,可以增加判断方法。

        

第一步是配置RS232的串口,这里使用USART1,测试printf函数的输出是否正常----------这里就不叙述了!


注意在主机的USART1中要打开接收中断-- USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//打开接收终端


第二步是配置RS485,这里使用USART2,配置如下---------

void USART2_Config(void)
{
        GPIO_InitTypeDef GPIO_InitStructure;
        USART_InitTypeDef USART_InitStructure;
  
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
        RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_Init(GPIOA, &GPIO_InitStructure);
       

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
        GPIO_Init(GPIOA, &GPIO_InitStructure);
      

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP ;
         //GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_Init(GPIOB, &GPIO_InitStructure);
 
        USART_InitStructure.USART_BaudRate = 115200;
        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_Init(USART2, &USART_InitStructure); 
        USART_Cmd(USART2, ENABLE);
  
        USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);//打开接收终端
        USART_ClearFlag(USART2, USART_FLAG_TC);
}

void MyPrintfByte(unsigned char byte)//发送一个字节
{
        USART_SendData(USART2, byte);  
        while( USART_GetFlagStatus(USART2,USART_FLAG_TC)!= SET);       
}
void MyPrintfStr(unsigned char *s)//发送字符串
{
        uint8_t i=0;

        while(s[i]!='\0')
        {
                USART_SendData(USART2,s[i]); 
                while( USART_GetFlagStatus(USART2,USART_FLAG_TC)!= SET);  
                i++;   
        }
}
void MyPrintfArray(uint8_t send_array[],uint8_t num)//发送数组中指定个数
{
        uint8_t i=0;

        while(i        {               
                USART_SendData(USART2,send_array[i]); 
                while( USART_GetFlagStatus(USART2,USART_FLAG_TC)!= SET);  
                i++;     
        }        
}

第三部是在中断里写接收函数--------------

void USART1_IRQHandler(void)//232接收中断函数---主要是在主机中
{        
        if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收到数据
        {        
                USART_ClearITPendingBit(USART1, USART_IT_RXNE);//清标志位
                RS232_RX_BUF[RS232_RX_CNT++]=USART_ReceiveData(USART1);
                if( RS232_RX_CNT==4)
                        RS232_RX_CNT=0;    
        }

void USART2_IRQHandler(void)  //485中断接收函数
{       
        if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET)//接收到数据
        {        
                USART_ClearITPendingBit(USART2, USART_IT_RXNE);//清标志位
                RS485_RX_BUF[RS485_RX_CNT++]=USART_ReceiveData(USART2); //将接收到的数据保存到指定数组
                if( RS485_RX_CNT==9)//接收指定长度为9个数据
                        RS485_RX_CNT=0;     
        }

第四步是对中断的优先级进行配置----

static void NVIC_Configuration(void)
{
      NVIC_InitTypeDef NVIC_InitStructure;
  
      NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
  
      NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
      NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
      NVIC_InitStructure.NVIC_IRQChannelSubPriority = ENABLE;
      NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
      NVIC_Init(&NVIC_InitStructure);
  
      NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
  
      NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
      NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
      NVIC_InitStructure.NVIC_IRQChannelSubPriority = ENABLE;
      NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
      NVIC_Init(&NVIC_InitStructure);
}

第五部是主函数----------

---------------------------主机----------------------------------

int main(void)
{
        uint8_t i,flag=0;
  

        USART2_Config(); 
        USART1_Config(); 
        GPIO_Config();
        NVIC_Configuration();

        printf("485主机\r\n");

        while(1)
        {
                GPIO_SetBits(GPIOB,GPIO_Pin_5);//发送模式
                Delay(50000);  //必要的延时
                if(RS232_RX_BUF[3]=='1')      //通过串口助手发送一组数据,根据不同数据下面向从机发送不同的指令
                        MyPrintfStr("12345678!");
                else if(RS232_RX_BUF[3]=='2')
                        MyPrintfStr("abcdefgh@");
                else if(RS232_RX_BUF[3]=='3')
                       MyPrintfStr("ABCDEFGH#");

                RS485_RX_BUF[8]='-';  //将接收数据的最后一位设置为其它
                Delay(50000);
                GPIO_ResetBits(GPIOB,GPIO_Pin_5);//接收模式
                Delay(50000);
                //USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);
                flag=0;
                while(!(RS485_RX_BUF[8]=='T')&&flag<20)//等待接收到数据,并且最后一位数据位---T---或者超时退出循环---超时数值比较重要,太小会使数据丢失。
                {
                       Delay(0x666ee);
                      flag++;
                }
                for(i=0;i<9;i++)  //打印出接收到从机的数据
                {
                       if(i<8)
                            printf("%c",RS485_RX_BUF[i]); 
                       else
                            printf("%c\r\n",RS485_RX_BUF[i]); 
                } 
                Delay(0x6666ee);
        }
}

------------------------------------------从机--------------------------------------

int main(void)
{
        uint8_t i;
     

        USART2_Config();
        GPIO_Config();

        USART1_Config();
        NVIC_Configuration();
  
        printf("\r\n485从机\r\n");

        while(1) 
        {
                RS485_RX_BUF[8]='+';
                Delay(50000);
                GPIO_ResetBits(GPIOB,GPIO_Pin_5); //接收模式
                Delay(50000);
                while(!(RS485_RX_BUF[8]=='!')&&!(RS485_RX_BUF[8]=='@')&&!(RS485_RX_BUF[8]=='#'));//简单判断最后一位

                for(i=0;i<9;i++)//打印出接收到的数据
                {
                        if(i<8)
                            printf("%c",RS485_RX_BUF[i]); 
                        else
                            printf("%c\r\n",RS485_RX_BUF[i]); 
                }
                Delay(0x6666ee);

                GPIO_SetBits(GPIOB,GPIO_Pin_5); //发送模式
                Delay(50000);
                if(RS485_RX_BUF[8]=='!')  //根据主机发送的不同指令返回发送不同数据给主机
                        MyPrintfStr("ascii789T");
                else if(RS485_RX_BUF[8]=='@')
                        MyPrintfStr("ASCII123T");
                else if(RS485_RX_BUF[8]=='#')
                        MyPrintfStr("CHINA456T");
        }
}

第六步是下载测试--------------------------------

01---复位两个开发板----------------


02----通过串口助手向主机发送指令1111------指令要求---必须4位----只判断最后一位是否为1或者2或者3------0001与1111表示一样的指令-----3312与8762表示一样。------------发现从机接收到了主机发送的指令12345678!------主机也接收到了从机返回的数据ascii789T。


03----通过串口助手向主机发送指令2222------指令要求---必须4位----只判断最后一位是否为1或者2或者3------0001与1111表示一样的指令-----3312与8762表示一样。------------发现从机接收到了主机发送的指令abcdefgh@------主机也接收到了从机返回的数据ASCII123T。


04----通过串口助手向主机发送指令3333------指令要求---必须4位----只判断最后一位是否为1或者2或者3------0001与1111表示一样的指令-----3312与8762表示一样。------------发现从机接收到了主机发送的指令ABCDEFGH#------主机也接收到了从机返回的数据CHINA456T。



-------------------目前是9个数据的收发----是正常的----------接下来将测试收发更多数据-----不知道效果如何-----------------

14-10-30

----------------------------------17个数据也是正常的-------------再多一些也是可以的----------




----------------------------在调试过程中有一些问题出现---------------------

1.---其中有一个是-----------主机接收时总是出错,想了很多办法,也仔细看了程序,最后发现从机接收正常,于是将主机与从机交换,发现主机正常了,而从机出错了,得出结论是硬件的问题,后来仔细观察发现焊接不是很好,虽然都是焊接上了的,最后重新加锡牢固焊接-----通信完全正常了!

2.---第二个就是线的长度问题----我试验的时候使用的是20cm的线,通讯完全正常。后来我直接上500米的线,发现暂时没有反应,现在我用3-4M的网线,通讯正常,后面再慢慢加长--------------听说要加上拉电阻---不知道对不对---打算试一试!

3.---线长有十几米,也没有加上拉电阻,通信时完全正常的,上面的程序需要在接收过程中实现一个纠错的简单功能,防止在长时间通信过程中偶尔丢失数据,从而影响后面的数据接收-------

------主机main------


-------从机main--------


4.---赶快焊接第三个板子------做多机通信实验---看效果!

14-11-1

5.---今天焊接了两个板子,加起来有四个,现在在多机通信,过程中有几个问题出现,-------上面双机通信时主机是不停发送指令的,但是在多机通信的时候不可用,会出现错误-----我简单理解是要释放总线-----这个度不好把握-----把握好可以增加通信频率---次数

6.---我现在是主机接收串口指令,发送指令给从机,3个从机根据指令地址判断,是否需要返回数据----恢复通信是没有问题的------就是反应慢点,串口给主机指令后,3秒钟才接收到从机返回的数据----------不过一步一步来,再调试一下!

---------------------图片---------------------

------------指令1111---------



------------指令2222---------


------------指令3333--------



7.---慢慢调试了延时,现在的反应快多了,一秒不到--------------!

14-11-2

8.---前几天在学习STM32的485通信,基本搞清楚了原理,也实际操作成功---后续还将进一步学习---与变频器通信,从而去控制380V电机!


---------------------实物图------------------


------14-11-7--------------------通过PC串口发送指令给主机,主机再向变频器发送指令,去控制380V电机的速度,启停,正反转!-----------------见下图----------------





关键字:STM32  485通信 引用地址:STM32 485通信 自我学习总结

上一篇:STM32 I2C总线 自我总结学习
下一篇:STM32 串口通讯 发送 接收

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

stm32 usb数据缓冲区疑问
USB不同的应用需要使用到的端点数和端点数据长度各不相同,如果为每个端点都单独规划一个存储区非常浪费。 所以STM32为USB模块提供了共512个字节的存储区,至于如何为每个端点分配使用这512B的空间,就是用户自己的事情了,这样存储区的分配就非常经济灵活。这512B空间的首地址是0x40006000。 你看到的#define ENDP2_RXADDR (0xD8),这个0xD8就是在这512B空间里的偏移地址,说明端点2收到的数据将放在0x40006000 + 0xD8 * 2的地址空间里。至于为什么要×2,是因为这部分存储区是按照2字节访问的,即每存放1个字节的数据要占据2个字节的空间。所以这段存储区的地址是从0
[单片机]
STM32——GPIO输入模式下上拉和下拉的设置
GPIO处于输入模式下,下拉输入和上拉输入的相关配置如下图所示。需要注意的是,下拉输入和上拉输入是通过端口输出寄存器GPIOx_ODR来区分的。因此,在进行上拉/下拉输入配置时候,虽然对GPIO进行的关于输入的操作,但是仍要对和输出相关的寄存器ODR进行配置。 上述注意事项在实际代码编写时表现为: (1)库函数 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; //PA0 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //PA0 下拉输入 GPIO_Init(GPIOA, &GPIO_InitStructure); GP
[单片机]
<font color='red'>STM32</font>——GPIO输入模式下上拉和下拉的设置
stm32定时器优先级
什么是优先级   优先级是具有高抢占式优先级的中断可以在具有低抢占式优先级的中断处理过程中被响应,即中断嵌套,或者说高抢占式优先级的中断可以嵌套低抢占式优先级的中断。   当两个中断源的抢占式优先级相同时,这两个中断将没有嵌套关系,当一个中断到来后,如果正在处理另一个中断,这个后到来的中断就要等到前一个中断处理完之后才能被处理。如果这两个中断同时到达,则中断控制器根据他们的响应优先级高低来决定先处理哪一个;如果他们的抢占式优先级和响应优先级都相等,则根据他们在中断表中的排位顺序决定先处理哪一个。 stm32定时器优先级   STM32 可以支持的 68 个外部中断通道,已经固定的分配给相应的外部设备。每个中断通道都具备自己
[单片机]
stm32液晶、RTC时间显示问题
最近调液晶屏日期和时间显示,修改RTC.c中的程序中if(HAL_RTCEx_BKUPRead(RTCHandle,RTC_BKP_DR0)!= 0xABCD) { sTime- Hours=14; sTime- Minutes=37; sTime- Seconds=20; sTime- SubSeconds=0; sTime- TimeFormat=RTC_HOURFORMAT12_AM; sTime- DayLightSaving=RTC_DAYLIGHTSAVING_NONE; sTime- StoreOperation=RTC_STOREOPERATION_RESET; HAL
[单片机]
ucosii在stm32上的移植详解4
详解3中有一个问题还没解释,就是stm32f10x_it.c中已经有SysTick中断函数的定义SysTick_Handler(),为什么官方版非要弄个OS_CPU_SysTickHandler()。答案就在启动文件上,一般我们自己开发基于stm32芯片的软件,都会使用标准外设库CMSIS中提供的启动文件,而官方移植的启动文件却是自己写的,在两个文件init.s,vectors.s中(Micrium\Software\EvalBoards\ST\STM3210B-EVAL\RVMDK)。init.s负责进入main(),vectors.s设置中断向量。OS_CPU_SysTickHandler和OS_CPU_PendSVHand
[单片机]
STM32串口——5个串口的使用方法
串口是我们常用的一个数据传输接口,STM32F103系列单片机共有5个串口,其中1-3是通用同步/异步串行接口USART(Universal Synchronous/Asynchronous Receiver/Transmitter),4,、5是通用异步串行接口UART(Universal Asynchronous Receiver/Transmitter)。 配置串口包括三部分内容: 1. I/O口配置:TXD配置为复用推挽输出(GPIO_Mode_AF_PP),RXD配置为浮空输入(GPIO_Mode_IN_FLOATING); 2. 串口配置:波特率等; 3. 中断向量配置:一般用中断方式接收数据。 注
[单片机]
<font color='red'>STM32</font>串口——5个串口的使用方法
STM32内置参照电压的使用
每个STM32芯片都有一个内部的参照电压,相当于一个标准电压测量点,在芯片内部连接到ADC1的通道17。 根据数据手册中的数据,这个参照电压的典型值是1.20V,最小值是1.16V,最大值是1.24V。这个电压基本不随外部供电电压的变化而变化。 不少人把这个参照电压与ADC的参考电压混淆。ADC的参考电压都是通过Vref+提供的。100脚以上的型号,Vref+引到了片外,引脚名称为Vref+;64脚和小于64脚的型号,Vref+在芯片内部与VCC信号线相连,没有引到片外,这样AD的参考电压就是VCC上的电压。 在ADC的外部参考电压波动,或因为Vref+在芯片内部与VCC相连而VCC变化的情况下,如果对于ADC测量的准
[单片机]
STM32电源管理系统浅谈
电源对电子设备的重要性不言而喻,它是保证系统稳定运行的基础,而保证系统能稳定运行后,又有低功耗的要求。在很多应用场合中都对电子设备的功耗要求非常苛刻,如某些传感器信息采集设备,仅靠小型的电池提供电源,要求工作长达数年之久,且期间不需要任何维护;由于智慧穿戴设备的小型化要求,电池体积不能太大导致容量也比较小,所以也很有必要从控制功耗入手,提高设备的续行时间。 STM32的电源管理系统主要分为: 1、备份域 2、调压器供电电路 3、ADC电源电路 备份域电路 STM32的备份域包括LSE振荡器、RTC、备份寄存器及备份SRAM这些器件,这部分的电路可以通过STM32的VBAT引脚获取供电电源,在实际应用中一般会使用3V的钮扣电
[单片机]
<font color='red'>STM32</font>电源管理系统浅谈
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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