我在网上下了一个USBPortDlg的程序读取USB接口信息,发现里面有挺多问题,改了一下,发现在调用CreateFile时总是返回INVALID_HANDLE_VALUE.请大家看看.
工程是对话框工程,有一个输出信息用的EDIT控件IDC_EDIT_REPORT,和四个显示USB信息的ComboList控件.按钮消息响应OnSearch查询USB信息,代码如下:
void CUSBPortDlg::OnSearch()
{
m_strLog = _T("正在查找系统中HID类的GUID标识...\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);//
// 查找本系统中HID类的GUID标识
GUID guidHID;
HidD_GetHidGuid(&guidHID);
m_strLog += _T("HID类的GUID标识为\r\n");
CString strShow;
strShow.Format("%08x-%04x-%04x-%02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x\r\n",
guidHID.Data1,guidHID.Data2,guidHID.Data3,guidHID.Data4[0],
guidHID.Data4[1],guidHID.Data4[2],guidHID.Data4[3],guidHID.Data4[4],
guidHID.Data4[5],guidHID.Data4[6],guidHID.Data4[7]);
m_strLog += strShow;
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
// 准备查找符合HID规范的USB设备
//
HDEVINFO hDevInfo = SetupDiGetClassDevs(&guidHID,NULL,0, DIGCF_PRESENT |DIGCF_DEVICEINTERFACE);
if(hDevInfo==INVALID_HANDLE_VALUE)
{
m_strLog+=_T("SetupDiGetClassDevs Erro!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
return;
}
m_strLog += _T("正在查找可用的USB设备...\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
// 查找USB设备接口
SP_DEVICE_INTERFACE_DATA strtInterfaceData;
strtInterfaceData.cbSize=sizeof(SP_DEVINFO_DATA);//device
int iNumber=0;
DWORD dRequairSize,dNeedSize;
PSP_DEVICE_INTERFACE_DETAIL_DATA pstrtDetailData=NULL;
while(TRUE)
{
if(SetupDiEnumDeviceInterfaces(hDevInfo,NULL,&guidHID,iNumber,&strtInterfaceData))
{
// 若找到了设备,则读取设备路径名
//call two times
//First call to get the requair size
SetupDiGetDeviceInterfaceDetail(hDevInfo,&strtInterfaceData,NULL,0,&dRequairSize,NULL);
dNeedSize=dRequairSize;
pstrtDetailData=(PSP_DEVICE_INTERFACE_DETAIL_DATA)malloc(dRequairSize);
pstrtDetailData->cbSize=sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA);
if(!SetupDiGetDeviceInterfaceDetail(hDevInfo,&strtInterfaceData,pstrtDetailData,dNeedSize,&dRequairSize,NULL))
{
if(GetLastError()== ERROR_INSUFFICIENT_BUFFER)
{
AfxMessageBox(_T("ERROR_INSUFFICIENT_BUFFER"));
}
m_strLog+=_T("查找设备路径时出错!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
//AfxMessageBox(_T("查找设备路径时出错!"));
SetupDiDestroyDeviceInfoList(hDevInfo);
free(pstrtDetailData);
return;
}
// 找到了设备路径
CString m_strPath = pstrtDetailData->DevicePath;
m_usbPathList.AddString(m_strPath);
strShow.Format("可以通过路径%s访问设备\r\n",m_strPath);
m_strLog += strShow;
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
// 开放与设备的通信
//\\?\hid#vid_15d9&pid_0a37#6&162e5b0b&0&0000#{4d1e55b2-f16f-11cf-88cb-001111000030}//这是我得到的路径数值,调用下面CreateFile出错
HANDLE hCom = CreateFile(pstrtDetailData->DevicePath,GENERIC_READ|GENERIC_WRITE,
FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL,OPEN_EXISTING,0,NULL);
if (hCom == INVALID_HANDLE_VALUE) {
//
m_strLog += _T("无法打开通信端口!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
//AfxMessageBox(_T("无法打开通信端口"));
SetupDiDestroyDeviceInfoList(hDevInfo);
free(pstrtDetailData);
return;
}
// 查询设备标识
HIDD_ATTRIBUTES strtAttrib; m_strLog += _T("正在读取设备的标识...\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog); if (!HidD_GetAttributes(hCom,&strtAttrib))
{
m_strLog += _T("查询设备状态时出错!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
//AfxMessageBox(_T("查询设备状态时出错!"));
CloseHandle(hCom);
SetupDiDestroyDeviceInfoList(hDevInfo);
free(pstrtDetailData);
return;
}
// 将有关该设备的标识显示出来
m_strPath += _T("已读取,见右边所示\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strPath);
// 显示供应商标识
strShow.Format("0x%x",strtAttrib.VendorID);
m_usbVendorList.AddString(strShow);//ComboList控件变量
// 显示产品标识
strShow.Format("0x%x",strtAttrib.ProductID);
m_usbIDList.AddString(strShow);//ComboList控件变量
// 显示产品版本号
strShow.Format("%d",strtAttrib.VersionNumber);
m_usbVersionList.AddString(strShow);//ComboList控件变量
// 释放资源
CloseHandle(hCom);
iNumber++;
free(pstrtDetailData);
}
else
{
m_strLog+=_T("SetupDiEnumDeviceInterfaces Return False!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
break;
}
}
m_strLog+=_T("查找完毕!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
SetupDiDestroyDeviceInfoList(hDevInfo);}
当进行到CreateFile时得到INVALID_HANDLE_VALUE.请问这是怎么回事呢?
工程是对话框工程,有一个输出信息用的EDIT控件IDC_EDIT_REPORT,和四个显示USB信息的ComboList控件.按钮消息响应OnSearch查询USB信息,代码如下:
void CUSBPortDlg::OnSearch()
{
m_strLog = _T("正在查找系统中HID类的GUID标识...\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);//
// 查找本系统中HID类的GUID标识
GUID guidHID;
HidD_GetHidGuid(&guidHID);
m_strLog += _T("HID类的GUID标识为\r\n");
CString strShow;
strShow.Format("%08x-%04x-%04x-%02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x\r\n",
guidHID.Data1,guidHID.Data2,guidHID.Data3,guidHID.Data4[0],
guidHID.Data4[1],guidHID.Data4[2],guidHID.Data4[3],guidHID.Data4[4],
guidHID.Data4[5],guidHID.Data4[6],guidHID.Data4[7]);
m_strLog += strShow;
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
// 准备查找符合HID规范的USB设备
//
HDEVINFO hDevInfo = SetupDiGetClassDevs(&guidHID,NULL,0, DIGCF_PRESENT |DIGCF_DEVICEINTERFACE);
if(hDevInfo==INVALID_HANDLE_VALUE)
{
m_strLog+=_T("SetupDiGetClassDevs Erro!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
return;
}
m_strLog += _T("正在查找可用的USB设备...\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
// 查找USB设备接口
SP_DEVICE_INTERFACE_DATA strtInterfaceData;
strtInterfaceData.cbSize=sizeof(SP_DEVINFO_DATA);//device
int iNumber=0;
DWORD dRequairSize,dNeedSize;
PSP_DEVICE_INTERFACE_DETAIL_DATA pstrtDetailData=NULL;
while(TRUE)
{
if(SetupDiEnumDeviceInterfaces(hDevInfo,NULL,&guidHID,iNumber,&strtInterfaceData))
{
// 若找到了设备,则读取设备路径名
//call two times
//First call to get the requair size
SetupDiGetDeviceInterfaceDetail(hDevInfo,&strtInterfaceData,NULL,0,&dRequairSize,NULL);
dNeedSize=dRequairSize;
pstrtDetailData=(PSP_DEVICE_INTERFACE_DETAIL_DATA)malloc(dRequairSize);
pstrtDetailData->cbSize=sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA);
if(!SetupDiGetDeviceInterfaceDetail(hDevInfo,&strtInterfaceData,pstrtDetailData,dNeedSize,&dRequairSize,NULL))
{
if(GetLastError()== ERROR_INSUFFICIENT_BUFFER)
{
AfxMessageBox(_T("ERROR_INSUFFICIENT_BUFFER"));
}
m_strLog+=_T("查找设备路径时出错!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
//AfxMessageBox(_T("查找设备路径时出错!"));
SetupDiDestroyDeviceInfoList(hDevInfo);
free(pstrtDetailData);
return;
}
// 找到了设备路径
CString m_strPath = pstrtDetailData->DevicePath;
m_usbPathList.AddString(m_strPath);
strShow.Format("可以通过路径%s访问设备\r\n",m_strPath);
m_strLog += strShow;
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
// 开放与设备的通信
//\\?\hid#vid_15d9&pid_0a37#6&162e5b0b&0&0000#{4d1e55b2-f16f-11cf-88cb-001111000030}//这是我得到的路径数值,调用下面CreateFile出错
HANDLE hCom = CreateFile(pstrtDetailData->DevicePath,GENERIC_READ|GENERIC_WRITE,
FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL,OPEN_EXISTING,0,NULL);
if (hCom == INVALID_HANDLE_VALUE) {
//
m_strLog += _T("无法打开通信端口!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
//AfxMessageBox(_T("无法打开通信端口"));
SetupDiDestroyDeviceInfoList(hDevInfo);
free(pstrtDetailData);
return;
}
// 查询设备标识
HIDD_ATTRIBUTES strtAttrib; m_strLog += _T("正在读取设备的标识...\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog); if (!HidD_GetAttributes(hCom,&strtAttrib))
{
m_strLog += _T("查询设备状态时出错!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
//AfxMessageBox(_T("查询设备状态时出错!"));
CloseHandle(hCom);
SetupDiDestroyDeviceInfoList(hDevInfo);
free(pstrtDetailData);
return;
}
// 将有关该设备的标识显示出来
m_strPath += _T("已读取,见右边所示\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strPath);
// 显示供应商标识
strShow.Format("0x%x",strtAttrib.VendorID);
m_usbVendorList.AddString(strShow);//ComboList控件变量
// 显示产品标识
strShow.Format("0x%x",strtAttrib.ProductID);
m_usbIDList.AddString(strShow);//ComboList控件变量
// 显示产品版本号
strShow.Format("%d",strtAttrib.VersionNumber);
m_usbVersionList.AddString(strShow);//ComboList控件变量
// 释放资源
CloseHandle(hCom);
iNumber++;
free(pstrtDetailData);
}
else
{
m_strLog+=_T("SetupDiEnumDeviceInterfaces Return False!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
break;
}
}
m_strLog+=_T("查找完毕!\r\n");
SetDlgItemText(IDC_EDIT_REPORT,m_strLog);
SetupDiDestroyDeviceInfoList(hDevInfo);}
当进行到CreateFile时得到INVALID_HANDLE_VALUE.请问这是怎么回事呢?
FILE_SHARE_READ|FILE_SHARE_WRITE,
NULL,OPEN_EXISTING,0,NULL);可能是没有权限,把GENERIC_READ|GENERIC_WRITE这一项改成0,以查询的方式打开试试看。因为像键盘,鼠标这样的HID设备是系统独占,无法以读写方式打开。我一般都是先以查询的方式枚举设备,再根据PID,VID,Usage Page,Usage找到自定义的设备后再以读写方式打开。
hDevHandle = CreateFile(MyDevPathName, NULL, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);但 GENERIC_READ不行,而GENERIC_WRITE 则可以,