ReactOS 0.4.16-dev-937-g7afcd2a
QueryDosVolumePaths.c File Reference
#include "precomp.h"
Include dependency graph for QueryDosVolumePaths.c:

Go to the source code of this file.

Macros

#define IS_DRIVE_LETTER_PFX(s)
 
#define IS_DRIVE_LETTER(s)    (IS_DRIVE_LETTER_PFX(s) && (s)->Length == 2*sizeof(WCHAR))
 

Functions

static VOID Call_QueryDosVolume_Path_Paths (_In_ HANDLE MountMgrHandle, _In_ PCWSTR NtVolumeName, _In_ ULONG IoctlPathOrPaths, _Out_ PMOUNTMGR_VOLUME_PATHS *pVolumePathPtr)
 Invokes either IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH or IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS for testing, given the volume device name, and returns an allocated volume paths buffer. This buffer must be freed by the caller via RtlFreeHeap(RtlGetProcessHeap(), ...) .
 
static VOID Call_QueryPoints (_In_ HANDLE MountMgrHandle, _In_ PCWSTR NtVolumeName, _Out_ PMOUNTMGR_MOUNT_POINTS *pMountPointsPtr)
 Invokes IOCTL_MOUNTMGR_QUERY_POINTS for testing, given the volume device name, and returns an allocated mount points buffer. This buffer must be freed by the caller via RtlFreeHeap(RtlGetProcessHeap(), ...) .
 
static VOID Test_QueryDosVolumePath (_In_ PCWSTR NtVolumeName, _In_ PMOUNTMGR_VOLUME_PATHS VolumePath)
 Tests the output of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH.
 
static VOID Test_QueryDosVolumePaths (_In_ PCWSTR NtVolumeName, _In_ PMOUNTMGR_VOLUME_PATHS VolumePaths, _In_opt_ PMOUNTMGR_VOLUME_PATHS VolumePath)
 Tests the output of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS.
 
static BOOLEAN doesPathExistInMountPoints (_In_ PMOUNTMGR_MOUNT_POINTS MountPoints, _In_ PUNICODE_STRING DosPath)
 
static VOID Test_QueryPoints (_In_ PCWSTR NtVolumeName, _In_ PMOUNTMGR_MOUNT_POINTS MountPoints, _In_opt_ PMOUNTMGR_VOLUME_PATHS VolumePath, _In_opt_ PMOUNTMGR_VOLUME_PATHS VolumePaths)
 Tests the output of IOCTL_MOUNTMGR_QUERY_POINTS.
 
static VOID Test_QueryDosVolumePathAndPaths (_In_ HANDLE MountMgrHandle, _In_ PCWSTR NtVolumeName)
 Tests the consistency of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH, IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS and IOCTL_MOUNTMGR_QUERY_POINTS.
 
 START_TEST (QueryDosVolumePaths)
 

Macro Definition Documentation

◆ IS_DRIVE_LETTER

#define IS_DRIVE_LETTER (   s)     (IS_DRIVE_LETTER_PFX(s) && (s)->Length == 2*sizeof(WCHAR))

Definition at line 249 of file QueryDosVolumePaths.c.

◆ IS_DRIVE_LETTER_PFX

#define IS_DRIVE_LETTER_PFX (   s)
Value:
((s)->Length >= 2*sizeof(WCHAR) && (s)->Buffer[0] >= 'A' && \
(s)->Buffer[0] <= 'Z' && (s)->Buffer[1] == ':')
Definition: bufpool.h:45
GLdouble s
Definition: gl.h:2039
_In_ ULONG _In_ ULONG _In_ ULONG Length
Definition: ntddpcm.h:102
__wchar_t WCHAR
Definition: xmlstorage.h:180

Definition at line 244 of file QueryDosVolumePaths.c.

Function Documentation

◆ Call_QueryDosVolume_Path_Paths()

static VOID Call_QueryDosVolume_Path_Paths ( _In_ HANDLE  MountMgrHandle,
_In_ PCWSTR  NtVolumeName,
_In_ ULONG  IoctlPathOrPaths,
_Out_ PMOUNTMGR_VOLUME_PATHS pVolumePathPtr 
)
static

Invokes either IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH or IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS for testing, given the volume device name, and returns an allocated volume paths buffer. This buffer must be freed by the caller via RtlFreeHeap(RtlGetProcessHeap(), ...) .

These IOCTLs return both the drive letter (if any) and the volume GUID symlink path, as well as any other file-system mount reparse points linking to the volume.

Definition at line 24 of file QueryDosVolumePaths.c.

