驱动安装知识储备

 最近需要实现自动安装手机或者平板驱动在PC上的功能。搜集了相关的资料,整理下:

硬件主导安装:

       硬件主导的设备安装,是指在安装程序尚未运行,系统尚无设备驱动信息的情况下,用户将设备连接到电脑上,这将导致“硬件主导”的设备安装方式。每当有新设备连接到系统中,系统PNP管理器都会尝试为设备寻找合适的驱动程序,如果能够找到,将尽可能地采用服务器端方式,悄无声息地为设备将驱动安装完毕。否则,将提示用户进行驱动安装指导。

软件主导安装:

    软件主导的安装方式,是软件安装先于设备接入。般来说,安装软件在运行之初,需要做一件事:判断当前系统中,PNP管理器是否正在为另一个设备安装忙碌中,如果这样的话,最好应当等待其完毕后,再进行本设备的安装。

 

安装设备:

 分两种情况处理此问题:当前正连接在系统上的设备,当前不连接到系统的设备。

UpdateDriverForPlugAndPlayDevices函数最大的作用是驱动更新。通过UpdateDriverForPlugAndPlayDevices并传入设备ID,通过返回值来判断设备是否已连接:函数返回TRUE;或函数返回FALSE,且GetLastError返回错误值ERROR_NO_MORE_ITEMS.

安装驱动需要依靠UpdateDriverForPlugAndPlayDevices这个函数。它需要的参数其实不多,主要是INF文件路径等。

BOOL WINAPI
UpdateDriverForPlugAndPlayDevices(
HWND  hwndParent,
LPCTSTR  HardwareId,
LPCTSTR  FullInfPath,
DWORD  InstallFlags,
     PBOOL  bRebootRequired OPTIONAL
);

eg:

bool blDriverInstall = DriverDAL.UpdateDriverForPlugAndPlayDevices(
                IntPtr.Zero,
                hardWareID,
                infDriverPath,
                DriverEntity.INSTALLFLAG_FORCE,
                false);执行这个函数相当于在设备管理器右键菜单上点击“扫描检测硬件改动”。它会查找当前已经添加到系统中但可能还没有安装驱动程序的硬件设备。由于没有对应的硬件,因此我们要另想办法,让它能检测到虚拟硬件的存在。

为此,我们需要创建一设备信息块,并将它弄到系统注册表中去。

创建设备信息块的函数是SetupDiCreateDeviceInfo

WINSETUPAPI BOOL WINAPI
SetupDiCreateDeviceInfo(
  IN HDEVINFO  DeviceInfoSet,
  IN PCTSTR  DeviceName,
  IN LPGUID  ClassGuid,
  IN PCTSTR  DeviceDescription,   OPTIONAL
  IN HWND  hwndParent,   OPTIONAL
  IN DWORD  CreationFlags,
     OUT PSP_DEVINFO_DATA  DeviceInfoData   OPTIONAL
  );

这个函数需要7个参数,其他的都好办,无非是设备ID字符串,设备GUID等,还有一个用于输出的结构体,记住填写其中的size字段,其余为0就可以了。问题是第一个参数HDEVINFO需要用另一个函数SetupDiCreateDeviceInfoList来创建。如果GUID不想写死,想通过INF读取,则需要另一个函数SetupDiGetINFClass。

HDEVINFO
SetupDiCreateDeviceInfoList(
  IN LPGUID  ClassGuid,   OPTIONAL
  IN HWND  hwndParent   OPTIONAL
  );

WINSETUPAPI BOOL WINAPI
SetupDiGetINFClass(
  IN PCTSTR  InfName,
  OUT LPGUID  ClassGuid,
  OUT PTSTR  ClassName,
  IN DWORD  ClassNameSize,
  OUT PDWORD  RequiredSize  OPTIONAL,
  );

以上函数调用的顺序是:SetupDiGetINFClass获取GUID,SetupDiCreateDeviceInfoList创建设备信息块列表,SetupDiCreateDeviceInfo创建设备信息块。完成这些步骤之后,我们就可以注册设备了。首先,我们要调用SetupDiSetDeviceRegistryProperty这个函数来设置设备在系统设备树上的路径,然后通过SetupDiCallClassInstaller这个函数来注册:


WINSETUPAPI BOOL WINAPI
 SetupDiSetDeviceRegistryProperty(
   IN HDEVINFO  DeviceInfoSet,
   IN OUT PSP_DEVINFO_DATA  DeviceInfoData,
   IN DWORD  Property,
   IN CONST BYTE  *PropertyBuffer,
   IN DWORD  PropertyBufferSize
   );

这个函数的作用是设置驱动信息块中的信息。对于我们最重要的是其中的硬件ID。这通过给第三个参数以SPDRP_HARDWAREID这个值就可以实现。这时候,第四个参数就是指向硬件ID字符串的指针,第五个参数就是字符串的字节长度。

WINSETUPAPI BOOL WINAPI
SetupDiCallClassInstaller(
  IN DI_FUNCTION  InstallFunction,
  IN HDEVINFO  DeviceInfoSet,
  IN PSP_DEVINFO_DATA  DeviceInfoData   OPTIONAL
  );


这个函数第一个参数要用DIF_REGISTERDEVICE填充以便注册设备,第二项填写设备信息块列表,第三项则填写设备信息块。

