C++ Builder 訪(fǎng)問(wèn) USB 口的方法
編譯環(huán)境: Borland C++ Builder 6.0, Windows Server 2003 DDK
運(yùn)行環(huán)境: Win98/2000/XP/2003
下面是訪(fǎng)問(wèn) USB 口必備的函數(shù):
#include <vcl.h>
#include <dir.h>
#include <setupapi.h>
#include "C:/WINDDK/3790/inc/ddk/w2k/usbdi.h"
#include "C:/WINDDK/3790/inc/ddk/w2k/devioctl.h"
#include <initguid.h>
//---------------------------------------------------------------------------
// 下面必須為驅(qū)動(dòng)程序的 GUID 值, 這里我亂寫(xiě)的數(shù)
DEFINE_GUID(USB_DRIVER_GUID, 0x12345678,0xabcd,0x1122,0x33,0x44,0x55,0x66,0x77,0x88,0x99,0x00);
//---------------------------------------------------------------------------
HANDLE OpenOneDevice(HDEVINFO hDvcInfo, PSP_INTERFACE_DEVICE_DATA DvcInfoData, char *sDevNameBuf)
{
HANDLE hOut = INVALID_HANDLE_VALUE;
ULONG iReqLen = 0;
SetupDiGetInterfaceDeviceDetail(hDvcInfo, DvcInfoData, NULL, 0, &iReqLen, NULL);
ULONG iDevDataLen = iReqLen; //sizeof(SP_FNCLASS_DEVICE_DATA) + 512;
PSP_INTERFACE_DEVICE_DETAIL_DATA pDevData = (PSP_INTERFACE_DEVICE_DETAIL_DATA)malloc(iDevDataLen);
pDevData->cbSize = sizeof(SP_INTERFACE_DEVICE_DETAIL_DATA);
if(SetupDiGetInterfaceDeviceDetail(hDvcInfo, DvcInfoData, pDevData, iDevDataLen, &iReqLen, NULL))
{
strcpy(sDevNameBuf, pDevData->DevicePath);
hOut = CreateFile(pDevData->DevicePath, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
}
free(pDevData);
return hOut;
}
//---------------------------------------------------------------------------
HANDLE OpenUsbDevice(const GUID *pGuid, char *sDevNameBuf)
{
HANDLE hOut = INVALID_HANDLE_VALUE;
HDEVINFO hDevInfo = SetupDiGetClassDevs(pGuid, NULL, NULL, DIGCF_PRESENT|DIGCF_INTERFACEDEVICE);
SP_INTERFACE_DEVICE_DATA deviceInfoData;
deviceInfoData.cbSize = sizeof (SP_INTERFACE_DEVICE_DATA);
ULONG nGuessCount = MAXLONG;
for(ULONG iDevIndex=0; iDevIndex<nGuessCount; iDevIndex++)
{
if(SetupDiEnumDeviceInterfaces(hDevInfo, 0, pGuid, iDevIndex, &deviceInfoData))
{
if((hOut=OpenOneDevice(hDevInfo, &deviceInfoData, sDevNameBuf)) != INVALID_HANDLE_VALUE)
break;
}
else if(GetLastError() == ERROR_NO_MORE_ITEMS) //No more items
{
break;
}
}
SetupDiDestroyDeviceInfoList(hDevInfo);
return hOut;
}
//---------------------------------------------------------------------------
bool GetUsbDeviceFileName(const GUID *pGuid, char *sDevNameBuf)
{
HANDLE hDev = OpenUsbDevice(pGuid, sDevNameBuf);
if(hDev != INVALID_HANDLE_VALUE)
{
CloseHandle(hDev);
return true;
}
return false;
}
//---------------------------------------------------------------------------
HANDLE OpenMyDevice()
{
char DeviceName[MAXPATH] = "";
return OpenUsbDevice(&USB_DRIVER_GUID, DeviceName);
}
//---------------------------------------------------------------------------
HANDLE OpenMyDevPipe(const char *PipeName)
{
char DeviceName[MAXPATH] = "";
if(GetUsbDeviceFileName(&USB_DRIVER_GUID, DeviceName))
{
strcat(DeviceName,"\");
strcat(DeviceName,PipeName);
return CreateFile(DeviceName, GENERIC_WRITE|GENERIC_READ, FILE_SHARE_WRITE|FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
}
return INVALID_HANDLE_VALUE;
}
//---------------------------------------------------------------------------
有了上面的函數(shù)就可以訪(fǎng)問(wèn) USB 口了:
//打開(kāi) USB 口讀寫(xiě), 由驅(qū)動(dòng)程序的 Pipe 名確定
HANDLE hPipe = OpenMyDevPipe("MyPipe1"); //驅(qū)動(dòng)程序里面的 Pipe 名, 對(duì)應(yīng)訪(fǎng)問(wèn)某個(gè)端點(diǎn)的 I/O, 這里我亂寫(xiě)的, 需要與驅(qū)動(dòng)一致
if(hPipe != INVALID_HANDLE_VALUE) //打開(kāi) Pipe 成功
{
ReadFile(hPipe, Buffer, BufSize, &nBytesRead, NULL); //從 hPipe 里讀取數(shù)據(jù)到 Buffer 里
//WriteFile(hPipe, Buffer, BytesToWrite, &nBytesWritten, NULL); //把 Buffer 里面的 BytesToWrite 字節(jié)寫(xiě)入 hPipe
CloseHandle(hPipe);
}
//使用 DeviceIoControl 訪(fǎng)問(wèn) USB 設(shè)備
HANDLE hDevice = OpenMyDevice();
if(hDevice != INVALID_HANDLE_VALUE) //打開(kāi)設(shè)備成功
{
//這些 DeviceIoControl 功能都是由設(shè)備定義的, 具體看設(shè)備和驅(qū)動(dòng)的資料
if(DeviceIoControl(hDevice, IOCTL_READ_xxxx, &IOBlock, sizeof(IOBLOCK), &c, 1, &nBytes, NULL))
{
//成功
}
CloseHandle(hDevice);
}
USB 設(shè)備、USB 驅(qū)動(dòng)、USB 應(yīng)用程序
1.USB 設(shè)備硬件部分
a.這個(gè)硬件的標(biāo)識(shí)是用的 Vender ID 和 Product ID, 即“廠(chǎng)家標(biāo)識(shí)”和“產(chǎn)品標(biāo)識(shí)”
b.這個(gè)硬件規(guī)定了各個(gè) End Point (端點(diǎn)) 的性質(zhì), 讀/寫(xiě) 及 類(lèi)型 (Control/Interrupt/Bulk/Isochronous)
c.這個(gè)硬件的固件里面有 DeviceIoControl 的實(shí)現(xiàn)部分, 規(guī)定了這個(gè)函數(shù)的具體參數(shù)和動(dòng)作
2.USB 設(shè)備驅(qū)動(dòng)
①硬件接口
a.需要識(shí)別 Vender ID 和 Product ID
b.對(duì)每個(gè) EndPoint 的每個(gè) I/O 分配一個(gè) Pipe, 并且起一個(gè)名字作為軟件接口
c.做 DeviceIoControl 的接口
②軟件接口
a.GUID, 驅(qū)動(dòng)程序的標(biāo)識(shí), 每個(gè)驅(qū)動(dòng)程序使用不同的 GUID, GUID 是識(shí)別驅(qū)動(dòng)的, 與硬件無(wú)關(guān) (驅(qū)動(dòng)程序升級(jí)版本 GUID 不能修改)
b.硬件接口里面的 b: Pipe 名字是軟件接口, 這個(gè) Pipe 名字純粹由驅(qū)動(dòng)定義的, 和硬件無(wú)關(guān), 升級(jí)驅(qū)動(dòng)不能改 Pipe 的名字
c.硬件接口里面的 c 的各個(gè)參數(shù)也是軟件的接口, 這些參數(shù)是由硬件帶來(lái)的, 不是驅(qū)動(dòng)規(guī)定的, 當(dāng)然也可以在驅(qū)動(dòng)里面轉(zhuǎn)義, 隱藏設(shè)備的真實(shí)情況
③這個(gè)驅(qū)動(dòng)程序是用 WinDDK 編譯的, 可以用文本編輯器或其他開(kāi)發(fā)工具的編輯器編程序代碼, 然后調(diào)用 WinDDK 編譯
3.讀寫(xiě) USB 口的程序
①與驅(qū)動(dòng)的接口
a.利用驅(qū)動(dòng)程序里面的 GUID 找出設(shè)備的文件名, 用 CreateFile 函數(shù)打開(kāi)設(shè)備。我前面的程序里面的 OpenUsbDevice 就是這個(gè)作用
b.通過(guò) a.得到的設(shè)備文件名和驅(qū)動(dòng)程序里面的 Pipe 名打開(kāi) Pipe, 訪(fǎng)問(wèn)這個(gè) Pipe 對(duì)應(yīng)的 USB 端點(diǎn) (讀寫(xiě)數(shù)據(jù))
c.使用 a.的 CreateFile 得到的句柄, 通過(guò) DeviceIoControl 實(shí)現(xiàn)設(shè)備規(guī)定的動(dòng)作
②有關(guān)需要的資料
a.Vender ID, Product ID 和 GUID 一般在驅(qū)動(dòng)程序的 .inf 文件里面能看到, 如果找不到就需要和廠(chǎng)家聯(lián)系
b.Pipe 的名字是驅(qū)動(dòng)程序規(guī)定的, 需要有驅(qū)動(dòng)程序的資料才能知道
c.DeviceIoControl 的參數(shù)需要有驅(qū)動(dòng)程序的資料或者硬件資料才能知道
③這個(gè)程序一般用 C/C++ 直接編寫(xiě), 如果使用其他語(yǔ)言(VB/PB等)需要調(diào)用 C/C++ 編的 DLL
其他相關(guān)內(nèi)容:
USB 驅(qū)動(dòng)程序可以到注冊(cè)表里面找到:
"HKEY_LOCAL_MACHINE\SYSTEM\ControlSet001\Enum\USB\Vid_廠(chǎng)家標(biāo)識(shí)&Pid_產(chǎn)品標(biāo)識(shí)\驅(qū)動(dòng)程序"
里面的 ClassGUID 就是驅(qū)動(dòng)程序的 GUID 標(biāo)識(shí), 例如 {36FC9E60-C465-11CF-805*44553540000}
相當(dāng)于程序的: DEFINE_GUID(USB_DRIVER_GUID, 0x36FC9E60,0xC465,0x11CF,0x80,0x56,0x44,0x45,0x53,0x54,0x00,0x00);
另外在這個(gè)注冊(cè)表鍵里面還可找到有關(guān)設(shè)備的其他描述, 例如 DeviceDesc = "USB Mass Storage Device" 等
來(lái)源:Etta0次