引言
Windows CE是一个开放的、可升级、可裁减的32位实时嵌入式操作系统,具有可靠性好、实时性高、内核体积小的特点,广泛应用于工业控制、信息家电、移动通信、汽车电子、个人电子消费品等领域。最新版本Windows Em-bedded CE 6.0于2006年11月发布,其特点有:最大进程数量到32K,且每个进程有最大2 GB的虚拟内存空间;将关键的驱动程序、文件系统和图形界面管理器移到了内核中,大大减少了CPU在内核态和用户态间切换造成的性能损失等。 LPC3250是NXP半导体公司(由Philips公司成立)推出的带有矢量浮点协处理器的ARM926EJ-SCPU内核的微控制器。它具有丰富的外围接口,包括7个UART,其中4个是标准UART,另外3个是高速UART,都带有64字节的接收和发送FIFO,最高可支持的速率达921 600 b/s。为了实现低功耗,LPC3250采用NXP半导体先进的开发技术来优化内在功率,并使用增强型的软件控制结构使基于功率管理的应用得到优化。在同时要求高性能和低功耗的嵌入式应用中,运行Win-dows CE的LPC3250平台将会有很好的市场前景,对于最常用到的串口的驱动开发显得尤为重要。
1 WindOWS CE的串口驱动模型
基于Windows CE有两种驱动程序模型:本机设备驱动程序和流接口驱动程序。串口驱动就属于分层的流接口驱动程序。分层驱动程序将设备的驱动程序分为两层:平台相关驱动 PDD(Platform Dependence Driver)层和模型设备驱动MDD(Model Device Driver)层。PDD层由特定于给定硬件设备或平台的代码组成,很多时候用户需要根据具体平台修改;MDD层包含平台无关的代码,它通过实现一些操作系统预先定义的接口来实现某一类设备的通用功能,通常由微软提供。操作系统与MDD层之间通过DDI(设备驱动接口)进行交互。MDD层也实现了中断处理线程IST,并定义一些与PDD层的接口函数,这些接口函数称为DDSI(设备驱动服务接口)。
用%_WINCEROOT%来表示Windows CE的安装根目录,符合‘550工业规范的串口驱动源码主要位于\%_WINCEROOT%\PUBLIC\COMMON\OAK\DRIVERS\SERIAL下,主要看表1所列的一些重要文件。
如图1所示,串口应用程序通过设备管理器调用mdd.c中MDD层的标准流设备驱动接口COM_XXX,在COM_XXX中通过结构体 HW_INDEP_INFO中HWOBJ结构体调用串口硬件操作函数HWxxx;然后在cserpdd.cpp中GetSeri-alObject函数通过HW_VTBL类型数组IoVTb1将HWxxx映射为Serxxx系列函数,Serxxx系列函数则调用CSerialPDD类中的成员函数(其中的纯虚函数由CserialP-DD的继承类CP-dd16550实现,真正与物理底层操作的是CPdd16550的数据成员CReg16550中的 Write_XXX、Read_XXX函数);最终通过调用WRITE_PORT_UCHAR和READ_PORT_UCHAR系统函数来实现。
2 WinCE6.0下的LPC3250串口驱动程序开发
Windows CE的串口驱动程序开发中最重要的是两点:配置串口相关的寄存器和处理中断。配置寄存器,包括实现与物理底层操作的函数,将寄存器地址映射到内核进程的虚拟地址,在串口操作的不同阶段配置好各种寄存器;处理中断,包括将物理中断映射为系统中断,将中断与事件绑定,中断发生时进行相应的中断处理。
LPC3250串口与‘550工业规范的串口有差异,为了保证程序的通用性和尽量减少代码量,在实现LPC3250串口驱动程序时,需要继承 CPdd16550和CReg16550类,根据实际的硬件特性实现它们的纯虚函数并扩展其虚函数的功能,配置硬件相关的寄存器和修改相关代码。首先实现 CReg16550的继承类CRegLPC32xx,主要实现与物理底层操作的函数Write_xxx和Read_xxx,对串口寄存器进行读写操作。这里要注意的是LPC3250串口寄存器地址间隔是32位,而不是标准的8位;CPdd16550的继承类Clpc32xxPdd16550UART本质还是个抽象类,同时为标准串口和高速串口服务,要重新实现Init、GetDivisorO-{Rate、GetWaterMark、 MapHardware、CreateHardwareAc-cess、CreateSerialObject、DeleteSerialObject等函数,其他的函数可以直接调用CPdd16550的成员函数,只需要修改相关串口寄存器的宏定义。
在Clpc32xxPdd16550UART的Init函数中,GetIsrInfo以串口的Active注册表键为依据查出物理中断号,并保存在 DDKISRINFO结构体的dwlrq成员中。KernelloCon-trol函数将物理中断号转换为逻辑中断号,符合条件就将逻辑中断号回写到注册表中。相关代码如下:
接着调用父类CPdd16550的Init函数,创建中断服务线程(IST)事件,并通过InterruptInitialize函数将事件与逻辑中断号关联起来,最后调用CreateHardwareAccess和MapHardware函数将串口基地址及相关寄存器片内地址映射到内核进程的虚拟地址。
在MapHardware中,用GetWindowInfo根据串口的Active注册表键获得串口的全部I/O端口和内存地址信息,然后用 MmMapIoSpace函数将串口物理地址和相关控制寄存器地址转换成内核进程的虚拟地址,以便后面对寄存器进行操作,部分代码如下:
CreateHardwareAccess函数根据MapHardware得到的m_pBaseAddress,构造一个CRegLPC32xx类实例,然后调用CRegLPC32xx类的Init函数确保串口控制器硬件进入稳定的工作状态。
根据LPC3250的数据手册,设置标准UART的波特率需要设置小数波特率预分频器和UART波特率发生器。当不用小数波特率预分频器(即X=Y=1) 时,将标准UART的{Baudrate,DLM:DLL}的值定义一个数组BaudPairs[]。GetDivisorOfRate根据这个数组得到分频系数,然后调用父类的成员函数SetBaudRate便可设置波特率。高速UART的波特率类似,只是波特率计算公式和分频系数与标准UART不同。
用GetWaterMark得到接收器FIFO的触发深度,分别为16、32、48和60位,然后在CPdd16550的InitReceive中设置FIFO控制寄存器,默认的FIFO触发深度是32位。
Clpc32xxPdd16550UART是个抽象类,实现通用功能,具体的要分别由继承的标准串口Clpc32xxPdd16550Stan- dardUART类和高速串口Clpc32xxPdd16550HighUART类实现。在各自初始化时,主要是配置各种寄存器,实现具体硬件差异化,包括:配置UART时钟控制寄存器、时钟模式寄存器和时钟选择寄存器,分别使能UART时钟、设置自动时钟模式、选择相应的时钟源作为分频器的输入时钟;禁止UART3 Modem和UART6 IrDA功能;禁止UART的回送功能。
特别要强调的是关于中断的处理,串口驱动中断可以用动态映射,也可以用静态映射。在OEMInter-ruptHandler、 Clpc32xxPdd16550UART::Init、CPdd16550::Init、CPdd16550::ThreadRun等处加入调试打印信息,可以较快地找到问题所在,确定硬件中断是否映射为系统中断、系统中断与中断事件是否绑定、中断产生时是否进入相应的处理程序。中断处理好了,串口驱动就基本完成了。
上述工作结束后,就要添加串口的注册表。以串口3为例,主要是设置动态链接库DLL、设备基地址、中断号、前缀名、被加载的顺序等。根据注册表的 DeviceArrayIn-dex、CreateSerialObject就可以构造标准串口或高速串口类实例了, DeleteSerialObject在退出驱动时删除实例。具体代码如下:
在广州致远电子有限公司的SmartARM3250开发板上,通过WinCE的串口应用程序与上位PC机进行发送接收实验,本驱动已经实现标准串口最高460 800 b/s、高速串口最高921 600 b/s的稳定传输。
结语
本文介绍了WinCE6.0下的串口驱动模型,结合LPC3250的硬件情况,详细说明了串口驱动开发过程,包括配置串口相关的寄存器和处理中断中重要函数的实现,以及注册表和Source文件编写等。本驱动程序在广州致远电子有限公司的SmartARM3250开发板上实验成功。在串口驱动开发中所用的思路,对其他类似的驱动设计有较高的参考价值。