找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

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

MODBUS協(xié)議框架程序,不過多占用MCU時間

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:145411 發(fā)表于 2019-7-1 10:01 | 只看該作者 |只看大圖 回帖獎勵 |倒序瀏覽 |閱讀模式
modbus做了03 04 的解析,不過多占用MCU時間,發(fā)送接收分時處理,有更新會及時上傳


單片機源程序如下:
  1. #include <all.h>

  2. unsigned char xdata Modbus_Read_Buffer[256] = 0;  //數(shù)據(jù)接收緩沖區(qū)
  3. unsigned char xdata Modbus_Write_Buffer[256] = 0; //數(shù)據(jù)寄存器
  4. unsigned char xdata Modbus_Send[256] = 0;//數(shù)據(jù)發(fā)送數(shù)組
  5. unsigned char xdata Modbus_CRC[20]= {0};

  6. unsigned int Modbus_Read_Value = 0;        //接收字節(jié)計數(shù)
  7. unsigned int Modbus_Write_Value = 0;  //發(fā)送字節(jié)計數(shù)                                    
  8. unsigned char Modbus_Read_Complete = 0; //幀接收完成標(biāo)志位
  9. unsigned int CRC_Reg = 0xffff;
  10. unsigned char CRC_Value = 0;
  11. unsigned int Modbus_CRC_Cecode(unsigned char *puchMsg, unsigned int usDataLen);        //CRC校驗
  12. void Modbus_Init();        //幀檢測定時器
  13. void Modbus_data_Decode();


  14. void Modbus_Init()//利用定時器2來計算數(shù)據(jù)幀 定時1MS   30MHz@1ms
  15. {
  16.         AUXR |= 0x04;//定時器2用作定時,不分頻
  17.         IE2 |= 0x04;//允許定時器2中斷
  18.         T2L = 0xd0;                //設(shè)置定時初值
  19.         T2H = 0x8a;                //設(shè)置定時初值
  20.         EA = 1;  //開總中斷
  21.         AUXR |= 0x10;
  22. }



  23. //****************************************************************************
  24. //函數(shù)名:Modbus_CRC_Cecode
  25. //函數(shù)作用:用于CRC_16的校驗計算
  26. //輸入?yún)?shù)1:要計算數(shù)據(jù)的序列名稱
  27. //輸入?yún)?shù)2:要計算數(shù)據(jù)的字節(jié)長度
  28. //輸出參數(shù):計算好的CRC值,16位數(shù)據(jù)
  29. //****************************************************************************
  30. unsigned int Modbus_CRC_Cecode(unsigned char *puchMsg, unsigned int usDataLen)
  31. {
  32.         
  33.          unsigned int i,j,CRC_Reg1,Check1;
  34.          CRC_Reg1 = 0xFFFF;
  35.          for(i=0;i<usDataLen;i++)
  36.          {
  37.                 CRC_Reg1 = CRC_Reg1 ^ (unsigned int)*(puchMsg + i);
  38.                 for(j=0;j<8;j++)
  39.                 {
  40.                         Check1 = CRC_Reg1 & 0x0001;
  41.                         CRC_Reg1 >>= 1;
  42.                         if(Check1==0x0001)
  43.                         {
  44.                                 CRC_Reg1 ^= 0xA001;
  45.                         }
  46.                 }
  47.          }
  48.          
  49.          return CRC_Reg1;
  50. }


  51. //****************************************************************************
  52. //函數(shù)名:Modbus_03
  53. //函數(shù)作用:用來執(zhí)行MODBUS協(xié)議中03碼的功能
  54. //輸入?yún)?shù):無
  55. //輸出參數(shù):無
  56. //備注:寄存器地址 0X00~0XFF;
  57. //****************************************************************************
  58. void Modbus_03(void)
  59. {
  60.         unsigned int i;
  61.         unsigned int CRC = 0;
  62.         P55 = ~P55;//調(diào)試板的狀態(tài)指示燈

  63.         Modbus_Read_Value = 0;//接收數(shù)據(jù)清零,很重要,根據(jù)MODBUS指令放在需要的位置,影響很大
  64. //        for(i = 0; i < 256; i++)//調(diào)試用給內(nèi)部寄存器 賦值   (使用時可以注釋)
  65. //        {
  66. //                Modbus_Write_Buffer[i] = i;        
  67. //        }                        

  68.         Modbus_Send[0] = Modbus_Read_Buffer[0];
  69.         Modbus_Send[1] = Modbus_Read_Buffer[1];
  70.         Modbus_Send[2] = (Modbus_Read_Buffer[5] * 2);

  71.          for(i = 3; i < (Modbus_Send[2]+3); i++)         
  72.         {

  73.                 Modbus_Send[i] = Modbus_Write_Buffer[(i + Modbus_Read_Buffer[3] - 3)];         
  74.         }
  75.         CRC = Modbus_CRC_Cecode(Modbus_Send,(Modbus_Send[2]+3));
  76.         Modbus_Send[(Modbus_Send[2] + 3)] = CRC & 0x00ff;                                                                  
  77.          Modbus_Send[(Modbus_Send[2] + 4)] = CRC >> 8;                                                   
  78.          UART_Byte_Value = (Modbus_Send[2] + 5);//啟用發(fā)送前要設(shè)置發(fā)送字節(jié)長度,每次發(fā)送前都要設(shè)置
  79.         while(!UART_OFF);//發(fā)送數(shù)據(jù)前檢測一下串口是否繁忙
  80.         UART_OFF = 0;//啟用發(fā)送前要把發(fā)送結(jié)束標(biāo)志位清零,每次發(fā)送前都要設(shè)置
  81.          SBUF = Modbus_Send[0];

  82.          for(i = 0; i < 256; i++)//清空發(fā)送緩存區(qū)數(shù)據(jù)
  83.         {
  84.                 Modbus_Read_Buffer[i] = 0;;        
  85.         }
  86.         Modbus_CRC[0] = 0;
  87.         Modbus_CRC[1] = 0;        
  88. }

  89. void Modbus_data_Decode()
  90. {
  91.                   
  92.         if(Modbus_Read_Complete)
  93.         {                 
  94.                  Modbus_Read_Complete = 0;
  95.         
  96.                  if((Modbus_CRC[1] == Modbus_Read_Buffer[Modbus_Read_Value-1]) && (Modbus_CRC[0] == Modbus_Read_Buffer[Modbus_Read_Value-2]))

  97.                  {
  98.                          //這里可以開始分析數(shù)據(jù) 程序進入這里就證明CRC檢驗已經(jīng)通過了
  99.                          if(Modbus_Read_Buffer[0] == (unsigned char)MODBUS_ADD)
  100.                          {
  101.                                   switch(Modbus_Read_Buffer[1])
  102.                                  {
  103.                                          case 03: Modbus_03(); break;
  104.                                  }
  105.                          }
  106.                          else
  107.                          {
  108.                                    Modbus_Read_Value = 0;//接收數(shù)據(jù)清零,很重要,根據(jù)MODBUS指令放在需要的位置,影響很大
  109.                          }
  110.                  }
  111.                  
  112.          }
  113. }

  114. /* 串口接收監(jiān)控,由空閑時間判定幀結(jié)束,需在定時中斷中調(diào)用,ms-定時間隔 */
  115. void UartRxMonitor(unsigned char ms)
  116. {
  117.     static unsigned char cntbkp = 0;
  118.     static unsigned char idletmr = 0;

  119.     if (Modbus_Read_Value > 0)  //接收計數(shù)器大于零時,監(jiān)控總線空閑時間
  120.     {
  121.         if (cntbkp != Modbus_Read_Value)  //接收計數(shù)器改變,即剛接收到數(shù)據(jù)時,清零空閑計時
  122.         {
  123.             cntbkp = Modbus_Read_Value;
  124.             idletmr = 0;
  125.         }
  126.         else                   //接收計數(shù)器未改變,即總線空閑時,累積空閑時間
  127.         {
  128.             if (idletmr < 3)  //空閑計時小于3ms時,持續(xù)累加
  129.             {
  130.                 idletmr += ms;
  131.                 if (idletmr >= 3)  //空閑時間達到3ms時,即判定為一幀接收完畢
  132. ……………………

  133. …………限于本文篇幅 余下代碼請從51黑下載附件…………
復(fù)制代碼

所有資料51hei提供下載:
MODBUS框架V2.0.zip (54.36 KB, 下載次數(shù): 58)

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

使用道具 舉報

沙發(fā)
ID:137005 發(fā)表于 2019-11-28 21:48 | 只看該作者
9600bps@30MHz?
樓主用的是外接晶振?
回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

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

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

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