VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/nt/pathint-nt.cpp@ 65282

Last change on this file since 65282 was 64640, checked in by vboxsync, 8 years ago

IPRT: Implemented native NT version of RTPathQueryInfoEx that provides INode number and link count.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 20.2 KB
Line 
1/* $Id: pathint-nt.cpp 64640 2016-11-10 15:27:09Z vboxsync $ */
2/** @file
3 * IPRT - Native NT, Internal Path stuff.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_FS
32#include "internal-r3-nt.h"
33
34#include <iprt/path.h>
35#include <iprt/string.h>
36#include <iprt/err.h>
37#include <iprt/assert.h>
38
39
40/*********************************************************************************************************************************
41* Global Variables *
42*********************************************************************************************************************************/
43static char const g_szPrefixUnc[] = "\\??\\UNC\\";
44static char const g_szPrefix[] = "\\??\\";
45
46
47/**
48 * Handles the pass thru case for UTF-8 input.
49 * Win32 path uses "\\?\" prefix which is converted to "\??\" NT prefix.
50 *
51 * @returns IPRT status code.
52 * @param pNtName Where to return the NT name.
53 * @param phRootDir Where to return the root handle, if applicable.
54 * @param pszPath The UTF-8 path.
55 */
56static int rtNtPathFromWinUtf8PassThru(struct _UNICODE_STRING *pNtName, PHANDLE phRootDir, const char *pszPath)
57{
58 PRTUTF16 pwszPath = NULL;
59 size_t cwcLen;
60 int rc = RTStrToUtf16Ex(pszPath, RTSTR_MAX, &pwszPath, 0, &cwcLen);
61 if (RT_SUCCESS(rc))
62 {
63 if (cwcLen < _32K - 1)
64 {
65 pwszPath[0] = '\\';
66 pwszPath[1] = '?';
67 pwszPath[2] = '?';
68 pwszPath[3] = '\\';
69
70 pNtName->Buffer = pwszPath;
71 pNtName->Length = (uint16_t)(cwcLen * sizeof(RTUTF16));
72 pNtName->MaximumLength = pNtName->Length + sizeof(RTUTF16);
73 *phRootDir = NULL;
74 return VINF_SUCCESS;
75 }
76
77 RTUtf16Free(pwszPath);
78 rc = VERR_FILENAME_TOO_LONG;
79 }
80 return rc;
81}
82
83
84/**
85 * Handles the pass thru case for UTF-16 input.
86 * Win32 path uses "\\?\" prefix which is converted to "\??\" NT prefix.
87 *
88 * @returns IPRT status code.
89 * @param pNtName Where to return the NT name.
90 * @param phRootDir Stores NULL here, as we don't use it.
91 * @param pwszWinPath The UTF-16 windows-style path.
92 * @param cwcWinPath The length of the windows-style input path.
93 */
94static int rtNtPathFromWinUtf16PassThru(struct _UNICODE_STRING *pNtName, PHANDLE phRootDir,
95 PCRTUTF16 pwszWinPath, size_t cwcWinPath)
96{
97 /* Check length and allocate memory for it. */
98 int rc;
99 if (cwcWinPath < _32K - 1)
100 {
101 PRTUTF16 pwszNtPath = (PRTUTF16)RTUtf16Alloc((cwcWinPath + 1) * sizeof(RTUTF16));
102 if (pwszNtPath)
103 {
104 /* Intialize the path. */
105 pwszNtPath[0] = '\\';
106 pwszNtPath[1] = '?';
107 pwszNtPath[2] = '?';
108 pwszNtPath[3] = '\\';
109 memcpy(pwszNtPath + 4, pwszWinPath + 4, (cwcWinPath - 4) * sizeof(RTUTF16));
110 pwszNtPath[cwcWinPath] = '\0';
111
112 /* Initialize the return values. */
113 pNtName->Buffer = pwszNtPath;
114 pNtName->Length = (uint16_t)(cwcWinPath * sizeof(RTUTF16));
115 pNtName->MaximumLength = pNtName->Length + sizeof(RTUTF16);
116 *phRootDir = NULL;
117
118 rc = VINF_SUCCESS;
119 }
120 else
121 rc = VERR_NO_UTF16_MEMORY;
122 }
123 else
124 rc = VERR_FILENAME_TOO_LONG;
125 return rc;
126}
127
128
129
130
131
132/**
133 * Converts the path to UTF-16 and sets all the return values.
134 *
135 * @returns IPRT status code.
136 * @param pNtName Where to return the NT name.
137 * @param phRootDir Where to return the root handle, if applicable.
138 * @param pszPath The UTF-8 path.
139 */
140static int rtNtPathUtf8ToUniStr(struct _UNICODE_STRING *pNtName, PHANDLE phRootDir, const char *pszPath)
141{
142 PRTUTF16 pwszPath = NULL;
143 size_t cwcLen;
144 int rc = RTStrToUtf16Ex(pszPath, RTSTR_MAX, &pwszPath, 0, &cwcLen);
145 if (RT_SUCCESS(rc))
146 {
147 if (cwcLen < _32K - 1)
148 {
149 pNtName->Buffer = pwszPath;
150 pNtName->Length = (uint16_t)(cwcLen * sizeof(RTUTF16));
151 pNtName->MaximumLength = pNtName->Length + sizeof(RTUTF16);
152 *phRootDir = NULL;
153 return VINF_SUCCESS;
154 }
155
156 RTUtf16Free(pwszPath);
157 rc = VERR_FILENAME_TOO_LONG;
158 }
159 return rc;
160}
161
162
163/**
164 * Converts a windows-style path to NT format and encoding.
165 *
166 * @returns IPRT status code.
167 * @param pNtName Where to return the NT name. Free using
168 * rtTNtPathToNative.
169 * @param phRootDir Where to return the root handle, if applicable.
170 * @param pszPath The UTF-8 path.
171 */
172static int rtNtPathToNative(struct _UNICODE_STRING *pNtName, PHANDLE phRootDir, const char *pszPath)
173{
174/** @todo This code sucks a bit performance wise, esp. calling
175 * generic RTPathAbs. Too many buffers involved, I think. */
176
177 /*
178 * Very simple conversion of a win32-like path into an NT path.
179 */
180 const char *pszPrefix = g_szPrefix;
181 size_t cchPrefix = sizeof(g_szPrefix) - 1;
182 size_t cchSkip = 0;
183
184 if ( RTPATH_IS_SLASH(pszPath[0])
185 && RTPATH_IS_SLASH(pszPath[1])
186 && !RTPATH_IS_SLASH(pszPath[2])
187 && pszPath[2])
188 {
189 if ( pszPath[2] == '?'
190 && RTPATH_IS_SLASH(pszPath[3]))
191 return rtNtPathFromWinUtf8PassThru(pNtName, phRootDir, pszPath);
192
193#ifdef IPRT_WITH_NT_PATH_PASSTHRU
194 /* Special hack: The path starts with "\\\\!\\", we will skip past the bang and pass it thru. */
195 if ( pszPath[2] == '!'
196 && RTPATH_IS_SLASH(pszPath[3]))
197 return rtNtPathUtf8ToUniStr(pNtName, phRootDir, pszPath + 3);
198#endif
199
200 if ( pszPath[2] == '.'
201 && RTPATH_IS_SLASH(pszPath[3]))
202 {
203 /*
204 * Device path.
205 * Note! I suspect \\.\stuff\..\otherstuff may be handled differently by windows.
206 */
207 cchSkip = 4;
208 }
209 else
210 {
211 /* UNC */
212 pszPrefix = g_szPrefixUnc;
213 cchPrefix = sizeof(g_szPrefixUnc) - 1;
214 cchSkip = 2;
215 }
216 }
217
218 /*
219 * Straighten out all .. and uncessary . references and convert slashes.
220 */
221 char szPath[RTPATH_MAX];
222 int rc = RTPathAbs(pszPath, &szPath[cchPrefix - cchSkip], sizeof(szPath) - (cchPrefix - cchSkip));
223 if (RT_FAILURE(rc))
224 return rc;
225
226 /*
227 * Add prefix and convert it to UTF16.
228 */
229 memcpy(szPath, pszPrefix, cchPrefix);
230 return rtNtPathUtf8ToUniStr(pNtName, phRootDir, szPath);
231}
232
233
234/**
235 * Converts a windows-style path to NT format and encoding.
236 *
237 * @returns IPRT status code.
238 * @param pNtName Where to return the NT name. Free using
239 * RTNtPathToNative.
240 * @param phRootDir Where to return the root handle, if applicable.
241 * @param pszPath The UTF-8 path.
242 */
243RTDECL(int) RTNtPathFromWinUtf8(struct _UNICODE_STRING *pNtName, PHANDLE phRootDir, const char *pszPath)
244{
245 return rtNtPathToNative(pNtName, phRootDir, pszPath);
246}
247
248
249/**
250 * Converts a UTF-16 windows-style path to NT format.
251 *
252 * @returns IPRT status code.
253 * @param pNtName Where to return the NT name. Free using
254 * RTNtPathFree.
255 * @param phRootDir Where to return the root handle, if applicable.
256 * @param pwszWinPath The UTF-16 windows-style path.
257 * @param cwcWinPath The max length of the windows-style path in
258 * RTUTF16 units. Use RTSTR_MAX if unknown and @a
259 * pwszWinPath is correctly terminated.
260 */
261RTDECL(int) RTNtPathFromWinUtf16Ex(struct _UNICODE_STRING *pNtName, HANDLE *phRootDir, PCRTUTF16 pwszWinPath, size_t cwcWinPath)
262{
263 /*
264 * Validate the input, calculating the correct length.
265 */
266 if (cwcWinPath == 0 || *pwszWinPath == '\0')
267 return VERR_INVALID_NAME;
268
269 RTUtf16NLenEx(pwszWinPath, cwcWinPath, &cwcWinPath);
270 int rc = RTUtf16ValidateEncodingEx(pwszWinPath, cwcWinPath, 0);
271 if (RT_FAILURE(rc))
272 return rc;
273
274 /*
275 * Very simple conversion of a win32-like path into an NT path.
276 */
277 const char *pszPrefix = g_szPrefix;
278 size_t cchPrefix = sizeof(g_szPrefix) - 1;
279 size_t cchSkip = 0;
280
281 if ( RTPATH_IS_SLASH(pwszWinPath[0])
282 && cwcWinPath >= 3
283 && RTPATH_IS_SLASH(pwszWinPath[1])
284 && !RTPATH_IS_SLASH(pwszWinPath[2]) )
285 {
286 if ( pwszWinPath[2] == '?'
287 && cwcWinPath >= 4
288 && RTPATH_IS_SLASH(pwszWinPath[3]))
289 return rtNtPathFromWinUtf16PassThru(pNtName, phRootDir, pwszWinPath, cwcWinPath);
290
291#ifdef IPRT_WITH_NT_PATH_PASSTHRU
292 /* Special hack: The path starts with "\\\\!\\", we will skip past the bang and pass it thru. */
293 if ( pwszWinPath[2] == '!'
294 && cwcWinPath >= 4
295 && RTPATH_IS_SLASH(pwszWinPath[3]))
296 {
297 pwszWinPath += 3;
298 cwcWinPath -= 3;
299 if (cwcWinPath < _32K - 1)
300 {
301 PRTUTF16 pwszNtPath = RTUtf16Alloc((cwcWinPath + 1) * sizeof(RTUTF16));
302 if (pwszNtPath)
303 {
304 memcpy(pwszNtPath, pwszWinPath, cwcWinPath * sizeof(RTUTF16));
305 pwszNtPath[cwcWinPath] = '\0';
306 pNtName->Buffer = pwszNtPath;
307 pNtName->Length = (uint16_t)(cwcWinPath * sizeof(RTUTF16));
308 pNtName->MaximumLength = pNtName->Length + sizeof(RTUTF16);
309 *phRootDir = NULL;
310 return VINF_SUCCESS;
311 }
312 rc = VERR_NO_UTF16_MEMORY;
313 }
314 else
315 rc = VERR_FILENAME_TOO_LONG;
316 return rc;
317 }
318#endif
319
320 if ( pwszWinPath[2] == '.'
321 && cwcWinPath >= 4
322 && RTPATH_IS_SLASH(pwszWinPath[3]))
323 {
324 /*
325 * Device path.
326 * Note! I suspect \\.\stuff\..\otherstuff may be handled differently by windows.
327 */
328 cchSkip = 4;
329 }
330 else
331 {
332 /* UNC */
333 pszPrefix = g_szPrefixUnc;
334 cchPrefix = sizeof(g_szPrefixUnc) - 1;
335 cchSkip = 2;
336 }
337 }
338
339 /*
340 * Straighten out all .. and unnecessary . references and convert slashes.
341 */
342 char szAbsPath[RTPATH_MAX];
343 char szRelPath[RTPATH_MAX];
344 char *pszRelPath = szRelPath;
345 size_t cchRelPath;
346 rc = RTUtf16ToUtf8Ex(pwszWinPath, cwcWinPath, &pszRelPath, sizeof(szRelPath), &cchRelPath);
347 if (RT_SUCCESS(rc))
348 rc = RTPathAbs(szRelPath, &szAbsPath[cchPrefix - cchSkip], sizeof(szAbsPath) - (cchPrefix - cchSkip));
349 if (RT_SUCCESS(rc))
350 {
351 /*
352 * Add prefix and convert it to UTF16.
353 */
354 memcpy(szAbsPath, pszPrefix, cchPrefix);
355 return rtNtPathUtf8ToUniStr(pNtName, phRootDir, szAbsPath);
356 }
357 return rc;
358}
359
360
361/**
362 * Ensures that the NT string has sufficient storage to hold @a cwcMin RTUTF16
363 * chars plus a terminator.
364 *
365 * The NT string must have been returned by RTNtPathFromWinUtf8 or
366 * RTNtPathFromWinUtf16Ex.
367 *
368 * @returns IPRT status code.
369 * @param pNtName The NT path string.
370 * @param cwcMin The minimum number of RTUTF16 chars. Max 32767.
371 * @sa RTNtPathFree
372 */
373RTDECL(int) RTNtPathEnsureSpace(struct _UNICODE_STRING *pNtName, size_t cwcMin)
374{
375 if (pNtName->MaximumLength / sizeof(RTUTF16) > cwcMin)
376 return VINF_SUCCESS;
377
378 AssertReturn(cwcMin < _64K / sizeof(RTUTF16), VERR_OUT_OF_RANGE);
379
380 size_t const cbMin = (cwcMin + 1) * sizeof(RTUTF16);
381 int rc = RTUtf16Realloc(&pNtName->Buffer, cbMin);
382 if (RT_SUCCESS(rc))
383 pNtName->MaximumLength = (uint16_t)cbMin;
384 return rc;
385}
386
387
388/**
389 * Frees the native path and root handle.
390 *
391 * @param pNtName The NT path after a successful rtNtPathToNative
392 * call.
393 * @param phRootDir The root handle variable from the same call.
394 */
395static void rtNtPathFreeNative(struct _UNICODE_STRING *pNtName, PHANDLE phRootDir)
396{
397 RTUtf16Free(pNtName->Buffer);
398 pNtName->Buffer = NULL;
399
400 RT_NOREF_PV(phRootDir); /* never returned by rtNtPathToNative */
401}
402
403
404/**
405 * Frees the native path and root handle.
406 *
407 * @param pNtName The NT path from a successful RTNtPathToNative
408 * or RTNtPathFromWinUtf16Ex call.
409 * @param phRootDir The root handle variable from the same call.
410 */
411RTDECL(void) RTNtPathFree(struct _UNICODE_STRING *pNtName, HANDLE *phRootDir)
412{
413 rtNtPathFreeNative(pNtName, phRootDir);
414}
415
416
417/**
418 * Wrapper around NtCreateFile.
419 *
420 * @returns IPRT status code.
421 * @param pszPath The UTF-8 path.
422 * @param fDesiredAccess See NtCreateFile.
423 * @param fFileAttribs See NtCreateFile.
424 * @param fShareAccess See NtCreateFile.
425 * @param fCreateDisposition See NtCreateFile.
426 * @param fCreateOptions See NtCreateFile.
427 * @param fObjAttribs The OBJECT_ATTRIBUTES::Attributes value, see
428 * NtCreateFile and InitializeObjectAttributes.
429 * @param phHandle Where to return the handle.
430 * @param puAction Where to return the action taken. Optional.
431 */
432RTDECL(int) RTNtPathOpen(const char *pszPath, ACCESS_MASK fDesiredAccess, ULONG fFileAttribs, ULONG fShareAccess,
433 ULONG fCreateDisposition, ULONG fCreateOptions, ULONG fObjAttribs,
434 PHANDLE phHandle, PULONG_PTR puAction)
435{
436 *phHandle = RTNT_INVALID_HANDLE_VALUE;
437
438 HANDLE hRootDir;
439 UNICODE_STRING NtName;
440 int rc = rtNtPathToNative(&NtName, &hRootDir, pszPath);
441 if (RT_SUCCESS(rc))
442 {
443 HANDLE hFile = RTNT_INVALID_HANDLE_VALUE;
444 IO_STATUS_BLOCK Ios = RTNT_IO_STATUS_BLOCK_INITIALIZER;
445 OBJECT_ATTRIBUTES ObjAttr;
446 InitializeObjectAttributes(&ObjAttr, &NtName, fObjAttribs, hRootDir, NULL);
447
448 NTSTATUS rcNt = NtCreateFile(&hFile,
449 fDesiredAccess,
450 &ObjAttr,
451 &Ios,
452 NULL /* AllocationSize*/,
453 fFileAttribs,
454 fShareAccess,
455 fCreateDisposition,
456 fCreateOptions,
457 NULL /*EaBuffer*/,
458 0 /*EaLength*/);
459 if (NT_SUCCESS(rcNt))
460 {
461 if (puAction)
462 *puAction = Ios.Information;
463 *phHandle = hFile;
464 rc = VINF_SUCCESS;
465 }
466 else
467 rc = RTErrConvertFromNtStatus(rcNt);
468 rtNtPathFreeNative(&NtName, &hRootDir);
469 }
470 return rc;
471}
472
473
474/**
475 * Wrapper around NtCreateFile.
476 *
477 * @returns IPRT status code.
478 * @param pszPath The UTF-8 path.
479 * @param fDesiredAccess See NtCreateFile.
480 * @param fShareAccess See NtCreateFile.
481 * @param fCreateOptions See NtCreateFile.
482 * @param fObjAttribs The OBJECT_ATTRIBUTES::Attributes value, see
483 * NtCreateFile and InitializeObjectAttributes.
484 * @param phHandle Where to return the handle.
485 * @param pfObjDir If not NULL, the variable pointed to will be set
486 * to @c true if we opened an object directory and
487 * @c false if we opened an directory file (normal
488 * directory).
489 */
490RTDECL(int) RTNtPathOpenDir(const char *pszPath, ACCESS_MASK fDesiredAccess, ULONG fShareAccess, ULONG fCreateOptions,
491 ULONG fObjAttribs, PHANDLE phHandle, bool *pfObjDir)
492{
493 *phHandle = RTNT_INVALID_HANDLE_VALUE;
494
495 HANDLE hRootDir;
496 UNICODE_STRING NtName;
497 int rc = rtNtPathToNative(&NtName, &hRootDir, pszPath);
498 if (RT_SUCCESS(rc))
499 {
500 HANDLE hFile = RTNT_INVALID_HANDLE_VALUE;
501 IO_STATUS_BLOCK Ios = RTNT_IO_STATUS_BLOCK_INITIALIZER;
502 OBJECT_ATTRIBUTES ObjAttr;
503 InitializeObjectAttributes(&ObjAttr, &NtName, fObjAttribs, hRootDir, NULL);
504
505 NTSTATUS rcNt = NtCreateFile(&hFile,
506 fDesiredAccess,
507 &ObjAttr,
508 &Ios,
509 NULL /* AllocationSize*/,
510 FILE_ATTRIBUTE_NORMAL,
511 fShareAccess,
512 FILE_OPEN,
513 fCreateOptions,
514 NULL /*EaBuffer*/,
515 0 /*EaLength*/);
516 if (NT_SUCCESS(rcNt))
517 {
518 if (pfObjDir)
519 *pfObjDir = false;
520 *phHandle = hFile;
521 rc = VINF_SUCCESS;
522 }
523#ifdef IPRT_WITH_NT_PATH_PASSTHRU
524 else if ( pfObjDir
525 && (rcNt == STATUS_OBJECT_NAME_INVALID || rcNt == STATUS_OBJECT_TYPE_MISMATCH)
526 && RTPATH_IS_SLASH(pszPath[0])
527 && RTPATH_IS_SLASH(pszPath[1])
528 && pszPath[2] == '!'
529 && RTPATH_IS_SLASH(pszPath[3]))
530 {
531 /* Strip trailing slash. */
532 if ( NtName.Length > 2
533 && RTPATH_IS_SLASH(NtName.Buffer[(NtName.Length / 2) - 1]))
534 NtName.Length -= 2;
535
536 /* Rought conversion of the access flags. */
537 ULONG fObjDesiredAccess = 0;
538 if (fDesiredAccess & (GENERIC_ALL | STANDARD_RIGHTS_ALL))
539 fObjDesiredAccess = DIRECTORY_ALL_ACCESS;
540 else
541 {
542 if (fDesiredAccess & (FILE_GENERIC_WRITE | GENERIC_WRITE | STANDARD_RIGHTS_WRITE))
543 fObjDesiredAccess |= DIRECTORY_CREATE_OBJECT | DIRECTORY_CREATE_OBJECT;
544 if ( (fDesiredAccess & (FILE_LIST_DIRECTORY | FILE_GENERIC_READ | GENERIC_READ | STANDARD_RIGHTS_READ))
545 || !fObjDesiredAccess)
546 fObjDesiredAccess |= DIRECTORY_QUERY | FILE_LIST_DIRECTORY;
547 }
548
549 rcNt = NtOpenDirectoryObject(&hFile, fObjDesiredAccess, &ObjAttr);
550 if (NT_SUCCESS(rcNt))
551 {
552 *pfObjDir = true;
553 *phHandle = hFile;
554 rc = VINF_SUCCESS;
555 }
556 else
557 rc = RTErrConvertFromNtStatus(rcNt);
558 }
559#endif
560 else
561 rc = RTErrConvertFromNtStatus(rcNt);
562 rtNtPathFreeNative(&NtName, &hRootDir);
563 }
564 return rc;
565}
566
567
568/**
569 * Closes an handled open by rtNtPathOpen.
570 *
571 * @returns IPRT status code
572 * @param hHandle The handle value.
573 */
574RTDECL(int) RTNtPathClose(HANDLE hHandle)
575{
576 NTSTATUS rcNt = NtClose(hHandle);
577 if (NT_SUCCESS(rcNt))
578 return VINF_SUCCESS;
579 return RTErrConvertFromNtStatus(rcNt);
580}
581
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette