VirtualBox

source: vbox/trunk/src/VBox/Runtime/nt/RTNtPathExpand8dot3Path.cpp@ 62956

Last change on this file since 62956 was 60481, checked in by vboxsync, 9 years ago

IPRT: Added testcase for RTNtPathExpand8dot3Path and RTNtPathFindPossible8dot3Name, fixing bug in the former. include\ src\VBox\Runtime\

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.5 KB
Line 
1/* $Id: RTNtPathExpand8dot3Path.cpp 60481 2016-04-13 20:14:41Z vboxsync $ */
2/** @file
3 * IPRT - Native NT, RTNtPathExpand8dot3Path.
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#ifdef IN_SUP_HARDENED_R3
33# include <iprt/nt/nt-and-windows.h>
34#else
35# include <iprt/nt/nt.h>
36#endif
37
38#include <iprt/mem.h>
39#include <iprt/string.h>
40
41
42
43/**
44 * Fixes up a path possibly containing one or more alternative 8-dot-3 style
45 * components.
46 *
47 * The path is fixed up in place. Errors are ignored.
48 *
49 * @returns VINF_SUCCESS if it all went smoothly, informational status codes
50 * indicating the nature of last problem we ran into.
51 *
52 * @param pUniStr The path to fix up. MaximumLength is the max buffer
53 * length.
54 * @param fPathOnly Whether to only process the path and leave the filename
55 * as passed in.
56 */
57RTDECL(int) RTNtPathExpand8dot3Path(PUNICODE_STRING pUniStr, bool fPathOnly)
58{
59 int rc = VINF_SUCCESS;
60
61 /*
62 * We could use FileNormalizedNameInformation here and slap the volume device
63 * path in front of the result, but it's only supported since windows 8.0
64 * according to some docs... So we expand all supicious names.
65 */
66 union fix8dot3tmp
67 {
68 FILE_BOTH_DIR_INFORMATION Info;
69 uint8_t abBuffer[sizeof(FILE_BOTH_DIR_INFORMATION) + 2048 * sizeof(WCHAR)];
70 } *puBuf = NULL;
71
72
73 PRTUTF16 pwszFix = pUniStr->Buffer;
74 while (*pwszFix)
75 {
76 pwszFix = RTNtPathFindPossible8dot3Name(pwszFix);
77 if (pwszFix == NULL)
78 break;
79
80 RTUTF16 wc;
81 PRTUTF16 pwszFixEnd = pwszFix;
82 while ((wc = *pwszFixEnd) != '\0' && wc != '\\' && wc != '/')
83 pwszFixEnd++;
84 if (wc == '\0' && fPathOnly)
85 break;
86
87 if (!puBuf)
88 {
89 puBuf = (union fix8dot3tmp *)RTMemAlloc(sizeof(*puBuf));
90 if (!puBuf)
91 break;
92 }
93
94 RTUTF16 const wcSaved = *pwszFix;
95 *pwszFix = '\0'; /* paranoia. */
96
97 UNICODE_STRING NtDir;
98 NtDir.Buffer = pUniStr->Buffer;
99 NtDir.Length = NtDir.MaximumLength = (USHORT)((pwszFix - pUniStr->Buffer) * sizeof(WCHAR));
100
101 HANDLE hDir = RTNT_INVALID_HANDLE_VALUE;
102 IO_STATUS_BLOCK Ios = RTNT_IO_STATUS_BLOCK_INITIALIZER;
103
104 OBJECT_ATTRIBUTES ObjAttr;
105 InitializeObjectAttributes(&ObjAttr, &NtDir, OBJ_CASE_INSENSITIVE, NULL /*hRootDir*/, NULL /*pSecDesc*/);
106#ifdef IN_RING0
107 ObjAttr.Attributes |= OBJ_KERNEL_HANDLE;
108#endif
109
110 NTSTATUS rcNt = NtCreateFile(&hDir,
111 FILE_READ_DATA | SYNCHRONIZE,
112 &ObjAttr,
113 &Ios,
114 NULL /* Allocation Size*/,
115 FILE_ATTRIBUTE_NORMAL,
116 FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
117 FILE_OPEN,
118 FILE_DIRECTORY_FILE | FILE_OPEN_FOR_BACKUP_INTENT | FILE_SYNCHRONOUS_IO_NONALERT,
119 NULL /*EaBuffer*/,
120 0 /*EaLength*/);
121 *pwszFix = wcSaved;
122 if (NT_SUCCESS(rcNt))
123 {
124 RT_ZERO(*puBuf);
125
126 IO_STATUS_BLOCK Ios = RTNT_IO_STATUS_BLOCK_INITIALIZER;
127 UNICODE_STRING NtFilterStr;
128 NtFilterStr.Buffer = pwszFix;
129 NtFilterStr.Length = (USHORT)((uintptr_t)pwszFixEnd - (uintptr_t)pwszFix);
130 NtFilterStr.MaximumLength = NtFilterStr.Length;
131 rcNt = NtQueryDirectoryFile(hDir,
132 NULL /* Event */,
133 NULL /* ApcRoutine */,
134 NULL /* ApcContext */,
135 &Ios,
136 puBuf,
137 sizeof(*puBuf) - sizeof(WCHAR),
138 FileBothDirectoryInformation,
139 FALSE /*ReturnSingleEntry*/,
140 &NtFilterStr,
141 FALSE /*RestartScan */);
142 if (NT_SUCCESS(rcNt) && puBuf->Info.NextEntryOffset == 0) /* There shall only be one entry matching... */
143 {
144 uint32_t offName = puBuf->Info.FileNameLength / sizeof(WCHAR);
145 while (offName > 0 && puBuf->Info.FileName[offName - 1] != '\\' && puBuf->Info.FileName[offName - 1] != '/')
146 offName--;
147 uint32_t cwcNameNew = (puBuf->Info.FileNameLength / sizeof(WCHAR)) - offName;
148 uint32_t cwcNameOld = (uint32_t)(pwszFixEnd - pwszFix);
149
150 if (cwcNameOld == cwcNameNew)
151 memcpy(pwszFix, &puBuf->Info.FileName[offName], cwcNameNew * sizeof(WCHAR));
152 else if ( pUniStr->Length + cwcNameNew * sizeof(WCHAR) - cwcNameOld * sizeof(WCHAR) + sizeof(WCHAR)
153 <= pUniStr->MaximumLength)
154 {
155 size_t cwcLeft = pUniStr->Length - (pwszFixEnd - pUniStr->Buffer) * sizeof(WCHAR) + sizeof(WCHAR);
156 memmove(&pwszFix[cwcNameNew], pwszFixEnd, cwcLeft * sizeof(WCHAR));
157 pUniStr->Length -= (USHORT)(cwcNameOld * sizeof(WCHAR));
158 pUniStr->Length += (USHORT)(cwcNameNew * sizeof(WCHAR));
159 pwszFixEnd -= cwcNameOld;
160 pwszFixEnd += cwcNameNew;
161 memcpy(pwszFix, &puBuf->Info.FileName[offName], cwcNameNew * sizeof(WCHAR));
162 }
163 else
164 rc = VINF_BUFFER_OVERFLOW;
165 }
166 else if (NT_SUCCESS(rcNt))
167 rc = -VERR_DUPLICATE;
168 else
169 {
170 rc = -RTErrConvertFromNtStatus(rcNt);
171 if (rc < 0)
172 rc = -rc;
173 }
174
175 NtClose(hDir);
176 }
177
178 /* Advance */
179 pwszFix = pwszFixEnd;
180 }
181
182 if (puBuf)
183 RTMemFree(puBuf);
184
185 if (pUniStr->Length < pUniStr->MaximumLength)
186 pUniStr->Buffer[pUniStr->Length / sizeof(WCHAR)] = '\0';
187
188 return rc;
189}
190
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