找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 489|回复: 0
打印 上一主题 下一主题
收起左侧

分享个自己写的自定义串口协议 附单片机代码

[复制链接]
跳转到指定楼层
楼主
ID:773929 发表于 2024-6-12 11:07 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
最近在写一个通过手控器控制设备动作的程序,手控器按键较多,所以通过串口来和外设设备进行通信,由于命令较多,所以做了个小协议以后也能用到,可做主机可做从机,适合近距离单机通讯,第二位功能码可做地址组成多机通讯,但最好加上校验位。新手入门,大佬请多多指教
附上代码
  1. //自定义串口发送协议 例:68 01 00 03 FF ,除去包头包尾,中间为 功能码,数据1,数据2
  2. // 发送数据包函数  功能码,第一位数据,第二位数据
  3. void SendDataPacket(uint8_t function, uint8_t data1, uint8_t data2) {  
  4.     // 发送包头  
  5.     USART1_SendData(PACKET_HEADER);  
  6.       
  7.     // 发送功能码  
  8.     USART1_SendData(function);  
  9.       
  10.     // 发送数据1和数据2  
  11.     USART1_SendData(data1);  
  12.     USART1_SendData(data2);  
  13.       
  14.     // 发送包尾  
  15.     USART1_SendData(PACKET_TAILER);  
  16. }


  17. void Anlsy_Data(uint8_t function, uint8_t data1, uint8_t data2)//所有按键接收解析在这里写
  18. {
  19.     if(function == KEY_STAUS)
  20.     {
  21.         SysChenyuan.mark = QX_MARK;
  22.     switch(data2)
  23.         {
  24.         case KEY1:
  25.             SysChenyuan.Uart_Comd = QX1;
  26.             printf("清洗1按键按下,data2 = %02d,function = %02d", data2, function);
  27.             break;
  28.         case KEY2:
  29.                         SysChenyuan.Uart_Comd = QX2;
  30.             printf("清洗2按键按下,data2 = %02d,function = %02d", data2, function);
  31.             break;
  32.         case KEY3:
  33.                         SysChenyuan.Uart_Comd = QX3;
  34.             printf("清洗3按键按下,data2 = %02d,function = %02d", data2, function);
  35.             break;
  36.                 case KEY_STOP:
  37.                         SysChenyuan.all_stopped = YES;//停止标志
  38.             printf("停止按键按下,data2 = %02d,function = %02d", data2, function);
  39.             break;
  40.                 case KEY_CHONGSHUI:
  41.                         SysChenyuan.QiTa = CHONGSHUI;//冲水标志
  42.             printf("冲水按键按下,data2 = %02d,function = %02d", data2, function);
  43.             break;
  44.                 case KEY_HONGGAN:
  45.                         SysChenyuan.QiTa = HONGGAN;//烘干标志
  46.             printf("烘干按键按下,data2 = %02d,function = %02d", data2, function);
  47.             break;
  48.         default:
  49.             // 处理未知按键情况
  50.             break;
  51.     }
  52.     }
  53. }
  54. extern uint8_t Serial_Rx3Flag,time_out3,time_out3_flag;
  55. // 解析数据包的函数  
  56. int ParseDataPacket(void) {  
  57.     static uint8_t expectingHeader = 1; // 标记是否正在等待包头  
  58.     static uint8_t packetIndex = 0, i = 0; // 当前数据包内的索引  
  59.     uint8_t function, data1, data2;   

  60.         if(Serial_Rx3Flag == 1)
  61.         {
  62.             time_out3 = 0;
  63.             time_out3_flag = 0;
  64.             // 遍历缓冲区中的所有数据(在实际情况中,你可能只处理新接收到的数据)  
  65.             for ( i = 0; i < rxIndex; ++i) {  
  66.                     if (expectingHeader) {  
  67.                             if (rxBuffer[i] == PACKET_HEADER) {  
  68.                                     expectingHeader = 0; // 找到了包头,开始解析数据包  
  69.                                     packetIndex = 0; // 重置数据包索引  
  70.                             }  
  71.                             // 如果不是包头,则忽略该字节或进行错误处理  
  72.                     } else {  
  73.                             switch (packetIndex) {  
  74.                                     case 0: // 功能码  
  75.                                             function = rxBuffer[i];  
  76.                                             break;  
  77.                                     case 1: // 数据1  
  78.                                             data1 = rxBuffer[i];  
  79.                                             break;  
  80.                                     case 2: // 数据2  
  81.                                             data2 = rxBuffer[i];  
  82.         
  83.                                             // 检查包尾  
  84.                                             if (rxBuffer[i+1] == PACKET_TAILER) {  
  85.                                                     // 完整的数据包已接收,可以处理数据  
  86.                                                 Anlsy_Data(function, data1, data2);
  87.                                                     // processPacket(function, data1, data2);  
  88.                                                    
  89.                                                     // 重置状态以接收下一个数据包  
  90.                                                     expectingHeader = 1;  
  91.                                                     packetIndex = 0;  
  92.                                                     rxIndex = 0; // 清除缓冲区(或者只清除已处理的部分)  
  93.                                             } else {  
  94.                                                     // 没有找到包尾,可能是数据损坏或丢失,进行错误处理  
  95.                                                     // TODO: 错误处理代码  
  96.                                                     expectingHeader = 1; // 重置状态以接收下一个数据包  
  97.                                                     packetIndex = 0;  
  98.                                             }  
  99.                                             break;  
  100.                                     default:  
  101.                                             // 无效的数据包索引,进行错误处理  
  102.                                             expectingHeader = 1; // 重置状态以接收下一个数据包  
  103.                                             packetIndex = 0;  
  104.                                             break;  
  105.                             }  
  106.                             if (packetIndex < 3) { // 还未到达包尾,递增索引  
  107.                                     packetIndex++;  
  108.                             }  
  109.                     }  
  110.             }  
  111.             rxIndex = 0;// 清除已处理的数据(如果rxIndex不是自动管理的)  
  112.             Serial_Rx3Flag = 0;
  113.     }  
  114.     return 0; // 返回值可以根据需要来定义,例如表示是否成功解析了一个数据包  
  115. }


  116. //下面是串口超时代码,我放在了10ms定时器里,看需要。
  117. // 定时器回调函数 10ms
  118. void vTimerCallback( TimerHandle_t xTimer )  
  119. {  
  120.     if(time_out3 < 10 && time_out3_flag == 1) {
  121.             time_out3++;
  122.     }

  123.     // 检查是否超时
  124.     if(time_out3 >= 10) {
  125.         time_out3 = 0;
  126.         time_out3_flag = 0;
  127.             // 执行超时处理,例如发送确认信息或者进行数据处理
  128.         Serial_Rx3Flag = 1;
  129.     }
  130.    
  131. }

  132. //串口中断

  133. void USART1_IRQHandler(void)
  134. {
  135.     if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET)   //RXNE 标志位为1 表示可以接收数据
  136.     {
  137.         uint8_t ucReceivedChar;
  138.          // 读取接收到的字符
  139.         ucReceivedChar = USART_ReceiveData(USART1);
  140.                 if (rxIndex < RX_BUFFER_SIZE) {
  141.                     rxBuffer[rxIndex++] = ucReceivedChar;
  142.                     time_out3 = 0;
  143.                     time_out3_flag = 1;
  144.                 }
  145. //                if(Serial_Rx3Flag == 1)
  146. //                {
  147. //                    rxIndex  = 0;
  148. //                }
  149.         USART_ClearITPendingBit(USART1,USART_IT_RXNE);  //清除RXNE标志位
  150.     }
  151. }
复制代码


评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏4 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表