《电子技术应用》
您所在的位置:首页 > 嵌入式技术 > 业界动态 > Windows 98下硬件中断驱动程序的开发

Windows 98下硬件中断驱动程序的开发

2009-02-20
作者:李 博 鲍 超

  摘 要: 介绍了Windows98的内核管理机制和应用程序权限级别,简述了在Windows98下进行虚拟驱动程序开发的几种工具和编程方法,并给出了借助VToolsD用C++语言编写的处理硬件中断的程序实例。
  关键词: 虚拟设备驱动程序 VToolsD 中断服务例程


  美国微软公司出品的Windows98以其友好的图形用户界面,在我国赢得了广泛的市场。在给广大办公环境工作人员带来方便的同时,也给不少工程技术人员带来了一些麻烦。一些原本在DOS下很容易编出的控制硬件的程序,现在在Windows98下就不那么容易实现了。作为一个完善的操作系统也必须能控制硬件,象DOS那样直接与硬件打交道是Windows 98不提倡的。它需要开发专门的硬件设备驱动程序,即通过一系列的虚拟设备驱动程序来管理硬件,如:进行中断响应、I/O端口读写或直接存储器存取(DMA)。Windows 98内核管理机制非常复杂,因而编写虚拟驱动程序也变得十分困难,要想编写虚拟驱动程序,就必须对Windows 98的内核有所了解。
1 Windows98的内核管理机制
  在Windows 95三年后推出的Windows 98虽然扩充了许多新的设备驱动特性,如对AGP、USB、DVD的支持,但在内核上却和Windows 95基本一样,它们都是基于DOS内核的操作系统。Windows98系统核心(Kernel)由虚拟机管理器(VMM)和VxD(Virtual Device Driver)的集合组成。Kernel提供了900多个服务函数来管理内存、控制物理设备、处理中断、创建网络协议栈、管理文件系统等,这些服务函数都可以被自己写的VxD调用。虚拟机(VM)是一个可运行的任务,包含应用程序、支撑软件、内存和CPU寄存器。在Windows98下有系统虚拟机和DOS虚拟机两种。虚拟机管理器(VMM)是在系统级核心运行的32位保护模式操作系统,它运行于Ring0,而且不可重入。VMM主要功能是创建、运行、监控和终止虚拟机。VxD即虚拟设备驱动程序,是用来扩展Windwos操作系统功能的一类程序。由于VxD运行在系统的Ring0级,拥有与操作系统同等的级别,所以我们可利用它来支持硬件设备的管理。虚拟可编程中断控制器(VPICD)是负责管理所有硬件中断事件的程序,它本身也是一种VxD,能提供缺省的中断处理函数或者允许其它VxD重载中断处理函数。
2 Windows 98下应用程序权限级别
  Intel的80x86CUP系列芯片可在三种模式下工作:实模式、保护模式和V86模式。实模式是MS-DOS的运行环境。Windows 98只利用了两种模式:保护模式和V86模式。保护模式给我们带来很多优越性,如应用程序不再受1M内存的限制,理论上,在保护模式下,CPU可以进行4096M内存的寻址。但在保护模式下,所有的应用程序都有权限级别(Privilege Level)。权限级别按优先次序分为四等:0、1、2、3。0级是最高级别,操作系统就运行在0级,运行在Ring0级的应用程序可以执行所有的指令并可直接对硬件、中断和文件系统进行物理访问。如果应用程序拥有的权限级别是第3级,那么它能执行的指令是有限的,对硬件的很多直接操作是不能实现的。在Windows中,一般的应用程序是运行在Ring3级的(如用Visual C++、Borland C++、Visual Basic、Delphi、C++ Builder等SDK工具开发出的应用程序)。它们享有的权限是最低的,受到了保护模式的“保护”,它们没有权限去绕过操作系统直接对硬件操作。
  有了权限级别,操作系统就有机会在中断和I/O操作上产生“虚拟”效果。由于操作系统的权限为0级,它就可以捕获权限不为0级的应用程序的中断和I/O请求,然后建立缓冲队列,再一一进行串行处理。为了使自己的应用程序也能直接处理硬件,就需要编写专门的VxD。由于VxD是作为操作系统的组件运行于第0级,因而可以利用它来捕获特定的硬件操作,完成我们需要的任务。
3 Windows 98下虚拟设备驱动程序的开发工具和基本编程方法
  微软为驱动程序的开发提供了设备驱动程序工具箱(DDK),基于汇编语言的编程方式和许多VMM服务都使用寄存器的调用方式,确实非常难学,没有深厚的汇编语言和硬件基础很难在短时间里开发出自己的VxD。
  美国Vireo Software公司推出的VToolsD为我们开发VxD提供了方便快捷的方法。程序员可利用C或C++语言编写自己的VxD,而不必操心许多繁琐的细节。它的基本编程方法是:用VToolsD自带的QuickVxD程序快速生成程序框架,在VC++或Borland C++中打开此框架的工程文件,并写进特定的处理代码,编译后就可得到所需的VxD文件。
4 一个中断程序实例
  用C或C++都可在VToolsD中进行开发,相比之下,由于VToolsD中封装许多C++类库,因而用C++语言进行VxD的开发更加容易和方便。
  在我们开发的小型实时光谱能量辐射仪中利用北京众人公司的PS-2129AD采集卡作为数据采集器件。卡上载有三路8253计数器,利用其中的两路产生光电二极管阵列(SSPD)的驱动信号,另外一路让它工作在方式0—计数结束产生中断,每当计数值变为0,便产生中断,在中断程序里进行数据采集。本例中中断号为9。
  定义好设备属性各参数后,在QuickVxD生成的头文件中定义如下:
  #define DEVICE_CLASS AdcardDevice
  #define ADCARD_DeviceID UNDEFINED_DEVICE_ID
  #define ADCARD_Init_Order UNDEFINED_INIT_ORDER
  #define ADCARD_Major 1
  #define ADCARD_Minor 0
  #define MY_IRQ 9 /*中断号为9*/
  其中ADCARD为在QuickVxD的对话框中键入的设备名。VToolsD提供类VHardwareInt来实现对某个IRQ端口的虚拟化,并处理该IRQ端口上的硬件中断。本实例中就用到这个类。
  类OnHaredwareInt的主要成员函数定义如下:
  VHardwareInt::VHaredwaredInt(int irq , DWORD flags,DWORD timeout,PVOID refdata)
  irq 要虚拟化的IRQ,其值从0~15;
  flag 一般设置为0;
  timeout 从虚拟中断申请(assert)到进入VM中的虚
  拟中断处理函数所允许的最大时延,设置0表示忽略 timeout;
  refdata refdata将存放在m_refdata变量中,在通知事件处理函数里,可以作为参考数据。
  VHardwareInt::OnHardwareInt(VMHANDLE hVM)
  当IRQ硬件中断发生时,VPICD将调用类VHardwareInt里的成员函数OnHardwardInt,在调用OnHardwareInt函数之前,VPICD首先发Cli指令,同时设置本IRQ的屏蔽位。所以可以在OnHardwareInt函数里直接处理中断服务。
  VHardwareInt::hook()
  hook成员函数把虚拟IRQ与OnHardwareInt函数相勾连。通过重载OnHardwareInt函数用来处理本IRQ的各种中断事件。返回TRUE表示IRQ虚拟化成功,返回FALSE则表示IRQ虚拟化失败。
  VHardwareInt::physicalUnmask()
  命令VPICD物理地不屏蔽本IRQ。
  VHardwareInt::sendPhysicalEOI()
  此成员函数通知VPICD中断处理结束。VPICD将物理地不屏蔽本IRQ。
  在头文件里派生类VHardwareInt,在派生类中定义构造函数,并重载OnHardwareInt函数。
  class MyHwInt:public VHardwareInt
  {
  public:MyHwInt():VHardwareInt(MY_IRQ,0,0,0){} /*定义构造函数*/
  virtual VOID OnHardwareInt(VMHANDLE); /*重载OnHardwareInt函数*/
  };
  并在头文件里定义一个派生类的实例,放在QuickVxD自动生成的AdcardDevice类里:
  class AdcardDevice : public VDevice
  {
  public:
  virtual BOOL OnSysDynamicDeviceInit();/*用来动态加载VxD */
  virtual BOOL OnSysDynamicDeviceExit();/*用来动态卸载VxD */
  MyHwInt pMyIRQ; /*定义MyHwInt的一个实例*/
  };
  为了让VxD能与Ring3级的应用程序通讯,本实例中采用共享内存的办法。在源文件里定义一个内存地址,Ring3级应用程序通过读取此内存地址来得到AD卡的采集结果。
  源文件以下所示:
  #define DEVICE_MAIN
  #include ″adcard.h″
  Declare_Virtual_Device(ADCARD)
  #undef DEVICE_MAIN
  PWORD pVal=(PWORD) 0x9F000;//定义一个内存地址
  BOOL AdcardDevice::OnSysDynamicDeviceInit()
  {
  pMyIRQ = new MyHwInt();/*创建类MyHwInt的一个实例*/
  if (pMyIRQ->hook()) /*判断本IRQ是否虚拟化成功*/
  {
  pMyIRQ->physicalUnmask(); //不屏蔽本IRQ
  return TRUE;
  }
  else return FALSE;
  }
  BOOL AdcardDevice::OnSysDynamicDeviceExit()
  {
  delete pMyIRQ; //删除类MyHwInt的实例
  return TRUE;
  }
  VOID MyHwInt::OnHardwareInt(VMHANDLE hVM)
  {
  ...... /*这里写上处理中断的代码*/
  *pVal=_inp(0x0102) /*读进AD转换结果,这里假定AD卡存放转换结果的端口为0x0102*/
  sendPhysicalEOI(); /*发中断结束信号*/
  }
  用VC++打开此工程文件,编译后就可得到adcard.vxd文件。
  在Ring3级的应有程序中为了调用adcard.vxd,可在其源文件中添加以下语句:
  HANDLE HVXD;
  HVXD=CreateFile(″\\\\.\\adcard.vxd″,0,0,0,CREATE_NEW,
  FILE_FLAG_DELETE_ON_CLOSE,0);
  if(HVXD==INVALID_HANDLE_VALUE)
  ......
  CreateFile()的详细用法可查阅VC++的帮助。这样应用程序中就加载了adcard.vxd文件。为了能使此应用程序能得到AD卡的转换结果,同样也要在源文件里定义一个内存地址:
  PWORD pVal=(PWORD) 0x9F000;
  在需要得到转换结果的地方加上以下语句即可:
  int data;
  data=*pVal; //假定把结果存在data变量里
  这样一个中断实例就完成了。
  以上实例我们已在VToolsD3.01和VC++6中调试通过,并已成功地在我们开发的小型实时光谱能量辐射仪中得到应用。
  VxD作为现在流行的编程技术已逐渐受到广泛的关注,在工程技术中必将有着广阔的应用前景。学习、使用此技术将在科学研究中给我们带来便利。
参考文献
1 杨强,李堂秋.Win9X虚拟设备驱动程序编程指南. 北京:清华大学出版社,1999
2 杨振钧,谢瑞和.Windows95下虚拟驱动程序的开发.电子技术,1999(4)

本站内容除特别声明的原创文章之外,转载内容只为传递更多信息,并不代表本网站赞同其观点。转载的所有的文章、图片、音/视频文件等资料的版权归版权所有权人所有。本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如涉及作品内容、版权和其它问题,请及时通过电子邮件或电话通知我们,以便迅速采取适当措施,避免给双方造成不必要的经济损失。联系电话:010-82306118;邮箱:aet@chinaaet.com。