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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2171|回復: 1
打印 上一主題 下一主題
收起左側

MCU51-PC通訊協議

[復制鏈接]
跳轉到指定樓層
樓主
ID:105323 發表于 2016-2-13 02:54 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
  1. /****************************************************************************/
  2. /*                                                                          */
  3. /*                Copyright (c) 2016, 老馬工作室                            */
  4. /*                     All rights reserved.                                 */
  5. /*                                                                          */
  6. /*      Email:pcwebmaster@163.com                                           */
  7. /****************************************************************************/
  8. /****************************************************************************/
  9. /* 文 件 名:comint.c                                                       */
  10. /* 版    本:Version 1.0                                                    */
  11. /* 描    述:串行口中斷帶通訊協議測試                                       */
  12. /* 主控芯片:STC89LE54RD*/
  13. /* 晶振頻率:22.1184MHz*/
  14. /* 作    者:pcwebmaster                                                    */
  15. /* 函    數:                                                               */
  16. /*         system_init                                                     */
  17. /*         com_send_command                                                */
  18. /*         com_command_receive*/
  19. /*CalCRC16_1021*/
  20. /*command_decoder*/
  21. /*send_command*/
  22. /* 測    試: 發送:16 16 02 01 02 01 00 35 03 94 BD接收:49 16 16 06 AA 55 */
  23. /* 測試現象:P05/P06/P07三個LED取反*/
  24. /* 歷史記錄:20016.02.09測試通過                                            */
  25. /* 北極狼          20016-02-09     Creat Inital version. (Version 1.0)      */
  26. /****************************************************************************/
  27. #include "comint.h"

  28. void main(void)
  29. {
  30.     u_char i = 0;
  31.     system_init();   
  32.     while(1)
  33.     {
  34.         com_command_receive();
  35.     }
  36. }


  37. #include "comint.h"

  38. u_char  data  pint_buf[MAX_RINTL];   /* 串口接收緩沖區       */
  39. u_char  data  pint_read;             /* 串口緩沖區讀指針     */
  40. u_char  data  pint_write;            /* 串口緩沖區寫指針     */
  41. u_char  data  psend_int;             /* 串口發送允許標志     */
  42. u_char serial_flag = 0;              /* 串口接收數據標志位   */


  43. u_char  idata prec_buf[MAX_COMMAND_LEN];/* 命令接收緩沖區    */
  44. u_char  prec_num;                    /* 命令接收緩沖區字節數 */

  45. u_char serial_lengthl = 0;           /* 消息命令長度低8位    */
  46. u_int  serial_length = 0;            /* 消息命令長度16位     */

  47. u_char ADDRESS[2]={ZU,ZHAN};       /* byte0:通訊組地址, byte1:開發板地址 */

  48. /* 串口發送一個字節 */
  49. void com_send_command(char onebyte)
  50. {
  51.     psend_int = 0;
  52.     SBUF = onebyte;
  53.     while (psend_int != 1);
  54. }

  55. /* 串口接收數據處理 */
  56. void com_command_receive(void)
  57. {
  58.     u_char var1,var4;
  59.     u_int crc_data = 0;
  60.     var4 = pint_read;

  61.     if (var4 != pint_write)
  62.     {
  63.         var1 = pint_buf[var4];
  64.         var4 = var4+1;
  65.         if (var4 >= MAX_RINTL)
  66.             var4=0;

  67.         pint_read = var4;

  68.         switch(serial_flag)
  69.         {
  70.             case 0: /*收到起始位*/
  71.                 if (var1 == SYN)
  72.                 {
  73.                     serial_flag = 1;
  74. //com_send_command(0x01); //測試
  75.                 }
  76.                 else
  77.                 {
  78.                     serial_flag = 0;
  79.                 }
  80.                 break;

  81.             case 1:/*收到起始位*/
  82.                 if (var1 == SYN)
  83.                 {
  84.                     serial_flag = 2;
  85. //com_send_command(0x02); //測試
  86.                 }
  87.                 else
  88.                 {
  89.                     serial_flag = 0;
  90.                 }
  91.                 break;

  92.             case 2:/*收到同步位*/
  93.                 if (var1 == STX)
  94.                 {
  95.                     serial_flag = 3;
  96. //com_send_command(0x03);//測試
  97.                 }
  98.                 else
  99.                 {
  100.                     serial_flag = 0;
  101.                 }
  102.                 break;

  103.             case 3: /*收到組地址*/
  104.                 if (var1 == ADDRESS[0])
  105.                 {
  106.                     serial_flag = 4;
  107.                     prec_num = 1;
  108.                     prec_buf[0] = var1;
  109. //com_send_command(0x04); //測試
  110.                 }
  111.                 else
  112.                 {
  113.                     serial_flag = 0;
  114.                 }
  115.                 break;

  116.             case 4:/*收到本機地址或者廣播地址*/
  117.                 if ( (var1 == ADDRESS[1]) || (var1 == 0) )
  118.                 {
  119.                     prec_num = 2;
  120.                     prec_buf[1] = var1;
  121.                     serial_flag = 5;
  122. //com_send_command(0x05); //測試
  123.                 }
  124.                 else
  125.                 {
  126.                     serial_flag = 0;
  127.                 }
  128.                 break;

  129.             case 5:/*數據長度*/
  130.                 prec_num = 3;
  131.                 prec_buf[2] = var1;
  132.                 serial_lengthl = var1;
  133.                 serial_flag = 6;
  134. //com_send_command(0x06);//測試
  135.                 break;

  136.             case 6:
  137.                 prec_num = 4;
  138.                 prec_buf[3] = var1;
  139.                 serial_length |= var1;
  140.                 serial_length = ( (serial_length << 8) & 0xff00 ) + serial_lengthl + 3;
  141.                 serial_flag = 7;
  142. //com_send_command(0x07);//測試
  143.                 break;

  144.             case 7:
  145.                 prec_buf[prec_num] = var1;
  146.                 prec_num++;
  147.                 serial_length--;
  148.                 if (serial_length == 0)
  149.                 {
  150.                     crc_data = CalCRC16_1021(prec_buf, prec_num - 2); /* 計算crc校驗和(從組地址開始到ETX )*/

  151.                     if ( ( (crc_data & 0x00ff) == prec_buf[prec_num - 2]) && ( ( (crc_data >>8) & 0x00ff) == prec_buf[prec_num - 1]) ) /* 校驗和正確 */
  152.                     {
  153.                         prec_num = 1;
  154.                         var1 = 0;

  155.                         if ( (prec_buf[4] >= 0x31) && (prec_buf[4] <= 0x3b) ) /* 命令有效 */
  156.                         {
  157.                             if (prec_buf[1] != 0x00) /* 如果不是廣播地址則回應ACK*/
  158. com_send_command(0x49);//測試                                msg_last_push(MSG_ACK,0);
  159. send_command(ACK); //測試   

  160.                             command_decoder(); /* 如果校驗和正確,則進行命令解碼 */
  161.                         }
  162.                         else
  163.                         {

  164. send_command(NAK); //測試
  165.                         }
  166.                     }
  167.                     else
  168.                     {

  169. send_command(NAK); //測試
  170.                     }
  171.                     serial_flag = 0;
  172.                     prec_num = 1;
  173.                 }
  174.                 break;

  175.             default:
  176.                 serial_flag = 0;
  177.                 prec_num = 1;
  178.                 break;
  179.         }
  180.     }
  181. }
  182. /* 命令解碼子程序 */
  183. void command_decoder(void)
  184. {
  185.     u_char i = 0;

  186.     if (prec_buf[4] == 0x31)       /* 設置報警閾值   */
  187.     {  
  188. P05 = !P05;//測試
  189.         return;
  190.     }
  191.     else if (prec_buf[4] == 0x32)  /* 請求報警閾值 */
  192.     {
  193. P05 = !P06;//測試
  194.         return;
  195.     }
  196.     else if (prec_buf[4] == 0x33)  /* 修改當前時間 */
  197.     {
  198. P07 = !P07;//測試
  199.         return;
  200.     }
  201.     else if (prec_buf[4] == 0x34)  /* 請求當前時間 */
  202.     {
  203. P05 = !P05;//測試
  204. P07 = !P07;//測試
  205.         return;
  206.     }
  207.     else if (prec_buf[4] == 0x35)  /* 請求當前數據 */
  208.     {
  209. com_send_command(0xAA);//測試
  210. com_send_command(0x55);//測試
  211. P0 = ~P0;//測試
  212.         return;
  213.     }
  214.     else if (prec_buf[4] == 0x36)  /* 請求看門狗信息*/
  215.     {
  216. com_send_command(0xFF);//測試
  217.         return;
  218.     }
  219.     else if (prec_buf[4] == 0x37)  /* 請求報警情況 */
  220.     {
  221. com_send_command(0xFF);//測試
  222.         return;
  223.     }
  224.     else if (prec_buf[4] == 0x38)  /* 配置設備地址 */
  225.     {
  226.         ADDRESS[0] = prec_buf[5];  /* 通訊組地址   */
  227.         ADDRESS[1] = prec_buf[6];  /* 站地址   */

  228. com_send_command(0xFF);//測試
  229.         return;
  230.     }
  231.     else if (prec_buf[4] == 0x39)  /* 請求設備地址 */
  232.     {
  233. com_send_command(0xFF);//測試
  234.         return;
  235.     }
  236.     else if (prec_buf[4] == 0x3a)  /* 控制模擬量輸出 */
  237.     {
  238. com_send_command(0xFF);//測試
  239.         return;
  240.     }
  241.     else if (prec_buf[4] == 0x3b)  /* 控制開關量輸出 */
  242.     {
  243. com_send_command(0xFF);//測試

  244.         return;
  245.     }
  246. }

  247. /* 向PC主機發送消息幀,入口參數:消息類型 */
  248. void send_command(u_char command)
  249. {

  250.     switch (command)
  251.     {
  252.         case ACK:
  253.             com_send_command(SYN);
  254.             com_send_command(SYN);
  255.             com_send_command(ACK);
  256.             break;

  257.         case NAK:
  258.             com_send_command(SYN);
  259.             com_send_command(SYN);
  260.             com_send_command(NAK);
  261.             break;

  262.         default:
  263.             break;
  264.     }
  265. }
  266. void system_init()
  267. {
  268.     u_char loop;

  269.     EA = 0;         /* CPU關中斷      */
  270.     pint_read  = 0; /* 串口緩沖讀指針 */
  271.     pint_write = 0; /* 串口緩沖寫指針 */

  272. //
  273.     SCON = 0x48; //主控芯片STC89LE54RD
  274.     PCON = 0x80;
  275.     TMOD = 0x20;
  276.     TCON = 0x50;
  277.     TH1  = 0xFD;    // 波特率為19200  
  278.     TL1  = 0xFD;
  279.     TR1  = 1;       // 定時器1啟動計數*/
  280.     ES   = 1;       /* 串口開中斷     */
  281.     PS   = 0;       /* 串口低優先級   */
  282.     REN  = 1;       /* 串口接收允許   */
  283.     EA   = 1;       /* 開CPU中斷      */

  284.     loop = SBUF;    /* 清串口緩沖區   */
  285.     for (loop=0; loop<MAX_RINTL; loop++)
  286.     {
  287.         pint_buf[loop] = 0;
  288.     }
  289. }
  290. /*計算CRC校驗和使用MTT(0X1021)
  291. 參數:
  292. pBuff 為需計算CRC的緩沖區的頭指針
  293. BufferLen 緩沖區長度(以字節計)
  294. */
  295. u_short CalCRC16_1021(u_char x[], u_short BufferLen)
  296. {
  297.     u_short i;
  298.     u_char  j;
  299.     u_short crc16 = 0;
  300.     u_short mask = 0x1021;
  301.     u_char *pByteBuffer;
  302.     u_char tmpbyte;
  303.     u_short calval;

  304.     pByteBuffer = &x[0];

  305.     for (i = 0; i < BufferLen; i++)
  306.     {
  307.         tmpbyte = *pByteBuffer;
  308.         calval = tmpbyte << 8;
  309.         for (j = 0; j < 8; j++)
  310.         {
  311.             if ((crc16 ^ calval) & 0x8000)
  312.                 crc16 = (crc16 << 1) ^ mask;
  313.             else
  314.                 crc16 <<= 1;

  315.             calval <<= 1;
  316.         }
  317.         pByteBuffer++;
  318.     }
  319.     return crc16;
  320. }
  321. /* 串口中斷服務程序,使用第3組寄存器 */
  322. void com_int_proc(void) interrupt 4 using 3
  323. {
  324.     u_char temp;
  325.     u_char temp1;

  326.     if (TI == 1)                /* 是發送中斷       */
  327.     {
  328.         TI = 0;
  329.         psend_int = 1;          /* 可以發送         */
  330.     }

  331.     if (RI == 1)                /* 是接收中斷       */
  332.     {
  333.         RI = 0;                 /* 清串口接收中斷   */
  334.         temp1 = SBUF;
  335.         temp  = pint_write + 1; /* 判斷是否可以寫入 */
  336.         if (temp == MAX_RINTL)
  337.         {
  338.             temp=0;
  339.         }

  340.         if (temp != pint_read)
  341.         {
  342.             pint_buf[pint_write] = temp1; /* 讀取數據 */
  343.             pint_write = temp;
  344.         }
  345.     }
  346. }


  347. #ifndef __COMINT_H__
  348. #define __COMINT_H__

  349. #define ZU0x01 /*組地址*///通訊地址修改這兩項
  350. #define ZHAN0x02 /*站地址*///通訊地址修改這兩項

  351. #include "STC.h"
  352. #include <absacc.h>
  353. #include <intrins.h>
  354. #include <string.h>

  355. typedef unsigned char u_char;
  356. typedef unsigned int  u_int;
  357. typedefunsigned shortu_short;

  358. #define MAX_RINTL        8   /* 串口接收緩沖區長度   */
  359. #define SYN           0x16   /* 通訊同步位*/
  360. #define STX           0x02   /* 通訊起始位*/
  361. #define ETX           0x03   /* 通訊結束位*/

  362. #define ACK           0x06
  363. #define NAK           0x15

  364. #define MSG_ACK          2    /* 正確應答信息         */
  365. #define MSG_NAK          3    /* 錯誤應答信息         */
  366. #define MAX_COMMAND_LEN  16   /* 串口接受命令長度     */

  367. extern char str_test[25]  ;
  368. extern u_char  data  pint_read;             // 串口緩沖區讀指針     */
  369. extern u_char  data  pint_write;            // 串口緩沖區寫指針
  370. //extern u_char  data  psend_int;             // 串口發送允許標志   
  371. extern u_char  data  pint_buf[MAX_RINTL];   // 串口接收緩沖區      
  372. extern u_char serial_flag;              /* 串口接收數據標志位   */

  373. extern u_char  idata prec_buf[MAX_COMMAND_LEN];/* 命令接收緩沖區 */

  374. /* 串口發送一個字節 */
  375. extern void com_send_command(char onebyte);
  376. /* 串口接收數據處理 */
  377. extern void com_command_receive(void);
  378. /* 系統初始化 */
  379. extern void system_init();

  380. ///* 串口接收一字節數據 */
  381. //unsigned char UartReadChar(void); //reentrant
  382. /*計算CRC校驗和使用MTT(0X1021)
  383. 參數:
  384. pBuff 為需計算CRC的緩沖區的頭指針
  385. BufferLen 緩沖區長度(以字節計)
  386. */
  387. u_short CalCRC16_1021(u_char x[], u_short BufferLen);

  388. /* 命令解碼子程序 */
  389. void command_decoder(void);
  390. /* 向主機發送消息幀,入口參數:消息類型 */
  391. void send_command(u_char command);

  392. #endif
