1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS FS utility tool 4 * FILE: modules/rosapps/applications/cmdutils/vcdcli/vcdcli.c 5 * PURPOSE: Virtual CD-ROM management application 6 * PROGRAMMERS: Pierre Schweitzer <pierre@reactos.org> 7 */ 8 9 #define WIN32_NO_STATUS 10 #include <windef.h> 11 #include <winbase.h> 12 #include <winsvc.h> 13 #include <winreg.h> 14 #include <ndk/rtltypes.h> 15 #include <ndk/rtlfuncs.h> 16 #include <tchar.h> 17 #include <stdio.h> 18 19 #include <vcdioctl.h> 20 21 #define IOCTL_CDROM_BASE FILE_DEVICE_CD_ROM 22 #define IOCTL_CDROM_EJECT_MEDIA CTL_CODE(IOCTL_CDROM_BASE, 0x0202, METHOD_BUFFERED, FILE_READ_ACCESS) 23 24 void 25 PrintUsage(int type) 26 { 27 if (type == 0) 28 { 29 _ftprintf(stdout, _T("vcdcli usage:\n")); 30 _ftprintf(stdout, _T("\tlist [/a]: list all the virtual drives\n")); 31 _ftprintf(stdout, _T("\tcreate: create a virtual drive\n")); 32 _ftprintf(stdout, _T("\tmount X path: mount path image on X virtual drive\n")); 33 _ftprintf(stdout, _T("\tremount X: remount image on X virtual drive\n")); 34 _ftprintf(stdout, _T("\tremount X: remount image on X virtual drive\n")); 35 _ftprintf(stdout, _T("\teject X: eject image on X virtual drive\n")); 36 _ftprintf(stdout, _T("\tremove X: remove virtual drive X\n")); 37 } 38 else if (type == 1) 39 { 40 _ftprintf(stdout, _T("mount usage:\n")); 41 _ftprintf(stdout, _T("\tmount <drive letter> <path.iso> [/u] [/j] [/p]\n")); 42 _ftprintf(stdout, _T("\tMount the ISO image given in <path.iso> on the previously created virtual drive <drive letter>\n")); 43 _ftprintf(stdout, _T("\t\tDo not use colon for drive letter\n")); 44 _ftprintf(stdout, _T("\t\tUse /u to make UDF volumes not appear as such\n")); 45 _ftprintf(stdout, _T("\t\tUse /j to make Joliet volumes not appear as such\n")); 46 _ftprintf(stdout, _T("\t\tUse /p to make the mounting persistent\n")); 47 } 48 else if (type == 2) 49 { 50 _ftprintf(stdout, _T("remount usage:\n")); 51 _ftprintf(stdout, _T("\tremount <drive letter>\n")); 52 _ftprintf(stdout, _T("\tRemount the ISO image that was previously mounted on the virtual drive <drive letter>\n")); 53 _ftprintf(stdout, _T("\t\tDo not use colon for drive letter\n")); 54 } 55 else if (type == 3) 56 { 57 _ftprintf(stdout, _T("eject usage:\n")); 58 _ftprintf(stdout, _T("\teject <drive letter>\n")); 59 _ftprintf(stdout, _T("\tEjects the ISO image that is mounted on the virtual drive <drive letter>\n")); 60 _ftprintf(stdout, _T("\t\tDo not use colon for drive letter\n")); 61 } 62 else if (type == 4) 63 { 64 _ftprintf(stdout, _T("remove usage:\n")); 65 _ftprintf(stdout, _T("\tremove <drive letter>\n")); 66 _ftprintf(stdout, _T("\tRemoves the virtual drive <drive letter> making it no longer usable\n")); 67 _ftprintf(stdout, _T("\t\tDo not use colon for drive letter\n")); 68 } 69 } 70 71 HANDLE 72 OpenLetter(WCHAR Letter) 73 { 74 TCHAR Device[255]; 75 76 /* Make name */ 77 _stprintf(Device, _T("\\\\.\\%c:"), Letter); 78 79 /* And open */ 80 return CreateFile(Device, GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE, 81 NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); 82 } 83 84 BOOLEAN 85 StartDriver(VOID) 86 { 87 SC_HANDLE hMgr, hSvc; 88 89 /* Open the SC manager */ 90 hMgr = OpenSCManager(NULL, NULL, SC_MANAGER_CREATE_SERVICE); 91 if (hMgr == NULL) 92 { 93 _ftprintf(stderr, _T("Failed opening service manager: %x\n"), GetLastError()); 94 return FALSE; 95 } 96 97 /* Open the service matching our driver */ 98 hSvc = OpenService(hMgr, _T("Vcdrom"), SERVICE_START); 99 if (hSvc == NULL) 100 { 101 _ftprintf(stderr, _T("Failed opening service: %x\n"), GetLastError()); 102 CloseServiceHandle(hMgr); 103 return FALSE; 104 } 105 106 /* Start it */ 107 /* FIXME: improve */ 108 StartService(hSvc, 0, NULL); 109 110 /* Cleanup */ 111 CloseServiceHandle(hSvc); 112 CloseServiceHandle(hMgr); 113 114 /* Always return true when service exists 115 * We don't care whether it was running or not 116 * We just need it 117 */ 118 return TRUE; 119 } 120 121 HANDLE 122 OpenMaster(VOID) 123 { 124 /* We'll always talk to master first, so we start it here */ 125 if (!StartDriver()) 126 { 127 return INVALID_HANDLE_VALUE; 128 } 129 130 /* And then, open it */ 131 return CreateFile(_T("\\\\.\\\\VirtualCdRom"), GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE, 132 NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); 133 } 134 135 BOOLEAN 136 IsLetterOwned(WCHAR Letter) 137 { 138 HANDLE hDev; 139 BOOLEAN Res; 140 DRIVES_LIST Drives; 141 DWORD i, BytesRead; 142 143 /* We've to deal with driver */ 144 hDev = OpenMaster(); 145 if (hDev == INVALID_HANDLE_VALUE) 146 { 147 _ftprintf(stderr, _T("Failed to open VCD: %x\n"), GetLastError()); 148 return FALSE; 149 } 150 151 /* Get the list of the managed drives */ 152 Res = DeviceIoControl(hDev, IOCTL_VCDROM_ENUMERATE_DRIVES, NULL, 0, &Drives, sizeof(Drives), &BytesRead, NULL); 153 if (!Res) 154 { 155 _ftprintf(stderr, _T("Failed to enumerate drives: %x\n"), GetLastError()); 156 CloseHandle(hDev); 157 return FALSE; 158 } 159 160 /* Don't leak ;-) */ 161 CloseHandle(hDev); 162 163 /* Do we find our letter in the list? */ 164 for (i = 0; i < Drives.Count; ++i) 165 { 166 if (Drives.Drives[i] == Letter) 167 { 168 break; 169 } 170 } 171 172 /* No? Fail */ 173 if (i == Drives.Count) 174 { 175 _ftprintf(stderr, _T("%c is not a drive owned by VCD\n"), Letter); 176 return FALSE; 177 } 178 179 /* Otherwise, that's fine! */ 180 return TRUE; 181 } 182 183 FORCEINLINE 184 DWORD 185 Min(DWORD a, DWORD b) 186 { 187 return (a > b ? b : a); 188 } 189 190 int 191 __cdecl 192 _tmain(int argc, const TCHAR *argv[]) 193 { 194 HANDLE hDev; 195 BOOLEAN Res; 196 DWORD BytesRead; 197 198 /* We need a command, at least */ 199 if (argc < 2) 200 { 201 PrintUsage(0); 202 return 1; 203 } 204 205 /* List will display all the managed drives */ 206 if (_tcscmp(argv[1], _T("list")) == 0) 207 { 208 DWORD i; 209 BOOLEAN All; 210 DRIVES_LIST Drives; 211 212 /* Open the driver for query */ 213 hDev = OpenMaster(); 214 if (hDev == INVALID_HANDLE_VALUE) 215 { 216 _ftprintf(stderr, _T("Failed to open VCD: %x\n"), GetLastError()); 217 return 1; 218 } 219 220 /* Query the virtual drives */ 221 Res = DeviceIoControl(hDev, IOCTL_VCDROM_ENUMERATE_DRIVES, NULL, 0, &Drives, sizeof(Drives), &BytesRead, NULL); 222 if (!Res) 223 { 224 _ftprintf(stderr, _T("Failed to create VCD: %x\n"), GetLastError()); 225 CloseHandle(hDev); 226 return 1; 227 } 228 229 /* Done with master */ 230 CloseHandle(hDev); 231 232 /* No drives? Display a pretty message */ 233 if (Drives.Count == 0) 234 { 235 _ftprintf(stdout, _T("No virtual drives\n")); 236 } 237 else 238 { 239 /* Do we have to display all the information? That's '/a' */ 240 All = FALSE; 241 if (argc > 2) 242 { 243 if (_tcscmp(argv[2], _T("/a")) == 0) 244 { 245 All = TRUE; 246 } 247 } 248 249 if (All) 250 { 251 _ftprintf(stdout, _T("Managed drives:\n")); 252 /* For each virtual drive... */ 253 for (i = 0; i < Drives.Count; ++i) 254 { 255 HANDLE hLet; 256 IMAGE_PATH Image; 257 258 /* Display its letter */ 259 _ftprintf(stdout, _T("%c: "), Drives.Drives[i]); 260 261 /* And open it to query more data */ 262 hLet = OpenLetter(Drives.Drives[i]); 263 if (hLet != INVALID_HANDLE_VALUE) 264 { 265 /* Get the image path along with mount status */ 266 Res = DeviceIoControl(hLet, IOCTL_VCDROM_GET_IMAGE_PATH, NULL, 0, &Image, sizeof(Image), &BytesRead, NULL); 267 /* If it succeed */ 268 if (Res) 269 { 270 UNICODE_STRING Path; 271 272 /* Display image if any, otherwise display "no image" */ 273 if (Image.Length != 0) 274 { 275 Path.Length = Image.Length; 276 Path.MaximumLength = Image.Length; 277 Path.Buffer = Image.Path; 278 } 279 else 280 { 281 Path.Length = sizeof(L"no image") - sizeof(UNICODE_NULL); 282 Path.MaximumLength = sizeof(L"no image"); 283 Path.Buffer = L"no image"; 284 } 285 286 /* Print everything including mount status */ 287 _ftprintf(stdout, _T("%wZ, %s"), &Path, (Image.Mounted == 0 ? _T("not mounted") : _T("mounted"))); 288 } 289 290 /* Close drive and move to the next one */ 291 CloseHandle(hLet); 292 } 293 294 /* EOL! */ 295 _ftprintf(stdout, _T("\n")); 296 } 297 } 298 else 299 { 300 /* Basic display, just display drives on a single line */ 301 _ftprintf(stdout, _T("Virtual drives:\n")); 302 for (i = 0; i < Drives.Count; ++i) 303 { 304 _ftprintf(stdout, _T("%c: "), Drives.Drives[i]); 305 } 306 _ftprintf(stdout, _T("\n")); 307 } 308 } 309 } 310 else if (_tcscmp(argv[1], _T("create")) == 0) 311 { 312 WCHAR Letter; 313 314 /* Open driver */ 315 hDev = OpenMaster(); 316 if (hDev == INVALID_HANDLE_VALUE) 317 { 318 _ftprintf(stderr, _T("Failed to open VCD: %x\n"), GetLastError()); 319 return 1; 320 } 321 322 /* Issue the IOCTL */ 323 Res = DeviceIoControl(hDev, IOCTL_VCDROM_CREATE_DRIVE, NULL, 0, &Letter, sizeof(WCHAR), &BytesRead, NULL); 324 if (!Res) 325 { 326 _ftprintf(stderr, _T("Failed to create drive: %x\n"), GetLastError()); 327 CloseHandle(hDev); 328 return 1; 329 } 330 331 /* And display the create drive letter to the user */ 332 _ftprintf(stdout, _T("The virtual drive '%c' has been created\n"), Letter); 333 334 CloseHandle(hDev); 335 } 336 else if (_tcscmp(argv[1], _T("mount")) == 0) 337 { 338 DWORD i; 339 HKEY hKey; 340 HANDLE hFile; 341 WCHAR Letter; 342 BOOLEAN bPersist; 343 TCHAR szBuffer[260]; 344 UNICODE_STRING NtPathName; 345 MOUNT_PARAMETERS MountParams; 346 347 /* We need two args */ 348 if (argc < 4) 349 { 350 PrintUsage(1); 351 return 1; 352 } 353 354 /* First, check letter is OK */ 355 if (!_istalpha(argv[2][0]) || argv[2][1] != 0) 356 { 357 PrintUsage(1); 358 return 1; 359 } 360 361 /* Now, check the ISO image is OK and reachable by the user */ 362 hFile = CreateFile(argv[3], FILE_READ_DATA, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); 363 if (hFile == INVALID_HANDLE_VALUE) 364 { 365 _ftprintf(stderr, _T("Failed to open file: %lu\n"), GetLastError()); 366 return 1; 367 } 368 369 /* Validate the drive is owned by vcdrom */ 370 Letter = _totupper(argv[2][0]); 371 if (!IsLetterOwned(Letter)) 372 { 373 CloseHandle(hFile); 374 return 1; 375 } 376 377 /* Get NT path for the driver */ 378 if (!RtlDosPathNameToNtPathName_U(argv[3], &NtPathName, NULL, NULL)) 379 { 380 _ftprintf(stderr, _T("Failed to convert path\n")); 381 CloseHandle(hFile); 382 return 1; 383 } 384 385 /* Copy it in the parameter structure */ 386 _tcsncpy(MountParams.Path, NtPathName.Buffer, 255); 387 MountParams.Length = Min(NtPathName.Length, 255 * sizeof(WCHAR)); 388 MountParams.Flags = 0; 389 390 /* Do we have to suppress anything? */ 391 bPersist = FALSE; 392 for (i = 4; i < argc; ++i) 393 { 394 /* Make UDF uneffective */ 395 if (_tcscmp(argv[i], _T("/u")) == 0) 396 { 397 MountParams.Flags |= MOUNT_FLAG_SUPP_UDF; 398 } 399 /* Make Joliet uneffective */ 400 else if (_tcscmp(argv[i], _T("/j")) == 0) 401 { 402 MountParams.Flags |= MOUNT_FLAG_SUPP_JOLIET; 403 } 404 /* Should it be persistent? */ 405 else if (_tcscmp(argv[i], _T("/p")) == 0) 406 { 407 bPersist = TRUE; 408 } 409 } 410 411 /* No longer needed */ 412 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathName.Buffer); 413 414 /* Open the drive */ 415 hDev = OpenLetter(Letter); 416 if (hDev == INVALID_HANDLE_VALUE) 417 { 418 _ftprintf(stderr, _T("Failed to open VCD %c: %x\n"), Letter, GetLastError()); 419 CloseHandle(hFile); 420 return 1; 421 } 422 423 /* We have to release image now, the driver will attempt to open it */ 424 CloseHandle(hFile); 425 426 /* Issue the mount IOCTL */ 427 Res = DeviceIoControl(hDev, IOCTL_VCDROM_MOUNT_IMAGE, &MountParams, sizeof(MountParams), NULL, 0, &BytesRead, NULL); 428 if (!Res) 429 { 430 _ftprintf(stderr, _T("Failed to mount %s on %c: %x\n"), argv[3], Letter, GetLastError()); 431 CloseHandle(hDev); 432 return 1; 433 } 434 435 /* Pretty print in case of a success */ 436 _ftprintf(stdout, _T("%s mounted on %c\n"), argv[3], Letter); 437 438 CloseHandle(hDev); 439 440 /* Should it persistent? */ 441 if (bPersist) 442 { 443 /* Create the registry key Device<Letter> */ 444 _stprintf(szBuffer, _T("SYSTEM\\CurrentControlSet\\Services\\Vcdrom\\Parameters\\Device%c"), Letter); 445 if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, szBuffer, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_CREATE_SUB_KEY | KEY_SET_VALUE, NULL, &hKey, NULL) == ERROR_SUCCESS) 446 { 447 /* Set the image path */ 448 _tcsncpy(szBuffer, MountParams.Path, MountParams.Length); 449 szBuffer[MountParams.Length / sizeof(TCHAR)] = 0; 450 RegSetValueExW(hKey, _T("IMAGE"), 0, REG_SZ, (BYTE *)szBuffer, MountParams.Length); 451 452 /* Set the drive letter */ 453 szBuffer[0] = Letter; 454 szBuffer[1] = _T(':'); 455 szBuffer[2] = 0; 456 RegSetValueExW(hKey, _T("DRIVE"), 0, REG_SZ, (BYTE *)szBuffer, 3 * sizeof(TCHAR)); 457 458 RegCloseKey(hKey); 459 } 460 else 461 { 462 _ftprintf(stderr, _T("Failed to make mounting persistent: %x\n"), GetLastError()); 463 } 464 } 465 } 466 else if (_tcscmp(argv[1], _T("remount")) == 0) 467 { 468 WCHAR Letter; 469 470 /* We need an arg */ 471 if (argc < 3) 472 { 473 PrintUsage(2); 474 return 1; 475 } 476 477 /* First, check letter is OK */ 478 if (!_istalpha(argv[2][0]) || argv[2][1] != 0) 479 { 480 PrintUsage(2); 481 return 1; 482 } 483 484 /* Validate the drive is owned by vcdrom */ 485 Letter = _totupper(argv[2][0]); 486 if (!IsLetterOwned(Letter)) 487 { 488 return 1; 489 } 490 491 /* Open the drive */ 492 hDev = OpenLetter(Letter); 493 if (hDev == INVALID_HANDLE_VALUE) 494 { 495 _ftprintf(stderr, _T("Failed to open VCD %c: %x\n"), Letter, GetLastError()); 496 return 1; 497 } 498 499 /* Issue the remount IOCTL */ 500 Res = DeviceIoControl(hDev, IOCTL_STORAGE_LOAD_MEDIA, NULL, 0, NULL, 0, &BytesRead, NULL); 501 if (!Res) 502 { 503 _ftprintf(stderr, _T("Failed to remount media on %c: %x\n"), Letter, GetLastError()); 504 CloseHandle(hDev); 505 return 1; 506 } 507 508 /* Pretty print in case of a success */ 509 _ftprintf(stdout, _T("Media remounted on %c\n"), Letter); 510 511 CloseHandle(hDev); 512 } 513 else if (_tcscmp(argv[1], _T("eject")) == 0) 514 { 515 WCHAR Letter; 516 517 /* We need an arg */ 518 if (argc < 3) 519 { 520 PrintUsage(3); 521 return 1; 522 } 523 524 /* First, check letter is OK */ 525 if (!_istalpha(argv[2][0]) || argv[2][1] != 0) 526 { 527 PrintUsage(3); 528 return 1; 529 } 530 531 /* Validate the drive is owned by vcdrom */ 532 Letter = _totupper(argv[2][0]); 533 if (!IsLetterOwned(Letter)) 534 { 535 return 1; 536 } 537 538 /* Open the drive */ 539 hDev = OpenLetter(Letter); 540 if (hDev == INVALID_HANDLE_VALUE) 541 { 542 _ftprintf(stderr, _T("Failed to open VCD %c: %x\n"), Letter, GetLastError()); 543 return 1; 544 } 545 546 /* Issue the eject IOCTL */ 547 Res = DeviceIoControl(hDev, IOCTL_CDROM_EJECT_MEDIA, NULL, 0, NULL, 0, &BytesRead, NULL); 548 if (!Res) 549 { 550 _ftprintf(stderr, _T("Failed to eject media on %c: %x\n"), Letter, GetLastError()); 551 CloseHandle(hDev); 552 return 1; 553 } 554 555 /* Pretty print in case of a success */ 556 _ftprintf(stdout, _T("Media ejected on %c\n"), Letter); 557 558 CloseHandle(hDev); 559 } 560 else if (_tcscmp(argv[1], _T("remove")) == 0) 561 { 562 WCHAR Letter; 563 564 /* We need an arg */ 565 if (argc < 3) 566 { 567 PrintUsage(4); 568 return 1; 569 } 570 571 /* First, check letter is OK */ 572 if (!_istalpha(argv[2][0]) || argv[2][1] != 0) 573 { 574 PrintUsage(4); 575 return 1; 576 } 577 578 /* Validate the drive is owned by vcdrom */ 579 Letter = _totupper(argv[2][0]); 580 if (!IsLetterOwned(Letter)) 581 { 582 return 1; 583 } 584 585 /* Open the drive */ 586 hDev = OpenLetter(Letter); 587 if (hDev == INVALID_HANDLE_VALUE) 588 { 589 _ftprintf(stderr, _T("Failed to open VCD %c: %x\n"), Letter, GetLastError()); 590 return 1; 591 } 592 593 /* Issue the remove IOCTL */ 594 Res = DeviceIoControl(hDev, IOCTL_VCDROM_DELETE_DRIVE, NULL, 0, NULL, 0, &BytesRead, NULL); 595 if (!Res) 596 { 597 _ftprintf(stderr, _T("Failed to remove virtual drive %c: %x\n"), Letter, GetLastError()); 598 CloseHandle(hDev); 599 return 1; 600 } 601 602 /* Pretty print in case of a success */ 603 _ftprintf(stdout, _T("Virtual drive %c removed\n"), Letter); 604 605 CloseHandle(hDev); 606 } 607 else 608 { 609 PrintUsage(0); 610 } 611 612 return 0; 613 } 614