自定义串口通信协议,如何实现?
作者 | strongerHuang
微信公众号 | 嵌入式专栏
1什么通信协议?
通信协议是指双方实体完成通信或服务所必须遵循的规则和约定。通过通信信道和设备互连起来的多个不同地理位置的数据通信系统,要使其能协同工作实现信息交换和资源共享,它们之间必须具有共同的语言。交流什么、怎样交流及何时交流,都必须遵循某种互相都能接受的规则。这个规则就是通信协议。
| 帧头 | 温度值 | 帧尾 |
|---|---|---|
| 5A | 一字节数值 | 3B |
2过于简单的通信协议引发的问题
3通信协议常见内容







4通信协议代码实现
/* DGUS寄存器地址 *///往DGDS屏指定寄存器写一字节数据void DGUS_REG_WriteWord(uint8_t RegAddr, uint16_t Data){DGUS_SendByte(DGUS_FRAME_HEAD1);DGUS_SendByte(DGUS_FRAME_HEAD2);DGUS_SendByte(0x04);DGUS_SendByte(DGUS_CMD_W_REG); //指令DGUS_SendByte(RegAddr); //地址DGUS_SendByte((uint8_t)(Data>>8)); //数据DGUS_SendByte((uint8_t)(Data&0xFF));}//往DGDS屏指定地址写一字节数据void DGUS_DATA_WriteWord(uint16_t DataAddr, uint16_t Data){DGUS_SendByte(DGUS_FRAME_HEAD1);DGUS_SendByte(DGUS_FRAME_HEAD2);DGUS_SendByte(0x05);DGUS_SendByte(DGUS_CMD_W_DATA); //指令DGUS_SendByte((uint8_t)(DataAddr>>8)); //地址DGUS_SendByte((uint8_t)(DataAddr&0xFF));DGUS_SendByte((uint8_t)(Data>>8)); //数据DGUS_SendByte((uint8_t)(Data&0xFF));}
static uint8_t sDGUS_SendBuf[DGUS_PACKAGE_LEN];//往DGDS屏指定寄存器写一字节数据void DGUS_REG_WriteWord(uint8_t RegAddr, uint16_t Data){sDGUS_SendBuf[0] = DGUS_FRAME_HEAD1; //帧头sDGUS_SendBuf[1] = DGUS_FRAME_HEAD2;sDGUS_SendBuf[2] = 0x06; //长度sDGUS_SendBuf[3] = DGUS_CMD_W_CTRL; //指令sDGUS_SendBuf[4] = RegAddr; //地址sDGUS_SendBuf[5] = (uint8_t)(Data>>8); //数据sDGUS_SendBuf[6] = (uint8_t)(Data&0xFF);DGUS_CRC16(&sDGUS_SendBuf[3], sDGUS_SendBuf[2] - 2, &sDGUS_CRC_H, &sDGUS_CRC_L);sDGUS_SendBuf[7] = sDGUS_CRC_H; //校验sDGUS_SendBuf[8] = sDGUS_CRC_L;DGUSSend_Packet_ToQueue(sDGUS_SendBuf, sDGUS_SendBuf[2] + 3);}//往DGDS屏指定地址写一字节数据void DGUS_DATA_WriteWord(uint16_t DataAddr, uint16_t Data){sDGUS_SendBuf[0] = DGUS_FRAME_HEAD1; //帧头sDGUS_SendBuf[1] = DGUS_FRAME_HEAD2;sDGUS_SendBuf[2] = 0x07; //长度sDGUS_SendBuf[3] = DGUS_CMD_W_DATA; //指令sDGUS_SendBuf[4] = (uint8_t)(DataAddr>>8); //地址sDGUS_SendBuf[5] = (uint8_t)(DataAddr&0xFF);sDGUS_SendBuf[6] = (uint8_t)(Data>>8); //数据sDGUS_SendBuf[7] = (uint8_t)(Data&0xFF);DGUS_CRC16(&sDGUS_SendBuf[3], sDGUS_SendBuf[2] - 2, &sDGUS_CRC_H, &sDGUS_CRC_L);sDGUS_SendBuf[8] = sDGUS_CRC_H; //校验sDGUS_SendBuf[9] = sDGUS_CRC_L;DGUSSend_Packet_ToQueue(sDGUS_SendBuf, sDGUS_SendBuf[2] + 3);}
typedef struct{uint8_t Head1; //帧头1uint8_t Head2; //帧头2uint8_t Len; //长度uint8_t Cmd; //命令uint8_t Data[DGUS_DATA_LEN]; //数据uint16_t CRC16; //CRC校验}DGUS_PACKAGE_TypeDef;
void DGUS_ISRHandler(uint8_t Data){static uint8_t sDgus_RxNum = 0; //数量static uint8_t sDgus_RxBuf[DGUS_PACKAGE_LEN];static portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE;sDgus_RxBuf[gDGUS_RxCnt] = Data;gDGUS_RxCnt++;/* 判断帧头 */if(sDgus_RxBuf[0] != DGUS_FRAME_HEAD1) //接收到帧头1{gDGUS_RxCnt = 0;return;}if((2 == gDGUS_RxCnt) && (sDgus_RxBuf[1] != DGUS_FRAME_HEAD2)){gDGUS_RxCnt = 0;return;}/* 确定一帧数据长度 */if(gDGUS_RxCnt == 3){sDgus_RxNum = sDgus_RxBuf[2] + 3;}/* 接收完一帧数据 */if((6 <= gDGUS_RxCnt) && (sDgus_RxNum <= gDGUS_RxCnt)){gDGUS_RxCnt = 0;if(xDGUSRcvQueue != NULL) //解析成功, 加入队列{xQueueSendFromISR(xDGUSRcvQueue, &sDgus_RxBuf[0], &xHigherPriorityTaskWoken);portEND_SWITCHING_ISR(xHigherPriorityTaskWoken);}}}
b.增加超时检测
接收数据有可能存在接收了一半,中断因为某种原因中断了,这时候,超时检测也很有必要。
比如:用多余的MCU定时器做一个超时计数的处理,接收到一个数据,开始计时,超过1ms没有接收到下一个数据,就丢掉这一包(前面接收的)数据。
static void DGUS_TimingAndUpdate(uint16_t Nms){sDGUSTiming_Nms_Num = Nms;TIM_SetCounter(DGUS_TIM, 0); //设置计数值为0TIM_Cmd(DGUS_TIM, ENABLE); //启动定时器}void DGUS_COM_IRQHandler(void){if((DGUS_COM->SR & USART_FLAG_RXNE) == USART_FLAG_RXNE){DGUS_TimingAndUpdate(5); //更新定时(防止超时)DGUS_ISRHandler((uint8_t)USART_ReceiveData(DGUS_COM));}}
c.更多
接收和发送一样,实现方法有很多种,比如接收同样也可以用结构体方式。但有一点,都需要结合你实际需求来编码。
5最后
‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧ END ‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧ 关注我的微信公众号,回复“加群”按规则加入技术交流群。
点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。
评论