29{
34 MOUNTMGR_VOLUME_PATHS VolumePath;
35 PMOUNTMGR_VOLUME_PATHS VolumePathPtr;
36 ULONG DeviceNameLength;
37 /*
38 * This variable is used to query the device name.
39 * It's based on MOUNTMGR_TARGET_NAME (mountmgr.h).
40 * Doing it this way prevents memory allocation.
41 * The device name won't be longer.
42 */
43 struct
44 {
45 USHORT NameLength;
46 WCHAR DeviceName[256];
47 } DeviceName;
48
49
50 *pVolumePathPtr = NULL;
51
52 /* First, build the corresponding device name */
53 RtlInitUnicodeString(&VolumeName, NtVolumeName);
54 DeviceName.NameLength = VolumeName.Length;
56 DeviceNameLength = FIELD_OFFSET(MOUNTMGR_TARGET_NAME, DeviceName) + DeviceName.NameLength;
57
58 /* Now, query the MountMgr for the DOS path(s) */
59 Status = NtDeviceIoControlFile(MountMgrHandle,
60 NULL, NULL, NULL,
62 IoctlPathOrPaths,
63 &DeviceName, DeviceNameLength,
64 &VolumePath, sizeof(VolumePath));
65
66 /* Check for unsupported device */
68 {
69 skip("Device '%S': Doesn't support MountMgr queries, Status 0x%08lx\n",
70 NtVolumeName, Status);
71 return;
72 }
73
74 /* The only tolerated failure here is buffer too small, which is expected */
76 "Device '%S': IOCTL 0x%lx failed unexpectedly, Status 0x%08lx\n",
77 NtVolumeName, IoctlPathOrPaths, Status);
79 {
80 skip("Device '%S': Wrong Status\n", NtVolumeName);
81 return;
82 }
83
84 /* Compute the needed size to store the DOS path(s).
85 * Even if MOUNTMGR_VOLUME_PATHS allows bigger name lengths
86 * than MAXUSHORT, we can't use them, because we have to return
87 * this in an UNICODE_STRING that stores length in a USHORT. */
88 Length = sizeof(VolumePath) + VolumePath.MultiSzLength;
90 "Device '%S': DOS volume path too large: %lu\n",
91 NtVolumeName, Length);
92 if (Length > MAXUSHORT)
93 {
94 skip("Device '%S': Wrong Length\n", NtVolumeName);
95 return;
96 }
97
98 /* Allocate the buffer and fill it with test pattern */
99 VolumePathPtr = RtlAllocateHeap(RtlGetProcessHeap(), 0, Length);
100 if (!VolumePathPtr)
101 {
102 skip("Device '%S': Failed to allocate buffer with size %lu)\n",
103 NtVolumeName, Length);
104 return;
105 }
106 RtlFillMemory(VolumePathPtr, Length, 0xCC);
107
108 /* Re-query the DOS path(s) with the proper size */
109 Status = NtDeviceIoControlFile(MountMgrHandle,
110 NULL, NULL, NULL,
112 IoctlPathOrPaths,
113 &DeviceName, DeviceNameLength,
114 VolumePathPtr, Length);
116 "Device '%S': IOCTL 0x%lx failed unexpectedly, Status 0x%08lx\n",
117 NtVolumeName, IoctlPathOrPaths, Status);
118
119 if (winetest_debug > 1)
120 {
121 trace("Buffer:\n");
122 DumpBuffer(VolumePathPtr, Length);
123 printf("\n");
124 }
125
126 /* Return the buffer */
127 *pVolumePathPtr = VolumePathPtr;
128}
#define trace
Definition: atltest.h:70
#define ok(value,...)
Definition: atltest.h:57
#define skip(...)
Definition: atltest.h:64
LONG NTSTATUS
Definition: precomp.h:26
PVOID NTAPI RtlAllocateHeap(IN PVOID HeapHandle, IN ULONG Flags, IN SIZE_T Size)
Definition: heap.c:616
#define NULL
Definition: types.h:112
#define NT_SUCCESS(StatCode)
Definition: apphelp.c:33
_Must_inspect_result_ _Inout_opt_ PUNICODE_STRING VolumeName
Definition: fltkernel.h:1117
#define printf
Definition: freeldr.h:97
Status
Definition: gdiplustypes.h:25
#define RtlFillMemory(Dest, Length, Fill)
Definition: winternl.h:599
VOID DumpBuffer(_In_ PVOID Buffer, _In_ ULONG Length)
Definition: utils.c:53
static OUT PIO_STATUS_BLOCK IoStatusBlock
Definition: pipe.c:75
NTSYSAPI VOID NTAPI RtlInitUnicodeString(PUNICODE_STRING DestinationString, PCWSTR SourceString)
NTSYSAPI NTSTATUS NTAPI NtDeviceIoControlFile(IN HANDLE hFile, IN HANDLE hEvent OPTIONAL, IN PIO_APC_ROUTINE IoApcRoutine OPTIONAL, IN PVOID IoApcContext OPTIONAL, OUT PIO_STATUS_BLOCK pIoStatusBlock, IN ULONG DeviceIoControlCode, IN PVOID InBuffer OPTIONAL, IN ULONG InBufferLength, OUT PVOID OutBuffer OPTIONAL, IN ULONG OutBufferLength)
unsigned short USHORT
Definition: pedump.c:61
int winetest_debug
#define STATUS_BUFFER_OVERFLOW
Definition: shellext.h:66
#define FIELD_OFFSET(t, f)
Definition: typedefs.h:255
#define RtlCopyMemory(Destination, Source, Length)
Definition: typedefs.h:263
#define MAXUSHORT
Definition: typedefs.h:83
uint32_t ULONG
Definition: typedefs.h:59
#define STATUS_INVALID_DEVICE_REQUEST
Definition: udferr_usr.h:138
#define STATUS_NO_MEDIA_IN_DEVICE
Definition: udferr_usr.h:141
_Must_inspect_result_ _In_ PWDFDEVICE_INIT _In_opt_ PCUNICODE_STRING DeviceName
Definition: wdfdevice.h:3275

