mirror of
https://github.com/kovidgoyal/calibre.git
synced 2025-06-22 23:11:00 -04:00
Eject individual drives before trying to eject the device. Hopefully, this fixes incomplete ejection with the Nook devices.
424 lines
12 KiB
C
424 lines
12 KiB
C
/*
|
|
* eject.c
|
|
* Copyright (C) 2013 Kovid Goyal <kovid at kovidgoyal.net>
|
|
*
|
|
* Distributed under terms of the GPL3 license.
|
|
*/
|
|
|
|
#include "Windows.h"
|
|
#include <stdio.h>
|
|
#include <wchar.h>
|
|
#include <winioctl.h>
|
|
#include <setupapi.h>
|
|
#include <devguid.h>
|
|
#include <cfgmgr32.h>
|
|
|
|
#define BUFSIZE 4096
|
|
#define LOCK_TIMEOUT 10000 // 10 Seconds
|
|
#define LOCK_RETRIES 20
|
|
|
|
#define BOOL2STR(x) ((x) ? L"True" : L"False")
|
|
|
|
// Error handling {{{
|
|
|
|
static void show_error(LPCWSTR msg) {
|
|
MessageBeep(MB_ICONERROR);
|
|
MessageBoxW(NULL, msg, L"Error", MB_OK|MB_ICONERROR);
|
|
}
|
|
|
|
static void show_detailed_error(LPCWSTR preamble, LPCWSTR msg, int code) {
|
|
LPWSTR buf;
|
|
buf = (LPWSTR)LocalAlloc(LMEM_ZEROINIT, sizeof(WCHAR)*
|
|
(wcslen(msg) + wcslen(preamble) + 80));
|
|
|
|
_snwprintf_s(buf,
|
|
LocalSize(buf) / sizeof(WCHAR), _TRUNCATE,
|
|
L"%s\r\n %s (Error Code: %d)\r\n",
|
|
preamble, msg, code);
|
|
|
|
show_error(buf);
|
|
LocalFree(buf);
|
|
}
|
|
|
|
static void print_detailed_error(LPCWSTR preamble, LPCWSTR msg, int code) {
|
|
fwprintf_s(stderr, L"%s\r\n %s (Error Code: %d)\r\n", preamble, msg, code);
|
|
fflush(stderr);
|
|
}
|
|
|
|
static void show_last_error_crt(LPCWSTR preamble) {
|
|
WCHAR buf[BUFSIZE];
|
|
int err = 0;
|
|
|
|
_get_errno(&err);
|
|
_wcserror_s(buf, BUFSIZE, err);
|
|
show_detailed_error(preamble, buf, err);
|
|
}
|
|
|
|
static void show_last_error(LPCWSTR preamble) {
|
|
WCHAR *msg = NULL;
|
|
DWORD dw = GetLastError();
|
|
|
|
FormatMessageW(
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
NULL,
|
|
dw,
|
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
(LPWSTR)&msg,
|
|
0, NULL );
|
|
|
|
show_detailed_error(preamble, msg, (int)dw);
|
|
}
|
|
|
|
static void print_last_error(LPCWSTR preamble) {
|
|
WCHAR *msg = NULL;
|
|
DWORD dw = GetLastError();
|
|
|
|
FormatMessageW(
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
NULL,
|
|
dw,
|
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
(LPWSTR)&msg,
|
|
0, NULL );
|
|
|
|
print_detailed_error(preamble, msg, (int)dw);
|
|
}
|
|
|
|
// }}}
|
|
|
|
static void print_help() {
|
|
fwprintf_s(stderr, L"Usage: calibre-eject.exe drive-letter1 [drive-letter2 drive-letter3 ...]");
|
|
}
|
|
|
|
static LPWSTR root_path = L"X:\\", device_path = L"X:", volume_access_path = L"\\\\.\\X:";
|
|
static wchar_t dos_device_name[MAX_PATH];
|
|
static UINT drive_type = 0;
|
|
static long device_number = -1;
|
|
static DEVINST dev_inst = 0, dev_inst_parent = 0;
|
|
|
|
// Unmount and eject volumes (drives) {{{
|
|
static HANDLE open_volume(wchar_t drive_letter) {
|
|
DWORD access_flags;
|
|
|
|
switch(drive_type) {
|
|
case DRIVE_REMOVABLE:
|
|
access_flags = GENERIC_READ | GENERIC_WRITE;
|
|
break;
|
|
case DRIVE_CDROM:
|
|
access_flags = GENERIC_READ;
|
|
break;
|
|
default:
|
|
fwprintf_s(stderr, L"Cannot eject %c: Drive type is incorrect.\r\n", drive_letter);
|
|
fflush(stderr);
|
|
return INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
return CreateFileW(volume_access_path, access_flags,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
NULL, OPEN_EXISTING, 0, NULL);
|
|
}
|
|
|
|
static BOOL lock_volume(HANDLE volume) {
|
|
DWORD bytes_returned;
|
|
DWORD sleep_amount = LOCK_TIMEOUT / LOCK_RETRIES;
|
|
int try_count;
|
|
|
|
// Do this in a loop until a timeout period has expired
|
|
for (try_count = 0; try_count < LOCK_RETRIES; try_count++) {
|
|
if (DeviceIoControl(volume,
|
|
FSCTL_LOCK_VOLUME,
|
|
NULL, 0,
|
|
NULL, 0,
|
|
&bytes_returned,
|
|
NULL))
|
|
return TRUE;
|
|
|
|
Sleep(sleep_amount);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static BOOL dismount_volume(HANDLE volume) {
|
|
DWORD bytes_returned;
|
|
|
|
return DeviceIoControl( volume,
|
|
FSCTL_DISMOUNT_VOLUME,
|
|
NULL, 0,
|
|
NULL, 0,
|
|
&bytes_returned,
|
|
NULL);
|
|
}
|
|
|
|
static BOOL disable_prevent_removal_of_volume(HANDLE volume) {
|
|
DWORD bytes_returned;
|
|
PREVENT_MEDIA_REMOVAL PMRBuffer;
|
|
|
|
PMRBuffer.PreventMediaRemoval = FALSE;
|
|
|
|
return DeviceIoControl( volume,
|
|
IOCTL_STORAGE_MEDIA_REMOVAL,
|
|
&PMRBuffer, sizeof(PREVENT_MEDIA_REMOVAL),
|
|
NULL, 0,
|
|
&bytes_returned,
|
|
NULL);
|
|
}
|
|
|
|
static BOOL auto_eject_volume(HANDLE volume) {
|
|
DWORD bytes_returned;
|
|
|
|
return DeviceIoControl( volume,
|
|
IOCTL_STORAGE_EJECT_MEDIA,
|
|
NULL, 0,
|
|
NULL, 0,
|
|
&bytes_returned,
|
|
NULL);
|
|
}
|
|
|
|
static BOOL unmount_drive(wchar_t drive_letter, BOOL *remove_safely, BOOL *auto_eject) {
|
|
// Unmount the drive identified by drive_letter. Code adapted from:
|
|
// http://support.microsoft.com/kb/165721
|
|
HANDLE volume;
|
|
*remove_safely = FALSE; *auto_eject = FALSE;
|
|
|
|
volume = open_volume(drive_letter);
|
|
if (volume == INVALID_HANDLE_VALUE) return FALSE;
|
|
|
|
// Lock and dismount the volume.
|
|
if (lock_volume(volume) && dismount_volume(volume)) {
|
|
*remove_safely = TRUE;
|
|
|
|
// Set prevent removal to false and eject the volume.
|
|
if (disable_prevent_removal_of_volume(volume) && auto_eject_volume(volume))
|
|
*auto_eject = TRUE;
|
|
}
|
|
CloseHandle(volume);
|
|
return TRUE;
|
|
|
|
}
|
|
// }}}
|
|
|
|
// Eject USB device {{{
|
|
static void get_device_number(wchar_t drive_letter) {
|
|
HANDLE volume;
|
|
DWORD bytes_returned = 0;
|
|
STORAGE_DEVICE_NUMBER sdn;
|
|
|
|
volume = CreateFileW(volume_access_path, 0,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
NULL, OPEN_EXISTING, 0, NULL);
|
|
if (volume == INVALID_HANDLE_VALUE) {
|
|
print_last_error(L"Failed to open volume while getting device number");
|
|
return;
|
|
}
|
|
|
|
if (DeviceIoControl(volume,
|
|
IOCTL_STORAGE_GET_DEVICE_NUMBER,
|
|
NULL, 0, &sdn, sizeof(sdn),
|
|
&bytes_returned, NULL))
|
|
device_number = sdn.DeviceNumber;
|
|
CloseHandle(volume);
|
|
}
|
|
|
|
static DEVINST get_dev_inst_by_device_number(long device_number, UINT drive_type, LPWSTR dos_device_name) {
|
|
GUID *guid;
|
|
HDEVINFO dev_info;
|
|
DWORD index, bytes_returned;
|
|
BOOL bRet, is_floppy;
|
|
BYTE Buf[1024];
|
|
PSP_DEVICE_INTERFACE_DETAIL_DATA pspdidd;
|
|
long res;
|
|
HANDLE drive;
|
|
STORAGE_DEVICE_NUMBER sdn;
|
|
SP_DEVICE_INTERFACE_DATA spdid;
|
|
SP_DEVINFO_DATA spdd;
|
|
DWORD size;
|
|
|
|
is_floppy = (wcsstr(dos_device_name, L"\\Floppy") != NULL); // is there a better way?
|
|
|
|
switch (drive_type) {
|
|
case DRIVE_REMOVABLE:
|
|
guid = ( (is_floppy) ? (GUID*)&GUID_DEVINTERFACE_FLOPPY : (GUID*)&GUID_DEVINTERFACE_DISK );
|
|
break;
|
|
case DRIVE_FIXED:
|
|
guid = (GUID*)&GUID_DEVINTERFACE_DISK;
|
|
break;
|
|
case DRIVE_CDROM:
|
|
guid = (GUID*)&GUID_DEVINTERFACE_CDROM;
|
|
break;
|
|
default:
|
|
fwprintf_s(stderr, L"Invalid drive type at line: %d\r\n", __LINE__);
|
|
fflush(stderr);
|
|
return 0;
|
|
}
|
|
|
|
// Get device interface info set handle
|
|
// for all devices attached to system
|
|
dev_info = SetupDiGetClassDevs(guid, NULL, NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE);
|
|
|
|
if (dev_info == INVALID_HANDLE_VALUE) {
|
|
fwprintf_s(stderr, L"Failed to setup class devs at line: %d\r\n", __LINE__);
|
|
fflush(stderr);
|
|
return 0;
|
|
}
|
|
|
|
// Retrieve a context structure for a device interface
|
|
// of a device information set.
|
|
index = 0;
|
|
bRet = FALSE;
|
|
|
|
pspdidd = (PSP_DEVICE_INTERFACE_DETAIL_DATA)Buf;
|
|
spdid.cbSize = sizeof(spdid);
|
|
|
|
while ( TRUE ) {
|
|
bRet = SetupDiEnumDeviceInterfaces(dev_info, NULL,
|
|
guid, index, &spdid);
|
|
if ( !bRet ) break;
|
|
|
|
size = 0;
|
|
SetupDiGetDeviceInterfaceDetail(dev_info,
|
|
&spdid, NULL, 0, &size, NULL);
|
|
|
|
if ( size!=0 && size<=sizeof(Buf) ) {
|
|
pspdidd->cbSize = sizeof(*pspdidd); // 5 Bytes!
|
|
|
|
ZeroMemory((PVOID)&spdd, sizeof(spdd));
|
|
spdd.cbSize = sizeof(spdd);
|
|
|
|
res = SetupDiGetDeviceInterfaceDetail(dev_info, &spdid, pspdidd, size, &size, &spdd);
|
|
if ( res ) {
|
|
drive = CreateFile(pspdidd->DevicePath,0,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
NULL, OPEN_EXISTING, 0, NULL);
|
|
if ( drive != INVALID_HANDLE_VALUE ) {
|
|
bytes_returned = 0;
|
|
res = DeviceIoControl(drive,
|
|
IOCTL_STORAGE_GET_DEVICE_NUMBER,
|
|
NULL, 0, &sdn, sizeof(sdn),
|
|
&bytes_returned, NULL);
|
|
if ( res ) {
|
|
if ( device_number == (long)sdn.DeviceNumber ) {
|
|
CloseHandle(drive);
|
|
SetupDiDestroyDeviceInfoList(dev_info);
|
|
return spdd.DevInst;
|
|
}
|
|
}
|
|
CloseHandle(drive);
|
|
}
|
|
}
|
|
}
|
|
index++;
|
|
}
|
|
|
|
SetupDiDestroyDeviceInfoList(dev_info);
|
|
fwprintf_s(stderr, L"Invalid device number at line: %d\r\n", __LINE__);
|
|
fflush(stderr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void get_parent_device(wchar_t drive_letter) {
|
|
get_device_number(drive_letter);
|
|
if (device_number == -1) return;
|
|
if (QueryDosDeviceW(device_path, dos_device_name, MAX_PATH) == 0) {
|
|
print_last_error(L"Failed to query DOS device name");
|
|
return;
|
|
}
|
|
|
|
dev_inst = get_dev_inst_by_device_number(device_number,
|
|
drive_type, dos_device_name);
|
|
if (dev_inst == 0) {
|
|
fwprintf_s(stderr, L"Failed to get device by device number");
|
|
fflush(stderr);
|
|
return;
|
|
}
|
|
if (CM_Get_Parent(&dev_inst_parent, dev_inst, 0) != CR_SUCCESS) {
|
|
fwprintf_s(stderr, L"Failed to get device parent from CM");
|
|
fflush(stderr);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static int eject_device() {
|
|
int tries;
|
|
CONFIGRET res;
|
|
PNP_VETO_TYPE VetoType;
|
|
WCHAR VetoNameW[MAX_PATH];
|
|
BOOL success;
|
|
|
|
for ( tries = 0; tries < 3; tries++ ) {
|
|
VetoNameW[0] = 0;
|
|
|
|
res = CM_Request_Device_EjectW(dev_inst_parent,
|
|
&VetoType, VetoNameW, MAX_PATH, 0);
|
|
|
|
success = (res==CR_SUCCESS &&
|
|
VetoType==PNP_VetoTypeUnknown);
|
|
if ( success ) {
|
|
break;
|
|
}
|
|
|
|
Sleep(500); // required to give the next tries a chance!
|
|
}
|
|
if (!success) {
|
|
fwprintf_s(stderr, L"CM_Request_Device_Eject failed after three tries\r\n");
|
|
fflush(stderr);
|
|
}
|
|
|
|
return (success) ? 0 : 1;
|
|
}
|
|
|
|
// }}}
|
|
|
|
int wmain(int argc, wchar_t *argv[ ]) {
|
|
int i = 0;
|
|
wchar_t drive_letter;
|
|
BOOL remove_safely, auto_eject;
|
|
|
|
// Validate command line arguments
|
|
if (argc < 2) { print_help(); return 1; }
|
|
for (i = 1; i < argc; i++) {
|
|
if (wcsnlen_s(argv[i], 2) != 1) { print_help(); return 1; }
|
|
}
|
|
|
|
// Unmount all mounted volumes and eject volume media
|
|
for (i = 1; i < argc; i++) {
|
|
drive_letter = *argv[i];
|
|
root_path[0] = drive_letter;
|
|
device_path[0] = drive_letter;
|
|
volume_access_path[4] = drive_letter;
|
|
drive_type = GetDriveTypeW(root_path);
|
|
if (i == 1 && device_number == -1) {
|
|
get_parent_device(drive_letter);
|
|
}
|
|
if (device_number != -1) {
|
|
unmount_drive(drive_letter, &remove_safely, &auto_eject);
|
|
fwprintf_s(stdout, L"Unmounting: %c: Remove safely: %s Media Ejected: %s\r\n",
|
|
drive_letter, BOOL2STR(remove_safely), BOOL2STR(auto_eject));
|
|
fflush(stdout);
|
|
}
|
|
}
|
|
|
|
// Eject the parent USB device
|
|
if (device_number == -1) {
|
|
fwprintf_s(stderr, L"Cannot eject, failed to get device number\r\n");
|
|
fflush(stderr);
|
|
return 1;
|
|
}
|
|
|
|
if (dev_inst_parent == 0) {
|
|
fwprintf_s(stderr, L"Cannot eject, failed to get device parent\r\n");
|
|
fflush(stderr);
|
|
return 1;
|
|
}
|
|
|
|
return eject_device();
|
|
}
|
|
|
|
|