找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3211|回复: 5
打印 上一主题 下一主题
收起左侧

单片机MODBUS_RTU的通讯程序 03码只能读连续的18个数据

[复制链接]
跳转到指定楼层
楼主
从大学毕业到现在,重新学习了下51单片机,最近站在各位前辈大佬的肩膀上,调试了一下STC15W408AS的一个MODBUS_RTU的通讯,程序能够调通,但是用modscan32的时候,只能读连续的18个数值,搞不懂了。贴上代码,请各位大神帮忙看看

4002是程序做的自加不用管,其他初始化为0


4002是程序做的自加不用管,其他初始化为0


****本行不是代码*****只是说明 ****** 以下是modbusRTU.c *************************************************************

单片机源程序如下:
  1. #include "config.h"
  2. #include "modbusRTU.h"
  3. #include "CRC16.h"
  4. bit flagFrame = 0; /*帧接收完成标志*/
  5. bit flagTxd = 0;    /*单字节发送完成标志*/
  6. unsigned char cntRxd = 0;  /*接收字节计数器*/
  7. unsigned char  bufRxd[128]; /*接收字节缓冲区*/

  8. /*预定义一个或多个实参数组,用于存放需要处理的数据*/
  9. unsigned int idata regGroup[60]={0};  //数据缓存区,全局变量,
  10. unsigned int idata modbusData[2]={0};   //该数组用来存放modbus数据,分别为波特率和地址

  11. /*串口配置函数,baud-波特率*/
  12. void UartInit(unsigned int baud)                //9600bps@22.1184MHz   16位自动重装载
  13. {
  14.         RS485_DIR = 0;
  15.         SCON = 0x50;                //8位数据,可变波特率
  16.         AUXR |= 0x01;                //串口1选择定时器2为波特率发生器
  17.         AUXR |= 0x04;                //定时器2时钟为Fosc,即1T
  18.         T2L = (65536-SYSclk/baud/4) & 0xff;                //设定定时初值,取低8位
  19.         T2H = (65536-SYSclk/baud/4) >> 8;                //设定定时初值,取高8位
  20.         AUXR |= 0x10;                //启动定时器2
  21.         AUXR1 = 0x40;       //将串口切换到P36P37,此处之所以要将串口切换,是因为P30/P31需要作为下载口
  22.         ES = 1;
  23. }


  24. /*串口数据写入,即串口发送函数,buf-待发送数据指针,len-指定的发送长度*/
  25. void UartWrite(unsigned char *buf, unsigned char len)
  26. {
  27.         RS485_DIR = 1;
  28.         while(len--)
  29.         {
  30.                 flagTxd = 0;   //清零发送标志
  31.                 SBUF = *buf++;  //发送一个字节数据
  32.                 while(!flagTxd);
  33.         }
  34. //        DelayX10us(5);
  35.         Delay5ms();   //延时5ms,等待一帧数据的最后3.5字符发送完成
  36.         RS485_DIR = 0;
  37. }


  38. /*串口数据读取,buf-接收数据指针,len-指定的读取长度,返回值,实际读到的数据长度*/
  39. unsigned char UartRead(unsigned char *buf, unsigned char len)
  40. {
  41.         unsigned char i;
  42.         if(len > cntRxd)  //指定读取长度 大于 实际接收到的长度时,
  43.         {                 //读取长度设置为实际接收到的数据长度
  44.                 len = cntRxd;
  45.         }
  46.         for(i=0;i<len;i++)
  47.         {
  48.                 *buf++ = bufRxd[i];
  49.         }
  50.         cntRxd = 0;
  51.         return len;
  52. }


  53. /*串口接收监控,由空闲时间判定帧结束,需在定时中断中调用,ms-定时间隔*/
  54. void UartRxMonitor(unsigned char ms)
  55. {
  56.         static unsigned char cntbkp = 0;
  57.         static unsigned char idlemr = 0;
  58.         if(cntRxd >0)
  59.         {
  60.                 if(cntbkp != cntRxd)
  61.                 {
  62.                         cntbkp = cntRxd;
  63.                         idlemr = 0;
  64.                 }
  65.                 else
  66.                 {
  67.                         if(idlemr < 50)
  68.                         {
  69.                                 idlemr = idlemr + ms;
  70.                         }
  71.                         if(idlemr >= 50)
  72.                         {
  73.                                 flagFrame = 1;
  74.                         }
  75.                 }
  76.         }
  77.         else
  78.         {
  79.                 cntbkp = 0;
  80.         }
  81. }


  82. /*串口驱动函数,监测数据帧的接收,调用函数功能,需要在主函数中调用*/
  83. void UartDriver()
  84. {
  85.         unsigned char len;
  86.         unsigned char xdata buf[40];
  87.         if(flagFrame)
  88.         {
  89.                 flagFrame = 0;
  90.                 len = UartRead(buf,sizeof(buf)-2);
  91.                 UartAction(buf,len);
  92.         }
  93. }


  94. void InterruptUART() interrupt 4
  95. {
  96.         if(RI)
  97.         {
  98.                 RI = 0;
  99.                 if(cntRxd < sizeof(bufRxd))
  100.                 {
  101.                         bufRxd[cntRxd++] = SBUF;               
  102.                 }
  103.                
  104.         }
  105.         if(TI)
  106.         {
  107.                 TI = 0;
  108.                 flagTxd = 1;
  109.         }
  110. }


  111. /*串口动作函数,根据接收到的信号执行相应的动作
  112.    buf-接收到的命令帧指针,len - 命令帧长度*/
  113. void UartAction(unsigned char *buf, unsigned char len)
  114. {
  115.         unsigned int crc;
  116.         unsigned char crcl,crch;
  117.         unsigned char i = 0;
  118.         unsigned char cnt = 0;
  119.         
  120.         if(buf[0] != modbusData[1])
  121.         {
  122.                 return;
  123.         }
  124.         crc = GetCRC16(buf,len-2);   //计算CRC
  125.         crcl = crc & 0xff;   //取低8位 CA
  126.         crch = crc >> 8;     //取高8位 D5   举例 01 03 00 01 00 01 D5 CA
  127.         if((buf[len-2] != crch) || (buf[len-1] != crcl)) //CRC校验不通过,则退出该函数
  128.         {
  129.                 return;
  130.         }
  131.         switch(buf[1])
  132.         {
  133.                 case 0x03:
  134.                     if ((buf[2]==0x00) && (buf[3]<=0x3c))
  135.                     {                                       
  136.                             i = buf[3];   //提取寄存器地址
  137.                             cnt = buf[5];  //提取寄存器数量                                       
  138.                             buf[2] = cnt*2;  //提取数据的字节数,为寄存器数量 *2                                       
  139.                             len = 3;  // 帧前部已经有地址、功能、字节数,所以len 从3开始算                                       
  140.                             while(cnt--)  //cnt=2
  141.                             {
  142.                                     buf[len] = regGroup[i] >> 8;   //高字节  len=3
  143.                                     buf[len+1] = regGroup[i] & 0xff; //低字节 len = 3+1
  144.                                     len = len + 2;
  145.                                     i = i + 1;
  146.                             }
  147.                             break;
  148.                     }
  149.                     else    //寄存器地址不被支持时,返回错误码
  150.                     {
  151.                             buf[1] = 0x83;  // 此处错误码 为 0x03(功能码) + 0x80(功能码出错时)
  152.                             buf[2] = 0x02;
  153.                             len = 3;
  154.                             break;
  155.                     }
  156.                 case 0x06:
  157.                     if((buf[2]==0x00) && (buf[3]<=0x3c))
  158.                         {
  159.                                 i = buf[3];   //需要写入的寄存器 的 地址低位
  160.                                 regGroup[i] = 256 * buf[4] + buf[5];
  161.                                 len = 6;
  162.                                 break;
  163.                         }
  164.                         else
  165.                         {
  166.                                 buf[1] = 0x86;  // 此处错误码 为 0x06(功能码) + 0x80(功能码出错时)
  167.                                 buf[2] = 0x02;
  168.                                 len = 3;
  169.                                 break;
  170.                         }
  171.                 default:
  172.                     buf[1] = 0x80;
  173.                         buf[2] = 0x01;
  174.                         len = 3;
  175.                         break;
  176.         }
  177.         crc = GetCRC16(buf,len);   //计算返回帧的CRC校验值
  178.         buf[len] = crc >> 8;             //高字节
  179.         buf[len+1] = crc & 0xFF;     //低字节               
  180.         UartWrite(buf, len+2);         
  181. }


  182. ****本行不是代码*****只是说明 ****** 以下是main.c *************************************************************
  183. #include "config.h"
  184. #include "power_led.h" /*开发板专有,启动STC的电源和LED*/
  185. #include "modbusRTU.h"

  186. void Timer0Init(void);
  187. void main()
  188. {
  189.         modbusData[0]=9600;   //此处暂时先用常数,后期从EEPROM中读取
  190.         modbusData[1]=2;      //此处暂时先用常数,后期从EEPROM中读取
  191.         STP_power();
  192.         EA = 1;
  193.         UartInit(modbusData[0]);
  194.         Timer0Init();
  195.         while(1)
  196.         {
  197.                 UartDriver();
  198.         }
  199. }

  200. void Timer0Init(void)                //1毫秒@22.1184MHz
  201. {
  202.         AUXR |= 0x80;                //定时器时钟1T模式
  203.         TMOD &= 0xF0;                //设置定时器模式
  204.         TL0 = 0x9A;                //设置定时初值
  205.         TH0 = 0xA9;                //设置定时初值
  206.         TF0 = 0;                //清除TF0标志
  207.         TR0 = 1;                //定时器0开始计时
  208.         ET0 = 1;
  209. }

  210. /*T0中断函数,执行串口接收监控和蜂鸣器驱动*/
  211. void InterruptTimer0() interrupt 1
  212. {
  213.         UartRxMonitor(1);
  214.         regGroup[1] ++;
  215.         if(regGroup[1]>65535)regGroup[1]=0;
  216. }
复制代码

STC15W408AS_Modbus.rar

197.29 KB, 下载次数: 75

完整的程序

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶1 踩
回复

使用道具 举报

沙发
ID:753989 发表于 2020-6-19 16:12 | 只看该作者
大部分代码来自本站下载,自己做了一点小改
回复

使用道具 举报

板凳
ID:753989 发表于 2020-6-19 16:19 | 只看该作者
另外说明一下,读取数据长度大于19的时候,我看 USB转485的转换模块上的灯,看下来正常的
回复

使用道具 举报

地板
ID:753989 发表于 2020-6-20 13:41 | 只看该作者
没有人回我吗??
回复

使用道具 举报

5#
ID:753989 发表于 2020-6-29 11:35 | 只看该作者
@ admin  没人回我,能帮帮忙吗?
回复

使用道具 举报

6#
ID:753989 发表于 2020-7-1 08:39 | 只看该作者
已经找到问题了,第96行 unsigned char xdata buf[40]; 这个地议定义的buf[40]的数组长度不够,当数据(字节)长度超过40时,就不能够存入这个缓存了。STC15W408AS的数据手册,xdata最大值为256,所以根据keil编译的结果,可以将这个缓存改成buf[110];
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表