|
- #include<reg51.h>
- #include<string.h>
- #define uchar unsigned char
- #define uint unsigned int
-
- /* 通信命令 */
- #define __ACTIVE_ 0x01 // 主機詢問從機是否存在
- #define __GETDATA_ 0x02 // 主機發送讀設備請求
- #define __OK_ 0x03 // 從機應答
- #define __STATUS_ 0x04 // 從機發送設備狀態信息
- /****************************幀格式****************************
- 目的地址-本機地址-命令字-數據-校驗碼-結束標志 至少5個字節
- *************************************************************/
- #define MAXSIZE 0x08 // 緩沖區長度
- #define ERRLEN 12 // 任何通信幀長度超過12則表示出錯
- uchar send_buf[MAXSIZE]={0xc0,0xdb}; // 該緩沖區用于保存設備狀態信息
- uchar rec_buf[ERRLEN]; // 保存接收到的幀
- uchar dev; // 該字節用于保存本機設備號
- sbit M_DE = P1^0; // 驅動器使能,1有效
- sbit M_RE = P1^1; // 接收器使能,0有效
- void get_status(); // 調用該函數獲得設備狀態信息,函數代碼未給出
- void send_data(uchar type, uchar len, uchar *send_buf); // 發送數據幀
- bit recv_cmd(uchar *type); // 接收主機命令,主機請求僅包含命令信息
- void send_byte(uchar da); // 該函數發送一幀數據中的一個字節,由send_data()函數調用
-
- void main()
- {
- uchar type;
- uchar len=2;
-
- /* 系統初始化 */
- dev =0xff; //設備號
- TMOD = 0x20; // 定時器T1使用工作方式2
- TH1 = 0xfd; // 設置初值
- TL1 = 0xfd;
- TR1 = 1; // 開始計時
- PCON = 0x00; // bps不倍增
- SCON = 0x50; // 工作方式1,波特率9600bps,允許接收
- ES = 1; // 開串口中斷
- IT0 = 0; // 外部中斷0使用電平觸發模式
- EX0 = 1; // 開啟外部中斷0
- EA = 1; // 開啟中斷
- /* 主程序流程 */
- while(1) // 主循環
- {
-
- if(recv_cmd(&type) == 0) // 發生幀錯誤或幀地址與本機地址不符,丟棄當前幀后返回
- continue;
- SBUF = rec_buf[0];
- while(!TI);
- TI = 0;
- SBUF = rec_buf[1];
- while(!TI);
- TI = 0;
- SBUF = rec_buf[2];
- while(!TI);
- TI = 0;
- SBUF = rec_buf[3];
- while(!TI);
- TI = 0;
- SBUF = rec_buf[4];
- while(!TI);
- TI = 0;
- /*
- switch(type)
- {
- case __ACTIVE_: // 主機詢問從機是否存在
- send_data(__OK_, 0, send_buf); // 發送應答信息,這里buf的內容并未用到
- break;
- case __GETDATA_:
- len = strlen(send_buf);
- send_data(__STATUS_, len, send_buf); // 發送設備狀態信息
- break;
- default:
- break; // 命令類型錯誤,丟棄當前幀后返回
- }
- */
- }
- }
- /* 該函數接收一幀數據并進行檢測,無論該幀是否錯誤,函數均會返回
- * 函數參數type保存接收到的命令字
- * 當接收到數據幀錯誤或其地址位不為0時(非主機發送幀),函數返回0,反之返回1
- */
- bit recv_cmd(uchar *type)
- {
- bit db = 0; // 當接收到的上一個字節為0xdb時,該位置位
- bit c0 = 0; // 當接收到的上一個字節為0xc0時,該位置位
-
- uchar tmp;
- uchar sum = 0;
- uchar i,j;
-
- M_DE = 0; // 置發送禁止,接收允許
- M_RE = 0;
-
- /* 接收一幀數據 */
- i = 0;
- while(!c0) // 循環直至幀接收完畢
- {
- RI = 0;
- while(!RI);
- tmp = SBUF;
- RI = 0;
- if(db == 1) // 接收到的上一個字節為0xdb
- {
- switch(tmp)
- {
- case 0xdd:
- rec_buf[i] = 0xdb; // 0xdbdd表示0xdb
-
- db = 0;
- break;
- case 0xdc:
- rec_buf[i] = 0xc0; // 0xdbdc表示0xc0
-
- db = 0;
- break;
- default :
- return 0; // 幀錯誤,返回
- }
- i++;
- }
- else{
- switch(tmp) // 正常情況
- {
- case 0xc0: // 幀結束
- c0 = 1;
-
- break;
- case 0xdb: // 檢測到轉義字符
- db = 1;
-
- break;
- default: // 普通數據
- rec_buf[i] = tmp; // 保存數據
- // 計算校驗字節
- i++;
- }
- if(i == ERRLEN) // 幀超長,錯誤,返回
- return 0;
- }
- }
-
- /* 判斷幀是否錯誤 */
- if(i<4) // 幀過短,錯誤,返回
- return 0;
- for(j=0;j<i-1;j++)
- sum+=rec_buf[j];
- sum=sum%256;
- if(sum !=rec_buf[i-1] ) // 校驗錯誤,返回
- return 0;
- if(rec_buf[0] != dev) // 非訪問本機命令,錯誤,返回
- return 0;
- *type = rec_buf[2]; // 獲得命令字
- return 1; // 函數成功返回
- }
-
- // 該函數發送一幀數據幀,參數type為命令字、len為數據長度、buf為要發送的數據內容
- void send_data(uchar type, uchar len, uchar *send_buf)
- {
- uchar i;
- uchar sum = 0; // 該字節用于保存校驗字節
-
- M_DE = 1; // 置發送允許,接收禁止
- M_RE = 1;
-
- send_byte(0x00); // 發送目的地址
- sum+=0x00; //假設主機地址為0x00
- send_byte(dev); // 發送本機地址
- sum+=dev;
- send_byte(type); // 發送命令字
- sum+=type;
- //send_byte(len); // 發送長度
-
- for(i=0; i<len; i++) // 發送數據
- {
- send_byte(*send_buf);
- sum =sum+ (*send_buf);
- send_buf++;
- }
- sum%=256;
- send_byte(sum); // 發送校驗字節
-
- TI = 0; // 發送幀結束標志
- SBUF = 0xc0;
- while(!TI);
- TI = 0;
- }
- //該函數發送一個數據字節,若該字節為0xdb,則發送0xdbdd,若該字節為0xc0則,發送0xdbdc
- void send_byte(uchar da)
- {
- switch(da)
- {
- case 0xdb: // 字節為0xdb,發送0xdbdd
- TI = 0;
- SBUF = 0xdb;
- while(!TI);
- TI = 0;
- SBUF = 0xdd;
- while(!TI)
- TI = 0;
- break;
- case 0xc0: // 字節為0xc0,發送0xdbdc
- TI = 0;
- SBUF = 0xdb;
- while(!TI);
- TI = 0;
- SBUF = 0xdc;
- while(!TI)
- TI = 0;
- break;
- default: // 普通數據則直接發送
- TI = 0;
- SBUF = da;
- while(!TI);
- TI = 0;
- }
- }
復制代碼
|
|