Referenced by Test_QueryDosVolumePathAndPaths().

◆ Call_QueryPoints()

static VOID Call_QueryPoints ( _In_ HANDLE  MountMgrHandle,
_In_ PCWSTR  NtVolumeName,
_Out_ PMOUNTMGR_MOUNT_POINTS pMountPointsPtr 
)
static

Invokes IOCTL_MOUNTMGR_QUERY_POINTS for testing, given the volume device name, and returns an allocated mount points buffer. This buffer must be freed by the caller via RtlFreeHeap(RtlGetProcessHeap(), ...) .

This IOCTL only returns both the drive letter (if any) and the volume GUID symlink path, but does NOT return file-system mount reparse points.

Definition at line 142 of file QueryDosVolumePaths.c.

146{
151 MOUNTMGR_MOUNT_POINTS MountPoints;
152 PMOUNTMGR_MOUNT_POINTS MountPointsPtr;
153 /*
154 * This variable is used to query the device name.
155 * It's based on MOUNTMGR_MOUNT_POINT (mountmgr.h).
156 * Doing it this way prevents memory allocation.
157 * The device name won't be longer.
158 */
159 struct
160 {
162 WCHAR DeviceName[256];
163 } DeviceName;
164
165
166 *pMountPointsPtr = NULL;
167
168 /* First, build the corresponding device name */
169 RtlInitUnicodeString(&VolumeName, NtVolumeName);
170 DeviceName.SymbolicLinkNameOffset = DeviceName.UniqueIdOffset = 0;
171 DeviceName.SymbolicLinkNameLength = DeviceName.UniqueIdLength = 0;
172 DeviceName.DeviceNameOffset = ((ULONG_PTR)&DeviceName.DeviceName - (ULONG_PTR)&DeviceName);
173 DeviceName.DeviceNameLength = VolumeName.Length;
175
176 /* Now, query the MountMgr for the mount points */
177 Status = NtDeviceIoControlFile(MountMgrHandle,
178 NULL, NULL, NULL,
181 &DeviceName, sizeof(DeviceName),
182 &MountPoints, sizeof(MountPoints));
183
184 /* Check for unsupported device */
186 {
187 skip("Device '%S': Doesn't support MountMgr queries, Status 0x%08lx\n",
188 NtVolumeName, Status);
189 return;
190 }
191
192 /* The only tolerated failure here is buffer too small, which is expected */
194 "Device '%S': IOCTL 0x%lx failed unexpectedly, Status 0x%08lx\n",
195 NtVolumeName, IOCTL_MOUNTMGR_QUERY_POINTS, Status);
197 {
198 skip("Device '%S': Wrong Status\n", NtVolumeName);
199 return;
200 }
201
202 /* Compute the needed size to retrieve the mount points */
203 Length = MountPoints.Size;
204
205 /* Allocate the buffer and fill it with test pattern */
206 MountPointsPtr = RtlAllocateHeap(RtlGetProcessHeap(), 0, Length);
207 if (!MountPointsPtr)
208 {
209 skip("Device '%S': Failed to allocate buffer with size %lu)\n",
210 NtVolumeName, Length);
211 return;
212 }
213 RtlFillMemory(MountPointsPtr, Length, 0xCC);
214
215 /* Re-query the mount points with the proper size */
216 Status = NtDeviceIoControlFile(MountMgrHandle,
217 NULL, NULL, NULL,
220 &DeviceName, sizeof(DeviceName),
221 MountPointsPtr, Length);
223 "Device '%S': IOCTL 0x%lx failed unexpectedly, Status 0x%08lx\n",
224 NtVolumeName, IOCTL_MOUNTMGR_QUERY_POINTS, Status);
225
226 if (winetest_debug > 1)
227 {
228 trace("IOCTL_MOUNTMGR_QUERY_POINTS returned:\n"
229 " Size: %lu\n"
230 " NumberOfMountPoints: %lu\n",
231 MountPointsPtr->Size,
232 MountPointsPtr->NumberOfMountPoints);
233
234 trace("Buffer:\n");
235 DumpBuffer(MountPointsPtr, Length);
236 printf("\n");
237 }
238
239 /* Return the buffer */
240 *pMountPointsPtr = MountPointsPtr;
241}
#define ULONG_PTR
Definition: config.h:101
struct _MOUNTMGR_MOUNT_POINT MOUNTMGR_MOUNT_POINT
#define IOCTL_MOUNTMGR_QUERY_POINTS
Definition: mountmgr.h:49
uint32_t ULONG_PTR
Definition: typedefs.h:65

