标题: STM32 RS485 modbus通讯源程序,亲测有效 [打印本页]

作者: 简铛铛铛铛    时间: 2018-10-16 08:56
标题: STM32 RS485 modbus通讯源程序,亲测有效
Modbus 一个工业上常用的通讯协议、一种通讯约定。Modbus协议包括RTU、ASCII、TCP。其中MODBUS-RTU最常用,比较简单,在单片机上很容易实现。modbus协议简单解析如下:
1、主机对从机写数据操作
如果单片机接收到一个报文那么就对报文进行解析执行相应的处理,如上面报文:
    01             06            00 01           00 17          98 04
  从机地址        功能号          数据地址          数据         CRC校验

假如本机地址是 1 ,那么单片机接收到这串数据根据数据计算CRC校验判断数据是否正确,如果判断数据无误,则结果是:
            HoldDataReg[1]  =  0x0017;
MODBUS主机就完成了一次对从机数据的写操作,实现了通讯。

2、主机对从机读数据操作
主机进行读HoldDataReg[1] 操作,则报文是:
    01             03            00 01           00 01          D5 CA
从机地址        功能号          数据地址      读取数据个数       CRC校验
那么单片机接收到这串数据根据数据计算CRC校验判断数据是否正确,如果判断数据无误,则结果是:返回信息给主机,返回的信息也是有格式的:
返回内容:  
    01         03            02             0017          F8 4A
  从机地址   功能号     数据字节个数    两个字节数据    CRC校验
MODBUS主机就完成了一次对从机数据的读操作,实现了通讯。


上传的程序是根据手把手教你51单片机的例程修改而来,能实现读/写操作(03,06码),成功通过modbus调试精灵测试,读写通讯正常!
由于刚接触modbus通讯协议,之前一直出现通讯超时的问题,找了网上很多资料,都不得其解,所以把程序拿出来和大家分享。希望和大家相互学习、共同进步!


