找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 7905|回復(fù): 12
收起左側(cè)

Modbus 51單片機例程與資料集合

  [復(fù)制鏈接]
ID:297890 發(fā)表于 2018-3-27 18:56 | 顯示全部樓層 |閱讀模式
分享關(guān)于modbus 51 例程,資料 工具:
0.jpg

Modbus兩種協(xié)議的編程方法:

1、LRC校驗
LRC域是一個包含一個8位二進制值的字節(jié)。LRC值由傳輸設(shè)備來計算并放到消息幀中,接收設(shè)備在接收消息的過程中計算LRC,并將它和接收到消息中LRC域中的值比較,如果兩值不等,說明有錯誤。
LRC校驗比較簡單,它在ASCII協(xié)議中使用,檢測了消息域中除開始的冒號及結(jié)束的回車換行號外的內(nèi)容。它僅僅是把每一個需要傳輸?shù)臄?shù)據(jù)按字節(jié)疊加后取反加1即可。下面是它對應(yīng)的代碼:

BYTE GetCheckCode(const char * pSendBuf, int nEnd)//獲得校驗碼
{
BYTE byLrc = 0;
char pBuf[4];
int nData = 0;
for(i=1; i<end; i+=2) //i初始為1,避開“開始標(biāo)記”冒號
{
//每兩個需要發(fā)送的ASCII碼轉(zhuǎn)化為一個十六進制數(shù)
pBuf [0] = pSendBuf [ i];
pBuf [1] = pSendBuf [i+1];
pBuf [2] = '\0';
sscanf(pBuf,"%x",& nData);
byLrc += nData;
}

byLrc = ~ byLrc;
byLrc ++;
return byLrc;
}

2、CRC校驗
CRC域是兩個字節(jié),包含一16位的二進制值。它由傳輸設(shè)備計算后加入到消息中。接收設(shè)備重新計算收到消息的CRC,并與接收到的CRC域中的值比較,如果兩值不同,則有誤。
CRC是先調(diào)入一值是全“1”的16位寄存器,然后調(diào)用一過程將消息中連續(xù)的8位字節(jié)各當(dāng)前寄存器中的值進行處理。僅每個字符中的8Bit數(shù)據(jù)對CRC有效,起始位和停止位以及奇偶校驗位均無效。
CRC產(chǎn)生過程中,每個8位字符都單獨和寄存器內(nèi)容相或(OR),結(jié)果向最低有效位方向移動,最高有效位以0填充。LSB被提取出來檢測,如果LSB為1,寄存器單獨和預(yù)置的值或一下,如果LSB為0,則不進行。整個過程要重復(fù)8次。在最后一位(第8位)完成后,下一個8位字節(jié)又單獨和寄存器的當(dāng)前值相或。最終寄存器中的值,是消息中所有的字節(jié)都執(zhí)行之后的CRC值。
CRC添加到消息中時,低字節(jié)先加入,然后高字節(jié)。下面是它對應(yīng)的代碼:
WORD GetCheckCode(const char * pSendBuf, int nEnd)//獲得校驗碼
{
WORD wCrc = WORD(0xFFFF);
for(int i=0; i<nEnd; i++)
{
wCrc ^= WORD(BYTE(pSendBuf[ i]));
for(int j=0; j<8; j++)
{
if(wCrc & 1)
{
wCrc >>= 1;
wCrc ^= 0xA001;
}
else
{
wCrc >>= 1;
}
}
}
return wCrc;
}

對于一條RTU協(xié)議的命令可以簡單的通過以下的步驟轉(zhuǎn)化為ASCII協(xié)議的命令:

1、 把命令的CRC校驗去掉,并且計算出LRC校驗取代。
2、 把生成的命令串的每一個字節(jié)轉(zhuǎn)化成對應(yīng)的兩個字節(jié)的ASCII碼,比如0x03轉(zhuǎn)化成0x30,0x33(0的ASCII碼和3的ASCII碼)。
3、 在命令的開頭加上起始標(biāo)記“:”,它的ASCII碼為0x3A。
4、 在命令的尾部加上結(jié)束標(biāo)記CR,LF(0xD,0xA),此處的CR,LF表示回車和換行的ASCII碼。