Referenced by Test_QueryDosVolumePathAndPaths().

◆ doesPathExistInMountPoints()

static BOOLEAN doesPathExistInMountPoints ( _In_ PMOUNTMGR_MOUNT_POINTS  MountPoints,
_In_ PUNICODE_STRING  DosPath 
)
static

Definition at line 390 of file QueryDosVolumePaths.c.

393{
394 UNICODE_STRING DosDevicesPrefix = RTL_CONSTANT_STRING(L"\\DosDevices\\");
395 ULONG i;
396 BOOLEAN IsDosVolName;
398
399 IsDosVolName = MOUNTMGR_IS_DOS_VOLUME_NAME(DosPath);
400 /* Temporarily patch \\?\ to \??\ in DosPath for comparison */
401 if (IsDosVolName)
402 DosPath->Buffer[1] = L'?';
403
404 for (i = 0; i < MountPoints->NumberOfMountPoints; ++i)
405 {
406 UNICODE_STRING SymLink;
407
408 SymLink.Length = SymLink.MaximumLength = MountPoints->MountPoints[i].SymbolicLinkNameLength;
409 SymLink.Buffer = (PWCHAR)((ULONG_PTR)MountPoints + MountPoints->MountPoints[i].SymbolicLinkNameOffset);
410
411 if (IS_DRIVE_LETTER(DosPath))
412 {
413 if (RtlPrefixUnicodeString(&DosDevicesPrefix, &SymLink, FALSE))
414 {
415 /* Advance past the prefix */
416 SymLink.Length -= DosDevicesPrefix.Length;
417 SymLink.MaximumLength -= DosDevicesPrefix.Length;
418 SymLink.Buffer += DosDevicesPrefix.Length / sizeof(WCHAR);
419
420 Found = RtlEqualUnicodeString(DosPath, &SymLink, FALSE);
421 }
422 }
423 else if (/*MOUNTMGR_IS_DOS_VOLUME_NAME(DosPath) ||*/ // See above
425 {
426 Found = RtlEqualUnicodeString(DosPath, &SymLink, FALSE);
427 }
428 else
429 {
430 /* Just test for simple string comparison, the path should not be found */
431 Found = RtlEqualUnicodeString(DosPath, &SymLink, FALSE);
432 }
433
434 /* Stop searching if we've found something */
435 if (Found)
436 break;
437 }
438
439 /* Revert \??\ back to \\?\ */
440 if (IsDosVolName)
441 DosPath->Buffer[1] = L'\\';
442
443 return Found;
444}
unsigned char BOOLEAN
#define IS_DRIVE_LETTER(s)
return Found
Definition: dirsup.c:1270
#define FALSE
Definition: types.h:117
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
Definition: glfuncs.h:248
#define MOUNTMGR_IS_DOS_VOLUME_NAME(s)
Definition: mountmgr.h:152
#define MOUNTMGR_IS_NT_VOLUME_NAME(s)
Definition: mountmgr.h:156
NTSYSAPI BOOLEAN NTAPI RtlEqualUnicodeString(PUNICODE_STRING String1, PUNICODE_STRING String2, BOOLEAN CaseInSensitive)
NTSYSAPI BOOLEAN NTAPI RtlPrefixUnicodeString(IN PUNICODE_STRING String1, IN PUNICODE_STRING String2, IN BOOLEAN CaseInSensitive)
#define L(x)
Definition: ntvdm.h:50
USHORT MaximumLength
Definition: env_spec_w32.h:370
#define RTL_CONSTANT_STRING(s)
Definition: tunneltest.c:14
uint16_t * PWCHAR
Definition: typedefs.h:56

Referenced by Test_QueryPoints().

◆ START_TEST()

START_TEST ( QueryDosVolumePaths  )

Definition at line 568 of file QueryDosVolumePaths.c.

569{
570 HANDLE MountMgrHandle;
571 HANDLE hFindVolume;
572 WCHAR szVolumeName[MAX_PATH];
573
574 MountMgrHandle = GetMountMgrHandle(FILE_READ_ATTRIBUTES);
575 if (!MountMgrHandle)
576 {
577 win_skip("MountMgr unavailable: %lu\n", GetLastError());
578 return;
579 }
580
581 hFindVolume = FindFirstVolumeW(szVolumeName, _countof(szVolumeName));
582 if (hFindVolume == INVALID_HANDLE_VALUE)
583 goto otherTests;
584 do
585 {
588
589 /*
590 * The Win32 FindFirst/NextVolumeW() functions convert the '\??\'
591 * prefix in '\??\Volume{...}' to '\\?\' and append a trailing
592 * backslash. Test this behaviour.
593 *
594 * NOTE: these functions actively filter out anything that is NOT
595 * '\??\Volume{...}' returned from IOCTL_MOUNTMGR_QUERY_POINTS.
596 * Thus, it also excludes mount-points specified as drive letters,
597 * like '\DosDevices\C:' .
598 */
599
600 RtlInitUnicodeString(&VolumeName, szVolumeName);
601 Length = VolumeName.Length / sizeof(WCHAR);
602 ok(Length >= 1 && VolumeName.Buffer[Length - 1] == L'\\',
603 "No trailing backslash found\n");
604
605 /* Remove the trailing backslash */
606 if (Length >= 1)
607 {
608 VolumeName.Length -= sizeof(WCHAR);
609 if (szVolumeName[Length - 1] == L'\\')
610 szVolumeName[Length - 1] = UNICODE_NULL;
611 }
612
614 "Invalid DOS volume path returned '%s'\n", wine_dbgstr_us(&VolumeName));
615
616 /* Patch '\\?\' back to '\??\' to convert to an NT path */
617 if (szVolumeName[0] == L'\\' && szVolumeName[1] == L'\\' &&
618 szVolumeName[2] == L'?' && szVolumeName[3] == L'\\')
619 {
620 szVolumeName[1] = L'?';
621 }
622
623 Test_QueryDosVolumePathAndPaths(MountMgrHandle, szVolumeName);
624 } while (FindNextVolumeW(hFindVolume, szVolumeName, _countof(szVolumeName)));
625 FindVolumeClose(hFindVolume);
626
627otherTests:
628 /* Test the drive containing SystemRoot */
629 wcscpy(szVolumeName, L"\\DosDevices\\?:");
630 szVolumeName[sizeof("\\DosDevices\\")-1] = SharedUserData->NtSystemRoot[0];
631 Test_QueryDosVolumePathAndPaths(MountMgrHandle, szVolumeName);
632
633 /* We are done */
634 CloseHandle(MountMgrHandle);
635}
static VOID Test_QueryDosVolumePathAndPaths(_In_ HANDLE MountMgrHandle, _In_ PCWSTR NtVolumeName)
Tests the consistency of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH, IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS ...
wcscpy
#define CloseHandle
Definition: compat.h:739
#define INVALID_HANDLE_VALUE
Definition: compat.h:731
#define MAX_PATH
Definition: compat.h:34
BOOL WINAPI FindNextVolumeW(IN HANDLE handle, IN LPWSTR volume, IN DWORD len)
Definition: volume.c:1082
HANDLE WINAPI FindFirstVolumeW(IN LPWSTR volume, IN DWORD len)
Definition: volume.c:660
BOOL WINAPI FindVolumeClose(IN HANDLE hFindVolume)
Definition: volume.c:741
HANDLE GetMountMgrHandle(_In_ ACCESS_MASK DesiredAccess)
Retrieves a handle to the MountMgr controlling device. The handle should be closed with NtClose() onc...
Definition: utils.c:22
LPCSTR wine_dbgstr_us(const UNICODE_STRING *us)
Definition: utils.c:10
#define FILE_READ_ATTRIBUTES
Definition: nt_native.h:647
#define UNICODE_NULL
#define win_skip
Definition: test.h:164
#define SharedUserData
#define _countof(array)
Definition: sndvol32.h:70
DWORD WINAPI GetLastError(void)
Definition: except.c:1042