单片机源程序如下:
  1. #include "rs485.h"
  2. #include "SysTick.h"
  3. #include "crc16.h"
  4. #include "led.h"

  5. /*******************************************************************************
  6. * 函 数 名         : RS485_Init
  7. * 函数功能                   : USART2初始化函数
  8. * 输    入         : bound:波特率
  9. * 输    出         : 无
  10. *******************************************************************************/  
  11. u8 USART2_RX_BUF[64];                   //接收缓冲,最大64字节
  12. u8 USART2_RX_CNT=0;                       //接收字节计数器
  13. u8 flagFrame=0;                         //帧接收完成标志,即接收到一帧新数据
  14. unsigned char regGroup[5];  //Modbus寄存器组,地址为0x00~0x04

  15. void RS485_Init(u32 bound)
  16. {
  17.         GPIO_InitTypeDef GPIO_InitStructure;
  18.         USART_InitTypeDef USART_InitStructure;
  19.         NVIC_InitTypeDef NVIC_InitStructure;
  20.        
  21.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOG|RCC_APB2Periph_GPIOA,ENABLE); //使能GPIOA\G时钟
  22.         RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);//使能USART2时钟
  23.        
  24.         /*  配置GPIO的模式和IO口 */
  25.         GPIO_InitStructure.GPIO_Pin=GPIO_Pin_2;        //TX-485        //串口输出PA2
  26.         GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;                  //复用推挽输出
  27.         GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;       
  28.         GPIO_Init(GPIOA,&GPIO_InitStructure);                /* 初始化串口输入IO */
  29.        
  30.         GPIO_InitStructure.GPIO_Pin=GPIO_Pin_3;        //RX-485           //串口输入PA3
  31.         GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;            //模拟输入
  32.         GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  33.         GPIO_Init(GPIOA,&GPIO_InitStructure);
  34.        
  35.         GPIO_InitStructure.GPIO_Pin=GPIO_Pin_3;        //CS-485
  36.         GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;           //推挽输出
  37.         GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  38.         GPIO_Init(GPIOG,&GPIO_InitStructure);
  39.        
  40.         //USART2 初始化设置
  41.         USART_InitStructure.USART_BaudRate = bound;//波特率设置
  42.         USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
  43.         USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
  44.         USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
  45.         USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
  46.         USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;        //收发模式
  47.         USART_Init(USART2, &USART_InitStructure); //初始化串口2
  48.        
  49.         USART_Cmd(USART2, ENABLE);  //使能串口 2
  50.        
  51.         USART_ClearFlag(USART2, USART_FLAG_TC);
  52.                
  53.         USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);//开启接受中断

  54.         //Usart2 NVIC 配置
  55.         NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
  56.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2;//抢占优先级3
  57.         NVIC_InitStructure.NVIC_IRQChannelSubPriority =2;                //子优先级2
  58.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                        //IRQ通道使能
  59.         NVIC_Init(&NVIC_InitStructure);        //根据指定的参数初始化VIC寄存器、
  60.        
  61.         RS485_TX_EN=0;                                //默认为接收模式       
  62. }

  63.        

  64. //1ms定时

  65. void TIM2_Init()
  66. {
  67.         TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
  68.         NVIC_InitTypeDef NVIC_InitStructure;
  69.        
  70.         RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);//使能TIM4时钟
  71.        
  72.         TIM_TimeBaseInitStructure.TIM_Period=1000;   //自动装载值
  73.         TIM_TimeBaseInitStructure.TIM_Prescaler=72-1; //分频系数
  74.         TIM_TimeBaseInitStructure.TIM_ClockDivision=TIM_CKD_DIV1;
  75.         TIM_TimeBaseInitStructure.TIM_CounterMode=TIM_CounterMode_Up; //设置向上计数模式
  76.         TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);
  77.        
  78.         TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE); //开启定时器中断
  79.         TIM_ClearITPendingBit(TIM2,TIM_IT_Update);
  80.        
  81.         NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;//定时器中断通道
  82.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2;//抢占优先级
  83.         NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;                //子优先级
  84.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                        //IRQ通道使能
  85.         NVIC_Init(&NVIC_InitStructure);       
  86.        
  87.         TIM_Cmd(TIM2,ENABLE); //使能定时器       
  88. }


  89. //计算发送的数据长度,并且将数据放到*buf数组中                     
  90. u8 UartRead(u8 *buf, u8 len)  
  91. {
  92.          u8 i;
  93.         if(len>USART2_RX_CNT)  //指定读取长度大于实际接收到的数据长度时
  94.         {
  95.                 len=USART2_RX_CNT; //读取长度设置为实际接收到的数据长度
  96.         }
  97.         for(i=0;i<len;i++)  //拷贝接收到的数据到接收指针中
  98.         {
  99.                 *buf=USART2_RX_BUF[i];  //将数据复制到buf中
  100.                 buf++;
  101.         }
  102.         USART2_RX_CNT=0;              //接收计数器清零
  103.         return len;                   //返回实际读取长度
  104. }


  105. u8 rs485_UartWrite(u8 *buf ,u8 len)                                                                                 //发送
  106. {
  107.         u8 i=0;
  108.    GPIO_SetBits(GPIOG,GPIO_Pin_3);                                            //发送模式
  109.    delay_ms(3);                                                               //3MS延时
  110.     for(i=0;i<=len;i++)
  111.     {
  112.         USART_SendData(USART2,buf[i]);                                              //通过USARTx外设发送单个数据
  113.         while(USART_GetFlagStatus(USART2,USART_FLAG_TXE)==RESET);             //检查指定的USART标志位设置与否,发送数据空位标志
  114.     }
  115.     GPIO_ResetBits(GPIOG,GPIO_Pin_3);                                         //设置为接收模式

  116. }





  117. //串口驱动函数,检测数据帧的接收,调度功能函数,需在主循环中调用
  118. void UartDriver()
  119. {
  120.         unsigned char i=0,cnt;
  121.         unsigned int crc;
  122.         unsigned char crch,crcl;
  123.         static u8 len;
  124.         static u8 buf[60];
  125.         if(flagFrame)            //帧接收完成标志,即接收到一帧新数据
  126.         {
  127.                 flagFrame=0;           //帧接收完成标志清零
  128.                 len = UartRead(buf,sizeof(buf));   //将接收到的命令读到缓冲区中
  129.                 if(buf[0]==0x01)                   //判断地址是不是0x01
  130.                 {
  131.                         crc=GetCRC16(buf,len-2);       //计算CRC校验值,出去CRC校验值
  132.                         crch=crc>>8;                                    //crc高位
  133.                         crcl=crc&0xFF;                                        //crc低位
  134.                         if((buf[len-2]==crch)&&(buf[len-1]==crcl))  //判断CRC校验是否正确
  135.                         {
  136.                                 switch (buf[1])  //按功能码执行操作
  137.                                 {
  138.                                         case 0x03:     //读数据
  139.                                                 if((buf[2]==0x00)&&(buf[3]<=0x05))  //寄存器地址支持0x0000~0x0005
  140.                                                 {
  141.                                                        
  142.                                                         if(buf[3]<=0x04)
  143.                                                         {
  144.                                                                 i=buf[3];//提取寄存器地址
  145.                                                                 cnt=buf[5];  //提取待读取的寄存器数量
  146.                                                                 buf[2]=cnt*2;  //读取数据的字节数,为寄存器*2,因modbus定义的寄存器为16位
  147.                                                                 len=3;                                                       
  148.                                                                 while(cnt--)
  149.                                                                 {
  150.                                                                         buf[len++]=0x00;                                //寄存器高字节补0
  151.                                                                         buf[len++]=regGroup[i++];                //低字节
  152.                                                         }
  153.                                                        
  154.                                                 }
  155.                                                         break;
  156.                                         }
  157.                                                 else  //寄存器地址不被支持时,返回错误码
  158.                                                 {   
  159.                                                         buf[1]=0x83;  //功能码最高位置1
  160.                                                         buf[2]=0x02;  //设置异常码为02-无效地址
  161.                                                         len=3;
  162.                                                         break;
  163.                                                 }
  164.                                         case 0x06:           //写入单个寄存器
  165.                                                 if((buf[2]==0x00)&&(buf[3]<=0x05))   //寄存器地址支持0x0000-0x0005
  166.                                                 {
  167.                                                         if(buf[3]<=0x04)
  168.                                                         {
  169.                                                                 i=buf[3];                                //提取寄存器地址
  170.                                                                 regGroup[i]=buf[5];                //保存寄存器数据
  171.                                                                 led3=0;
  172.                                                         }
  173.                                                         len -=2;                 //长度-2以重新计算CRC并返回原帧
  174.                                                         break;
  175.                                                 }
  176.                                                 else  
  177.                                                 {                                                        //寄存器地址不被支持,返回错误码
  178.                                                         buf[1]=0x86;           //功能码最高位置1
  179.                                                         buf[2]=0x02;           //设置异常码为02-无效地址
  180.                                                         len=3;
  181.                                                         break;
  182.                                         }
  183.                                         default:    //其他不支持的功能码
  184.                                                     buf[1]=0x80;     //功能码最高位置1
  185.                                                         buf[2]=0x01;     //设置异常码为01—无效功能
  186.                                                         len=3;
  187.                                                         break;
  188.                                 }
  189.                             crc=GetCRC16(buf,len);           //计算CRC校验值
  190.                                 buf[len++]=crc>>8;           //CRC高字节
  191.                                 buf[len++]=crc&0xff;        //CRC低字节
  192.                                 rs485_UartWrite(buf,len);  //发送响应帧
  193.                         }
  194.                 }
  195.         }
  196. }


  197.                                
  198. void UartRxMonitor(u8 ms) //串口接收监控
  199. {
  200.         static u8 USART2_RX_BKP=0;  //定义USART2_RC_BKP暂时存储诗句长度与实际长度比较
  201.         static u8 idletmr=0;        //定义监控时间
  202.         if(USART2_RX_CNT>0)//接收计数器大于零时,监控总线空闲时间
  203.         {
  204.                 if(USART2_RX_BKP!=USART2_RX_CNT) //接收计数器改变,即刚接收到数据时,清零空闲计时
  205.                 {
  206.                         USART2_RX_BKP=USART2_RX_CNT;  //赋值操作,将实际长度给USART2_RX_BKP
  207.                         idletmr=0;                    //将监控时间清零
  208.                 }
  209.                 else                              ////接收计数器未改变,即总线空闲时,累计空闲时间
  210.                 {
  211.                         //如果在一帧数据完成之前有超过3.5个字节时间的停顿,接收设备将刷新当前的消息并假定下一个字节是一个新的数据帧的开始
  212.                         if(idletmr<5)                  //空闲时间小于1ms时,持续累加
  213.                         {
  214.                                 idletmr +=ms;
  215.                                 if(idletmr>=5)             //空闲时间达到1ms时,即判定为1帧接收完毕
  216.                                 {
  217.                                         flagFrame=1;//设置命令到达标志,帧接收完毕标志
  218.                                 }
  219.                         }
  220.                 }
  221.         }
  222.         else
  223.         {
  224.                 USART2_RX_BKP=0;
  225.         }
  226. }
  227.                                                        
  228.                                                
  229.                                                        
  230.                        
  231.                
  232. /*******************************************************************************
  233. * 函 数 名         : USART2_IRQHandler
  234. * 函数功能                   : USART2中断函数
  235. * 输    入         : 无
  236. * 输    出         : 无
  237. *******************************************************************************/
  238. void USART2_IRQHandler(void)
  239. ……………………

  240. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:
