久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 3608|回復: 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 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 97国产精品 | 一级欧美黄色片 | 中文字幕一区二区视频 | 成人av一区二区三区 | 性一爱一乱一交一视频 | 精品日韩在线 | 国产精品伦理一区 | 欧美日韩黄色一级片 | 一级一级毛片免费看 | 夜夜摸夜夜操 | 久久久亚洲一区 | 青娱乐自拍 | 久久精品免费 | 亚洲美女一区 | 欧美综合在线视频 | 国产福利在线视频 | 三级黄视频在线观看 | 欧美一区二区三区 | 国产精品久久久久久福利一牛影视 | 国产一区二区 | 91视视频在线观看入口直接观看 | 国产伦精品一区二区三区照片91 | 欧美一级在线观看 | 成人性视频免费网站 | 在线91 | 99热在线播放 | 日韩中文视频 | 91精品国产综合久久久久久丝袜 | 国产亚洲精品美女久久久久久久久久 | 性色视频| 国产一区二区三区久久久久久久久 | 日韩精品免费在线观看 | 欧美综合在线观看 | 亚洲系列第一页 | 久久免费精品 | 在线观看av网站永久 | 午夜午夜精品一区二区三区文 | 久久久久精 | 亚洲综合大片69999 | 97国产一区二区精品久久呦 | 国产精品久久 |