◆ Test_QueryDosVolumePath()

static VOID Test_QueryDosVolumePath ( _In_ PCWSTR  NtVolumeName,
_In_ PMOUNTMGR_VOLUME_PATHS  VolumePath 
)
static

Tests the output of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH.

Definition at line 257 of file QueryDosVolumePaths.c.

260{
261 UNICODE_STRING DosPath;
262
263 UNREFERENCED_PARAMETER(NtVolumeName);
264
265 /* The VolumePath should contain one NUL-terminated string (always there?),
266 * plus one final NUL-terminator */
267 ok(VolumePath->MultiSzLength >= 2 * sizeof(UNICODE_NULL),
268 "DOS volume path string too short (length: %lu)\n",
269 VolumePath->MultiSzLength / sizeof(WCHAR));
270 ok(VolumePath->MultiSz[VolumePath->MultiSzLength / sizeof(WCHAR) - 2] == UNICODE_NULL,
271 "Missing NUL-terminator (2)\n");
272 ok(VolumePath->MultiSz[VolumePath->MultiSzLength / sizeof(WCHAR) - 1] == UNICODE_NULL,
273 "Missing NUL-terminator (1)\n");
274
275 /* Build the result string */
276 // RtlInitUnicodeString(&DosPath, VolumePath->MultiSz);
277 DosPath.Length = (USHORT)VolumePath->MultiSzLength - 2 * sizeof(UNICODE_NULL);
278 DosPath.MaximumLength = DosPath.Length + sizeof(UNICODE_NULL);
279 DosPath.Buffer = VolumePath->MultiSz;
280
281 /* The returned DOS path is either a drive letter (*WITHOUT* any
282 * '\DosDevices\' prefix present) or a Win32 file-system reparse point
283 * path, or a volume GUID name in Win32 format, i.e. prefixed by '\\?\' */
285 "Invalid DOS volume path returned '%s'\n", wine_dbgstr_us(&DosPath));
286}
#define IS_DRIVE_LETTER_PFX(s)
#define UNREFERENCED_PARAMETER(P)
Definition: ntbasedef.h:325