29. RS485通信实验.rar (273.59 KB, 下载次数: 4563)



作者: 17351371760    时间: 2018-10-25 10:11
你好! 这个功能模块能实现多机进行485通信吗
作者: 简铛铛铛铛    时间: 2018-10-25 13:27
17351371760 发表于 2018-10-25 10:11
你好! 这个功能模块能实现多机进行485通信吗

单主机,多从机?这个是从机源程序,多机通信可以在源程序的基础上扩展一下哦
作者: 1911733869    时间: 2018-12-11 14:13
为什么会提示通讯超时
作者: 简铛铛铛铛    时间: 2018-12-19 12:50
1911733869 发表于 2018-12-11 14:13
为什么会提示通讯超时

那就是没通讯上,你是把程序移植过去,还是下到开发板直接用的?程序是我试验过的,是能正常通讯的,或者你是不是读写码的格式错了,仔细检查看看
作者: 清泥    时间: 2018-12-24 10:26
楼主我把这程序下载到stm32 用的c8t6,结果通讯超时。为什么呀
作者: 清泥    时间: 2018-12-24 11:12
清泥 发表于 2018-12-24 10:26
楼主我把这程序下载到stm32 用的c8t6,结果通讯超时。为什么呀

读写码和你图上的一样的。就是通讯不上
作者: 周振宇    时间: 2018-12-24 12:29
楼主用的哪块板子,是stm32f103吗,我还是通讯超时,望指教
作者: 简铛铛铛铛    时间: 2018-12-26 09:37
周振宇 发表于 2018-12-24 12:29
楼主用的哪块板子,是stm32f103吗,我还是通讯超时,望指教

