dxdiagn: Fix string copy behavior with an excessively short buffer in IDxDiagContaine...
[wine/hramrach.git] / programs / eject / eject.c
blob02ff4ba7c284adda53f0a6e04efde6d07c2d6586
1 /*
2 * Eject CDs
4 * Copyright 2005 Alexandre Julliard for CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #define WIN32_LEAN_AND_MEAN
23 #include "config.h"
25 #include <windows.h>
26 #include <winioctl.h>
27 #include <ntddstor.h>
28 #include <stdio.h>
29 #include <stdlib.h>
31 #include "wine/debug.h"
33 WINE_DEFAULT_DEBUG_CHANNEL(eject);
35 /* options */
36 static int unmount_only;
37 static int eject_all;
39 /* wrapper for GetDriveTypeW */
40 static DWORD get_drive_type( WCHAR drive )
42 static const WCHAR rootW[] = {'a',':','\\',0};
43 WCHAR path[16];
45 memcpy( path, rootW, sizeof(rootW) );
46 path[0] = drive;
47 return GetDriveTypeW( path );
50 static BOOL eject_cd( WCHAR drive )
52 static const WCHAR deviceW[] = {'\\','\\','.','\\','a',':',0};
53 PREVENT_MEDIA_REMOVAL removal;
54 WCHAR buffer[16];
55 HANDLE handle;
56 DWORD result;
58 if (get_drive_type( drive ) != DRIVE_CDROM)
60 WINE_MESSAGE( "Drive %c: is not a CD or is not mounted\n", (char)drive );
61 return FALSE;
64 memcpy( buffer, deviceW, sizeof(deviceW) );
65 buffer[4] = drive;
66 handle = CreateFileW( buffer, 0, FILE_SHARE_READ|FILE_SHARE_WRITE,
67 NULL, OPEN_EXISTING, 0, 0 );
68 if (handle == INVALID_HANDLE_VALUE)
70 WINE_MESSAGE( "Cannot open device for drive %c:\n", (char)drive );
71 return FALSE;
74 WINE_TRACE( "ejecting %c:\n", (char)drive );
76 if (!DeviceIoControl( handle, FSCTL_DISMOUNT_VOLUME, NULL, 0, NULL, 0, &result, NULL ))
77 WINE_WARN( "FSCTL_DISMOUNT_VOLUME failed with err %d\n", GetLastError() );
79 removal.PreventMediaRemoval = FALSE;
80 if (!DeviceIoControl( handle, IOCTL_STORAGE_MEDIA_REMOVAL, &removal, sizeof(removal), NULL, 0, &result, NULL ))
81 WINE_WARN( "IOCTL_STORAGE_MEDIA_REMOVAL failed with err %d\n", GetLastError() );
83 if (!unmount_only)
85 if (!DeviceIoControl( handle, IOCTL_STORAGE_EJECT_MEDIA, NULL, 0, NULL, 0, &result, NULL ))
86 WINE_WARN( "IOCTL_STORAGE_EJECT_MEDIA failed with err %d\n", GetLastError() );
89 CloseHandle( handle );
90 return TRUE;
93 /* find the CD drive, and die if we find more than one */
94 static WCHAR find_cd_drive(void)
96 WCHAR ret = 0, drive;
98 for (drive = 'c'; drive <= 'z'; drive++)
100 if (get_drive_type( drive ) != DRIVE_CDROM) continue;
101 if (ret)
103 WINE_MESSAGE( "Multiple CD drives found (%c: and %c:), you need to specify the one you want.\n",
104 (char)ret, (char)drive );
105 exit(1);
107 ret = drive;
109 return ret;
112 static void usage(void)
114 WINE_MESSAGE( "Usage: eject [-u] [-a] [-h] [x:]...\n" );
115 WINE_MESSAGE( " -a Eject all the CD drives we find\n" );
116 WINE_MESSAGE( " -h Display this help message\n" );
117 WINE_MESSAGE( " -u Unmount only, don't eject the CD\n" );
118 WINE_MESSAGE( " x: Eject drive x:\n" );
119 exit(1);
122 static void parse_options( int *argc, char *argv[] )
124 int i;
125 char *opt;
127 for (i = 1; i < *argc; i++)
129 if (argv[i][0] != '-')
131 /* check for valid drive argument */
132 if (strlen(argv[i]) != 2 || argv[i][1] != ':') usage();
133 continue;
135 for (opt = argv[i] + 1; *opt; opt++) switch(*opt)
137 case 'a': eject_all = 1; break;
138 case 'u': unmount_only = 1; break;
139 case 'h': usage(); break;
140 default:
141 WINE_MESSAGE( "Unknown option -%c\n", *opt );
142 usage();
144 memmove( argv + i, argv + i + 1, (*argc - i) * sizeof(*argv) );
145 (*argc)--;
146 i--;
150 int main( int argc, char *argv[] )
152 parse_options( &argc, argv );
154 if (eject_all)
156 WCHAR drive;
158 for (drive = 'c'; drive <= 'z'; drive++)
160 if (get_drive_type( drive ) != DRIVE_CDROM) continue;
161 if (!eject_cd( drive )) exit(1);
164 else if (argc > 1)
166 int i;
168 for (i = 1; i < argc; i++)
169 if (!eject_cd( argv[i][0] )) exit(1);
171 else
173 WCHAR drive = find_cd_drive();
175 if (!drive)
177 WINE_MESSAGE( "No CD drive found\n" );
178 exit(1);
180 if (!eject_cd( drive )) exit(1);
182 exit(0);