Referenced by Test_QueryDosVolumePathAndPaths().

◆ Test_QueryDosVolumePathAndPaths()

static VOID Test_QueryDosVolumePathAndPaths ( _In_ HANDLE  MountMgrHandle,
_In_ PCWSTR  NtVolumeName 
)
static

Tests the consistency of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH, IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS and IOCTL_MOUNTMGR_QUERY_POINTS.

Definition at line 524 of file QueryDosVolumePaths.c.

527{
528 PMOUNTMGR_VOLUME_PATHS VolumePath = NULL;
529 PMOUNTMGR_VOLUME_PATHS VolumePaths = NULL;
530 PMOUNTMGR_MOUNT_POINTS MountPoints = NULL;
531
532 if (winetest_debug > 1)
533 trace("%S\n", NtVolumeName);
534
535 Call_QueryDosVolume_Path_Paths(MountMgrHandle,
536 NtVolumeName,
538 &VolumePath);
539 if (VolumePath)
540 Test_QueryDosVolumePath(NtVolumeName, VolumePath);
541 else
542 skip("Device '%S': IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH failed\n", NtVolumeName);
543
544 Call_QueryDosVolume_Path_Paths(MountMgrHandle,
545 NtVolumeName,
547 &VolumePaths);
548 if (VolumePaths)
549 Test_QueryDosVolumePaths(NtVolumeName, VolumePaths, VolumePath);
550 else
551 skip("Device '%S': IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS failed\n", NtVolumeName);
552
553 Call_QueryPoints(MountMgrHandle, NtVolumeName, &MountPoints);
554 if (MountPoints)
555 Test_QueryPoints(NtVolumeName, MountPoints, VolumePath, VolumePaths);
556 else
557 skip("Device '%S': IOCTL_MOUNTMGR_QUERY_POINTS failed\n", NtVolumeName);
558
559 if (MountPoints)
560 RtlFreeHeap(RtlGetProcessHeap(), 0, MountPoints);
561 if (VolumePaths)
562 RtlFreeHeap(RtlGetProcessHeap(), 0, VolumePaths);
563 if (VolumePath)
564 RtlFreeHeap(RtlGetProcessHeap(), 0, VolumePath);
565}
static VOID Call_QueryDosVolume_Path_Paths(_In_ HANDLE MountMgrHandle, _In_ PCWSTR NtVolumeName, _In_ ULONG IoctlPathOrPaths, _Out_ PMOUNTMGR_VOLUME_PATHS *pVolumePathPtr)
Invokes either IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH or IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS for test...
static VOID Test_QueryPoints(_In_ PCWSTR NtVolumeName, _In_ PMOUNTMGR_MOUNT_POINTS MountPoints, _In_opt_ PMOUNTMGR_VOLUME_PATHS VolumePath, _In_opt_ PMOUNTMGR_VOLUME_PATHS VolumePaths)
Tests the output of IOCTL_MOUNTMGR_QUERY_POINTS.
static VOID Test_QueryDosVolumePaths(_In_ PCWSTR NtVolumeName, _In_ PMOUNTMGR_VOLUME_PATHS VolumePaths, _In_opt_ PMOUNTMGR_VOLUME_PATHS VolumePath)
Tests the output of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS.
static VOID Test_QueryDosVolumePath(_In_ PCWSTR NtVolumeName, _In_ PMOUNTMGR_VOLUME_PATHS VolumePath)
Tests the output of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH.
static VOID Call_QueryPoints(_In_ HANDLE MountMgrHandle, _In_ PCWSTR NtVolumeName, _Out_ PMOUNTMGR_MOUNT_POINTS *pMountPointsPtr)
Invokes IOCTL_MOUNTMGR_QUERY_POINTS for testing, given the volume device name, and returns an allocat...
BOOLEAN NTAPI RtlFreeHeap(IN PVOID HeapHandle, IN ULONG Flags, IN PVOID HeapBase)
Definition: heap.c:634
#define IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATH
Definition: mountmgr.h:147
#define IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS
Definition: mountmgr.h:149