我用的是开发板,stm32f103zet6,你程序有改动吗?或者是端口设置之类的,你好好检查下
作者: 简铛铛铛铛    时间: 2018-12-26 09:42
清泥 发表于 2018-12-24 10:26
楼主我把这程序下载到stm32 用的c8t6,结果通讯超时。为什么呀

我用的开发板是stm32f103zet6,你要看看两个芯片的程序能否兼容,我看到网上说启动文件之类的要做些修改,具体的你网上搜搜看,看看用的端口和程序是否一致,一般也没有别的问题了
作者: 123按时到岗    时间: 2018-12-26 12:56
最近在用stm32f103搞RES485 MODBUS通信没弄好,,楼主发的不错,学习了,测试一下可以,谢谢了
作者: mzb007    时间: 2018-12-27 00:41
非常有用的程序,谢谢楼主程序
作者: 简铛铛铛铛    时间: 2018-12-27 08:10
123按时到岗 发表于 2018-12-26 12:56
最近在用stm32f103搞RES485 MODBUS通信没弄好,,楼主发的不错,学习了,测试一下可以,谢谢了


作者: pm1981    时间: 2018-12-28 14:15
不错,学习了
作者: pm1981    时间: 2018-12-28 14:35
很实用
作者: xin111222333    时间: 2018-12-28 17:15
非常好,谢谢分享
作者: h283369481    时间: 2019-1-9 16:07
下载来看看,怎么样!
作者: lsf002    时间: 2019-1-24 10:28
最近在用stm32f103搞RES485 MODBUS通信没弄好,,楼主发的不错,学习了,测试一下可以,谢谢了
作者: jeasey    时间: 2019-1-25 08:44
这个协议。如果读写数据是一个32bit的怎么操作呢?弄2个寄存器吗?
作者: 1051695085    时间: 2019-2-16 11:50
楼主有主机源程序吗?
作者: 计量王工    时间: 2019-2-16 12:45
谢谢分享
作者: 望月的兔    时间: 2019-2-17 14:02
感谢感谢
作者: staticx    时间: 2019-2-28 13:25
感谢感谢
作者: Xiaohaha424    时间: 2019-3-1 13:14
楼主,这个程序移植到C51上能行吗,要注意些啥啊?
作者: lijunxing    时间: 2019-3-4 20:56
感谢楼主的无私奉献
作者: chwncy    时间: 2019-3-6 09:29

