以文本方式查看主題 - 曙海教育集團論壇 (http://www.hufushizhe.com/bbs/index.asp) -- WinCE系統(tǒng)定制與驅(qū)動開發(fā) (http://www.hufushizhe.com/bbs/list.asp?boardid=36) ---- 詳解WinCE下USB Host驅(qū)動開發(fā) (http://www.hufushizhe.com/bbs/dispbbs.asp?boardid=36&id=1822) |
-- 作者:wangxinxin -- 發(fā)布時間:2010-11-26 11:00:37 -- 詳解WinCE下USB Host驅(qū)動開發(fā) 詳解WinCE下USB Host驅(qū)動開發(fā)(2) 當(dāng)用戶需要卸載USB Host設(shè)備驅(qū)動時,將會調(diào)用USBUnInstallDriver函數(shù)BOOL USBUnInstallDriver() 它與USBInstallDriver類似,不過是調(diào)用如下兩個函數(shù)UnRegisterClientSettings BOOL UnRegisterClientSettings(LPCWSTR szUniqueDriverId, LPCWSTR szReserved, LPCUSB_DRIVER_SETTINGS lpDriverSettings) BOOL UnRegisterClientDriverID(LPCWSTR szUniqueDriverId) 其中szUniqueDriverId是注冊時,使用的ID,szReserved保留,故設(shè)置為NULL,lpDriverSettings則是驅(qū)動程序設(shè)置信息。例程如下:BOOL USBUnInstallDriver(){RETAILMSG(1,(TEXT("USBUninstallDriver\\r\\n")))BOOL fRet = FALSEUSB_DRIVER_SETTINGS DriverSettingsDriverSettings.dwCount = sizeof(DriverSettings)DriverSettings.dwVendorId = 0x10C4DriverSettings.dwProductId = 0x0003DriverSettings.dwReleaseNumber = USB_NO_INFODriverSettings.dwDeviceClass = USB_NO_INFODriverSettings.dwDeviceSubClass = USB_NO_INFODriverSettings.dwDeviceProtocol = USB_NO_INFODriverSettings.dwInterfaceClass = 0DriverSettings.dwInterfaceSubClass = 0DriverSettings.dwInterfaceProtocol = 0fRet = UnRegisterClientSettings(L"USBTest", NULL, DriverSettings)if(fRet) {fRet = UnRegisterClientDriverID(L"USBTest")if(!fRet)RETAILMSG(1,(TEXT("UnRegisterClientDriverID error\\r\\n")))} elseRETAILMSG(1,(TEXT("UnRegisterClientSettings error\\r\\n")))return fRet} 其中DriverSettings必須與USBInstallDriver的DriverSettings一致。 回到原來的流程,WinCE注冊表中已經(jīng)包含了驅(qū)動信息,WinCE系統(tǒng)自動查找注冊表,在找到設(shè)備對應(yīng)鍵值的DLL后,將會調(diào)用該DLL的USBDeviceAttach函數(shù)。BOOL USBDeviceAttach(USB_HANDLE hDevice,LPCUSB_FUNCS lpUsbFuncs,LPCUSB_INTERFACE lpInterface,LPCWSTR szUniqueDriverId,LPBOOL fAcceptControl,DWORD dwUnused) hDevice 設(shè)備句柄,操作USB設(shè)備時,需要使用該句柄 lpUsbFuncs 指向一個包含各種USB操作的函數(shù)指針 lpInterface USB接口信息,這里需要注意的是,如果在DriverSettings里dwInterfaceClass、dwInterfaceSubClass、dwInterfaceProtocol設(shè)置為USB_NO_INFO,則該指針為NULL szUniqueDriverId 注冊設(shè)備ID fAcceptControl 該值被賦值為TRUE,表示該驅(qū)動能操作該設(shè)備。如果不能操作該設(shè)備,則“未能識別的USB設(shè)備”對話框會再次出現(xiàn),要求用戶輸入驅(qū)動程序名稱 dwUnused 未使用 在該函數(shù)內(nèi),主要是做一些檢查,判斷是否能驅(qū)動設(shè)備,還有就是注冊USB事件通知回調(diào)函數(shù),以及激活流驅(qū)動。對于檢查部分,這里不再詳細說明。 首先,介紹一下激活流驅(qū)動。 流驅(qū)動為應(yīng)用程序提供了一個訪問設(shè)備的接口,利用該接口可以像訪問文件一樣訪問設(shè)備。USB設(shè)備同樣可以使用該接口來為應(yīng)用程序提供支持。在注冊表的 HKEY_LOCAL_MACHINE\\Drivers\\BuiltIn鍵下,保存了各種WinCE內(nèi)建流驅(qū)動程序的入口。這些驅(qū)動通過device.exe在系統(tǒng)啟動時被激活。像USB這樣的設(shè)備,只有插入時,才存在流 驅(qū)動接口,所以我們需要手動激活流驅(qū)動。激活流驅(qū)動的函數(shù)是: HANDLE ActivateDevice(LPCWSTR lpszDevKey, DWORD dwClientInfo)lpszDevKey 字符串指明了流驅(qū)動所在注冊表的鍵。獲悉流驅(qū)動的人都知道,流驅(qū)動在注冊表中必須包含兩個鍵Prefix和Dll。 流驅(qū)動中所有接口函數(shù)都有類似XXX_的前綴,而這個Prefix則指明XXX對應(yīng)的字符串,如Prefix為COM,則流驅(qū)動包含如COM_Open、COM_Close、COM_Write、COM_Read這樣接口函數(shù)。Dll則說明了這些函數(shù)所在的動態(tài)鏈接庫。在我的例子中存在如下的注冊表鍵:[HKEY_LOCAL_MACHINE\\Drivers\\USB\\ClientDrivers\\USBTest] "Prefix"="TST" "Dll"="MyUSBTest.dll" 通過dwClientInfo,可以把參數(shù)間接傳給驅(qū)動的XXX_init。我們可以把hDevice、lpUsbFuncs、lpInterface這樣信息放置在一個結(jié)構(gòu)體中,通過該函數(shù)傳遞給流驅(qū)動使用。 USB通知回調(diào)函數(shù),可以用來判斷各種USB事件的發(fā)生,如USB拔出。當(dāng)發(fā)生事件后,系統(tǒng)會根據(jù)注冊的回調(diào)函數(shù)做相應(yīng)的處理,在USB設(shè)備拔出后,所要做的事情,就是卸載流驅(qū)動,并釋放占用的各種資源。 注冊回調(diào)函數(shù)是一個包含在lpUsbFuncs中的函數(shù)指針:LPUN_REGISTER_NOTIFICATION_ROUTINE lpUnRegisterNotificationRoutine該函數(shù)的聲明如下:typedef BOOL (* LPREGISTER_NOTIFICATION_ROUTINE)( USB_HANDLE hDevice, LPDEVICE_NOTIFY_ROUTINE lpNotifyRoutine, LPVOID lpvNotifyParameter)hDevice 設(shè)備句柄lpNotifyRoutine 回調(diào)函數(shù)lpvNotifyParameter 傳遞給回調(diào)函數(shù)的參數(shù) 在回調(diào)函數(shù)中卸載流驅(qū)動使用BOOL DeactivateDevice(HANDLE hDevice)其中,hDevice 傳入ActivateDevice時返回的句柄。 下面是具體的示例: typedef struct {DWORD dwSizeUSB_HANDLE hDevice,LPCUSB_FUNCS lpUsbFuncs,LPCUSB_INTERFACE lpInterface,HANDLE hStreamDevice} TESTUSBINFO, PTESTUSBINFO //回調(diào)函數(shù)extern "C" BOOL USBDeviceNotifications(LPVOID lpvNotifyParameter,DWORD dwCode,LPDWORD *dwInfo1,LPDWORD *dwInfo2,LPDWORD *dwInfo3,LPDWORD *dwInfo4){if (dwCode == USB_CLOSE_DEVICE) {PTESTUSBINFO pDrv = (PDRVCONTEXT) lpvNotifyParameterDeactivateDevice(pDrv-hStreamDevice) //卸載流驅(qū)動LocalFree(pDrv) //釋放資源}RETAILMSG(1,(TEXT("Free Driver Resources!\\r\\n")))return TRUE} BOOL USBDeviceAttach(USB_HANDLE hDevice,LPCUSB_FUNCS lpUsbFuncs,LPCUSB_INTERFACE lpInterface,LPCWSTR szUniqueDriverId,LPBOOL fAcceptControl,DWORD dwUnused){RETAILMSG(1,(TEXT("USBDeviceAttach\\r\\n")))*fAcceptControl = FALSE //顯示USB設(shè)備的一些信息if(lpInterface != NULL) {RETAILMSG(1,(TEXT("usbserialhost: DeviceAttach, IF %u, #EP:%u, Class:%u, Sub:%u, Prot:%u\\r\\n"),lpInterface-Descriptor.bInterfaceNumber,lpInterface-Descriptor.bNumEndpoints,lpInterface-Descriptor.bInterfaceClass,lpInterface-Descriptor.bInterfaceSubClass,lpInterface-Descriptor.bInterfaceProtocol))RETAILMSG(1,(TEXT("Endpoint 1:%u\\r\\n"), lpInterface-lpEndpoints.Descriptor.bmAttributes))RETAILMSG(1,(TEXT("Endpoint 2:%u\\r\\n"), lpInterface-lpEndpoints.Descriptor.bmAttributes))RETAILMSG(1,(TEXT("Endpoint 3:%u\\r\\n"), lpInterface-lpEndpoints.Descriptor.bmAttributes))} LPCUSB_DEVICE lpUsbDev = (lpUsbFuncs-lpGetDeviceInfo)(hDevice)if(!lpUsbDev){RETAILMSG(1,(TEXT("Unable to get USB device!\\r\\n")))return FALSE} //保存必要的信息供驅(qū)動程序其他部分使用PTESTUSBINFO pDrv = (PTESTUSBINFO)LocalAlloc (LPTR, sizeof (PTESTUSBINFO))pDrv-dwSize = sizeof (DRVCONTEXT)pDrv-hDevice = hDevicepDrv-lpUsbFuncs = lpUsbFuncspDrv-lpInterface = lpInterface //激活流驅(qū)動pDrv-hStreamDevice = ActivateDevice (L"Drivers\\\\USB\\\\ClientDrivers\\\\USBTest", (DWORD)pDrv)if (pDrv-hStreamDevice) {//注冊回調(diào)函數(shù)(*lpUsbFuncs-lpRegisterNotificationRoutine)(hDevice,USBDeviceNotifications,pDrv)} else {RETAILMSG(1, (TEXT("Can\'t activate stream device! rc=%d\\r\\n"), GetLastError()))LocalFree(pDrv)return FALSE}//驅(qū)動可以操作該設(shè)備*fAcceptControl = TRUEreturn TRUE} 至此,USB Host端設(shè)備驅(qū)動程序所必須實現(xiàn)的功能都已經(jīng)實現(xiàn)。并且和流驅(qū)動相連接。應(yīng)用程序已經(jīng)可以使用流驅(qū)動的接口來操作USB設(shè)備了。 |