Referenced by START_TEST().

◆ Test_QueryDosVolumePaths()

static VOID Test_QueryDosVolumePaths ( _In_ PCWSTR  NtVolumeName,
_In_ PMOUNTMGR_VOLUME_PATHS  VolumePaths,
_In_opt_ PMOUNTMGR_VOLUME_PATHS  VolumePath 
)
static

Tests the output of IOCTL_MOUNTMGR_QUERY_DOS_VOLUME_PATHS.

Definition at line 292 of file QueryDosVolumePaths.c.

296{
297 UNICODE_STRING DosPath;
298 PCWSTR pMountPoint;
299
300 /* The VolumePaths should contain zero or more NUL-terminated strings,
301 * plus one final NUL-terminator */
302
303 ok(VolumePaths->MultiSzLength >= sizeof(UNICODE_NULL),
304 "DOS volume path string too short (length: %lu)\n",
305 VolumePaths->MultiSzLength / sizeof(WCHAR));
306
307 /* Check for correct double-NUL-termination, if there is at least one string */
308 if (VolumePaths->MultiSzLength >= 2 * sizeof(UNICODE_NULL),
309 VolumePaths->MultiSz[0] != UNICODE_NULL)
310 {
311 ok(VolumePaths->MultiSz[VolumePaths->MultiSzLength / sizeof(WCHAR) - 2] == UNICODE_NULL,
312 "Missing NUL-terminator (2)\n");
313 }
314 /* Check for the final NUL-terminator */
315 ok(VolumePaths->MultiSz[VolumePaths->MultiSzLength / sizeof(WCHAR) - 1] == UNICODE_NULL,
316 "Missing NUL-terminator (1)\n");
317
318 if (winetest_debug > 1)
319 {
320 trace("\n%S =>\n", NtVolumeName);
321 for (pMountPoint = VolumePaths->MultiSz; *pMountPoint;
322 pMountPoint += wcslen(pMountPoint) + 1)
323 {
324 printf(" '%S'\n", pMountPoint);
325 }
326 printf("\n");
327 }
328
329 for (pMountPoint = VolumePaths->MultiSz; *pMountPoint;
330 pMountPoint += wcslen(pMountPoint) + 1)
331 {
332 /* The returned DOS path is either a drive letter (*WITHOUT* any
333 * '\DosDevices\' prefix present) or a Win32 file-system reparse point
334 * path, or a volume GUID name in Win32 format, i.e. prefixed by '\\?\' */
335 RtlInitUnicodeString(&DosPath, pMountPoint);
337 "Invalid DOS volume path returned '%s'\n", wine_dbgstr_us(&DosPath));
338 }
339
340 /*
341 * If provided, verify that the single VolumePath is found at the
342 * first position in the volume paths list, *IF* this is a DOS path;
343 * otherwise if it's a Volume{GUID} path, this means there is no
344 * DOS path associated, and none is listed in the volume paths list.
345 */
346 if (VolumePath)
347 {
348 RtlInitUnicodeString(&DosPath, VolumePath->MultiSz);
349 if (IS_DRIVE_LETTER_PFX(&DosPath))
350 {
351 /*
352 * The single path is a DOS path (single drive letter or Win32
353 * file-system reparse point path). It has to be listed first
354 * in the volume paths list.
355 */
356 UNICODE_STRING FirstPath;
357 BOOLEAN AreEqual;
358
359 ok(VolumePaths->MultiSzLength >= 2 * sizeof(UNICODE_NULL),
360 "DOS VolumePaths list isn't long enough\n");
361 ok(*VolumePaths->MultiSz != UNICODE_NULL,
362 "Empty DOS VolumePaths list\n");
363
364 RtlInitUnicodeString(&FirstPath, VolumePaths->MultiSz);
365 AreEqual = RtlEqualUnicodeString(&DosPath, &FirstPath, FALSE);
366 ok(AreEqual, "DOS paths '%s' and '%s' are not the same!\n",
367 wine_dbgstr_us(&DosPath), wine_dbgstr_us(&FirstPath));
368 }
369 else if (MOUNTMGR_IS_DOS_VOLUME_NAME(&DosPath))
370 {
371 /*
372 * The single "DOS" path is actually a volume name. This means
373 * that it wasn't really mounted, and the volume paths list must
374 * be empty. It contains only the last NUL-terminator.
375 */
376 ok(VolumePaths->MultiSzLength == sizeof(UNICODE_NULL),
377 "DOS VolumePaths list isn't 1 WCHAR long\n");
378 ok(*VolumePaths->MultiSz == UNICODE_NULL,
379 "Non-empty DOS VolumePaths list\n");
380 }
381 else
382 {
383 /* The volume path is invalid (shouldn't happen) */
384 ok(FALSE, "Invalid DOS volume path returned '%s'\n", wine_dbgstr_us(&DosPath));
385 }
386 }
387}
_CRTIMP size_t __cdecl wcslen(_In_z_ const wchar_t *_Str)
const uint16_t * PCWSTR
Definition: typedefs.h:57