感谢楼主的无私奉献
作者: pm1981    时间: 2019-3-6 14:10
不错,下来看看
作者: liujiacheng    时间: 2019-3-6 17:40
谢谢分享,学习了
作者: yangjiyou1969    时间: 2019-3-7 22:20
谢谢分享
作者: 简铛铛铛铛    时间: 2019-3-8 09:50
jeasey 发表于 2019-1-25 08:44
这个协议。如果读写数据是一个32bit的怎么操作呢?弄2个寄存器吗?

我是这样做的
作者: 简铛铛铛铛    时间: 2019-3-8 09:52
1051695085 发表于 2019-2-16 11:50
楼主有主机源程序吗?

你可以根据这个从机修改一下,主机就是发命令,把你要读或者要写的数据发出去,发出去再接收数据
作者: xtuwz    时间: 2019-3-11 08:17
正好需要,楼主真好。我需要学习了解一下
作者: 938434518    时间: 2019-3-14 11:16
感谢分享,试一试
作者: 唱歌睡觉打豆豆    时间: 2019-3-15 10:06
感谢楼主分享,学到很多。
作者: zlbxuziming    时间: 2019-3-15 15:50
可以多机通讯吗?
作者: 简铛铛铛铛    时间: 2019-3-16 10:00
zlbxuziming 发表于 2019-3-15 15:50
可以多机通讯吗?

这是一台主机,一台从机的,多机通讯的话要扩展一下哦
作者: lx159753    时间: 2019-3-18 16:21
为什么我用STM32C8板子出现通信超时,根据程序移植过来都没问题
作者: 简铛铛铛铛    时间: 2019-3-19 15:50
lx159753 发表于 2019-3-18 16:21
为什么我用STM32C8板子出现通信超时,根据程序移植过来都没问题

肯定是哪里没改好,好好看看修改的部分,好好检查一下。主机从机的设置能不能对的上,引脚,配置
作者: cqx9999    时间: 2019-3-20 10:03
正好需要,下载下来看看.
作者: 西点钟灵毓秀    时间: 2019-3-21 15:49
标记一下,写的不错
作者: 白歌    时间: 2019-3-22 17:57
感谢分享
作者: 梦然    时间: 2019-3-30 14:31
it's useful
作者: nkenen    时间: 2019-3-31 10:06
谢谢分享
作者: ship123    时间: 2019-4-1 19:45
谢谢楼主最近在看modbus
作者: kuangzxccxz    时间: 2019-4-8 09:45
谢谢分享
作者: 51hi/com    时间: 2019-4-8 22:52
非常感谢楼主,正在测试中,能否发一个主机程序?
作者: y8830600    时间: 2019-4-9 15:16
你好,我想问一下为什么我发送报文,01 06 00 01 00 01 19 CA,回复的是7F BE FD FF FD CD 6B这个。
作者: 简铛铛铛铛    时间: 2019-4-10 17:01
本帖最后由 简铛铛铛铛 于 2019-4-11 10:07 编辑
y8830600 发表于 2019-4-9 15:16
你好,我想问一下为什么我发送报文,01 06 00 01 00 01 19 CA,回复的是7F BE FD FF FD CD 6B这个。

是不是有一些信号的干扰,导致返回的数据出现异常。或者你可以发一个命令,控制指示灯亮灭,如果发送指令控制指示灯亮,指示灯亮了,说明接收程序是没有问题的,再去看看发送部分的程序。
作者: 简铛铛铛铛    时间: 2019-4-11 10:10
51hi/com 发表于 2019-4-8 22:52
非常感谢楼主,正在测试中,能否发一个主机程序?

