欢迎来到天天文库
浏览记录
ID:11124384
大小:35.11 KB
页数:6页
时间:2018-07-10
《缓冲队列的串口通信》由会员上传分享,免费在线阅读,更多相关内容在行业资料-天天文库。
1、缓冲队列的串口通信缓冲队列的串口通信2010-06-0516:451常用的串口处理方法串口部分的底层软件可以认为是串口的驱动程序,对上层软件而言,它应该提供一种比较自然而简洁的使用方式。以串口的发送为例,使用者可以直接调用一个函数输出一个字符串或者就像在通用平台上使用标准C中的printf函数一样。对上的接口已经确定了,下面我们讨论实现的问题。串口驱动的实现方法通常有两种:基于查询的办法。发送过程中不断检测串口硬件的发送缓冲区是不是为空,如果是,发送一个字节。如果还有数据没有发完,继续上述过程。下面以三星的S3C44B0XMCU为例,
2、给出了基于查询方法的串口发送的示意代码。voidUart_SendStr(char*pt)char*p;p=pt;while(*p!='[message]')while(!(rUTRSTAT0&0x2));//等待,直到串口缓冲区为空WrUTXH0(*p++);//发送一个字节基于中断的方法。在上面的基于查询的方法中,有一个很明显的弊病,那就是在发送一个字符串的过程中,CPU不能去做其它的事情,必须等待全部字符发送完成后返回。以上述MCU为例,其最高主频为66MHz,由于采用的是ARMv4体系结构,可以达到0.9指令每周期,而其串口最
3、高波特率为115200bps,这样就有大量的指令周期被浪费,而且在发送较长的字符串时会严重影响系统的实时性。所以在实际的系统中一般更多的采用中断的方法,发送一个字节之后,转去做其它的处理,发送完后自动进入发送中断,再发送下一个字节。这种方式比查询法提高了CPU的利用率,在串口部分进行发送和移位等操作时,CPU不用干预,但是同时也使串口的软件部分变得比较复杂,需要增加相应的中断服务程序(ISR)以及相关的软件缓冲区的管理。由于中断是由硬件触发的,为了使中断进入后能找到要发送的数据,最直接的办法就是设置一个全局的数组和一个指向待发送数据的
4、指针,这样每次中断进入后就发送指针指向的字节,直到发完。2基于双缓冲队列的方法在采用上述的中断方式之后,进一步考虑整个的处理流程,以及中断服务程序和上层程序交互的方便性,对缓冲区进行了仔细设计。由于串口发送和接收的数据是相对独立的,故将其分开,设置两个缓冲区,一个是发送缓冲区TxBuf,一个是接收缓冲区RxBuf,并为每个缓冲区分别设置两个指针,一个记录中断服务程序将处理的字节,另一个记录使用串口服务的上层程序将处理的字节。以串口发送为例,两个指针分别为inTxBuf,outTxBuf。outTxBuf指向发送中断将要发送的数据,in
5、TxBuf指向上层程序将数据放入缓冲区的起始位置。这种方式我们称之为采用双缓冲队列的方法。这种方法,保证了数据的顺序。在缓冲区够大的情况下,上层程序可以一次将要发送的数据全部放入发送缓冲区中,而不是一次发送一个字节,而且如果多个上层程序调用发送函数也不会造成混乱,因为每次调用时放入了要发送的全部数据。其原理类似于打印机对打印任务队列的管理,多个用户共享一个打印机,并发出各自的打印任务,但是不会出现不同任务的输出交叉的情况。串口的发送和接收这时成为公共的后台任务,只要发送缓冲区中有待发送的数据,就采用中断间歇性的进行发送,产生接收中断时
6、也类似的进行接收,并通知上层程序。下面开始分析具体的实现。以下是关于缓冲区和相关指针的声明:#defineTxBufLen1000#defineRxBufLen200charTxBuf[TxBufLen],RxBuf[RxBufLen];char*inTxBuf,*outTxBuf,*inRxBuf,*outRxBuf;intUartTxCount,UartRxCount;以下是串口发送的相关示例代码://发送中断的ISR采用汇编实现,进行现场保护之后调用UartTx进行后续的处理,//Uart返回后再恢复现场。voidUartTx(
7、void)if(outTxBuf==inTxBuf)//TxBuf空return;WrUartBUF(*outTxBuf);//将待发送数据写入串口的寄存器outTxBuf++;//指向下一个要发送的数据if(outTxBuf==(TxBuf+TxBufLen))//缓冲区回绕outTxBuf=TxBuf;UartTxCount++;//发送计数//上层程序调用串口发送数据的接口,一次发送一个'[message]'结尾的字符串。voidUart_PrintStr(char*pt)char*t,*p;t=inTxBuf;p=pt;whi
8、le(*p!='[message]')//逐个放入缓冲区t++;if(t==(TxBuf+TxBufLen))//回绕t=TxBuf;if(t==outTxBuf)//TxBuf满return;*inTxBuf=*p++
此文档下载收益归作者所有