復制代碼


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

使用道具 舉報

沙發
ID:83138 發表于 2016-2-15 10:04 | 只看該作者
51黑有你更精彩...................
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 国产免费拔擦拔擦8x高清 | av不卡一区| 日韩久久精品电影 | 中文字幕视频在线观看 | 天天操天天干天天爽 | 在线看亚洲 | 91偷拍精品一区二区三区 | 999免费网站| 日韩精品一区二区三区在线观看 | 精品欧美一区二区三区久久久 | 毛片视频观看 | 日本天天色 | 毛片一级片| 在线观看欧美一区 | 91亚洲国产成人久久精品网站 | 日韩精品成人一区二区三区视频 | 一区二区三区视频 | 欧美一二区 | 亚洲免费视频在线观看 | 欧美一区 | 久久一 | 日韩理论电影在线观看 | 91精品国产综合久久久久 | 青青草原精品99久久精品66 | 色爱av| 蜜桃免费av| 欧美精品二区三区 | 精品国产欧美一区二区 | 99精品国产一区二区三区 | 天天干在线播放 | 成人欧美一区二区 | 久久久久国产 | 国产精品视频入口 | 超碰在线人人 | 欧美久久久久久久久中文字幕 | 国产成人在线视频 | 成av在线 | 91久久婷婷 | 精品国产乱码久久久久久蜜柚 | 亚洲综合无码一区二区 | 91视频在线网站 |