主机程序,其实跟从机类似的,可以在从机的基础上改动。主机是发送数据,再接收。从机是接收数据再发送到主机
作者: 小罗同学    时间: 2019-4-13 16:28
楼主大佬。这个程序stm32f407能用吗
作者: canfly    时间: 2019-4-16 20:24
非常感谢楼主分享,学习了!
作者: flzt0898    时间: 2019-4-27 22:47
显示通信超时怎么解决啊

作者: ltfltfltfltf    时间: 2019-4-28 16:15
不错,学习一下
作者: w_smilove    时间: 2019-4-28 17:04
谢谢分享
作者: ltfltfltfltf    时间: 2019-4-29 17:37
楼主,我的也是STM32F103ZET6,结果通讯超时,请问是为什么呀?谢谢楼主
作者: 541312351    时间: 2019-5-2 17:16
谢谢分享,下载学习一下
作者: 一只想飞的猪    时间: 2019-5-5 16:37
可以在c8t6上运行吗
作者: 一只想飞的猪    时间: 2019-5-5 16:40
请问楼主,485是不是只要配置串口(例如usart3)就可以实现了?我用的板子是已经集成了一个485模块,我配置完之后还是不能通讯,但是串口可以收发数据
作者: 简铛铛铛铛    时间: 2019-5-6 10:52
一只想飞的猪 发表于 2019-5-5 16:40
请问楼主,485是不是只要配置串口(例如usart3)就可以实现了?我用的板子是已经集成了一个485模块,我配置 ...

485能正常通讯是吗?那你要仔细看看你根据这个程序更改了哪些东西,程序本身是能正常通讯的。
作者: yywd4    时间: 2019-5-6 23:59
下载学习,谢谢发帖。
作者: 一只想飞的猪    时间: 2019-5-7 16:15
简铛铛铛铛 发表于 2019-5-6 10:52
485能正常通讯是吗?那你要仔细看看你根据这个程序更改了哪些东西,程序本身是能正常通讯的。

是配置好的usart3能正常收发数据,485不能用
作者: 一只想飞的猪    时间: 2019-5-7 16:18
而且为什么我下载不了你的压缩文件·····
作者: mhunter    时间: 2019-5-7 20:58
谢谢分享!
作者: 简铛铛铛铛    时间: 2019-5-8 08:08
一只想飞的猪 发表于 2019-5-7 16:18
而且为什么我下载不了你的压缩文件·····

别人都可以下哦,是不是你黑币不够,485通讯就是在串口通讯的基础上加一个收发控制引脚,你可以找一下例程看看
作者: 开心豆    时间: 2019-5-8 08:29
您好,请问为啥我连程序都打不开



作者: 开心豆    时间: 2019-5-8 08:52
为什么我编译的时候出现了一个警告
warning:  #940-D: missing return statement at end of non-void function "rs485_UartWrite"
作者: 简铛铛铛铛    时间: 2019-5-8 11:06
开心豆 发表于 2019-5-8 08:52
为什么我编译的时候出现了一个警告
warning:  #940-D: missing return statement at end of non-void fun ...

缺少返回值,在语句后面加return o,不加也不会影响程序运行
作者: 一只想飞的猪    时间: 2019-5-9 09:39
为什么串口驱动函数中的flagFrame帧接收完成标志位一直是0,进入不了接收模式
作者: 一只想飞的猪    时间: 2019-5-11 18:08
lx159753 发表于 2019-3-18 16:21
为什么我用STM32C8板子出现通信超时,根据程序移植过来都没问题

你好,请问你的问题解决了吗?
作者: hahhah000    时间: 2019-5-12 17:59
小罗同学 发表于 2019-4-13 16:28
楼主大佬。这个程序stm32f407能用吗

改一下就可以用啊,我用的也是SMT32F407的开发板。
作者: AveryLRG    时间: 2019-5-14 10:49
谢谢分享
作者: 15626168335    时间: 2019-5-15 13:06
请问楼主,我也是zet6的板,为什么第一次用的时候显示可以,然后再次发送的时候就显示通讯超时了呢
作者: 15626168335    时间: 2019-5-15 13:15
楼主,我用的也是这个芯片,为什么一开始是可以进行读写操作,之后就是通讯超时了

作者: 简铛铛铛铛    时间: 2019-5-15 15:46
15626168335 发表于 2019-5-15 13:15
楼主,我用的也是这个芯片,为什么一开始是可以进行读写操作,之后就是通讯超时了

