Elaborazione di una richiesta per rimuovere un dispositivo
Un'applicazione riceve un evento del dispositivo DBT_DEVICEQUERYREMOVE quando una funzionalità nel sistema ha deciso di rimuovere un dispositivo specificato. Quando l'applicazione riceve questo evento, deve determinare se usa il dispositivo specificato e annullare o preparare la rimozione.
Nell'esempio seguente un'applicazione mantiene un handle aperto, hFile, nel file o nel dispositivo rappresentato da FileName. L'applicazione esegue la registrazione per la notifica degli eventi del dispositivo nel dispositivo sottostante chiamando la funzione RegisterDeviceNotification usando un filtro di notifica del tipo DBT_DEVTYP_HANDLE e specificando la variabile hFile nel membro dbch_handle del filtro.
L'applicazione elabora l'evento del dispositivo DBT_DEVICEQUERYREMOVE chiudendo l'handle di file aperto al dispositivo da rimuovere. Nel caso in cui la rimozione del dispositivo venga annullata, l'applicazione elabora l'evento del dispositivo DBT_DEVICEQUERYREMOVEFAILED per riaprire l'handle al dispositivo. Dopo che il dispositivo è stato rimosso dal sistema, l'applicazione elabora gli eventi del dispositivo DBT_DEVICEREMOVECOMPLETE e DBT_DEVICEREMOVEPENDING annullando la registrazione dell'handle di notifica per il dispositivo e chiudendo eventuali handle ancora aperti al dispositivo.
#include <windows.h>
#include <dbt.h>
#include <strsafe.h>
// ...
INT_PTR WINAPI WinProcCallback( HWND hWnd,
UINT message,
WPARAM wParam,
LPARAM lParam )
{
LPCTSTR FileName = NULL; // path to the file or device of interest
HANDLE hFile = INVALID_HANDLE_VALUE; // handle to the file or device
PDEV_BROADCAST_HDR pDBHdr;
PDEV_BROADCAST_HANDLE pDBHandle;
TCHAR szMsg[80];
switch (message)
{
//...
case WM_DEVICECHANGE:
switch (wParam)
{
case DBT_DEVICEQUERYREMOVE:
pDBHdr = (PDEV_BROADCAST_HDR) lParam;
switch (pDBHdr->dbch_devicetype)
{
case DBT_DEVTYP_HANDLE:
// A request has been made to remove the device;
// close any open handles to the file or device
pDBHandle = (PDEV_BROADCAST_HANDLE) pDBHdr;
if (hFile != INVALID_HANDLE_VALUE)
{
CloseHandle(hFile);
hFile = INVALID_HANDLE_VALUE;
}
}
return TRUE;
case DBT_DEVICEQUERYREMOVEFAILED:
pDBHdr = (PDEV_BROADCAST_HDR) lParam;
switch (pDBHdr->dbch_devicetype)
{
case DBT_DEVTYP_HANDLE:
// Removal of the device has failed;
// reopen a handle to the file or device
pDBHandle = (PDEV_BROADCAST_HANDLE) pDBHdr;
hFile = CreateFile(FileName,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);
if (hFile == INVALID_HANDLE_VALUE)
{
StringCchPrintf( szMsg, sizeof(szMsg)/sizeof(szMsg[0]),
TEXT("CreateFile failed: %lx.\n"),
GetLastError());
MessageBox(hWnd, szMsg, TEXT("CreateFile"), MB_OK);
}
}
return TRUE;
case DBT_DEVICEREMOVEPENDING:
pDBHdr = (PDEV_BROADCAST_HDR) lParam;
switch (pDBHdr->dbch_devicetype)
{
case DBT_DEVTYP_HANDLE:
// The device is being removed;
// close any open handles to the file or device
if (hFile != INVALID_HANDLE_VALUE)
{
CloseHandle(hFile);
hFile = INVALID_HANDLE_VALUE;
}
}
return TRUE;
case DBT_DEVICEREMOVECOMPLETE:
pDBHdr = (PDEV_BROADCAST_HDR) lParam;
switch (pDBHdr->dbch_devicetype)
{
case DBT_DEVTYP_HANDLE:
pDBHandle = (PDEV_BROADCAST_HANDLE) pDBHdr;
// The device has been removed from the system;
// unregister its notification handle
UnregisterDeviceNotification(
pDBHandle->dbch_hdevnotify);
// The device has been removed;
// close any remaining open handles to the file or device
if (hFile != INVALID_HANDLE_VALUE)
{
CloseHandle(hFile);
hFile = INVALID_HANDLE_VALUE;
}
}
return TRUE;
default:
return TRUE;
}
}
default:
return TRUE;
}
Argomenti correlati