aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--winsup/cygwin/fhandler/dev_disk.cc159
-rw-r--r--winsup/cygwin/local_includes/fhandler.h3
2 files changed, 141 insertions, 21 deletions
diff --git a/winsup/cygwin/fhandler/dev_disk.cc b/winsup/cygwin/fhandler/dev_disk.cc
index 5f79ab5..016b4c7 100644
--- a/winsup/cygwin/fhandler/dev_disk.cc
+++ b/winsup/cygwin/fhandler/dev_disk.cc
@@ -14,33 +14,62 @@ details. */
#include <wctype.h>
#include <winioctl.h>
-/* Replace spaces, non-printing and unexpected characters. Remove
- leading and trailing spaces. Return remaining string length. */
+/* Replace invalid characters. Optionally remove leading and trailing
+ characters. Return remaining string length. */
+template <typename char_type, typename func_type>
static int
-sanitize_id_string (char *s)
+sanitize_string (char_type *s, char_type leading, char_type trailing,
+ char_type replace, func_type valid)
{
int first = 0;
- while (s[first] == ' ')
- first++;
- int last = -1, i;
+ if (leading)
+ while (s[first] == leading)
+ first++;
+ int len = -1, i;
for (i = 0; s[first + i]; i++)
{
- char c = s[first + i];
- if (c != ' ')
- last = -1;
- else if (last < 0)
- last = i;
- if (!(('0' <= c && c <= '9') || c == '.' || c == '-'
- || ('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z')))
- c = '_';
+ char_type c = s[first + i];
+ if (c != trailing)
+ len = -1;
+ else if (len < 0)
+ len = i;
+ if (!valid (c))
+ c = replace;
else if (!first)
continue;
s[i] = c;
}
- if (last < 0)
- last = i;
- s[last] = '\0';
- return last;
+ if (len < 0)
+ len = i;
+ s[len] = (char_type) 0;
+ return len;
+}
+
+/* Variant for device identify strings. */
+static int
+sanitize_id_string (char *s)
+{
+ return sanitize_string (s, ' ', ' ', '_', [] (char c) -> bool
+ {
+ return (('0' <= c && c <= '9') || c == '.' || c == '-'
+ || ('A' <= c && c <= 'Z') || ('a' <= c && c <= 'z'));
+ }
+ );
+}
+
+/* Variant for volume labels. */
+static int
+sanitize_label_string (WCHAR *s)
+{
+ /* Linux does not skip leading spaces. */
+ return sanitize_string (s, L'\0', L' ', L'_', [] (WCHAR c) -> bool
+ {
+ /* Labels may contain characters not allowed in filenames.
+ Linux replaces spaces with \x20 which is not an option here. */
+ return !((0 <= c && c <= L' ') || c == L':' || c == L'/' || c == L'\\'
+ || c == L'"');
+ }
+ );
}
/* Fetch storage properties and create the ID string.
@@ -244,6 +273,79 @@ partition_to_voluuid(const UNICODE_STRING *drive_uname, DWORD part_num,
return true;
}
+/* ("HarddiskN", PART_NUM) -> "VOLUME_LABEL" or "VOLUME_SERIAL" */
+static bool
+partition_to_label_or_uuid(bool uuid, const UNICODE_STRING *drive_uname,
+ DWORD part_num, char *ioctl_buf, char *name)
+{
+ WCHAR wpath[MAX_PATH];
+ /* Trailing backslash is required. */
+ size_t len = __small_swprintf (wpath, L"\\Device\\%S\\Partition%u\\",
+ drive_uname, part_num);
+ len *= sizeof (WCHAR);
+ UNICODE_STRING upath = {(USHORT) len, (USHORT) (len + 1), wpath};
+ OBJECT_ATTRIBUTES attr;
+ InitializeObjectAttributes (&attr, &upath, OBJ_CASE_INSENSITIVE, nullptr,
+ nullptr);
+ IO_STATUS_BLOCK io;
+ HANDLE volhdl;
+ NTSTATUS status = NtOpenFile (&volhdl, READ_CONTROL, &attr, &io,
+ FILE_SHARE_VALID_FLAGS, 0);
+ if (!NT_SUCCESS (status))
+ {
+ /* Fails with STATUS_UNRECOGNIZED_VOLUME (0xC000014F) if the
+ partition/filesystem type is unsupported. */
+ debug_printf ("NtOpenFile(%S), status %y", upath, status);
+ return false;
+ }
+
+ /* Check for possible 64-bit NTFS serial number first. */
+ DWORD bytes_read;
+ const NTFS_VOLUME_DATA_BUFFER *nvdb =
+ reinterpret_cast<const NTFS_VOLUME_DATA_BUFFER *>(ioctl_buf);
+ if (uuid && DeviceIoControl (volhdl, FSCTL_GET_NTFS_VOLUME_DATA, nullptr, 0,
+ ioctl_buf, NT_MAX_PATH, &bytes_read, nullptr)
+ && nvdb->VolumeSerialNumber.QuadPart)
+ {
+ /* Print without any separator as on Linux. */
+ __small_sprintf (name, "%16X", nvdb->VolumeSerialNumber.QuadPart);
+ NtClose(volhdl);
+ return true;
+ }
+
+ /* Get label and 32-bit serial number. */
+ status = NtQueryVolumeInformationFile (volhdl, &io, ioctl_buf,
+ NT_MAX_PATH, FileFsVolumeInformation);
+ NtClose(volhdl);
+ if (!NT_SUCCESS (status))
+ {
+ debug_printf ("NtQueryVolumeInformationFile(%S), status %y", upath,
+ status);
+ return false;
+ }
+
+ FILE_FS_VOLUME_INFORMATION *ffvi =
+ reinterpret_cast<FILE_FS_VOLUME_INFORMATION *>(ioctl_buf);
+ if (uuid)
+ {
+ if (!ffvi->VolumeSerialNumber)
+ return false;
+ /* Print with separator as on Linux. */
+ __small_sprintf (name, "%04x-%04x", ffvi->VolumeSerialNumber >> 16,
+ ffvi->VolumeSerialNumber & 0xffff);
+ }
+ else
+ {
+ /* Label is not null terminated. */
+ ffvi->VolumeLabel[ffvi->VolumeLabelLength / sizeof(WCHAR)] = L'\0';
+ int len = sanitize_label_string (ffvi->VolumeLabel);
+ if (!len)
+ return false;
+ __small_sprintf (name, "%W", ffvi->VolumeLabel);
+ }
+ return true;
+}
+
struct by_id_entry
{
char name[NAME_MAX + 1];
@@ -324,7 +426,11 @@ get_by_id_table (by_id_entry * &table, fhandler_dev_disk::dev_disk_location loc)
unsigned alloc_size = 0, table_size = 0;
tmp_pathbuf tp;
char *ioctl_buf = tp.c_get ();
- WCHAR *w_buf = tp.w_get ();
+ char *ioctl_buf2 = (loc == fhandler_dev_disk::disk_by_label
+ || loc == fhandler_dev_disk::disk_by_uuid ?
+ tp.c_get () : nullptr);
+ WCHAR *w_buf = (loc == fhandler_dev_disk::disk_by_drive ?
+ tp.w_get () : nullptr);
DIRECTORY_BASIC_INFORMATION *dbi_buf =
reinterpret_cast<DIRECTORY_BASIC_INFORMATION *>(tp.w_get ());
@@ -462,11 +568,23 @@ get_by_id_table (by_id_entry * &table, fhandler_dev_disk::dev_disk_location loc)
__small_sprintf (name, "%s-part%u", drive_name, part_num);
break;
+ case fhandler_dev_disk::disk_by_label:
+ if (!partition_to_label_or_uuid (false, &dbi->ObjectName,
+ part_num, ioctl_buf2, name))
+ continue;
+ break;
+
case fhandler_dev_disk::disk_by_partuuid:
if (!format_partuuid (name, pix))
continue;
break;
+ case fhandler_dev_disk::disk_by_uuid:
+ if (!partition_to_label_or_uuid (true, &dbi->ObjectName,
+ part_num, ioctl_buf2, name))
+ continue;
+ break;
+
case fhandler_dev_disk::disk_by_voluuid:
if (!partition_to_voluuid (&dbi->ObjectName, part_num, name))
continue;
@@ -521,7 +639,8 @@ const size_t by_drive_len = sizeof(by_drive) - 1;
/* Keep this in sync with enum fhandler_dev_disk::dev_disk_location starting
at disk_by_drive. */
static const char * const by_dir_names[] {
- "/by-drive", "/by-id", "/by-partuuid", "/by-voluuid"
+ "/by-drive", "/by-id", "/by-label",
+ "/by-partuuid", "/by-uuid", "/by-voluuid"
};
const size_t by_dir_names_size = sizeof(by_dir_names) / sizeof(by_dir_names[0]);
static_assert((size_t) fhandler_dev_disk::disk_by_drive + by_dir_names_size - 1
diff --git a/winsup/cygwin/local_includes/fhandler.h b/winsup/cygwin/local_includes/fhandler.h
index 86c7b20..ca685a6 100644
--- a/winsup/cygwin/local_includes/fhandler.h
+++ b/winsup/cygwin/local_includes/fhandler.h
@@ -3198,7 +3198,8 @@ public:
enum dev_disk_location {
unknown_loc, invalid_loc, disk_dir,
/* Keep these in sync with dev_disk.cc:by_dir_names array: */
- disk_by_drive, disk_by_id, disk_by_partuuid, disk_by_voluuid
+ disk_by_drive, disk_by_id, disk_by_label,
+ disk_by_partuuid, disk_by_uuid, disk_by_voluuid
};
private: