本文分三部分来介绍如何构造一个简单的USB过滤驱动程序,包括“基本原理”、“程序的实现”、“使用INF安装”。此文的目的在于希望读者了解基本原理后,可以使用除DDK以外最流行也最方便的驱动开发工具DriverStudio来实现一个自己的过滤驱动,并正确地安装。



一、基本原理    

我们知道,WDM(和KDM)是分层的,在构造设备栈时,IO管理器可以使一个设备对象附加到另外一个初始驱动程序创建的设备对象上。与初始设备对象相关的驱动程序决定的IRP,也将被发送到附加的设备对象相关的驱动程序上。这个被附加的驱动程序便是过滤驱动程序。如右图,过滤驱动可以在设备栈的任何层次中插入。IO管理器发出的IRP将会沿着右图的顺序从上往下传递并返回。因此,我们可以使用过滤驱动程序来检查、修改、完成它接收到的IRP,或者构造自己的IRP。    

     上面这种文字是很枯燥的,好在“前人”已经写过一些范例以供我们更好地理解这些概念。读过Waltz Oney的《Programming Windows Driver Mode》一书的读者大概都知道Waltz Oney提供的范例中有一个关于USB过滤器(第九章)的例子,而在此基础上,《USB Design By Example》(http://www.usb-by-example.com)的作者John Hyde实现了一个USB键盘过滤驱动程序,即给此程序增加了一个“拦截(Intercept)”功能来处理USB键盘的Report以实现特定的功能:当驱动程序在IRP_MJ_INTERNAL_DEVICE_CONTROL设置的完成例程从USB设备拦截到一个Get_Report_Descriptor时,拦截程序将此Descriptor中的USAGE值从“Keyboard”改为“UserDefined”,再返回给系统。

我们可以从这个例子中获得一些灵感,比如,在Win2k下,键盘是由OS独占访问的,我们可以通过这种方式使之可以让用户自由访问;我们也可以拦截其他Report_Descriptor,将部分键重新定义,以满足特殊的要求;如果你愿意再做一个用户态的程序,你还可以将你拦截到的键值传递给你的用户态程序,以实现象联想、实达等国内电脑大厂出品的那些键盘上的各种实用的功能。



二、程序的实现

Waltz Oney和John Hyde的例子已经写得很详细了,读者可以不用修改一个字节便顺利地编译生成一个过滤驱动程序。本文的目的在于使用DriverStudio组件Driverworks来实现同样的功能。

相信读者读到这篇文章时,已经对DriverStudio有了很多的了解。DriverStudio作为一个以C++为基础的“快速”驱动开发工具,它封装了基本上所有的DDK的函数,其集成在VC++中的DriverWizard,可以很方便地引导你完成设备驱动程序开发的全过程,能根据你的硬件种类自动生成设备驱动程序源代码,并提供了很多范例程序。当然,这些例子中便包含一个USB Filter驱动程序的框架。在不侵犯版权的前提下,充分利用现有共享的、免费的、授权的代码是我们的一贯作法。我们下面便以此范例为基础来作修改。

我们的目的是做一个HID小驱动程序hidusb.sys的Lower Filter,它附加在“人机接口设备” ,通过拦截USB的Get_Report_Descriptor来修改其返回值,当它发现该Descriptor的Usage 为“Keyboard”时,将其改为“UserDefined”,如此我们便可以完全控制这只键盘。具体做法是,拦截IRP_MJ_INTERNAL_DEVICE_CONTROL,并检查其IOCTL代码及URB,如果满足IOCTRL功能代码为IOCTL_INTERNAL_USB_SUBMIT_URB以及URB功能代码为URB_FUNCTION_GET_DESCRIPTOR_FROM_INTERFACE的条件,即上层驱动发来Get_Report_Descriptor请求时,设置一个完成例程,在这个完成例程中,我们将判断Usage的值,将Usage由“6(Keyboard)”时,将其改为“0(UserDefined)”。

打开C:\Program Files\NuMega\DriverStudio\DriverWorks\Examples\wdm\usbfilt目录(具体目录依你的DriverStudio所安装的目录不同而不同) ,再打开工程文件usbfilt.dsw,我们先看一下代码。

程序由两个类组成,一个是Driver类,一个是Device类。Driver类包括:

    入口函数DriverEntry:

DECLARE_DRIVER_CLASS(UsbFilterDriver, NULL)

/////////////////////////////////////////////////////////////////////

// Driver Entry

//

NTSTATUS UsbFilterDriver::DriverEntry(PUNICODE_STRING RegistryPath)

{

     T << "UsbFilterDriver::DriverEntry\n";



     m_Unit = 0;

     return STATUS_SUCCESS;



     // The following macro simply allows compilation at Warning Level 4

     // If you reference this parameter in the function simply remove the macro.

     UNREFERENCED_PARAMETER(RegistryPath);

}

    AddDevice函数

NTSTATUS UsbFilterDriver::AddDevice(PDEVICE_OBJECT Pdo)

{

     T << "UsbFilterDriver::AddDevice\n";

     UsbFilterDevice * pFilterDevice = new (

             static_cast<PCWSTR>(NULL),

             FILE_DEVICE_UNKNOWN,

             static_cast<PCWSTR>(NULL),

             0,

             DO_DIRECT_IO

             )

         UsbFilterDevice(Pdo, m_Unit);

     if (pFilterDevice)

     {

         NTSTATUS status = pFilterDevice->ConstructorStatus();

         if ( !NT_SUCCESS(status) )

         {

             T << "Failed to construct UsbFilterDevice"

               << (ULONG) m_Unit

               << " status = "

               << status

               << "\n";



             delete pFilterDevice;

         }

         else

         {

             m_Unit++;

         }

         return status;

     }

     else

     {

         T << "Failed to allocate UsbFilterDevice"

           << (ULONG) m_Unit

           << "\n";

         return STATUS_INSUFFICIENT_RESOURCES;

     }

}



     这两段代码基本上和自动生成的代码差不多。AddDevice的作用是构造一个过滤器的实例。

关键的代码在Device类。在这个类里,我们把过滤器插入设备栈,并拦截IRP,用自己的完成例程来实现特定的功能。

Device构造函数

UsbFilterDevice::UsbFilterDevice(PDEVICE_OBJECT Pdo, ULONG Unit) :

     KWdmFilterDevice(Pdo, NULL)

{

     T << "UsbFilterDevice::UsbFilterDevice\n";

     // Check constructor status

     if ( ! NT_SUCCESS(m_ConstructorStatus) )

     {

         return;

     }

     // Remember our unit number

     m_Unit = Unit;

     // initialize the USB lower device

     m_Usb.Initialize(this, Pdo);

     NTSTATUS status = AttachFilter(&m_Usb); //Attach the filter

     if(!NT_SUCCESS(status))

         {

         m_ConstructorStatus = status;

         return;

         }

     SetFilterPowerPolicy();

     SetFilterPnpPolicy();

}

在DDK中,我们用IoAttachDevice将设备对象插入设备栈中。DriverStudio封装了这个函数。在DriverStudio中,其他驱动程序需要用Initialize来初始化设备对象和接口,对于过滤驱动,我们关键是需要Attachfilter将其附加在堆栈中。

对于大部分如IRP_MJ_SYSTEM_CONTROL等IRP,我们所做的只需用PassThrough(Irp)将其直接往设备栈下层传递,不需要做任何工作。这些代码我们就不一一列举了。下面的部分才是本文的关键。

     我们知道,HIDUSB.SYS是使用内部IOCTRL发出URB给USB类驱动程序(USBD)读取数据的,那么,HIDUSB首先必须构造一个IRP_MJ_INTERNAL_DEVICE_CONTROL,它的IOCTL功能码为IOCTL_INTERNAL_USB_SUBMIT_URB(发出URB的内部IOCTL)。另外,因为我们要检查并修改的是USB键盘某个接口的报告描述,那么这个URB应该是URB_FUNCTION_GET_DESCRIPTOR_FROM_INTERFACE,如下:

NTSTATUS UsbFilterDevice::InternalDeviceControl(KIrp I)

{

     T << "UsbFilterDevice::InternalDeviceControl\n";

     // Pass through IOCTLs that are not submitting an URB

//不是我们感兴趣的IOCTL不要理它

     if (I.IoctlCode() != IOCTL_INTERNAL_USB_SUBMIT_URB)

         return DefaultPnp(I);



     PURB p = I.Urb(CURRENT);     // get URB pointer from IRP



//不是我们感兴趣的URB,也不要理它,

     if (p->UrbHeader.Function !=

URB_FUNCTION_GET_DESCRIPTOR_FROM_INTERFACE)

         return DefaultPnp(I);

//符合要求的IRP才被设置完成例程

     return PassThrough(I, LinkTo(DeviceControlComplete), this);

}

在设置好条件以后,再来实现完成例程。所有的检查、修改等动作都是在完成例程里面完成的。

NTSTATUS UsbFilterDevice::DeviceControlComplete(KIrp I)

{

     PURB p = I.Urb(CURRENT);

     if(p)

     {

//拦截到设备返回的描述表,

         char* DescriptorBuffer = (char*)p->UrbControlDescriptorRequest.TransferBuffer;

//指向第三个字节,表示设备Usage属性的值

         DescriptorBuffer += 3;

//如果值为6则改成0,6表示hid键盘,0表示未知设备

//在设备管理器里面,原来的hid兼容键盘就不复存在了,取而代之的则是hid兼容设备

         if ((*DescriptorBuffer&0xff) == 6)

             *DescriptorBuffer = 0;

     }

     return I.Status();

}

读者可以对照DriverWorks中的例子,直接替换掉(或者修改)上面这两个函数,再编译一下,便可以得到一个完整的键盘过滤器驱动程序。

其实,只要弄清楚了我们需要做些什么动作,在DriverStudio里面只需要写少量的关键代码,便可实现我们的要求,其余的大部分工作,或有范例可供参考,或有Driver Wizard自动生成。

     从上面可以看出,我们只需要修改这两个函数,拦截合适的IRP,便可以在完成例程里面实现我们特定的要求。正如开头所说,我们也可以拦截其他的IRP,拦截其他的URB,或者拦截特定键盘的按键键值,将之传递到用户态,以方便实现联想、实达等随机配备的多功能键盘的功能。



三、使用INF安装驱动

     在完成了驱动以后,还必须把它安装到系统里面,驱动程序才会起作用。一般来说,我们都必须为我们的驱动程序提供一个inf文件,以便于用户安装或者维护。对于新手来说,过滤驱动程序的inf或许有些棘手。所以,针对本文所描述的驱动,我们提供一个Win98下的安装范例usbkey.inf,范例中“;”后的文字是注解,以方便读者理解。







; usbkey.INF  

;

; Installs Lower Level Filter for a HID keyboard device

;

; (c) Copyright 2001 SINO Co., Ltd.

;     

[Version]

;”CHICAGO”表示Win9x平台

Signature="$CHICAGO$"

;键盘所属类名

Class=HID

ClassGUID={745a17a0-74d3-11d0-b6fe-00a0c90f57da}

;驱动程序提供者,此信息会显示在设备属性的“常规”页

Provider=%USBDBE%

LayoutFile=layout.inf

;显示在驱动程序文件详细资料窗口

DriverVer=11/12/2001,4.10.2222.12





;[ControlFlags]

;ExcludeFromSelect = *



;驱动程序安装目录,inf会将我们的驱动程序安装到如下目录

;记得Destinationdir后面一定要带一个“s”

[DestinationDirs]

DefaultDestDir = 10,system32\drivers



;要增加的注册表项

[ClassInstall]

Addreg=HIDClassReg



[HIDClassReg]

HKR,,,,%HID.ClassName%

HKR,,Icon,,-20



;制造商

[Manufacturer]

%USBDBE%=USBDBE



[USBDBE]

;我们所要附加过滤驱动程序的设备ID。这个ID可以从IC的规范上得来,也可以

;用hidview.exe读出,或者从注册表HKLM\Enum\hid和usb项找出

%HID.DeviceDesc%     = Keypad_Inst, USB\VID_05AF&PID_0805&MI_00



;要安装的文件和需要修改的注册表项

;Install usbkey driver

[Keypad_Inst]

CopyFiles=Keypad_Inst.CopyFiles

AddReg=Keypad_Inst.AddReg



[Keypad_Inst.CopyFiles]

hidusb.sys

hidparse.sys

hidclass.sys

usbfilt.sys



[Keypad_Inst.AddReg]

HKR,,DevLoader,,*ntkern

HKR,,NTMPDriver,,"hidusb.sys"





[Keypad_Inst.HW]

AddReg=Keypad_Inst.AddReg.HW



;Lowerfilters表示是低层过滤驱动,如果是上层过滤驱动,则必须改为upperfilters

[Keypad_Inst.AddReg.HW]

HKR,,"LowerFilters",0x00010000,"usbfilt.sys"



;HID设备所需要安装的文件和注册表中需要修改的地方

;Install USBHIDDevice

[USBHIDDevice]

CopyFiles=USBHIDDevice.Copy

AddReg=USBHIDDevice.AddReg



[USBHIDDevice.Copy]

hidclass.sys

hidusb.sys

hidparse.sys



[USBHIDDevice.AddReg]

HKR,,DevLoader,,*ntkern

HKR,,NTMPDriver,,"hidusb.sys"



;以下定义需要在上面某些地方使用时替换的字符串

[strings]

USBDBE              = "SINO Co., Ltd."

HID.DeviceDesc        = "SINO USB MultiKeyboard"

HID.HIDDeviceDesc     = "Human Interface Devices"

HID.DefaultDevice     = "HID Default Device"

HID.ClassName         = "Human Input Devices (HID)"

HID.SvcDesc           = "Microsoft HID Class Driver"



其实最简单的写inf的方式,是找一些类似设备的inf文件或范例来修改。在不侵权的前提下,充分利用现有资源是我们的一贯原则。

如何构造一个简单的USB过滤驱动程序的更多相关文章

  1. Linux内核分析第三周学习总结:构造一个简单的Linux系统MenuOS

    韩玉琪 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.Linux内 ...

  2. Linux内核分析-构造一个简单的Linux系统MenuOS

    构造一个简单的Linux系统MenuOS linux内核目录结构 arch目录包括了所有和体系结构相关的核心代码.它下面的每一个子目录都代表一种Linux支持的体系结构,例如i386就是Intel C ...

  3. Linux内核设计第三周——构造一个简单的Linux系统

    Linux内核设计第三周 ——构造一个简单的Linux系统 一.知识点总结 计算机三个法宝: 存储程序计算机 函数调用堆栈 中断 操作系统两把宝剑: 中断上下文的切换 进程上下文的切换 linux内核 ...

  4. 第三节 构造一个简单的Linux系统MenuOS——20135203齐岳

    第三节 构造一个简单的Linux系统MenuOS By 20135203齐岳 Linux内核源代码 arch/ 支持不同cpu的源代码 Documentations/ 文档存储 init/ 内核启动相 ...

  5. tensorflow笔记(二)之构造一个简单的神经网络

    tensorflow笔记(二)之构造一个简单的神经网络 版权声明:本文为博主原创文章,转载请指明转载地址 http://www.cnblogs.com/fydeblog/p/7425200.html ...

  6. Linux内核分析— —构造一个简单的Linux系统MenuOS(20135213林涵锦)

    Linux内核分析— —构造一个简单的Linux系统MenuOS 实验内容 Linux内核的启动过程,从start_kernel到init进程启动 使用实验楼的虚拟机打开shell cd LinuxK ...

  7. 《Linux内核分析》第三周笔记 构造一个简单的Linux系统MenuOS

    构造一个简单的Linux系统MenuOS 一.linux内核源代码简介 三大法宝(存储程序计算机.函数调用堆栈.中断)和两把宝剑(中断上下文的切换:保存现场和恢复现场.进程上下文的切换) 1.在lin ...

  8. 20135202闫佳歆--week3 构造一个简单的Linux系统MenuOs--学习笔记

    此为个人学习笔记存档 week 3 构造一个简单的Linux系统MenuOs 复习: 计算机有三个法宝:存储程序计算机,函数调用堆栈,中断 操作系统有两把剑: 1.中断上下文的切换,保存现场和恢复现场 ...

  9. 《Linux内核分析》第三周学习小结 构造一个简单的Linux系统OS

    郝智宇 无转载 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 第三周 构造一个简单的Linux系统Me ...

随机推荐

  1. Lambda表达式详解 (转)

    前言 1.天真热,程序员活着不易,星期天,也要顶着火辣辣的太阳,总结这些东西. 2.夸夸lambda吧:简化了匿名委托的使用,让你让代码更加简洁,优雅.据说它是微软自C#1.0后新增的最重要的功能之一 ...

  2. python脚本获取主机Mac地址

    #!/usr/bin/python import re import subprocess ARP = "arp" IP = "192.168.128.27" ...

  3. js和java中使用正则表达式校验邮箱

    问题:经常在项目中要校验邮箱? 邮箱格式:首位必须为字母,必须包含一个@符号,并且@之后有个名字,之后还有个.,再有一个后缀名 例如:wyp55023@163.com 一.java中代码如下: Str ...

  4. 使用FileReader实现前端预览所选图片

    需求描述 在浏览器环境下进开发网站(也就是B/S架构的应用),获取到当前设备上的图片后,希望可以在上传到服务器前简单预览一下图片内容. 具体实现 主要代码 html 部分 <input type ...

  5. 使用命令行生成jar包

    测试用类 public class Hello { public static void main(String[] args) { System.out.println("hello wo ...

  6. javascript中的字符串对象和数组对象

    1.javascript的对象的概念 在javascript中,除了null和undefined以处,其他的数据类型都被定义成了对象 也可以用创建对象的方法定义变量,string,math,array ...

  7. PHPExcel生成或读取excel文件

    以下是详细代码示例: <?php /** * 以下是使用示例,对于以 //// 开头的行是不同的可选方式,请根据实际需要 * 打开对应行的注释. * 如果使用 Excel5 ,输出的内容应该是G ...

  8. 洛谷 [P2761] 软件补丁问题

    并不是网络流 状压+SPFA 通过题目中的描述及数据范围可知,我们状压当前的漏洞,以每个二进制位表示是否有这个漏洞,并以状压的结果为顶点,以补丁的时间为边跑SPFA即可 #include <io ...

  9. DirectSound---简易Wav播放器

    这篇文章主要给大家介绍下如何用DirectSound打造一个简易播放器,因为篇幅有限且代码逻辑较为复杂,我们只介绍下核心技术内容.该播放器主要包括以下功能: 播放.暂停 播放进度提示. 1. Dire ...

  10. 读书共享 Primer Plus C-part11

    第十四章结构和其他数据形式 关于fread以及fwrite fread(char* buff,int size,int count,FILE* fp) fwrite(char* buff,int si ...