Referenced by Test_QueryDosVolumePathAndPaths().

◆ Test_QueryPoints()

static VOID Test_QueryPoints ( _In_ PCWSTR  NtVolumeName,
_In_ PMOUNTMGR_MOUNT_POINTS  MountPoints,
_In_opt_ PMOUNTMGR_VOLUME_PATHS  VolumePath,
_In_opt_ PMOUNTMGR_VOLUME_PATHS  VolumePaths 
)
static

Tests the output of IOCTL_MOUNTMGR_QUERY_POINTS.

Definition at line 450 of file QueryDosVolumePaths.c.

455{
456 UNICODE_STRING DosPath;
457 PCWSTR pMountPoint;
458 BOOLEAN ExpectedFound, Found;
459
460 if (winetest_debug > 1)
461 {
462 ULONG i;
463 trace("\n%S =>\n", NtVolumeName);
464 for (i = 0; i < MountPoints->NumberOfMountPoints; ++i)
465 {
466 UNICODE_STRING DevName, SymLink;
467
468 DevName.Length = DevName.MaximumLength = MountPoints->MountPoints[i].DeviceNameLength;
469 DevName.Buffer = (PWCHAR)((ULONG_PTR)MountPoints + MountPoints->MountPoints[i].DeviceNameOffset);
470
471 SymLink.Length = SymLink.MaximumLength = MountPoints->MountPoints[i].SymbolicLinkNameLength;
472 SymLink.Buffer = (PWCHAR)((ULONG_PTR)MountPoints + MountPoints->MountPoints[i].SymbolicLinkNameOffset);
473
474 printf(" '%s' -- '%s'\n", wine_dbgstr_us(&DevName), wine_dbgstr_us(&SymLink));
475 }
476 printf("\n");
477 }
478
479 /*
480 * The Win32 file-system reparse point paths are NOT listed amongst
481 * the mount points. Only the drive letter and the volume GUID name
482 * are, but in an NT format (using '\DosDevices\' or '\??\' prefixes).
483 */
484
485 if (VolumePath)
486 {
487 /* VolumePath can either be a drive letter (usual case), a Win32
488 * reparse point path (if the volume is mounted only with these),
489 * or a volume GUID name (if the volume is NOT mounted). */
490 RtlInitUnicodeString(&DosPath, VolumePath->MultiSz);
491 ExpectedFound = (IS_DRIVE_LETTER(&DosPath) || MOUNTMGR_IS_DOS_VOLUME_NAME(&DosPath));
492 Found = doesPathExistInMountPoints(MountPoints, &DosPath);
493 ok(Found == ExpectedFound,
494 "DOS path '%s' %sfound in the mount points list, expected %sto be found\n",
495 wine_dbgstr_us(&DosPath), Found ? "" : "NOT ", ExpectedFound ? "" : "NOT ");
496 }
497
498 if (VolumePaths)
499 {
500 /* VolumePaths only contains a drive letter (usual case) or a Win32
501 * reparse point path (if the volume is mounted only with these).
502 * If the volume is NOT mounted, VolumePaths does not list the
503 * volume GUID name, contrary to VolumePath. */
504 for (pMountPoint = VolumePaths->MultiSz; *pMountPoint;
505 pMountPoint += wcslen(pMountPoint) + 1)
506 {
507 /* Only the drive letter (but NOT the volume GUID name!) can be found in the list */
508 RtlInitUnicodeString(&DosPath, pMountPoint);
509 ExpectedFound = IS_DRIVE_LETTER(&DosPath);
510 Found = doesPathExistInMountPoints(MountPoints, &DosPath);
511 ok(Found == ExpectedFound,
512 "DOS path '%s' %sfound in the mount points list, expected %sto be found\n",
513 wine_dbgstr_us(&DosPath), Found ? "" : "NOT ", ExpectedFound ? "" : "NOT ");
514 }
515 }
516}
static BOOLEAN doesPathExistInMountPoints(_In_ PMOUNTMGR_MOUNT_POINTS MountPoints, _In_ PUNICODE_STRING DosPath)

Referenced by Test_QueryDosVolumePathAndPaths().