1

以下の編集を参照してください。

テレビやプロジェクターなどの接続デバイスのクラス/インターフェイス 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;
4

0 に答える 0