2011-12-01 17:54:35 +00:00
|
|
|
/*
|
2011-12-05 11:36:02 +00:00
|
|
|
* Rufus: The Reliable USB Formatting Utility
|
2011-12-01 17:54:35 +00:00
|
|
|
* Drive access function calls
|
2014-01-05 01:39:41 +00:00
|
|
|
* Copyright © 2011-2014 Pete Batard <pete@akeo.ie>
|
2011-12-01 17:54:35 +00:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
#ifdef _CRTDBG_MAP_ALLOC
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <crtdbg.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
#include "msapi_utf8.h"
|
|
|
|
#include "rufus.h"
|
2014-01-05 01:39:41 +00:00
|
|
|
#include "drive.h"
|
2011-12-01 17:54:35 +00:00
|
|
|
#include "resource.h"
|
2013-01-22 02:40:43 +00:00
|
|
|
#include "sys_types.h"
|
2014-01-05 01:39:41 +00:00
|
|
|
#include "br.h"
|
|
|
|
#include "fat16.h"
|
|
|
|
#include "fat32.h"
|
|
|
|
#include "ntfs.h"
|
2013-10-15 21:58:27 +00:00
|
|
|
#include "localization.h"
|
2011-12-01 17:54:35 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Globals
|
|
|
|
*/
|
|
|
|
RUFUS_DRIVE_INFO SelectedDrive;
|
|
|
|
|
2013-11-13 01:07:32 +00:00
|
|
|
// TODO: add a DetectSectorSize()?
|
|
|
|
// http://msdn.microsoft.com/en-us/library/ff800831.aspx
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
/*
|
|
|
|
* Working with drive indexes quite risky (left unchecked,inadvertently passing 0 as
|
|
|
|
* index would return a handle to C:, which we might then proceed to unknowingly
|
|
|
|
* clear the MBR of!), so we mitigate the risk by forcing our indexes to belong to
|
|
|
|
* the specific range [DRIVE_INDEX_MIN; DRIVE_INDEX_MAX].
|
|
|
|
*/
|
|
|
|
#define CheckDriveIndex(DriveIndex) do { \
|
|
|
|
if ((DriveIndex < DRIVE_INDEX_MIN) || (DriveIndex > DRIVE_INDEX_MAX)) { \
|
|
|
|
uprintf("WARNING: Bad index value. Please check the code!\n"); \
|
|
|
|
goto out; \
|
|
|
|
} \
|
|
|
|
DriveIndex -= DRIVE_INDEX_MIN; } while (0)
|
|
|
|
|
2011-12-01 17:54:35 +00:00
|
|
|
/*
|
|
|
|
* Open a drive or volume with optional write and lock access
|
2013-04-07 23:10:58 +00:00
|
|
|
* Return INVALID_HANDLE_VALUE (/!\ which is DIFFERENT from NULL /!\) on failure.
|
2011-12-01 17:54:35 +00:00
|
|
|
*/
|
2013-04-07 23:10:58 +00:00
|
|
|
static HANDLE GetHandle(char* Path, BOOL bWriteAccess, BOOL bLockDrive)
|
2011-12-01 17:54:35 +00:00
|
|
|
{
|
2013-06-25 17:39:07 +00:00
|
|
|
int i;
|
2011-12-01 17:54:35 +00:00
|
|
|
DWORD size;
|
|
|
|
HANDLE hDrive = INVALID_HANDLE_VALUE;
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
if (Path == NULL)
|
|
|
|
goto out;
|
|
|
|
hDrive = CreateFileA(Path, GENERIC_READ|(bWriteAccess?GENERIC_WRITE:0),
|
2013-11-23 22:39:54 +00:00
|
|
|
FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
|
2013-04-07 23:10:58 +00:00
|
|
|
if (hDrive == INVALID_HANDLE_VALUE) {
|
|
|
|
uprintf("Could not open drive %s: %s\n", Path, WindowsErrorString());
|
|
|
|
goto out;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
if (bWriteAccess) {
|
|
|
|
uprintf("Caution: Opened drive %s for write access\n", Path);
|
|
|
|
}
|
|
|
|
|
2013-06-25 17:39:07 +00:00
|
|
|
if (bLockDrive) {
|
2013-11-23 22:39:54 +00:00
|
|
|
if (DeviceIoControl(hDrive, FSCTL_ALLOW_EXTENDED_DASD_IO, NULL, 0, NULL, 0, &size, NULL)) {
|
|
|
|
uprintf("I/O boundary checks disabled\n");
|
|
|
|
}
|
|
|
|
|
2013-06-25 17:39:07 +00:00
|
|
|
for (i = 0; i < DRIVE_ACCESS_RETRIES; i++) {
|
|
|
|
if (DeviceIoControl(hDrive, FSCTL_LOCK_VOLUME, NULL, 0, NULL, 0, &size, NULL))
|
|
|
|
goto out;
|
|
|
|
if (IS_ERROR(FormatStatus)) // User cancel
|
|
|
|
break;
|
|
|
|
Sleep(DRIVE_ACCESS_TIMEOUT/DRIVE_ACCESS_RETRIES);
|
|
|
|
}
|
|
|
|
// If we reached this section, either we didn't manage to get a lock or the user cancelled
|
2013-04-07 23:10:58 +00:00
|
|
|
uprintf("Could not get exclusive access to device %s: %s\n", Path, WindowsErrorString());
|
|
|
|
safe_closehandle(hDrive);
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return hDrive;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return the path to access the physical drive, or NULL on error.
|
|
|
|
* The string is allocated and must be freed (to ensure concurrent access)
|
|
|
|
*/
|
|
|
|
char* GetPhysicalName(DWORD DriveIndex)
|
|
|
|
{
|
|
|
|
BOOL success = FALSE;
|
|
|
|
char physical_name[24];
|
|
|
|
|
|
|
|
CheckDriveIndex(DriveIndex);
|
|
|
|
safe_sprintf(physical_name, sizeof(physical_name), "\\\\.\\PHYSICALDRIVE%d", DriveIndex);
|
|
|
|
success = TRUE;
|
|
|
|
out:
|
|
|
|
return (success)?safe_strdup(physical_name):NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return a handle to the physical drive identified by DriveIndex
|
|
|
|
*/
|
|
|
|
HANDLE GetPhysicalHandle(DWORD DriveIndex, BOOL bWriteAccess, BOOL bLockDrive)
|
|
|
|
{
|
|
|
|
HANDLE hPhysical = INVALID_HANDLE_VALUE;
|
|
|
|
char* PhysicalPath = GetPhysicalName(DriveIndex);
|
|
|
|
hPhysical = GetHandle(PhysicalPath, bWriteAccess, bLockDrive);
|
|
|
|
safe_free(PhysicalPath);
|
|
|
|
return hPhysical;
|
|
|
|
}
|
|
|
|
|
2013-11-17 01:39:43 +00:00
|
|
|
/*
|
|
|
|
* Return the first GUID volume name for the associated drive or NULL if not found
|
|
|
|
* See http://msdn.microsoft.com/en-us/library/cc542456.aspx
|
|
|
|
* The returned string is allocated and must be freed
|
|
|
|
* TODO: a drive may have multiple volumes - should we handle those?
|
|
|
|
*/
|
2013-06-25 17:39:07 +00:00
|
|
|
#define suprintf(...) if (!bSilent) uprintf(__VA_ARGS__)
|
|
|
|
char* GetLogicalName(DWORD DriveIndex, BOOL bKeepTrailingBackslash, BOOL bSilent)
|
2013-04-07 23:10:58 +00:00
|
|
|
{
|
|
|
|
BOOL success = FALSE;
|
|
|
|
char volume_name[MAX_PATH];
|
|
|
|
HANDLE hDrive = INVALID_HANDLE_VALUE, hVolume = INVALID_HANDLE_VALUE;
|
|
|
|
size_t len;
|
|
|
|
char path[MAX_PATH];
|
|
|
|
VOLUME_DISK_EXTENTS DiskExtents;
|
|
|
|
DWORD size;
|
2013-11-17 01:39:43 +00:00
|
|
|
UINT drive_type;
|
2013-04-07 23:10:58 +00:00
|
|
|
int i, j;
|
|
|
|
static const char* ignore_device[] = { "\\Device\\CdRom", "\\Device\\Floppy" };
|
2013-05-19 22:02:16 +00:00
|
|
|
static const char* volume_start = "\\\\?\\";
|
2013-04-07 23:10:58 +00:00
|
|
|
|
|
|
|
CheckDriveIndex(DriveIndex);
|
|
|
|
|
|
|
|
for (i=0; hDrive == INVALID_HANDLE_VALUE; i++) {
|
|
|
|
if (i == 0) {
|
|
|
|
hVolume = FindFirstVolumeA(volume_name, sizeof(volume_name));
|
|
|
|
if (hVolume == INVALID_HANDLE_VALUE) {
|
2013-06-25 17:39:07 +00:00
|
|
|
suprintf("Could not access first GUID volume: %s\n", WindowsErrorString());
|
2013-04-07 23:10:58 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!FindNextVolumeA(hVolume, volume_name, sizeof(volume_name))) {
|
|
|
|
if (GetLastError() != ERROR_NO_MORE_FILES) {
|
2013-06-25 17:39:07 +00:00
|
|
|
suprintf("Could not access next GUID volume: %s\n", WindowsErrorString());
|
2013-04-07 23:10:58 +00:00
|
|
|
}
|
|
|
|
goto out;
|
|
|
|
}
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
2013-04-07 23:10:58 +00:00
|
|
|
|
|
|
|
// Sanity checks
|
|
|
|
len = safe_strlen(volume_name);
|
2013-05-19 22:02:16 +00:00
|
|
|
if ((len <= 1) || (safe_strnicmp(volume_name, volume_start, 4) != 0) || (volume_name[len-1] != '\\')) {
|
2013-06-25 17:39:07 +00:00
|
|
|
suprintf("'%s' is not a GUID volume name\n", volume_name);
|
2013-04-07 23:10:58 +00:00
|
|
|
continue;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
2013-04-07 23:10:58 +00:00
|
|
|
|
|
|
|
drive_type = GetDriveTypeA(volume_name);
|
2013-11-14 01:21:50 +00:00
|
|
|
if ((drive_type != DRIVE_REMOVABLE) && (drive_type != DRIVE_FIXED))
|
2013-04-07 23:10:58 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
volume_name[len-1] = 0;
|
|
|
|
|
|
|
|
if (QueryDosDeviceA(&volume_name[4], path, sizeof(path)) == 0) {
|
2013-06-25 17:39:07 +00:00
|
|
|
suprintf("Failed to get device path for GUID volume '%s': %s\n", volume_name, WindowsErrorString());
|
2013-04-07 23:10:58 +00:00
|
|
|
continue;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
2013-04-07 23:10:58 +00:00
|
|
|
|
|
|
|
for (j=0; (j<ARRAYSIZE(ignore_device)) &&
|
|
|
|
(safe_strnicmp(path, ignore_device[j], safe_strlen(ignore_device[j])) != 0); j++);
|
|
|
|
if (j < ARRAYSIZE(ignore_device)) {
|
2013-06-25 17:39:07 +00:00
|
|
|
suprintf("Skipping GUID volume for '%s'\n", path);
|
2013-04-07 23:10:58 +00:00
|
|
|
continue;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
// If we can't have FILE_SHARE_WRITE, forget it
|
2013-11-23 22:39:54 +00:00
|
|
|
hDrive = CreateFileA(volume_name, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
|
2013-04-07 23:10:58 +00:00
|
|
|
if (hDrive == INVALID_HANDLE_VALUE) {
|
2013-06-25 17:39:07 +00:00
|
|
|
suprintf("Could not open GUID volume '%s': %s\n", volume_name, WindowsErrorString());
|
2013-04-07 23:10:58 +00:00
|
|
|
continue;
|
|
|
|
}
|
2012-02-02 19:47:08 +00:00
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
if ((!DeviceIoControl(hDrive, IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS, NULL, 0,
|
|
|
|
&DiskExtents, sizeof(DiskExtents), &size, NULL)) || (size <= 0)) {
|
2013-06-25 17:39:07 +00:00
|
|
|
suprintf("Could not get Disk Extents: %s\n", WindowsErrorString());
|
2013-04-07 23:10:58 +00:00
|
|
|
safe_closehandle(hDrive);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
safe_closehandle(hDrive);
|
|
|
|
if ((DiskExtents.NumberOfDiskExtents >= 1) && (DiskExtents.Extents[0].DiskNumber == DriveIndex)) {
|
|
|
|
if (bKeepTrailingBackslash)
|
|
|
|
volume_name[len-1] = '\\';
|
|
|
|
success = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-02-02 19:47:08 +00:00
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
out:
|
|
|
|
if (hVolume != INVALID_HANDLE_VALUE)
|
|
|
|
FindVolumeClose(hVolume);
|
|
|
|
return (success)?safe_strdup(volume_name):NULL;
|
|
|
|
}
|
2011-12-01 17:54:35 +00:00
|
|
|
|
2013-06-25 17:39:07 +00:00
|
|
|
/* Wait for a logical drive to reappear - Used when a drive has just been repartitioned */
|
|
|
|
BOOL WaitForLogical(DWORD DriveIndex)
|
|
|
|
{
|
|
|
|
DWORD i;
|
|
|
|
char* LogicalPath = NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < DRIVE_ACCESS_RETRIES; i++) {
|
|
|
|
LogicalPath = GetLogicalName(DriveIndex, FALSE, TRUE);
|
|
|
|
if (LogicalPath != NULL) {
|
|
|
|
free(LogicalPath);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
if (IS_ERROR(FormatStatus)) // User cancel
|
|
|
|
return FALSE;
|
|
|
|
Sleep(DRIVE_ACCESS_TIMEOUT/DRIVE_ACCESS_RETRIES);
|
|
|
|
}
|
|
|
|
uprintf("Timeout while waiting for logical drive\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
/*
|
2013-06-25 01:55:25 +00:00
|
|
|
* Obtain a handle to the first logical volume on the disk identified by DriveIndex
|
|
|
|
* Returns INVALID_HANDLE_VALUE on error or NULL if no logical path exists (typical
|
|
|
|
* of unpartitioned drives)
|
2013-04-07 23:10:58 +00:00
|
|
|
*/
|
|
|
|
HANDLE GetLogicalHandle(DWORD DriveIndex, BOOL bWriteAccess, BOOL bLockDrive)
|
|
|
|
{
|
|
|
|
HANDLE hLogical = INVALID_HANDLE_VALUE;
|
2013-06-25 17:39:07 +00:00
|
|
|
char* LogicalPath = GetLogicalName(DriveIndex, FALSE, FALSE);
|
|
|
|
|
2013-06-25 01:55:25 +00:00
|
|
|
if (LogicalPath == NULL) {
|
2013-06-25 17:39:07 +00:00
|
|
|
uprintf("No logical drive found (unpartitioned?)\n");
|
2013-06-25 01:55:25 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
hLogical = GetHandle(LogicalPath, bWriteAccess, bLockDrive);
|
|
|
|
safe_free(LogicalPath);
|
|
|
|
return hLogical;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2014-01-14 20:17:07 +00:00
|
|
|
* Returns the drive letters for all volumes located on the drive identified by DriveIndex,
|
2013-11-17 01:39:43 +00:00
|
|
|
* as well as the drive type. This is used as base for the 2 function calls that follow.
|
2013-04-07 23:10:58 +00:00
|
|
|
*/
|
2014-01-14 20:17:07 +00:00
|
|
|
static BOOL _GetDriveLettersAndType(DWORD DriveIndex, char* drive_letters, UINT* drive_type)
|
2013-04-07 23:10:58 +00:00
|
|
|
{
|
|
|
|
DWORD size;
|
2013-11-14 01:21:50 +00:00
|
|
|
BOOL r = FALSE;
|
2013-04-07 23:10:58 +00:00
|
|
|
STORAGE_DEVICE_NUMBER_REDEF device_number = {0};
|
|
|
|
HANDLE hDrive = INVALID_HANDLE_VALUE;
|
2013-11-17 01:39:43 +00:00
|
|
|
UINT _drive_type;
|
2014-01-14 20:17:07 +00:00
|
|
|
int i = 0;
|
2013-04-07 23:10:58 +00:00
|
|
|
char *drive, drives[26*4]; /* "D:\", "E:\", etc. */
|
|
|
|
char logical_drive[] = "\\\\.\\#:";
|
2013-11-14 01:21:50 +00:00
|
|
|
|
2014-01-14 20:17:07 +00:00
|
|
|
if (drive_letters != NULL)
|
|
|
|
drive_letters[0] = 0;
|
2013-11-17 01:39:43 +00:00
|
|
|
if (drive_type != NULL)
|
|
|
|
*drive_type = DRIVE_UNKNOWN;
|
2013-04-07 23:10:58 +00:00
|
|
|
CheckDriveIndex(DriveIndex);
|
|
|
|
|
|
|
|
size = GetLogicalDriveStringsA(sizeof(drives), drives);
|
|
|
|
if (size == 0) {
|
|
|
|
uprintf("GetLogicalDriveStrings failed: %s\n", WindowsErrorString());
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (size > sizeof(drives)) {
|
|
|
|
uprintf("GetLogicalDriveStrings: buffer too small (required %d vs %d)\n", size, sizeof(drives));
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2013-11-14 01:21:50 +00:00
|
|
|
r = TRUE;
|
2013-04-07 23:10:58 +00:00
|
|
|
for (drive = drives ;*drive; drive += safe_strlen(drive)+1) {
|
|
|
|
if (!isalpha(*drive))
|
|
|
|
continue;
|
|
|
|
*drive = (char)toupper((int)*drive);
|
|
|
|
if (*drive < 'C') {
|
|
|
|
continue;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
2013-04-07 23:10:58 +00:00
|
|
|
|
|
|
|
/* IOCTL_STORAGE_GET_DEVICE_NUMBER's STORAGE_DEVICE_NUMBER.DeviceNumber is
|
|
|
|
not unique! An HDD, a DVD and probably other drives can have the same
|
|
|
|
value there => Use GetDriveType() to filter out unwanted devices.
|
|
|
|
See https://github.com/pbatard/rufus/issues/32 for details. */
|
2013-11-17 01:39:43 +00:00
|
|
|
_drive_type = GetDriveTypeA(drive);
|
2013-11-14 01:21:50 +00:00
|
|
|
|
2013-11-17 01:39:43 +00:00
|
|
|
if ((_drive_type != DRIVE_REMOVABLE) && (_drive_type != DRIVE_FIXED))
|
2013-04-07 23:10:58 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
safe_sprintf(logical_drive, sizeof(logical_drive), "\\\\.\\%c:", drive[0]);
|
2013-11-23 22:39:54 +00:00
|
|
|
hDrive = CreateFileA(logical_drive, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
|
2011-12-01 17:54:35 +00:00
|
|
|
if (hDrive == INVALID_HANDLE_VALUE) {
|
2013-04-07 23:10:58 +00:00
|
|
|
uprintf("Warning: could not open drive %c: %s\n", drive[0], WindowsErrorString());
|
|
|
|
continue;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
2013-04-07 23:10:58 +00:00
|
|
|
|
|
|
|
r = DeviceIoControl(hDrive, IOCTL_STORAGE_GET_DEVICE_NUMBER, NULL,
|
2013-11-14 01:21:50 +00:00
|
|
|
0, &device_number, sizeof(device_number), &size, NULL) && (size > 0);
|
2013-04-07 23:10:58 +00:00
|
|
|
safe_closehandle(hDrive);
|
2013-11-14 01:21:50 +00:00
|
|
|
if (!r) {
|
2013-04-07 23:10:58 +00:00
|
|
|
uprintf("Could not get device number for device %s: %s\n",
|
|
|
|
logical_drive, WindowsErrorString());
|
|
|
|
} else if (device_number.DeviceNumber == DriveIndex) {
|
2014-01-14 20:17:07 +00:00
|
|
|
if (drive_letters != NULL)
|
|
|
|
drive_letters[i++] = *drive;
|
|
|
|
// The drive type should be the same for all volumes, so we can overwrite
|
2013-11-17 01:39:43 +00:00
|
|
|
if (drive_type != NULL)
|
|
|
|
*drive_type = _drive_type;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
out:
|
2014-01-14 20:17:07 +00:00
|
|
|
if (drive_letters != NULL)
|
|
|
|
drive_letters[i] = 0;
|
2013-11-14 01:21:50 +00:00
|
|
|
return r;
|
2013-04-07 23:10:58 +00:00
|
|
|
}
|
|
|
|
|
2013-11-17 01:39:43 +00:00
|
|
|
// Could have used a #define, but this is clearer
|
2014-01-14 20:17:07 +00:00
|
|
|
BOOL GetDriveLetters(DWORD DriveIndex, char* drive_letters)
|
2013-11-17 01:39:43 +00:00
|
|
|
{
|
2014-01-14 20:17:07 +00:00
|
|
|
return _GetDriveLettersAndType(DriveIndex, drive_letters, NULL);
|
2013-11-17 01:39:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// There's already a GetDriveType in the Windows API
|
|
|
|
UINT GetDriveTypeFromIndex(DWORD DriveIndex)
|
|
|
|
{
|
|
|
|
UINT drive_type;
|
2014-01-14 20:17:07 +00:00
|
|
|
_GetDriveLettersAndType(DriveIndex, NULL, &drive_type);
|
2013-11-17 01:39:43 +00:00
|
|
|
return drive_type;
|
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
/*
|
|
|
|
* Return the next unused drive letter from the system
|
|
|
|
*/
|
|
|
|
char GetUnusedDriveLetter(void)
|
|
|
|
{
|
|
|
|
DWORD size;
|
2013-06-03 23:44:49 +00:00
|
|
|
char drive_letter = 'Z'+1, *drive, drives[26*4]; /* "D:\", "E:\", etc. */
|
2013-04-07 23:10:58 +00:00
|
|
|
|
|
|
|
size = GetLogicalDriveStringsA(sizeof(drives), drives);
|
|
|
|
if (size == 0) {
|
|
|
|
uprintf("GetLogicalDriveStrings failed: %s\n", WindowsErrorString());
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (size > sizeof(drives)) {
|
|
|
|
uprintf("GetLogicalDriveStrings: buffer too small (required %d vs %d)\n", size, sizeof(drives));
|
2011-12-01 17:54:35 +00:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
for (drive_letter = 'C'; drive_letter < 'Z'; drive_letter++) {
|
|
|
|
for (drive = drives ;*drive; drive += safe_strlen(drive)+1) {
|
|
|
|
if (!isalpha(*drive))
|
|
|
|
continue;
|
|
|
|
if (drive_letter == (char)toupper((int)*drive))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!*drive)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2011-12-01 17:54:35 +00:00
|
|
|
out:
|
2014-01-14 20:17:07 +00:00
|
|
|
return (drive_letter>'Z')?0:drive_letter;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return the drive letter and volume label
|
2013-04-07 23:10:58 +00:00
|
|
|
* If the drive doesn't have a volume assigned, space is returned for the letter
|
2011-12-01 17:54:35 +00:00
|
|
|
*/
|
2014-01-14 20:17:07 +00:00
|
|
|
BOOL GetDriveLabel(DWORD DriveIndex, char* letters, char** label)
|
2011-12-01 17:54:35 +00:00
|
|
|
{
|
2013-04-07 23:10:58 +00:00
|
|
|
HANDLE hPhysical;
|
2012-03-09 01:38:52 +00:00
|
|
|
DWORD size;
|
|
|
|
char AutorunPath[] = "#:\\autorun.inf", *AutorunLabel = NULL;
|
2012-02-21 19:46:28 +00:00
|
|
|
wchar_t wDrivePath[] = L"#:\\";
|
|
|
|
wchar_t wVolumeLabel[MAX_PATH+1];
|
|
|
|
static char VolumeLabel[MAX_PATH+1];
|
2011-12-01 17:54:35 +00:00
|
|
|
|
|
|
|
*label = STR_NO_LABEL;
|
|
|
|
|
2014-01-14 20:17:07 +00:00
|
|
|
if (!GetDriveLetters(DriveIndex, letters))
|
2013-11-14 01:21:50 +00:00
|
|
|
return FALSE;
|
2014-01-14 20:17:07 +00:00
|
|
|
if (letters[0] == 0) {
|
2013-11-14 01:21:50 +00:00
|
|
|
// Drive without volume assigned - always enabled
|
|
|
|
return TRUE;
|
2013-02-10 21:54:47 +00:00
|
|
|
}
|
2014-01-14 20:17:07 +00:00
|
|
|
// We only care about an autorun.inf if we have a single volume
|
|
|
|
AutorunPath[0] = letters[0];
|
|
|
|
wDrivePath[0] = letters[0];
|
2011-12-01 17:54:35 +00:00
|
|
|
|
2012-03-01 19:19:12 +00:00
|
|
|
// Try to read an extended label from autorun first. Fallback to regular label if not found.
|
2012-03-09 01:38:52 +00:00
|
|
|
// In the case of card readers with no card, users can get an annoying popup asking them
|
|
|
|
// to insert media. Use IOCTL_STORAGE_CHECK_VERIFY to prevent this
|
2013-04-07 23:10:58 +00:00
|
|
|
hPhysical = GetPhysicalHandle(DriveIndex, FALSE, FALSE);
|
2012-03-09 01:38:52 +00:00
|
|
|
if (DeviceIoControl(hPhysical, IOCTL_STORAGE_CHECK_VERIFY, NULL, 0, NULL, 0, &size, NULL))
|
2012-11-12 01:53:34 +00:00
|
|
|
AutorunLabel = get_token_data_file("label", AutorunPath);
|
2012-03-09 01:38:52 +00:00
|
|
|
else if (GetLastError() == ERROR_NOT_READY)
|
2014-01-14 20:17:07 +00:00
|
|
|
uprintf("Ignoring autorun.inf label for drive %c: %s\n", letters[0],
|
2012-03-09 01:38:52 +00:00
|
|
|
(HRESULT_CODE(GetLastError()) == ERROR_NOT_READY)?"No media":WindowsErrorString());
|
|
|
|
safe_closehandle(hPhysical);
|
2012-03-01 19:19:12 +00:00
|
|
|
if (AutorunLabel != NULL) {
|
2014-01-14 20:17:07 +00:00
|
|
|
uprintf("Using autorun.inf label for drive %c: '%s'\n", letters[0], AutorunLabel);
|
2012-03-01 19:19:12 +00:00
|
|
|
strncpy(VolumeLabel, AutorunLabel, sizeof(VolumeLabel));
|
|
|
|
safe_free(AutorunLabel);
|
|
|
|
*label = VolumeLabel;
|
2013-01-09 21:54:28 +00:00
|
|
|
} else if (GetVolumeInformationW(wDrivePath, wVolumeLabel, ARRAYSIZE(wVolumeLabel),
|
2012-02-21 19:46:28 +00:00
|
|
|
NULL, NULL, NULL, NULL, 0) && *wVolumeLabel) {
|
|
|
|
wchar_to_utf8_no_alloc(wVolumeLabel, VolumeLabel, sizeof(VolumeLabel));
|
|
|
|
*label = VolumeLabel;
|
2011-12-01 17:54:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
2011-12-11 02:19:38 +00:00
|
|
|
|
2013-11-17 01:39:43 +00:00
|
|
|
/*
|
|
|
|
* Return the drive size
|
|
|
|
*/
|
|
|
|
uint64_t GetDriveSize(DWORD DriveIndex)
|
|
|
|
{
|
|
|
|
BOOL r;
|
|
|
|
HANDLE hPhysical;
|
|
|
|
DWORD size;
|
|
|
|
BYTE geometry[128];
|
|
|
|
void* disk_geometry = (void*)geometry;
|
|
|
|
PDISK_GEOMETRY_EX DiskGeometry = (PDISK_GEOMETRY_EX)disk_geometry;
|
|
|
|
|
|
|
|
hPhysical = GetPhysicalHandle(DriveIndex, FALSE, FALSE);
|
|
|
|
if (hPhysical == INVALID_HANDLE_VALUE)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
r = DeviceIoControl(hPhysical, IOCTL_DISK_GET_DRIVE_GEOMETRY_EX,
|
|
|
|
NULL, 0, geometry, sizeof(geometry), &size, NULL);
|
|
|
|
safe_closehandle(hPhysical);
|
|
|
|
if (!r || size <= 0)
|
|
|
|
return 0;
|
|
|
|
return DiskGeometry->DiskSize.QuadPart;
|
|
|
|
}
|
|
|
|
|
2013-12-20 18:32:10 +00:00
|
|
|
/*
|
|
|
|
* GET_DRIVE_GEOMETRY is used to tell if there is an actual media
|
|
|
|
*/
|
|
|
|
BOOL IsMediaPresent(DWORD DriveIndex)
|
|
|
|
{
|
|
|
|
BOOL r;
|
|
|
|
HANDLE hPhysical;
|
|
|
|
DWORD size;
|
|
|
|
BYTE geometry[128];
|
|
|
|
|
|
|
|
hPhysical = GetPhysicalHandle(DriveIndex, FALSE, FALSE);
|
|
|
|
r = DeviceIoControl(hPhysical, IOCTL_DISK_GET_DRIVE_GEOMETRY_EX,
|
|
|
|
NULL, 0, geometry, sizeof(geometry), &size, NULL) || (size <= 0);
|
|
|
|
safe_closehandle(hPhysical);
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2014-01-05 01:39:41 +00:00
|
|
|
// TODO: use an (fn,str) table and simplify this whole thing
|
|
|
|
BOOL AnalyzeMBR(HANDLE hPhysicalDrive)
|
|
|
|
{
|
|
|
|
FILE fake_fd = { 0 };
|
|
|
|
|
|
|
|
fake_fd._ptr = (char*)hPhysicalDrive;
|
|
|
|
fake_fd._bufsiz = SelectedDrive.Geometry.BytesPerSector;
|
|
|
|
|
|
|
|
if (!is_br(&fake_fd)) {
|
|
|
|
uprintf("Drive does not have an x86 master boot record\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (is_dos_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a DOS/NT/95A master boot record\n");
|
|
|
|
} else if (is_dos_f2_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a DOS/NT/95A master boot record "
|
|
|
|
"with the undocumented F2 instruction\n");
|
|
|
|
} else if (is_95b_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a Windows 95B/98/98SE/ME master boot record\n");
|
|
|
|
} else if (is_2000_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a Windows 2000/XP/2003 master boot record\n");
|
|
|
|
} else if (is_vista_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a Windows Vista master boot record\n");
|
|
|
|
} else if (is_win7_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a Windows 7 master boot record\n");
|
|
|
|
} else if (is_rufus_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a Rufus master boot record\n");
|
|
|
|
} else if (is_syslinux_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a Syslinux master boot record\n");
|
|
|
|
} else if (is_reactos_mbr(&fake_fd)) {
|
|
|
|
uprintf("Drive has a ReactOS master boot record\n");
|
|
|
|
} else if (is_zero_mbr(&fake_fd)) {
|
2014-01-11 00:19:03 +00:00
|
|
|
uprintf("Drive has a zeroed master boot record\n");
|
2014-01-05 01:39:41 +00:00
|
|
|
} else {
|
|
|
|
uprintf("Drive has an unknown master boot record\n");
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: use an (fn,str) table and simplify this whole thing
|
|
|
|
BOOL AnalyzePBR(HANDLE hLogicalVolume)
|
|
|
|
{
|
|
|
|
FILE fake_fd = { 0 };
|
|
|
|
|
|
|
|
fake_fd._ptr = (char*)hLogicalVolume;
|
|
|
|
fake_fd._bufsiz = SelectedDrive.Geometry.BytesPerSector;
|
|
|
|
|
|
|
|
if (!is_br(&fake_fd)) {
|
|
|
|
uprintf("Volume does not have an x86 partition boot record\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (is_fat_16_br(&fake_fd) || is_fat_32_br(&fake_fd)) {
|
|
|
|
if (entire_fat_16_br_matches(&fake_fd)) {
|
|
|
|
uprintf("Drive has a FAT16 DOS partition boot record\n");
|
|
|
|
} else if (entire_fat_16_fd_br_matches(&fake_fd)) {
|
|
|
|
uprintf("Drive has a FAT16 FreeDOS partition boot record\n");
|
|
|
|
} else if (entire_fat_16_ros_br_matches(&fake_fd)) {
|
|
|
|
uprintf("Drive has a FAT16 ReactOS partition boot record\n");
|
|
|
|
} else if (entire_fat_32_br_matches(&fake_fd)) {
|
|
|
|
uprintf("Drive has a FAT32 DOS partition boot record\n");
|
|
|
|
} else if (entire_fat_32_nt_br_matches(&fake_fd)) {
|
|
|
|
uprintf("Drive has a FAT32 NT partition boot record\n");
|
|
|
|
} else if (entire_fat_32_fd_br_matches(&fake_fd)) {
|
|
|
|
uprintf("Drive has a FAT32 FreeDOS partition boot record\n");
|
|
|
|
} else if (entire_fat_32_ros_br_matches(&fake_fd)) {
|
|
|
|
uprintf("Drive has a FAT32 ReactOS partition boot record\n");
|
|
|
|
} else {
|
|
|
|
uprintf("Drive has an unknown FAT16 or FAT32 partition boot record\n");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
uprintf("Drive has an unknown partition boot record\n");
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-01-24 21:30:11 +00:00
|
|
|
/*
|
|
|
|
* Fill the drive properties (size, FS, etc)
|
|
|
|
*/
|
2014-01-14 20:17:07 +00:00
|
|
|
int GetDrivePartitionData(DWORD DriveIndex, char* FileSystemName, DWORD FileSystemNameSize)
|
2013-01-24 21:30:11 +00:00
|
|
|
{
|
2014-01-14 20:17:07 +00:00
|
|
|
BOOL r, hasRufusExtra = FALSE;
|
2013-04-07 23:10:58 +00:00
|
|
|
HANDLE hPhysical;
|
2013-01-24 21:30:11 +00:00
|
|
|
DWORD size;
|
2013-12-03 23:56:43 +00:00
|
|
|
BYTE geometry[128], layout[4096], part_type;
|
2013-01-24 21:30:11 +00:00
|
|
|
void* disk_geometry = (void*)geometry;
|
|
|
|
void* drive_layout = (void*)layout;
|
|
|
|
PDISK_GEOMETRY_EX DiskGeometry = (PDISK_GEOMETRY_EX)disk_geometry;
|
|
|
|
PDRIVE_LAYOUT_INFORMATION_EX DriveLayout = (PDRIVE_LAYOUT_INFORMATION_EX)drive_layout;
|
2013-04-07 23:10:58 +00:00
|
|
|
char* volume_name;
|
|
|
|
char tmp[256];
|
2013-01-24 21:30:11 +00:00
|
|
|
DWORD i, nb_partitions = 0;
|
|
|
|
|
2013-07-08 23:14:29 +00:00
|
|
|
// Populate the filesystem data
|
|
|
|
FileSystemName[0] = 0;
|
|
|
|
volume_name = GetLogicalName(DriveIndex, TRUE, FALSE);
|
|
|
|
if ((volume_name == NULL) || (!GetVolumeInformationA(volume_name, NULL, 0, NULL, NULL, NULL, FileSystemName, FileSystemNameSize))) {
|
|
|
|
uprintf("No volume information for disk 0x%02x\n", DriveIndex);
|
|
|
|
}
|
|
|
|
safe_free(volume_name);
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
hPhysical = GetPhysicalHandle(DriveIndex, FALSE, FALSE);
|
|
|
|
if (hPhysical == INVALID_HANDLE_VALUE)
|
2014-01-14 20:17:07 +00:00
|
|
|
return 0;
|
2013-01-24 21:30:11 +00:00
|
|
|
|
2013-11-17 01:39:43 +00:00
|
|
|
r = DeviceIoControl(hPhysical, IOCTL_DISK_GET_DRIVE_GEOMETRY_EX,
|
2013-01-24 21:30:11 +00:00
|
|
|
NULL, 0, geometry, sizeof(geometry), &size, NULL);
|
|
|
|
if (!r || size <= 0) {
|
2013-06-06 22:40:37 +00:00
|
|
|
uprintf("Could not get geometry for drive 0x%02x: %s\n", DriveIndex, WindowsErrorString());
|
2013-04-07 23:10:58 +00:00
|
|
|
safe_closehandle(hPhysical);
|
2014-01-14 20:17:07 +00:00
|
|
|
return 0;
|
2013-01-24 21:30:11 +00:00
|
|
|
}
|
|
|
|
SelectedDrive.DiskSize = DiskGeometry->DiskSize.QuadPart;
|
|
|
|
memcpy(&SelectedDrive.Geometry, &DiskGeometry->Geometry, sizeof(DISK_GEOMETRY));
|
|
|
|
uprintf("Sector Size: %d bytes\n", DiskGeometry->Geometry.BytesPerSector);
|
|
|
|
uprintf("Cylinders: %lld, TracksPerCylinder: %d, SectorsPerTrack: %d\n",
|
|
|
|
DiskGeometry->Geometry.Cylinders, DiskGeometry->Geometry.TracksPerCylinder, DiskGeometry->Geometry.SectorsPerTrack);
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
r = DeviceIoControl(hPhysical, IOCTL_DISK_GET_DRIVE_LAYOUT_EX,
|
2013-01-24 21:30:11 +00:00
|
|
|
NULL, 0, layout, sizeof(layout), &size, NULL );
|
|
|
|
if (!r || size <= 0) {
|
2013-06-06 22:40:37 +00:00
|
|
|
uprintf("Could not get layout for drive 0x%02x: %s\n", DriveIndex, WindowsErrorString());
|
2014-01-14 20:17:07 +00:00
|
|
|
return 0;
|
2013-01-24 21:30:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (DriveLayout->PartitionStyle) {
|
|
|
|
case PARTITION_STYLE_MBR:
|
|
|
|
SelectedDrive.PartitionType = PARTITION_STYLE_MBR;
|
|
|
|
for (i=0; i<DriveLayout->PartitionCount; i++) {
|
|
|
|
if (DriveLayout->PartitionEntry[i].Mbr.PartitionType != PARTITION_ENTRY_UNUSED) {
|
|
|
|
nb_partitions++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
uprintf("Partition type: MBR, NB Partitions: %d\n", nb_partitions);
|
2013-01-27 20:56:57 +00:00
|
|
|
SelectedDrive.has_mbr_uefi_marker = (DriveLayout->Mbr.Signature == MBR_UEFI_MARKER);
|
|
|
|
uprintf("Disk ID: 0x%08X %s\n", DriveLayout->Mbr.Signature, SelectedDrive.has_mbr_uefi_marker?"(UEFI target)":"");
|
2014-01-05 01:39:41 +00:00
|
|
|
AnalyzeMBR(hPhysical);
|
2013-01-24 21:30:11 +00:00
|
|
|
for (i=0; i<DriveLayout->PartitionCount; i++) {
|
|
|
|
if (DriveLayout->PartitionEntry[i].Mbr.PartitionType != PARTITION_ENTRY_UNUSED) {
|
2014-01-31 02:51:28 +00:00
|
|
|
uprintf("Partition %d:\n", i+1);
|
2013-01-24 21:30:11 +00:00
|
|
|
part_type = DriveLayout->PartitionEntry[i].Mbr.PartitionType;
|
|
|
|
uprintf(" Type: %s (0x%02x)\r\n Size: %s (%lld bytes)\r\n Start Sector: %d, Boot: %s, Recognized: %s\n",
|
2013-07-08 23:14:29 +00:00
|
|
|
((part_type==0x07)&&(FileSystemName[0]!=0))?FileSystemName:GetPartitionType(part_type), part_type,
|
|
|
|
SizeToHumanReadable(DriveLayout->PartitionEntry[i].PartitionLength),
|
2013-01-24 21:30:11 +00:00
|
|
|
DriveLayout->PartitionEntry[i].PartitionLength, DriveLayout->PartitionEntry[i].Mbr.HiddenSectors,
|
|
|
|
DriveLayout->PartitionEntry[i].Mbr.BootIndicator?"Yes":"No",
|
|
|
|
DriveLayout->PartitionEntry[i].Mbr.RecognizedPartition?"Yes":"No");
|
2014-01-14 20:17:07 +00:00
|
|
|
if (part_type == RUFUS_EXTRA_PARTITION_TYPE) // This is a partition Rufus created => we can safely ignore it
|
|
|
|
hasRufusExtra = TRUE;
|
2013-01-24 21:30:11 +00:00
|
|
|
if (part_type == 0xee) // Flag a protective MBR for non GPT platforms (XP)
|
|
|
|
SelectedDrive.has_protective_mbr = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case PARTITION_STYLE_GPT:
|
|
|
|
SelectedDrive.PartitionType = PARTITION_STYLE_GPT;
|
|
|
|
uprintf("Partition type: GPT, NB Partitions: %d\n", DriveLayout->PartitionCount);
|
|
|
|
uprintf("Disk GUID: %s\n", GuidToString(&DriveLayout->Gpt.DiskId));
|
|
|
|
uprintf("Max parts: %d, Start Offset: %lld, Usable = %lld bytes\n",
|
|
|
|
DriveLayout->Gpt.MaxPartitionCount, DriveLayout->Gpt.StartingUsableOffset.QuadPart, DriveLayout->Gpt.UsableLength.QuadPart);
|
|
|
|
for (i=0; i<DriveLayout->PartitionCount; i++) {
|
|
|
|
nb_partitions++;
|
|
|
|
tmp[0] = 0;
|
|
|
|
wchar_to_utf8_no_alloc(DriveLayout->PartitionEntry[i].Gpt.Name, tmp, sizeof(tmp));
|
|
|
|
uprintf("Partition %d:\r\n Type: %s\r\n Name: '%s'\n", DriveLayout->PartitionEntry[i].PartitionNumber,
|
|
|
|
GuidToString(&DriveLayout->PartitionEntry[i].Gpt.PartitionType), tmp);
|
|
|
|
uprintf(" ID: %s\r\n Size: %s (%lld bytes)\r\n Start Sector: %lld, Attributes: 0x%016llX\n",
|
|
|
|
GuidToString(&DriveLayout->PartitionEntry[i].Gpt.PartitionId), SizeToHumanReadable(DriveLayout->PartitionEntry[i].PartitionLength),
|
|
|
|
DriveLayout->PartitionEntry[i].PartitionLength, DriveLayout->PartitionEntry[i].StartingOffset.QuadPart / DiskGeometry->Geometry.BytesPerSector,
|
|
|
|
DriveLayout->PartitionEntry[i].Gpt.Attributes);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SelectedDrive.PartitionType = PARTITION_STYLE_MBR;
|
|
|
|
uprintf("Partition type: RAW\n");
|
|
|
|
break;
|
|
|
|
}
|
2013-04-07 23:10:58 +00:00
|
|
|
safe_closehandle(hPhysical);
|
2013-01-24 21:30:11 +00:00
|
|
|
|
2014-01-14 20:17:07 +00:00
|
|
|
if (hasRufusExtra)
|
|
|
|
nb_partitions--;
|
|
|
|
return (int)nb_partitions;
|
2013-07-08 23:14:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Flush file data
|
|
|
|
*/
|
|
|
|
static BOOL FlushDrive(char drive_letter)
|
|
|
|
{
|
|
|
|
HANDLE hDrive = INVALID_HANDLE_VALUE;
|
|
|
|
BOOL r = FALSE;
|
|
|
|
char logical_drive[] = "\\\\.\\#:";
|
|
|
|
|
|
|
|
logical_drive[4] = drive_letter;
|
|
|
|
hDrive = CreateFileA(logical_drive, GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE,
|
2013-11-23 22:39:54 +00:00
|
|
|
NULL, OPEN_EXISTING, 0, NULL);
|
2013-07-08 23:14:29 +00:00
|
|
|
if (hDrive == INVALID_HANDLE_VALUE) {
|
|
|
|
uprintf("Failed to open %c: for flushing: %s\n", drive_letter, WindowsErrorString());
|
|
|
|
goto out;
|
2013-01-24 21:30:11 +00:00
|
|
|
}
|
2013-07-08 23:14:29 +00:00
|
|
|
r = FlushFileBuffers(hDrive);
|
|
|
|
if (r == FALSE)
|
|
|
|
uprintf("Failed to flush %c: %s\n", drive_letter, WindowsErrorString());
|
2013-01-24 21:30:11 +00:00
|
|
|
|
2013-07-08 23:14:29 +00:00
|
|
|
out:
|
|
|
|
safe_closehandle(hDrive);
|
|
|
|
return r;
|
2013-01-24 21:30:11 +00:00
|
|
|
}
|
|
|
|
|
2013-04-07 23:10:58 +00:00
|
|
|
/*
|
|
|
|
* Unmount of volume using the DISMOUNT_VOLUME ioctl
|
|
|
|
*/
|
|
|
|
BOOL UnmountVolume(HANDLE hDrive)
|
2011-12-11 02:19:38 +00:00
|
|
|
{
|
|
|
|
DWORD size;
|
|
|
|
|
|
|
|
if (!DeviceIoControl(hDrive, FSCTL_DISMOUNT_VOLUME, NULL, 0, NULL, 0, &size, NULL)) {
|
|
|
|
uprintf("Could not ummount drive: %s\n", WindowsErrorString());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
2013-01-22 02:40:43 +00:00
|
|
|
|
2013-07-08 23:14:29 +00:00
|
|
|
/*
|
|
|
|
* Mount the volume identified by drive_guid to mountpoint drive_name
|
|
|
|
*/
|
|
|
|
BOOL MountVolume(char* drive_name, char *drive_guid)
|
|
|
|
{
|
|
|
|
char mounted_guid[52]; // You need at least 51 characters on XP
|
2013-12-10 19:30:12 +00:00
|
|
|
char mounted_letter[16] = {0};
|
|
|
|
DWORD size;
|
|
|
|
|
|
|
|
// For fixed disks, Windows may already have remounted the volume, but with a different letter
|
|
|
|
// than the one we want. If that's the case, we need to unmount first.
|
|
|
|
if ( (GetVolumePathNamesForVolumeNameA(drive_guid, mounted_letter, sizeof(mounted_letter), &size))
|
2014-01-11 00:19:03 +00:00
|
|
|
&& (size > 1) && (mounted_letter[0] != drive_name[0]) ) {
|
|
|
|
uprintf("Volume is already mounted, but as %c: instead of %c: - Unmounting...\n", mounted_letter[0], drive_name[0]);
|
2013-12-10 19:30:12 +00:00
|
|
|
if (!DeleteVolumeMountPointA(mounted_letter))
|
|
|
|
uprintf("Failed to unmount volume: %s", WindowsErrorString());
|
|
|
|
Sleep(200);
|
|
|
|
}
|
2013-07-08 23:14:29 +00:00
|
|
|
|
|
|
|
if (!SetVolumeMountPointA(drive_name, drive_guid)) {
|
|
|
|
// If the OS was faster than us at remounting the drive, this operation can fail
|
|
|
|
// with ERROR_DIR_NOT_EMPTY. If that's the case, just check that mountpoints match
|
|
|
|
if (GetLastError() == ERROR_DIR_NOT_EMPTY) {
|
|
|
|
if (!GetVolumeNameForVolumeMountPointA(drive_name, mounted_guid, sizeof(mounted_guid))) {
|
|
|
|
uprintf("%s already mounted, but volume GUID could not be checked: %s\n",
|
|
|
|
drive_name, WindowsErrorString());
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (safe_strcmp(drive_guid, mounted_guid) != 0) {
|
|
|
|
uprintf("%s already mounted, but volume GUID doesn't match:\r\n expected %s, got %s\n",
|
|
|
|
drive_name, drive_guid, mounted_guid);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
uprintf("%s was already mounted as %s\n", drive_guid, drive_name);
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Issue a complete remount of the volume
|
|
|
|
*/
|
|
|
|
BOOL RemountVolume(char* drive_name)
|
|
|
|
{
|
|
|
|
char drive_guid[51];
|
|
|
|
|
|
|
|
// UDF requires a sync/flush, and it's also a good idea for other FS's
|
|
|
|
FlushDrive(drive_name[0]);
|
|
|
|
if (GetVolumeNameForVolumeMountPointA(drive_name, drive_guid, sizeof(drive_guid))) {
|
|
|
|
if (DeleteVolumeMountPointA(drive_name)) {
|
|
|
|
Sleep(200);
|
|
|
|
if (MountVolume(drive_name, drive_guid)) {
|
|
|
|
uprintf("Successfully remounted %s on %s\n", &drive_guid[4], drive_name);
|
|
|
|
} else {
|
|
|
|
uprintf("Failed to remount %s on %s\n", &drive_guid[4], drive_name);
|
|
|
|
// This will leave the drive inaccessible and must be flagged as an error
|
|
|
|
FormatStatus = ERROR_SEVERITY_ERROR|FAC(FACILITY_STORAGE)|APPERR(ERROR_CANT_REMOUNT_VOLUME);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
uprintf("Could not remount %s %s\n", drive_name, WindowsErrorString());
|
|
|
|
// Try to continue regardless
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-01-22 02:40:43 +00:00
|
|
|
/* MinGW is unhappy about accessing partitions beside the first unless we redef */
|
|
|
|
typedef struct _DRIVE_LAYOUT_INFORMATION_EX4 {
|
|
|
|
DWORD PartitionStyle;
|
|
|
|
DWORD PartitionCount;
|
|
|
|
union {
|
|
|
|
DRIVE_LAYOUT_INFORMATION_MBR Mbr;
|
|
|
|
DRIVE_LAYOUT_INFORMATION_GPT Gpt;
|
|
|
|
} Type;
|
|
|
|
PARTITION_INFORMATION_EX PartitionEntry[4];
|
|
|
|
} DRIVE_LAYOUT_INFORMATION_EX4,*PDRIVE_LAYOUT_INFORMATION_EX4;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Create a partition table
|
2013-01-27 20:56:57 +00:00
|
|
|
* See http://technet.microsoft.com/en-us/library/cc739412.aspx for some background info
|
|
|
|
* NB: if you modify the MBR outside of using the Windows API, Windows still uses the cached
|
2013-06-25 01:55:25 +00:00
|
|
|
* copy it got from the last IOCTL, and ignores your changes until you replug the drive
|
|
|
|
* or issue an IOCTL_DISK_UPDATE_PROPERTIES.
|
|
|
|
*/
|
2013-01-22 02:40:43 +00:00
|
|
|
#if !defined(PARTITION_BASIC_DATA_GUID)
|
|
|
|
const GUID PARTITION_BASIC_DATA_GUID = { 0xebd0a0a2, 0xb9e5, 0x4433, {0x87, 0xc0, 0x68, 0xb6, 0xb7, 0x26, 0x99, 0xc7} };
|
|
|
|
#endif
|
2013-01-27 20:56:57 +00:00
|
|
|
BOOL CreatePartition(HANDLE hDrive, int partition_style, int file_system, BOOL mbr_uefi_marker)
|
2013-01-22 02:40:43 +00:00
|
|
|
{
|
|
|
|
const char* PartitionTypeName[2] = { "MBR", "GPT" };
|
|
|
|
CREATE_DISK CreateDisk = {PARTITION_STYLE_RAW, {{0}}};
|
|
|
|
DRIVE_LAYOUT_INFORMATION_EX4 DriveLayoutEx = {0};
|
|
|
|
BOOL r;
|
|
|
|
DWORD size;
|
|
|
|
LONGLONG size_in_sectors;
|
|
|
|
|
2013-12-19 23:56:40 +00:00
|
|
|
PrintStatus(0, TRUE, MSG_238, PartitionTypeName[partition_style]);
|
2013-01-22 02:40:43 +00:00
|
|
|
|
|
|
|
if ((partition_style == PARTITION_STYLE_GPT) || (!IsChecked(IDC_EXTRA_PARTITION))) {
|
|
|
|
// Go with the MS 1 MB wastage at the beginning...
|
|
|
|
DriveLayoutEx.PartitionEntry[0].StartingOffset.QuadPart = 1024*1024;
|
|
|
|
} else {
|
|
|
|
// Align on Cylinder
|
|
|
|
DriveLayoutEx.PartitionEntry[0].StartingOffset.QuadPart =
|
|
|
|
SelectedDrive.Geometry.BytesPerSector * SelectedDrive.Geometry.SectorsPerTrack;
|
|
|
|
}
|
2013-07-08 23:14:29 +00:00
|
|
|
// TODO: should we try to align the following to the cluster size as well?
|
2013-01-22 02:40:43 +00:00
|
|
|
size_in_sectors = (SelectedDrive.DiskSize - DriveLayoutEx.PartitionEntry[0].StartingOffset.QuadPart) / SelectedDrive.Geometry.BytesPerSector;
|
|
|
|
|
|
|
|
switch (partition_style) {
|
|
|
|
case PARTITION_STYLE_MBR:
|
|
|
|
CreateDisk.PartitionStyle = PARTITION_STYLE_MBR;
|
2013-01-27 20:56:57 +00:00
|
|
|
// If MBR+UEFI is selected, write an UEFI marker in lieu of the regular MBR signature.
|
|
|
|
// This helps us reselect the partition scheme option that was used when creating the
|
|
|
|
// drive in Rufus. As far as I can tell, Windows doesn't care much if this signature
|
|
|
|
// isn't unique for USB drives.
|
|
|
|
CreateDisk.Mbr.Signature = mbr_uefi_marker?MBR_UEFI_MARKER:GetTickCount();
|
2013-01-22 02:40:43 +00:00
|
|
|
|
|
|
|
DriveLayoutEx.PartitionStyle = PARTITION_STYLE_MBR;
|
|
|
|
DriveLayoutEx.PartitionCount = 4; // Must be multiple of 4 for MBR
|
|
|
|
DriveLayoutEx.Type.Mbr.Signature = CreateDisk.Mbr.Signature;
|
|
|
|
DriveLayoutEx.PartitionEntry[0].PartitionStyle = PARTITION_STYLE_MBR;
|
|
|
|
// TODO: CHS fixup (32 sectors/track) through a cheat mode, if requested
|
|
|
|
// NB: disk geometry is computed by BIOS & co. by finding a match between LBA and CHS value of first partition
|
|
|
|
// ms-sys's write_partition_number_of_heads() and write_partition_start_sector_number() can be used if needed
|
|
|
|
|
|
|
|
// Align on sector boundary if the extra part option is checked
|
|
|
|
if (IsChecked(IDC_EXTRA_PARTITION)) {
|
|
|
|
size_in_sectors = ((size_in_sectors / SelectedDrive.Geometry.SectorsPerTrack)-1) * SelectedDrive.Geometry.SectorsPerTrack;
|
|
|
|
if (size_in_sectors <= 0)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case PARTITION_STYLE_GPT:
|
2013-07-05 21:20:46 +00:00
|
|
|
// TODO: (?) As per MSDN: "When specifying a GUID partition table (GPT) as the PARTITION_STYLE of the CREATE_DISK
|
|
|
|
// structure, an application should wait for the MSR partition arrival before sending the IOCTL_DISK_SET_DRIVE_LAYOUT_EX
|
|
|
|
// control code. For more information about device notification, see RegisterDeviceNotification."
|
|
|
|
|
2013-01-22 02:40:43 +00:00
|
|
|
CreateDisk.PartitionStyle = PARTITION_STYLE_GPT;
|
|
|
|
IGNORE_RETVAL(CoCreateGuid(&CreateDisk.Gpt.DiskId));
|
|
|
|
CreateDisk.Gpt.MaxPartitionCount = MAX_GPT_PARTITIONS;
|
|
|
|
|
|
|
|
DriveLayoutEx.PartitionStyle = PARTITION_STYLE_GPT;
|
|
|
|
DriveLayoutEx.PartitionCount = 1;
|
|
|
|
// At the very least, a GPT disk has atv least 34 reserved (512 bytes) blocks at the beginning
|
|
|
|
// and 33 at the end.
|
|
|
|
DriveLayoutEx.Type.Gpt.StartingUsableOffset.QuadPart = 34*512;
|
|
|
|
DriveLayoutEx.Type.Gpt.UsableLength.QuadPart = SelectedDrive.DiskSize - (34+33)*512;
|
|
|
|
DriveLayoutEx.Type.Gpt.MaxPartitionCount = MAX_GPT_PARTITIONS;
|
|
|
|
DriveLayoutEx.Type.Gpt.DiskId = CreateDisk.Gpt.DiskId;
|
|
|
|
DriveLayoutEx.PartitionEntry[0].PartitionStyle = PARTITION_STYLE_GPT;
|
|
|
|
|
|
|
|
size_in_sectors -= 33; // Need 33 sectors at the end for secondary GPT
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
DriveLayoutEx.PartitionEntry[0].PartitionLength.QuadPart = size_in_sectors * SelectedDrive.Geometry.BytesPerSector;
|
|
|
|
DriveLayoutEx.PartitionEntry[0].PartitionNumber = 1;
|
|
|
|
DriveLayoutEx.PartitionEntry[0].RewritePartition = TRUE;
|
|
|
|
|
|
|
|
switch (partition_style) {
|
|
|
|
case PARTITION_STYLE_MBR:
|
2013-05-01 22:43:44 +00:00
|
|
|
DriveLayoutEx.PartitionEntry[0].Mbr.BootIndicator = IsChecked(IDC_BOOT);
|
2013-01-22 02:40:43 +00:00
|
|
|
DriveLayoutEx.PartitionEntry[0].Mbr.HiddenSectors = SelectedDrive.Geometry.SectorsPerTrack;
|
|
|
|
switch (file_system) {
|
|
|
|
case FS_FAT16:
|
|
|
|
DriveLayoutEx.PartitionEntry[0].Mbr.PartitionType = 0x0e; // FAT16 LBA
|
|
|
|
break;
|
|
|
|
case FS_NTFS:
|
|
|
|
case FS_EXFAT:
|
2013-07-08 23:14:29 +00:00
|
|
|
case FS_UDF:
|
2013-01-22 02:40:43 +00:00
|
|
|
DriveLayoutEx.PartitionEntry[0].Mbr.PartitionType = 0x07; // NTFS
|
|
|
|
break;
|
|
|
|
case FS_FAT32:
|
|
|
|
DriveLayoutEx.PartitionEntry[0].Mbr.PartitionType = 0x0c; // FAT32 LBA
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
uprintf("Unsupported file system\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
// Create an extra partition on request - can improve BIOS detection as HDD for older BIOSes
|
|
|
|
if (IsChecked(IDC_EXTRA_PARTITION)) {
|
|
|
|
DriveLayoutEx.PartitionEntry[1].PartitionStyle = PARTITION_STYLE_MBR;
|
|
|
|
// Should end on a sector boundary
|
|
|
|
DriveLayoutEx.PartitionEntry[1].StartingOffset.QuadPart = DriveLayoutEx.PartitionEntry[0].StartingOffset.QuadPart +
|
|
|
|
DriveLayoutEx.PartitionEntry[0].PartitionLength.QuadPart;
|
|
|
|
DriveLayoutEx.PartitionEntry[1].PartitionLength.QuadPart = SelectedDrive.Geometry.SectorsPerTrack*SelectedDrive.Geometry.BytesPerSector;
|
|
|
|
DriveLayoutEx.PartitionEntry[1].PartitionNumber = 2;
|
|
|
|
DriveLayoutEx.PartitionEntry[1].RewritePartition = TRUE;
|
2013-04-07 23:14:28 +00:00
|
|
|
DriveLayoutEx.PartitionEntry[1].Mbr.BootIndicator = FALSE;
|
2013-01-22 02:40:43 +00:00
|
|
|
DriveLayoutEx.PartitionEntry[1].Mbr.HiddenSectors = SelectedDrive.Geometry.SectorsPerTrack*SelectedDrive.Geometry.BytesPerSector;
|
2014-01-14 20:10:09 +00:00
|
|
|
DriveLayoutEx.PartitionEntry[1].Mbr.PartitionType = RUFUS_EXTRA_PARTITION_TYPE;
|
2013-01-22 02:40:43 +00:00
|
|
|
}
|
|
|
|
// For the remaining partitions, PartitionStyle & PartitionType have already
|
|
|
|
// been zeroed => already set to MBR/unused
|
|
|
|
break;
|
|
|
|
case PARTITION_STYLE_GPT:
|
|
|
|
DriveLayoutEx.PartitionEntry[0].Gpt.PartitionType = PARTITION_BASIC_DATA_GUID;
|
|
|
|
wcscpy(DriveLayoutEx.PartitionEntry[0].Gpt.Name, L"Microsoft Basic Data");
|
|
|
|
IGNORE_RETVAL(CoCreateGuid(&DriveLayoutEx.PartitionEntry[0].Gpt.PartitionId));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If you don't call IOCTL_DISK_CREATE_DISK, the next call will fail
|
|
|
|
size = sizeof(CreateDisk);
|
|
|
|
r = DeviceIoControl(hDrive, IOCTL_DISK_CREATE_DISK,
|
|
|
|
(BYTE*)&CreateDisk, size, NULL, 0, &size, NULL );
|
|
|
|
if (!r) {
|
2013-04-07 23:10:58 +00:00
|
|
|
uprintf("Could not reset disk: %s\n", WindowsErrorString());
|
2013-01-22 02:40:43 +00:00
|
|
|
safe_closehandle(hDrive);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
size = sizeof(DriveLayoutEx) - ((partition_style == PARTITION_STYLE_GPT)?(3*sizeof(PARTITION_INFORMATION_EX)):0);
|
|
|
|
r = DeviceIoControl(hDrive, IOCTL_DISK_SET_DRIVE_LAYOUT_EX,
|
|
|
|
(BYTE*)&DriveLayoutEx, size, NULL, 0, &size, NULL );
|
|
|
|
if (!r) {
|
2013-04-07 23:10:58 +00:00
|
|
|
uprintf("Could not set drive layout: %s\n", WindowsErrorString());
|
2013-01-22 02:40:43 +00:00
|
|
|
safe_closehandle(hDrive);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2013-06-25 01:55:25 +00:00
|
|
|
// Diskpart does call the following IOCTL this after updating the partition table, so we do too
|
|
|
|
r = DeviceIoControl(hDrive, IOCTL_DISK_UPDATE_PROPERTIES, NULL, 0, NULL, 0, &size, NULL );
|
|
|
|
if (!r) {
|
|
|
|
uprintf("Could not refresh drive layout: %s\n", WindowsErrorString());
|
|
|
|
safe_closehandle(hDrive);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-07-05 21:20:46 +00:00
|
|
|
/* Delete the disk partition table */
|
|
|
|
BOOL DeletePartitions(HANDLE hDrive)
|
|
|
|
{
|
|
|
|
BOOL r;
|
|
|
|
DWORD size;
|
|
|
|
CREATE_DISK CreateDisk = {PARTITION_STYLE_RAW, {{0}}};
|
|
|
|
|
2013-12-19 23:56:40 +00:00
|
|
|
PrintStatus(0, TRUE, MSG_239);
|
2013-07-05 21:20:46 +00:00
|
|
|
|
|
|
|
size = sizeof(CreateDisk);
|
|
|
|
r = DeviceIoControl(hDrive, IOCTL_DISK_CREATE_DISK,
|
|
|
|
(BYTE*)&CreateDisk, size, NULL, 0, &size, NULL );
|
|
|
|
if (!r) {
|
|
|
|
uprintf("Could not delete drive layout: %s\n", WindowsErrorString());
|
|
|
|
safe_closehandle(hDrive);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
r = DeviceIoControl(hDrive, IOCTL_DISK_UPDATE_PROPERTIES, NULL, 0, NULL, 0, &size, NULL );
|
|
|
|
if (!r) {
|
|
|
|
uprintf("Could not refresh drive layout: %s\n", WindowsErrorString());
|
|
|
|
safe_closehandle(hDrive);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-01-22 02:40:43 +00:00
|
|
|
/*
|
|
|
|
* Convert a partition type to its human readable form using
|
|
|
|
* (slightly modified) entries from GNU fdisk
|
|
|
|
*/
|
|
|
|
const char* GetPartitionType(BYTE Type)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i=0; i<ARRAYSIZE(msdos_systypes); i++) {
|
|
|
|
if (msdos_systypes[i].type == Type)
|
|
|
|
return msdos_systypes[i].name;
|
|
|
|
}
|
|
|
|
return "Unknown";
|
|
|
|
}
|