虽然说串口没有标准协议,但是我们却可以把它们的相似部分提取出来,做成模块化的程序,方便代码的移植和理解。下面我们简单谈到串口数据的处理方法。。。。。
定义数据结构如下:
- typedef struct
- {
- u16 WtCnt; // 写指针
- u16 RdCnt;// 读指针
- u16 BufLen;缓冲尺寸
- u8 *RwBuf;// 读写缓冲
- } DF_RCV;
1. 初始化函数
本函数用于对串口结构体中的各种数据进行初始化。
- /**************************************************************************************
- * FunctionName : DFInit()
- * Description : 初始化
- * EntryParameter : None
- * ReturnValue : None
- **************************************************************************************/
- void DFInit(DF_RCV *pRcv)
- {
- u16 i;
- pRcv->WtCnt = 0x0000;
- pRcv->RdCnt = 0x0000;
- for (i=0; i
BufLen; i++) - {
- pRcv->RwBuf[i] = 0x00;
- }
- }
本函数用于把串口中断接收的数据放入数据缓冲区中,并且接收计数器加1.
- /**************************************************************************************
- * FunctionName : DFWriteByte()
- * Description : 数据接收(接收中断调用)
- * EntryParameter : None
- * ReturnValue : None
- **************************************************************************************/
- void DFWriteByte(u8 dat, DF_RCV *pRcv)
- {
- pRcv->RwBuf[pRcv->WtCnt] = dat; // 数据存入
- if (++(pRcv->WtCnt) >= pRcv->BufLen) // 缓冲判断
- {
- pRcv->WtCnt = 0;
- }
- }
本函数用于从接收缓冲区中读取未处理的一字节数据,读计数器加1.
- /**************************************************************************************
- * FunctionName : DFReadByte()
- * Description : 从接受缓冲中读取一字节数据
- * EntryParameter : None
- * ReturnValue : 返回读取数据
- **************************************************************************************/
- u8 DFReadByte(DF_RCV *pRcv)
- {
- u8 val = 0x00;
- val = pRcv->RwBuf[pRcv->RdCnt]; // 读取一字节
- if (++(pRcv->RdCnt) >= pRcv->BufLen)
- {
- pRcv->RdCnt = 0; // 清零
- }
- return val; // 返回数据
- }
本函数用于读取串口缓冲区中还未处理的数据的大小。
- /**************************************************************************************
- * FunctionName : DFGetLen()
- * Description : 获取缓冲区中未读数据长度
- * EntryParameter : None
- * ReturnValue : 返回数据长度
- **************************************************************************************/
- u16 DFGetLen(DF_RCV *pRcv)
- {
- return ((pRcv->WtCnt >= pRcv->RdCnt) ? ((pRcv->WtCnt - pRcv->RdCnt)) :
- ((pRcv->WtCnt + pRcv->BufLen) - pRcv->RdCnt));
- }
为了保证数据的相对独立和模块化,下面代码将写入应用代码中,和上面的程序不能放在相同的文件中。
1. 数据定义
首先需要定义一个缓冲区,这个缓冲区的大小根据实际应用定义,其大小一般为数据帧的最大值的2倍。之后需要定义一个DF_RCV数据,在这个数据中需要初始化这个结构图的参数。特别需要注意,缓冲的大小,和缓冲区指针赋值。
- u8 AU_Buf[AU_BUF_ZISE] = {0};
- DF_RCV AU_Rvc = {0, 0, AU_BUF_ZISE, AU_Buf};
本函数把串口数据放入缓冲区中,此函数必须在串口中断中调用。
- /**************************************************************************************
- * FunctionName : AURcvDat()
- * Description : 串口数据接收(串口中断服务调用)
- * EntryParameter : None
- * ReturnValue : None
- **************************************************************************************/
- void AURcvDat(u8 dat)
- {
- DFWriteByte(dat, &AU_Rvc);
- }
本函数判断缓冲区中是否有数据,如果有,逐个读取并处理。
- /**************************************************************************************
- * FunctionName : AUTaskCtrl()
- * Description : 通信数据处理
- * EntryParameter : None
- * ReturnValue : None
- **************************************************************************************/
- void AUTaskCtrl(void)
- {
- u8 tmpDat;
- u16 i, len = 0;
- static u8 sendMark = 0;
- len = DFGetLen(&AU_Rvc); // 获取未读数据长度
- for (i=0; i < len; i++)
- {
- tmpDat = DFReadByte(&AU_Rvc); // 读一字节数据
- AU_PrcRcvDat(tmpDat);
- }
- }
上一篇:通过日期计算出星期
下一篇:浅谈如果通过程序读取AT24系列芯片型号
推荐阅读最新更新时间:2024-03-16 15:00