Public Declare Function_
SetupDiEnumDeviceInterfaces Lib_
"setupapi.dll" _
(ByVal DeviceInfoSet As Long, _
ByVal DeviceInfoData As Long, _
ByRef InterfaceClassGuid As GUID, _
ByVal MenberIndex As Long, _
ByRef DeviceInterfaceData As_
SP_DEVICE_INTERFACE_DATA) _
As Long
這樣通過上面函數(shù)SetupDiGetClassDevs給出的設備信息得到的設備的一個接口的地址,每一次調用必需傳遞一個數(shù)組的索引來指定一個接口。上例中SP_DEVICE_INTERFACE
_DATA包含的結構用來識別每一個HID的接口。要與設備通信還需要一些更詳細的信息,其中最重要的是設備路徑,它可以通過函數(shù)SetupDiGetDeviceInterfaceDetail得到。其定義格式如下:
Public Declare Function_
SetupDiGetDeviceInterfaceDetail Lib_
"setupapi.dll" _
Alias "SetupDiGetDeviceInterfaceDetailA" _
(ByVal DeviceInfoSet As Long, _
ByRef DeviceInterfaceData As_
SP_DEVICE_INTERFACE_DATA, _
ByVal DeviceInterfaceDetailData As Long, _
ByVal DeviceInterfaceDetailDataSize As Long, _
ByRef RequiredSize As Long, _
ByVal DeviceInfoData As Long) _
As Long
其中的DeviceInterfaceData是自定義類型:
Public Type_
SP_DEVICE_INTERFACE_DETAIL_DATA
cbSize As Long
DevicePath As String
End Type
這樣函數(shù)就可傳回與前一函數(shù)所識別的接口有關的結構體,其中DevicePath是一個設備路徑,應用通過上面函數(shù)SetupDiEnumDevice-
Interfaces給出的設備接口數(shù)據(jù)設定設備接口的詳細信息。第一次調用該函數(shù)時,其中的DeviceInterfaceDetailDataSize無法預知,故可以兩次調用該函數(shù),第一次調用出錯,但可以返回正確的DeviceInterfaceDetailDataSize ,第二次調用傳遞返回值,調用即可成功。通過以上步驟基本可以建立與設備的連接了。
如想獲得更多關于設備能力的信息,還可以使用HidD_GetAttributes函數(shù),HidP_Get-
PreparsedData函數(shù),HidP_GetPreparsedData函數(shù),他們都是包含在hid.dll文件中的。分別可以實現(xiàn)獲得廠商ID,產品ID,Usage, Usage Page,報文長度等,不再一一贅述。
為了開啟一個HID設備需用到函數(shù)CreateFile,以取得設備代號,并用此代號與設備交換數(shù)據(jù)。定義形式如下:
Public Declare Function CreateFile Lib_
"kernel32" Alias "CreateFileA" _
(ByVal lpFilename As Long, _
ByVal dwDesiredAccess As Long, _
ByVal dwShareMode As Long, _
ByRef lpSecurityAttributes As Long, _
ByVal dwCreationDisposition As Long, _
ByVal dwFlagsAndAttributes As Long, _
ByVal hTemplateFile As Long) _
As Long
調用函數(shù)CreateFile以A結尾,是因為處理字符串時采用了8位的ANSI碼。
當應用程序取得HID設備的代號,此時就可以讀寫報文,利用通用函數(shù)WriteFile與ReadFile
定義如下所示:
Public Declare Function WriteFile Lib_
"kernel32" _
(ByVal hFile As Long, _
ByRef lpBuffer As Byte, _
ByVal nNumberOfBytesToWrite As Long, _
ByRef lpNumberOfBytesWritten As Long, _
ByVal lpOverlapped As Long) _
As Long
Declare Function ReadFile Lib "kernel32" _
(ByVal hFile As Long, _
ByRef lpBuffer As Any, _
ByVal nNumberOfBytesToRead As Long, _
ByRef lpNumberOfBytesRead As Long, _
lpOverlapped As Long) _
As Long
讀寫報文緩沖區(qū)時,第一個字節(jié)是Report ID,其后是報文數(shù)據(jù)。報文緩沖區(qū)默認是八個報文,并且環(huán)狀排列。因為數(shù)據(jù)讀寫是發(fā)生在主機輪訓設備的時候,并不是由設備觸發(fā)產生硬件中斷,所以如不能及時讀寫,新的數(shù)據(jù)會覆蓋舊的數(shù)據(jù),導致生報文丟失。當數(shù)據(jù)讀寫頻繁時應使用特征報文,它可以保證當報文數(shù)據(jù)沒有變化時,HID不會傳送新的數(shù)據(jù)。
當應用程序結束與HID的通信后,必須釋放所有之前保留的資源。所涉及到的幾個API函數(shù)如下:其中包括HidD_FreePreparsedData ,如下定義:
Public Declare Function_
HidD_FreePreparsedData Lib "hid.dll" _
(ByRef PreparsedData As Long) _
As Long
其作用是清除函數(shù)HidP_GetPreparsedData傳回的PreparsedData數(shù)據(jù)所占用的緩沖區(qū);函數(shù)SetupDiDestroyDeviceInfoLis定義為:
Public Declare Function_
SetupDiDestroyDeviceInfoList Lib_
"setupapi.dll" _
(ByVal DeviceInfoSet As Long) _
As Long
當不再使用SetupDiGetClassDevs時,應用上述函數(shù)釋放其返回的數(shù)組hDevInfo數(shù)組。還有需要使用函數(shù)CloseHandle,它是一個非常通用的API函數(shù),可以用于關閉通信。
4 小結
實際編程表明使用VB開發(fā)HID類USB設備是一條非常方便快捷的途徑,不需要編寫底層驅動,涉及到的API函數(shù)多是大家熟知的通用用函數(shù)。但同時還必須注意到一點,HID類只支持控制傳輸與中斷傳輸?刂苽鬏斖ǔ2挥糜跀(shù)據(jù)的傳輸,而中斷傳輸?shù)奶攸c是保證最大延遲,也就是事務之間的時間。他沒有保證傳輸速率,而是保證每筆事務之間的時間不會超過最大延時。所以中斷傳輸適用于數(shù)據(jù)量不太大,但需要及時快速的傳送,實時性要求較高的場合。對于數(shù)據(jù)量比較大的傳輸,是不宜使用中斷傳輸?shù)摹R虼薍ID設備應用的場合也必須根據(jù)該特點靈活使用。