掌握兩種協(xié)議的編程方法,剩下的就是C語言的問題了。

悉雨辰寂

單片機源程序如下:
  1. #include "main.h"
  2. /******************************
  3. modbus RTU 的C51程序
  4. 單片機89S52
  5. 通信波特率 9600 8位數(shù)據(jù) 1位停止位 偶校驗 485通位接口
  6. 單片機控制板地址 localAddr(變量)
  7. 通信可設(shè)置數(shù)據(jù)的地址:
  8. 字地址 0 - 255 (只取16位的低8位)
  9. 位地址 0 - 255 (只取16位的低8位)
  10. *******************************/

  11. uint32        dwTickCount,dwIntTick;        //時鐘
  12. uint8        idata sendBuf[16],receBuf[16]; //發(fā)送接收緩沖區(qū)
  13. uint8        idata checkoutError;        // ==2 偶校驗錯  
  14. uint8        idata receTimeOut;                //接收超時
  15. uint8        idata c10ms;                        //10ms 計時
  16. bit                b1ms,bt1ms,b10ms,bt10ms,b100ms,bt100ms;        //定時標(biāo)志位

  17. // 串行中斷程序
  18. void commIntProc() interrupt 4
  19. {
  20.         if(TI)
  21.         {
  22.                 TI = 0;
  23.                 if(sendPosi < sendCount)
  24.                 {
  25.                         sendPosi++;
  26.                         ACC = sendBuf[sendPosi];
  27.                         TB8 = P;        //加上校驗位
  28.                         SBUF = sendBuf[sendPosi];
  29.                 }
  30.                 else
  31.                 {
  32.                         b485Send = 0;    //發(fā)送完后將485置于接收狀態(tài)
  33.                         receCount = 0;   //清接收地址偏移寄存器
  34.                         checkoutError = 0;
  35.                 }
  36.         }
  37.         else if(RI)
  38.         {
  39.                 RI = 0;
  40.                 receTimeOut = 10;    //通訊超時值
  41.                 receBuf[receCount] = SBUF;
  42.                 ACC = receBuf[receCount];
  43.                 if(P != RB8)
  44.                         checkoutError = 2;        //偶校驗出錯
  45.                 receCount++;          //接收地址偏移寄存器加1
  46.                 receCount &= 0x0f;    //最多一次只能接收16個字節(jié)
  47.         }

  48. }   // void CommIntProc()

  49. //定時器0 1ms 中斷
  50. void timer0IntProc() interrupt 1
  51. {
  52.         TL0 = TIMER_LOW;
  53.     TH0 = TIMER_HIGHT;
  54.     dwIntTick++;
  55.         bt1ms = 1;
  56.     c10ms++;
  57.     if(c10ms >= 10)
  58.     {
  59.         c10ms = 0;      //10ms計時器清零
  60.         bt10ms = 1;
  61.     }
  62. }   // void Timer0IntProc()

  63. //外部中斷0
  64. void intEx0Proc(void) interrupt 0
  65. {

  66. }

  67. //計數(shù)器1中斷
  68. void counter1IntProc(void) interrupt 3 using 1
  69. {

  70. }


  71. //定時處理
  72. void timeProc(void)
  73. {
  74.         static uint8 c200ms;

  75.     bWatchDog = ~ bWatchDog;    //看門狗取反
  76.         b1ms = 0;
  77.         b10ms = 0;
  78.         b100ms = 0;
  79.         
  80.         ET0 = 0;
  81.         dwTickCount = dwIntTick;
  82.         ET0 = 1;

  83.         if(bt1ms)
  84.         {
  85.                 bt1ms = 0;
  86.                 b1ms = 1;

  87.         if(receTimeOut>0)
  88.         {
  89.             receTimeOut--;
  90.             if(receTimeOut==0 && receCount>0)   //判斷通訊接收是否超時
  91.             {
  92.                 b485Send = 0;       //將485置為接收狀態(tài)
  93.                 receCount = 0;      //將接收地址偏移寄存器清零
  94.                                 checkoutError = 0;
  95.             }
  96.         }
  97.         }
  98.         
  99.         if(bt100ms)
  100.         {
  101.                 bt100ms = 0;
  102.                 b100ms = 1;
  103.         }
  104.     if(bt10ms)      //判斷中斷10ms標(biāo)志位是否1
  105.     {
  106.         bt10ms = 0;     //清中斷10ms標(biāo)志位
  107.                 b10ms = 1;

  108.         c200ms++;                   //200ms計時器加1
  109.         if(c200ms >= 20)            //判斷是否計時到200ms
  110.         {
  111.             c200ms = 0;             //清200ms計時器
  112.             bRunLED = ~bRunLED;     //取反運行指示燈         
  113.         }
  114.     }
  115. }   // void TimerProc(void)

  116. //初始化串口
  117. void initUart(void)
  118. {
  119.         //T2 用于波特率 9600
  120.         T2CON = 0x30;
  121.         RCAP2H = 0xff;
  122.         RCAP2L = 0xb8;
  123.         TR2 = 1;

  124.         //偶校驗                                                
  125.         SCON = 0xd0;
  126.     PCON = 0;
  127.     ES = 1;
  128. }//void initUart(void)

  129. //初始化中斷
  130. void initInt(void)
  131. {
  132.         TMOD = 0x51;
  133.         TH0 = TIMER_HIGHT;
  134.         TL0 = TIMER_LOW;
  135.         TR0 = 1;        
  136.     ET0 = 1;
  137.         TH1 = 0;                        //9600
  138.     TL1 = 0;
  139.         TR1 = 0;                        //定時器1用于計數(shù)定時器2用于波特
  140.         ET1 = 1;
  141.         //PT1 = 1;

  142.         IT0 = 1;        
  143.     IT1 = 1;
  144.         EX0 = 0;        
  145.         PX0 = 1;
  146.     EX1 = 0;

  147.         initUart();
  148. ……………………

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

完整的Word格式文檔51黑下載地址:
Modbus兩種協(xié)議的編程方法.doc (26 KB, 下載次數(shù): 160)
51單片機modbus例程.rar (3.54 MB, 下載次數(shù): 293)


評分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

回復(fù)

使用道具 舉報

ID:337813 發(fā)表于 2018-6-19 13:16 | 顯示全部樓層
謝謝無私分享
回復(fù)

使用道具 舉報

ID:20154 發(fā)表于 2018-7-7 10:52 | 顯示全部樓層

謝謝樓主[
回復(fù)

使用道具 舉報

ID:466032 發(fā)表于 2019-1-10 18:07 | 顯示全部樓層
謝謝分享
回復(fù)

使用道具 舉報

ID:449522 發(fā)表于 2019-1-10 19:03 來自手機 | 顯示全部樓層
學(xué)習(xí)學(xué)習(xí),謝謝樓主分享。
回復(fù)

使用道具 舉報

ID:548322 發(fā)表于 2019-12-5 13:17 | 顯示全部樓層
謝謝無私分享
回復(fù)

使用道具 舉報

ID:368583 發(fā)表于 2019-12-9 16:12 | 顯示全部樓層
謝謝無私分享
回復(fù)

使用道具 舉報

ID:676027 發(fā)表于 2019-12-28 14:49 | 顯示全部樓層
很頂?shù)馁Y源!
回復(fù)

使用道具 舉報

ID:283261 發(fā)表于 2019-12-29 18:53 | 顯示全部樓層
單片機用處真是大,學(xué)習(xí)中
回復(fù)

使用道具 舉報

ID:674198 發(fā)表于 2020-4-18 17:42 | 顯示全部樓層
感謝樓主
回復(fù)

使用道具 舉報

ID:663975 發(fā)表于 2020-4-22 22:37 | 顯示全部樓層
感謝樓主分享!
回復(fù)

使用道具 舉報

ID:437175 發(fā)表于 2020-5-4 17:25 | 顯示全部樓層
什么下載網(wǎng)頁,下載三次都下載不成功,第一次35%,第二次34%,第三次31%,扣了15黑幣還手機流量用了幾百兆。
回復(fù)

使用道具 舉報

ID:437175 發(fā)表于 2020-5-29 17:34 | 顯示全部樓層
下載四次都在35%就跑流量不下數(shù)據(jù)(50-100M),20積分泡水了還損三百多兆流量。垃圾下載連接。
回復(fù)

使用道具 舉報

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

本版積分規(guī)則

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

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

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