USB作为一种串行通信总线,在嵌入式系统与PC机通信中有着广泛的应用。Ill=J=USB总线是主从式结构,通常采用主从式通信方式。这种方式的缺点在于从设备只能被动响应来自主设备的请求,不能主动发起请求,因而在交互性上受到了限制。但随着嵌入式系统技术的发展,对交互性操作要求越来越迫切,而采用USB双向通信可以很好地解决上述问题。本文介绍一种基于S1C33L11芯片利用嵌入式操作系统的同步机制通过对循环队列及自定义控制包的操作来实现双向通信的方法。
1嵌入式操作系统中USB双向通信系统整体层次结构
嵌入式操作系统中USB双向通信系统整体层次结构如图1所示。
3 USB双向通信的设计与实现
本文USB双向通信在基本传输方式上采用USB块传输。他由USB初始化、USB中断处理、控制传输和块传输几部分组成。在实现双向通信上,具体通信机制是:嵌入式应用程序通过读写循环队列和信号量状态与USB硬件模块中的OUT和IN FIFO相互通信,而USB下位机与上位机(PC)的读写通信则通过上位机对控制包的读写来实现,最后通过循环队列、信号量、控制包3者结合达到USB双向通信的目的。
3.1 USB双向通信固件程序的设计与实现
(1)循环队列
采用IN传输一个循环队列,OUT传输一个循环队列(以下简称队列),每队列动态分配32 kB。OUT队列做为OUT传输时的二级缓冲,即OUT输时fl~FIFO的数据必须先放人OUT队列才能由嵌入式操作系统读写;IN队列做为IN传输时的二级缓冲,即IN传输时的FIFO数据必须来自IN队列;嵌入式操作系统只对二级缓冲进行读写,操作系统对队列的管理是采用信号量通知机制来实现。
(2)控制包
为实现双向通信,规定一种控制包格式,读控制包是在USB协议之外自定义的。
控制包固定为5字节。从左到右第一字节为状态字,剩下4字节传送要收发的数据字节数。当控制包由上位机发出时,状态字规定有3种:0x4F:上位机请求OUT传输,0x49:上位机请求IN传输,0x52:上位机请求读取下位机状态;当上位机收到控制包时,状态字规定有5种:0x00:USB空闲态,0x01:下位机OUT循环队列满(I~POUT超时),0x02:下位机IN循环队列空(即IN超时),0x04:OUT传送成功,0x08:IN传送成功。
(3)嵌入式操作系统端应用程序读写USB过程
读函数:void ReadUSB(unsigned char *ReadBuffer,DWORD size)函数:
功能:嵌入式系统应用程序通过USB接口读取上位机(PC)的数据。
参数说明:unsigned char*ReadBuffer存放数据的指针,DWORD size为要读出的数据的尺寸(单位:B)。
实现过程:首先判断循环队列是否为空,不为空则判断自身信号量是否可用,若可用,则从队列中读取一字节,每读一字节后向USB任务中的BulkOutGet函数(直接读取OUT的FIFO函数)发出一个信号量,通知BulkOutGet函数队列此时可以向OUT循环队列中写入数据,接着重新判断,依次逐字节从OUT循环队列中读取数据,直到读完要求数据大小为止。当循环队列为空时,首先发一个信号量,通知BulkOutGet函数应向本队列中写入数据了,然后复位自身信号量,接着调用等待信号量的函数,直到信号量到时才接着读取。若超时,则向嵌入式操作系统发出超时通知,同时通过向控制包中写入超时状态(Ox01)来向上位机(PC)发出超时信号。
写函数:void WriteUSB(unsigned char*WriteBuffer,DWORD size)函数:
功能:嵌入式系统应用程序通过USB接口向上位机(PC)发送数据。
参数说明:unsigned char*WriteBuffer存放数据的指针,DWORD size为要写入的数据的尺寸(单位:B)。
实现过程:首先判断循环队列是否满,不为满则判断自身信号量是否可用,若可用,则向队列中写入一字节,每写入一字节后向USB任务中的BulkInDataSet(直接写IN的F.IFO函数)函数发出一个信号量通知此函数此时可以从IN循环队列中读取数据;然后接着重新判断依次逐字节向IN循环队列写人数据,直到写完要求数据大小的数据为止。当循环队列满时,先发一个信号量通知BulkInDataSet函数应从队列中取走数据,再复位自身信号量,接着调用等待信号量的函数,直到信号量到时才接着写入,若超时,则向嵌入式操作系统发出超时通知,同时通过向控制包中写入超时状态(0x02)来(PC)发出超时信号。
(4)USB块传输函数
USB块传输函数是直接和USB硬件打交道的函数,他们直接读取IN和OUT传输通道的FIFO。void BulkInDataSet(void):其功能是IN传输过程,即从IN循环队列中读取数据并向IN FIFO中写入数据,再对嵌入式操作系统信号量做相应处理。
void BulkOutDataGet(void)其功能是OUT传输过程,即从OUT FIFO中读出数据并向OUT循环队列中写人数据,再对嵌入式操作系统信号量做相应处理。
(5)嵌入式操作系统USB任务调用函数void Systemlnit(void):MCU初始化(微处理器各控制寄存器和状态初始化过程)
void USBInit(void):USB初始化(包括对循环队列分配内存等)
void USBThread(void):USB运行体(USB工作过程对USB中断进行处理主要包括USB块传输函数、USB中断状态分析处理等)。
void FreeUSB(void):关闭USB和释放由malloc函数分配的循环队列所占内存
3.2 上位机(PC)部分
USB函数层(USBD及HCD)由Windows98提供,负责管理USB设备驱动程序与USB控制器之间的通信、加载及卸载USB驱动程序等。具体方法是通过DriverW0rks软件生成上位机(PC)机端USB驱动程序模板,根据下位机的情况处理相应的读写部分,最后通过封装基本API函数ReadFile,WriteFile来实现用户态应用程序与PC机USB驱动程序的隔离使PC的应用层对USB的使用如同对串口的使用一样方便,给用户态应用程序提供有了3个接口函数:
unsigned char Read(void*pReadBuffer,DWORD
其中每次Read或Write函数的调用被分为若干次读/写发送。具体处理是:设待读写的数据字节数为X B,当X=5 B时,分割为X1=4 B和X2=1 B两次发送(由于自定义包是5 B,为了与自定义控制包区分开);当5 B<X<1 6 kB时以X字节一次发送;当X>1 6 kB时则分割以16 kB为单位的数据进行发送,不足16 kB的部分再发送一次。每次读/写发送分3个阶段:发控制包,读/写数据,读控制包状态。
4 结 语
基于S1C33L11芯片在嵌入式操作系统基础上实现的USB双向通信严格遵循USBl.1协议,充分利用了S1C33I.,11芯片的内置功能和嵌入式操作系统的作用,具有交互作用强、嵌入式操作系统中设备无关性好的特点。