亚洲春色中文字幕久久久-三上亚,一吻二脱三床四吻胸,国产真实伦对白视频全集,在线毛片观看,精品成品入口黄网,国产毛aⅴ片久久久,亚洲AV色香蕉一区二区三区老师,萧皇后A级艳片,色情日本视频更新,99久久亚洲精品日本无码

 找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開(kāi)始

搜索
查看: 3975|回復(fù): 5
打印 上一主題 下一主題
收起左側(cè)

單片機(jī)MODBUS_RTU的通訊程序 03碼只能讀連續(xù)的18個(gè)數(shù)據(jù)

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
從大學(xué)畢業(yè)到現(xiàn)在,重新學(xué)習(xí)了下51單片機(jī),最近站在各位前輩大佬的肩膀上,調(diào)試了一下STC15W408AS的一個(gè)MODBUS_RTU的通訊,程序能夠調(diào)通,但是用modscan32的時(shí)候,只能讀連續(xù)的18個(gè)數(shù)值,搞不懂了。貼上代碼,請(qǐng)各位大神幫忙看看

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


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


****本行不是代碼*****只是說(shuō)明 ****** 以下是modbusRTU.c *************************************************************

單片機(jī)源程序如下:
  1. #include "config.h"
  2. #include "modbusRTU.h"
  3. #include "CRC16.h"
  4. bit flagFrame = 0; /*幀接收完成標(biāo)志*/
  5. bit flagTxd = 0;    /*單字節(jié)發(fā)送完成標(biāo)志*/
  6. unsigned char cntRxd = 0;  /*接收字節(jié)計(jì)數(shù)器*/
  7. unsigned char  bufRxd[128]; /*接收字節(jié)緩沖區(qū)*/

  8. /*預(yù)定義一個(gè)或多個(gè)實(shí)參數(shù)組,用于存放需要處理的數(shù)據(jù)*/
  9. unsigned int idata regGroup[60]={0};  //數(shù)據(jù)緩存區(qū),全局變量,
  10. unsigned int idata modbusData[2]={0};   //該數(shù)組用來(lái)存放modbus數(shù)據(jù),分別為波特率和地址

  11. /*串口配置函數(shù),baud-波特率*/
  12. void UartInit(unsigned int baud)                //9600bps@22.1184MHz   16位自動(dòng)重裝載
  13. {
  14.         RS485_DIR = 0;
  15.         SCON = 0x50;                //8位數(shù)據(jù),可變波特率
  16.         AUXR |= 0x01;                //串口1選擇定時(shí)器2為波特率發(fā)生器
  17.         AUXR |= 0x04;                //定時(shí)器2時(shí)鐘為Fosc,即1T
  18.         T2L = (65536-SYSclk/baud/4) & 0xff;                //設(shè)定定時(shí)初值,取低8位
  19.         T2H = (65536-SYSclk/baud/4) >> 8;                //設(shè)定定時(shí)初值,取高8位
  20.         AUXR |= 0x10;                //啟動(dòng)定時(shí)器2
  21.         AUXR1 = 0x40;       //將串口切換到P36P37,此處之所以要將串口切換,是因?yàn)镻30/P31需要作為下載口
  22.         ES = 1;
  23. }


  24. /*串口數(shù)據(jù)寫(xiě)入,即串口發(fā)送函數(shù),buf-待發(fā)送數(shù)據(jù)指針,len-指定的發(fā)送長(zhǎng)度*/
  25. void UartWrite(unsigned char *buf, unsigned char len)
  26. {
  27.         RS485_DIR = 1;
  28.         while(len--)
  29.         {
  30.                 flagTxd = 0;   //清零發(fā)送標(biāo)志
  31.                 SBUF = *buf++;  //發(fā)送一個(gè)字節(jié)數(shù)據(jù)
  32.                 while(!flagTxd);
  33.         }
  34. //        DelayX10us(5);
  35.         Delay5ms();   //延時(shí)5ms,等待一幀數(shù)據(jù)的最后3.5字符發(fā)送完成
  36.         RS485_DIR = 0;
  37. }


  38. /*串口數(shù)據(jù)讀取,buf-接收數(shù)據(jù)指針,len-指定的讀取長(zhǎng)度,返回值,實(shí)際讀到的數(shù)據(jù)長(zhǎng)度*/
  39. unsigned char UartRead(unsigned char *buf, unsigned char len)
  40. {
  41.         unsigned char i;
  42.         if(len > cntRxd)  //指定讀取長(zhǎng)度 大于 實(shí)際接收到的長(zhǎng)度時(shí),
  43.         {                 //讀取長(zhǎng)度設(shè)置為實(shí)際接收到的數(shù)據(jù)長(zhǎng)度
  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. /*串口接收監(jiān)控,由空閑時(shí)間判定幀結(jié)束,需在定時(shí)中斷中調(diào)用,ms-定時(shí)間隔*/
  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. /*串口驅(qū)動(dòng)函數(shù),監(jiān)測(cè)數(shù)據(jù)幀的接收,調(diào)用函數(shù)功能,需要在主函數(shù)中調(diào)用*/
  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. /*串口動(dòng)作函數(shù),根據(jù)接收到的信號(hào)執(zhí)行相應(yīng)的動(dòng)作
  112.    buf-接收到的命令幀指針,len - 命令幀長(zhǎng)度*/
  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);   //計(jì)算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校驗(yàn)不通過(guò),則退出該函數(shù)
  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];  //提取寄存器數(shù)量                                       
  138.                             buf[2] = cnt*2;  //提取數(shù)據(jù)的字節(jié)數(shù),為寄存器數(shù)量 *2                                       
  139.                             len = 3;  // 幀前部已經(jīng)有地址、功能、字節(jié)數(shù),所以len 從3開(kāi)始算                                       
  140.                             while(cnt--)  //cnt=2
  141.                             {
  142.                                     buf[len] = regGroup[i] >> 8;   //高字節(jié)  len=3
  143.                                     buf[len+1] = regGroup[i] & 0xff; //低字節(jié) len = 3+1
  144.                                     len = len + 2;
  145.                                     i = i + 1;
  146.                             }
  147.                             break;
  148.                     }
  149.                     else    //寄存器地址不被支持時(shí),返回錯(cuò)誤碼
  150.                     {
  151.                             buf[1] = 0x83;  // 此處錯(cuò)誤碼 為 0x03(功能碼) + 0x80(功能碼出錯(cuò)時(shí))
  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];   //需要寫(xiě)入的寄存器 的 地址低位
  160.                                 regGroup[i] = 256 * buf[4] + buf[5];
  161.                                 len = 6;
  162.                                 break;
  163.                         }
  164.                         else
  165.                         {
  166.                                 buf[1] = 0x86;  // 此處錯(cuò)誤碼 為 0x06(功能碼) + 0x80(功能碼出錯(cuò)時(shí))
  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);   //計(jì)算返回幀的CRC校驗(yàn)值
  178.         buf[len] = crc >> 8;             //高字節(jié)
  179.         buf[len+1] = crc & 0xFF;     //低字節(jié)               
  180.         UartWrite(buf, len+2);         
  181. }


  182. ****本行不是代碼*****只是說(shuō)明 ****** 以下是main.c *************************************************************
  183. #include "config.h"
  184. #include "power_led.h" /*開(kāi)發(fā)板專有,啟動(dòng)STC的電源和LED*/
  185. #include "modbusRTU.h"

  186. void Timer0Init(void);
  187. void main()
  188. {
  189.         modbusData[0]=9600;   //此處暫時(shí)先用常數(shù),后期從EEPROM中讀取
  190.         modbusData[1]=2;      //此處暫時(shí)先用常數(shù),后期從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;                //定時(shí)器時(shí)鐘1T模式
  203.         TMOD &= 0xF0;                //設(shè)置定時(shí)器模式
  204.         TL0 = 0x9A;                //設(shè)置定時(shí)初值
  205.         TH0 = 0xA9;                //設(shè)置定時(shí)初值
  206.         TF0 = 0;                //清除TF0標(biāo)志
  207.         TR0 = 1;                //定時(shí)器0開(kāi)始計(jì)時(shí)
  208.         ET0 = 1;
  209. }

  210. /*T0中斷函數(shù),執(zhí)行串口接收監(jiān)控和蜂鳴器驅(qū)動(dòng)*/
  211. void InterruptTimer0() interrupt 1
  212. {
  213.         UartRxMonitor(1);
  214.         regGroup[1] ++;
  215.         if(regGroup[1]>65535)regGroup[1]=0;
  216. }