驱动程序更新:

     我们知道,系统为所有的设备建立了一棵设备树,每个设备都是数上的一个节点,叫DevNode。设备首次连接到系统中,设备节点就被建立了。即使将设备移除,系统为了效率起见,并不会将此设备节点也从设备树上删除,这将利于设备再次插入的时候,快速识别,省略了驱动寻找的过程。这个特点不利于我们这里的更新工作,因为当设备再次插入时,更新信息不会被系统使用。故而安装程序必须要找到这个节点,并为他打上“重新安装”的标记。这样设备再次插入,PNP管理器将为他重新寻找驱动。

1> 首先安装新驱动,调用SetupCopyOEMInf函数将安装文件拷贝到系统中。这是必要的,下面的三个步骤用来修改标记。

BOOL WINAPI SetupCopyOEMInf(

__in          PCTSTR SourceInfFileName,          //INF文件路径

__in          PCTSTR OEMSourceMediaLocation,     //其他驱动文件或源媒体路径

__in          DWORD OEMSourceMediaType,          //源媒体类型

__in          DWORD CopyStyle,                   //拷贝类型,建议NOOVERWRITE

__out_opt    PTSTR DestinationInfFileName,      //系统创建的OEMxx.inf路径

__in          DWORD DestinationInfFileNameSize,  //上面缓冲区大小

__out_opt    PDWORD RequiredSize,               //实际使用大小

__out_opt    PTSTR DestinationInfFileNameComponent//下述

);

eg:

SetupCopyOEMInf(
 
“c:...inf”, 

NULL,      // 和SPOST_PATH一起使用时设为NULL,表示和inf文件同目录即c:/tmp
 
SPOST_PATH, 

SP_COPY_NOOVERWRITE, 

asDesPath, 

MAX_PATH, NULL, 

&asDesName))


找到要更新的设备。调用SetupDiGetClassDevs函数。由于设备未连接到系统,所以调用时要去除DIGCF_PRESENT标志。SetupDiGetClassDevs将返回一个所有符合条件的设备集。此时继续调用SetupDiEnumDeviceInfo对此设备集进行遍历操作,以得到每个设备的信息,通过设备ID或者兼容ID,来判断是否是要找的设备。

2>找到要更新的设备。调用SetupDiGetClassDevs函数。由于设备未连接到系统,所以调用时要去除DIGCF_PRESENT标志。SetupDiGetClassDevs将返回一个所有符合条件的设备集。此时继续调用SetupDiEnumDeviceInfo对此设备集进行遍历操作,以得到每个设备的信息,通过设备ID或者兼容ID,来判断是否是要找的设备。

 

检查设备节点DevNode状态。调用配置管理器函数CM_Get_DevNode_Status,并传入在第二步中得到的SP_DRVINFO_DATA结构体中的DevInst变量作为关键参数。不出意外的话,CM_Get_DevNode_Status的返回值应该是CR_NO_SUCH_DEVINST,表明DevNode虽存在设备却已移除。

4> 为DevNode打标记。首先获取现有的配置值,调用SetupDiGetDeviceRegistryProperty,并将属性参数设置为SPDRP_CONFIGFLAGS。将取得的配置值与CONFIGFLAG_REINSTALL(重安装也!)进行或操作,并调用相反的设置函数SetupDiSetDeviceRegistryProperty将新的配置值写入。

 

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
DPInst.exe: installs and uninstalls driver packages. DPInst.exe: 安装及删除驱动程序包。 By default, the tools searches the current directory and tries to install all driver packages found. 默认情况下,此工具会搜索当前目录并尝试安装找到的驱动程序包。 Usage: DPInst.exe [/U INF-file][/S | /Q][/LM][/P][/F][/SH][/SA][/A][/PATH Path][/EL][/L LanguageID][/C][/D][/LogTitle Title][/SW][/? | /h | /help] 用法:DPInst.exe [/U INF-file][/S | /Q][/LM][/P][/F][/SH][/SA][/A][/PATH Path][/EL][/L LanguageID][/C][/D][/LogTitle Title][/SW][/? | /h | /help] /U path to INF file Uninstall a driver package (INF-file). /U 指定 INF 文件 删除一个驱动程序包(INF-文件)。 /S | /Q Silent (Quiet) mode. Suppresses the Device Installation Wizard and any dialogs popped-up by the operating system. /LM Legacy mode. Accepts unsigned driver packages and packages with missing files. These packages won't install on the latest version of Windows. /LM 传统模式。接受未签名的驱动包(缺少.cat文件)以及包中有文件丢失。 /P Prompt if the driver package to be installed is not better than the current one. /P 如果驱动包已经安装以及比当前使用的版本要旧就发出提示。 /F Force install inf the driver package is not better than the current one. /F 强制安装比当前使用的版本要旧的驱动包。 /SH Scans hardware for matching devices and only copies and installs those drivers for which a device is present. Only valid for Plug and Play drivers. /SH 仅仅对已插入的匹配的硬件设备进行扫描。 /SA Suppress the Add/Remove Programs entry normally created for each driver package. /SA 禁止“添加或删除程序”创建卸载项。 /A Install all or none. /PATH Path Search for driver packages under the given path. /PATH 路径 在指定的路径下搜索驱动程序包。 /EL Enables all languages not explicitly listed in the XML file. /L LanguageID Tries to use the given language in all UI. Useful for localization tests. /SE Suppress the EULA. /SE 禁止最终用户许可协议。 /C Dump logging output to attached Console (Windows XP and above). /C 转存日志文件输出到附加控

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值