你看看你改动的部分
作者: 871862599    时间: 2019-5-15 20:06
楼主我的板子是stm32f103zet6的,这个程序可以直接移植直接用吗?
作者: 871862599    时间: 2019-5-15 20:11
楼主我也是stm32f103zet6的板子,这个程序可以直接移植直接用吗,我基础不太好,谢谢了&#128514;
作者: 简铛铛铛铛    时间: 2019-5-18 08:05
871862599 发表于 2019-5-15 20:11
楼主我也是stm32f103zet6的板子,这个程序可以直接移植直接用吗,我基础不太好,谢谢了&#128514;

可以直接用,串口要根据你的板子改一下,别的没有什么要改的

作者: joy95611    时间: 2019-5-18 09:58

非常好,谢谢分享
作者: newhada    时间: 2019-5-18 11:19
写得很好,谢谢分享
作者: irisice008    时间: 2019-5-19 22:02
谢谢分享,学习了
作者: 张君波    时间: 2019-5-20 07:55
楼主,我用的是STM32F103ZET6的精英版开发板,为什么也显示通信超时
作者: itnoss    时间: 2019-5-21 11:55
正好需要,感谢楼主分享
作者: 11哈哈哈    时间: 2019-5-21 22:37
楼主,这个没有实现modbus呢?

作者: fouroee    时间: 2019-5-23 13:16
感谢楼主,STM32F103VE上测试可用。发现了一个小问题,rs485.c里面定义Modbus寄存器组的时候只定义了5个,但后面程序是按6个寄存器写的,不过这个问题也并不影响使用……
作者: coding1    时间: 2019-5-24 10:28
可以
作者: popkart123    时间: 2019-5-24 13:37
谢谢楼主分享
作者: 871862599    时间: 2019-5-25 15:36
简铛铛铛铛 发表于 2019-5-18 08:05
可以直接用,串口要根据你的板子改一下,别的没有什么要改的

[img][/img]真的很感谢楼主分享。移植程序以后接收端有数据,但却提示你读写错误,那到底是对还是错啊,不太懂
作者: 0212    时间: 2019-5-26 10:09
楼主 阴险  故意给代码挖坑
作者: mod1595632    时间: 2019-5-27 19:06
请问楼主485转ttl用的什么芯片啊,是max485么
作者: mod1595632    时间: 2019-5-27 19:08
fouroee 发表于 2019-5-23 13:16
感谢楼主,STM32F103VE上测试可用。发现了一个小问题,rs485.c里面定义Modbus寄存器组的时候只定义了5个, ...

您好,请问485转ttl怎么实现的?用的什么芯片
作者: zhangyingpen    时间: 2019-5-29 07:56
谢谢,学习了,
作者: mod1595632    时间: 2019-5-29 11:15
zhangyingpen 发表于 2019-5-29 07:56
谢谢,学习了,

你好,你跑这个程序成功了么
作者: s08c820    时间: 2019-5-30 14:35
我的也提示通讯超时呢
作者: YZM168    时间: 2019-5-31 17:03
我的成功了,谢谢楼主
作者: 2018_51hei    时间: 2019-6-6 16:49
不错

作者: xy51hei    时间: 2019-6-11 14:08
程序很完整,按块划分,正在弄rs485通讯,虽然不是stc8的单片机,但是值得借鉴,感谢楼主分享。
作者: 张君波    时间: 2019-6-12 21:31
Xiaohaha424 发表于 2019-3-1 13:14
楼主,这个程序移植到C51上能行吗,要注意些啥啊?

定时器的配置,485挂接在哪个串口上,从机地址,定时器中断配置,485配置,两款单片机移植,比较难
作者: liyuloveyou1    时间: 2019-6-26 21:34
楼主你好有联系方式吗,我用了原子的精英版开发板芯片一样的改了GPIO口还是通讯超时
作者: liyuloveyou1    时间: 2019-6-26 21:36
楼主,我用的是STM32F103ZET6的精英版开发板,改了GPIO口,为什么也显示通信超时
作者: liyuloveyou1    时间: 2019-6-27 20:58
只能写入不能读出是为什么




欢迎光临 (http://www.51hei.com/bbs/) Powered by Discuz! X3.1