復(fù)制代碼

STC15W408AS_Modbus.rar

197.29 KB, 下載次數(shù): 75

完整的程序

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂1 踩
回復(fù)

使用道具 舉報(bào)

沙發(fā)
ID:753989 發(fā)表于 2020-6-19 16:12 | 只看該作者
大部分代碼來(lái)自本站下載,自己做了一點(diǎn)小改
回復(fù)

使用道具 舉報(bào)

板凳
ID:753989 發(fā)表于 2020-6-19 16:19 | 只看該作者
另外說(shuō)明一下,讀取數(shù)據(jù)長(zhǎng)度大于19的時(shí)候,我看 USB轉(zhuǎn)485的轉(zhuǎn)換模塊上的燈,看下來(lái)正常的
回復(fù)

使用道具 舉報(bào)

地板
ID:753989 發(fā)表于 2020-6-20 13:41 | 只看該作者
沒(méi)有人回我嗎??
回復(fù)

使用道具 舉報(bào)

5#
ID:753989 發(fā)表于 2020-6-29 11:35 | 只看該作者
@ admin  沒(méi)人回我,能幫幫忙嗎?
回復(fù)

使用道具 舉報(bào)

6#
ID:753989 發(fā)表于 2020-7-1 08:39 | 只看該作者
已經(jīng)找到問(wèn)題了,第96行 unsigned char xdata buf[40]; 這個(gè)地議定義的buf[40]的數(shù)組長(zhǎng)度不夠,當(dāng)數(shù)據(jù)(字節(jié))長(zhǎng)度超過(guò)40時(shí),就不能夠存入這個(gè)緩存了。STC15W408AS的數(shù)據(jù)手冊(cè),xdata最大值為256,所以根據(jù)keil編譯的結(jié)果,可以將這個(gè)緩存改成buf[110];
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表