以下の編集を参照してください。
テレビやプロジェクターなどの接続デバイスのクラス/インターフェイス GUID を取得しようとしています。テレビが接続されているときに、関数 RegisterDeviceNotification() を使用して GUID を取得しています。
私の問題: TV が接続されている場合は正常に通知されますが、(lParam 構造を調べて) デバイスの GUID にアクセスしようとすると、アクセス違反またはエラーが発生します。
何が問題なのかわかりませんか?GUID にアクセスしたり使用したりするたびに、プログラムが失敗することに注意してください。プログラムはクラッシュしませんが、出力ウィンドウには次のようなメッセージが表示され、何が問題なのFirst-chance exception at 0x001c1a9d in myProgram.exe...
かわかりませんか?
HDEVNOTIFY *hDeviceNotify;
GUID interfaceClassGuid = { 0x25dbce51, 0x6c8f, 0x4a72, 0x8a,0x6d,0xb5,0x4c,0x2b,0x4f,0xc8,0x35 };
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
int wmId, wmEvent;
PAINTSTRUCT ps;
HDC hdc;
switch (message)
{
case WM_CREATE:
{
openConsoleWindow();
DEV_BROADCAST_DEVICEINTERFACE NotificationFilter;
ZeroMemory( &NotificationFilter, sizeof(NotificationFilter) );
NotificationFilter.dbcc_size = sizeof(DEV_BROADCAST_DEVICEINTERFACE);
NotificationFilter.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;
NotificationFilter.dbcc_classguid = interfaceClassGuid;
*hDeviceNotify = RegisterDeviceNotification( hWnd, &NotificationFilter, DEVICE_NOTIFY_WINDOW_HANDLE );
if (*hDeviceNotify == NULL)
printf("hDeviceNotify = NULL \n");
}
break;
case WM_DEVICECHANGE:
{
PDEV_BROADCAST_DEVICEINTERFACE b = (PDEV_BROADCAST_DEVICEINTERFACE) lParam;
// The following printf NEVER prints out.
// When accessing the below GUID I get an error:
// "First-chance exception at 0x001c1a9d in myProgram.exe: 0xC0000005: Access violation writing location 0x00000000."
printf("GUID: Data1: %x, Data2: %x, Data3: %x, Data4: %s \n", b->dbcc_classguid.Data1,
b->dbcc_classguid.Data2, b->dbcc_classguid.Data3, (char*)b->dbcc_classguid.Data4);
// The following function ALWAYS fails.
// GetLastError() gives the error "The parameter is incorrect"
HDEVINFO hDevInfo = SetupDiGetClassDevs(&b->dbcc_classguid, NULL, NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
if (hDevInfo == INVALID_HANDLE_VALUE) {
//printf("hDevInfo == INVALID_HANDLE_VALUE \n");
outputLastError(_T("hDevInfo == INVALID_HANDLE_VALUE \n"));
}
}
break;
...
編集:ハンスのアドバイスを適用しましたが、それでも同じ問題が発生します(printf()は印刷されず、例外が発生します):
case WM_DEVICECHANGE:
{
if (wParam != DBT_DEVNODES_CHANGED) {
printf("1: \n");
break;
}
PDEV_BROADCAST_HDR h = (PDEV_BROADCAST_HDR) lParam;
if (h->dbch_devicetype != DBT_DEVTYP_DEVICEINTERFACE) {
printf("2\n");
break;
}
PDEV_BROADCAST_DEVICEINTERFACE b = (PDEV_BROADCAST_DEVICEINTERFACE) h;
// When accessing the below GUID I get an error:
// "First-chance exception at 0x001c1a9d in TEST GUID Error.exe: 0xC0000005: Access violation writing location 0x00000000."
printf("GUID: Data1: %x, Data2: %x, Data3: %x, Data4: %s \n", b->dbcc_classguid.Data1,
b->dbcc_classguid.Data2, b->dbcc_classguid.Data3, (char*)b->dbcc_classguid.Data4);
// The following function ALWAYS fails.
// GetLastError() gives the error "The parameter is incorrect"
HDEVINFO hDevInfo = SetupDiGetClassDevs(&b->dbcc_classguid, NULL, NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
if (hDevInfo == INVALID_HANDLE_VALUE) {
//printf("hDevInfo == INVALID_HANDLE_VALUE \n");
outputLastError(_T("hDevInfo == INVALID_HANDLE_VALUE \n"));
}
}
break;