VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/SharedFolders/driver/vbsfhlp.cpp@ 91546

Last change on this file since 91546 was 82968, checked in by vboxsync, 5 years ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.0 KB
Line 
1/* $Id: vbsfhlp.cpp 82968 2020-02-04 10:35:17Z vboxsync $ */
2/** @file
3 * VirtualBox Windows Guest Shared Folders - File System Driver system helpers
4 */
5
6/*
7 * Copyright (C) 2012-2020 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
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include "vbsf.h"
23#include <iprt/err.h>
24
25
26/*********************************************************************************************************************************
27* Global Variables *
28*********************************************************************************************************************************/
29#ifdef DEBUG
30static int volatile g_cAllocations = 0;
31#endif
32
33
34/**
35 * Convert VBox error code to NT status code
36 *
37 * @returns NT status code
38 * @param vrc VBox status code.
39 *
40 */
41NTSTATUS vbsfNtVBoxStatusToNt(int vrc)
42{
43 NTSTATUS Status;
44
45 switch (vrc)
46 {
47 case VINF_SUCCESS:
48 Status = STATUS_SUCCESS;
49 break;
50
51 case VERR_ACCESS_DENIED:
52 Status = STATUS_ACCESS_DENIED;
53 break;
54
55 case VERR_NO_MORE_FILES:
56 Status = STATUS_NO_MORE_FILES;
57 break;
58
59 case VERR_PATH_NOT_FOUND:
60 Status = STATUS_OBJECT_PATH_NOT_FOUND;
61 break;
62
63 case VERR_FILE_NOT_FOUND:
64 Status = STATUS_OBJECT_NAME_NOT_FOUND;
65 break;
66
67 case VERR_DIR_NOT_EMPTY:
68 Status = STATUS_DIRECTORY_NOT_EMPTY;
69 break;
70
71 case VERR_SHARING_VIOLATION:
72 Status = STATUS_SHARING_VIOLATION;
73 break;
74
75 case VERR_FILE_LOCK_VIOLATION:
76 Status = STATUS_FILE_LOCK_CONFLICT;
77 break;
78
79 case VERR_FILE_LOCK_FAILED:
80 Status = STATUS_LOCK_NOT_GRANTED;
81 break;
82
83 case VINF_BUFFER_OVERFLOW:
84 Status = STATUS_BUFFER_OVERFLOW;
85 break;
86
87 case VERR_EOF:
88 case VINF_EOF:
89 Status = STATUS_END_OF_FILE;
90 break;
91
92 case VERR_READ_ERROR:
93 case VERR_WRITE_ERROR:
94 case VERR_FILE_IO_ERROR:
95 Status = STATUS_UNEXPECTED_IO_ERROR;
96 break;
97
98 case VERR_WRITE_PROTECT:
99 Status = STATUS_MEDIA_WRITE_PROTECTED;
100 break;
101
102 case VERR_ALREADY_EXISTS:
103 Status = STATUS_OBJECT_NAME_COLLISION;
104 break;
105
106 case VERR_NOT_A_DIRECTORY:
107 Status = STATUS_NOT_A_DIRECTORY;
108 break;
109
110 case VERR_SEEK:
111 Status = STATUS_INVALID_PARAMETER;
112 break;
113
114 case VERR_INVALID_PARAMETER:
115 Status = STATUS_INVALID_PARAMETER;
116 break;
117
118 case VERR_NOT_SUPPORTED:
119 Status = STATUS_NOT_SUPPORTED;
120 break;
121
122 case VERR_INVALID_NAME:
123 Status = STATUS_OBJECT_NAME_INVALID;
124 break;
125
126 default:
127 Status = STATUS_INVALID_PARAMETER;
128 Log(("Unexpected vbox error %Rrc\n",
129 vrc));
130 break;
131 }
132 return Status;
133}
134
135/**
136 * Wrapper around ExAllocatePoolWithTag.
137 */
138PVOID vbsfNtAllocNonPagedMem(ULONG cbMemory)
139{
140 /* Tag is reversed (a.k.a "SHFL") to display correctly in debuggers, so search for "SHFL" */
141 PVOID pMemory = ExAllocatePoolWithTag(NonPagedPool, cbMemory, 'LFHS');
142 if (NULL != pMemory)
143 {
144 RtlZeroMemory(pMemory, cbMemory);
145#ifdef DEBUG
146 int const cAllocations = g_cAllocations += 1;
147 Log(("vbsfNtAllocNonPagedMem: Allocated %u bytes of memory at %p (g_iAllocRefCount=%d)\n", cbMemory, pMemory, cAllocations));
148#endif
149 }
150#ifdef DEBUG
151 else
152 Log(("vbsfNtAllocNonPagedMem: ERROR: Could not allocate %u bytes of memory!\n", cbMemory));
153#endif
154 return pMemory;
155}
156
157/**
158 * Wrapper around ExFreePoolWithTag.
159 */
160void vbsfNtFreeNonPagedMem(PVOID pvMemory)
161{
162#ifdef DEBUG
163 int cAllocations = g_cAllocations -= 1;
164 Log(("vbsfNtFreeNonPagedMem: %p (g_cAllocations=%d)\n", pvMemory, cAllocations));
165#endif
166 AssertPtr(pvMemory);
167
168 /* Tagged allocations must be freed using the same tag as used when allocating the memory. */
169 ExFreePoolWithTag(pvMemory, 'LFHS');
170}
171
172/** Allocate and initialize a SHFLSTRING from a UNICODE string.
173 *
174 * @param ppShflString Where to store the pointer to the allocated SHFLSTRING structure.
175 * The structure must be deallocated with vbsfNtFreeNonPagedMem.
176 * @param pwc The UNICODE string. If NULL then SHFL is only allocated.
177 * @param cb Size of the UNICODE string in bytes without the trailing nul.
178 *
179 * @return Status code.
180 */
181NTSTATUS vbsfNtShflStringFromUnicodeAlloc(PSHFLSTRING *ppShflString, const WCHAR *pwc, uint16_t cb)
182{
183 NTSTATUS Status;
184
185 /* Calculate length required for the SHFL structure: header + chars + nul. */
186 ULONG const cbShflString = SHFLSTRING_HEADER_SIZE + cb + sizeof(WCHAR);
187 PSHFLSTRING pShflString = (PSHFLSTRING)vbsfNtAllocNonPagedMem(cbShflString);
188 if (pShflString)
189 {
190 if (ShflStringInitBuffer(pShflString, cbShflString))
191 {
192 if (pwc)
193 {
194 RtlCopyMemory(pShflString->String.ucs2, pwc, cb);
195 pShflString->String.ucs2[cb / sizeof(WCHAR)] = 0;
196 pShflString->u16Length = cb; /* without terminating null */
197 AssertMsg(pShflString->u16Length + sizeof(WCHAR) == pShflString->u16Size,
198 ("u16Length %d, u16Size %d\n", pShflString->u16Length, pShflString->u16Size));
199 }
200 else
201 {
202 /** @todo r=bird: vbsfNtAllocNonPagedMem already zero'ed it... */
203 RtlZeroMemory(pShflString->String.ucs2, cb + sizeof(WCHAR));
204 pShflString->u16Length = 0; /* without terminating null */
205 AssertMsg(pShflString->u16Size >= sizeof(WCHAR),
206 ("u16Size %d\n", pShflString->u16Size));
207 }
208
209 *ppShflString = pShflString;
210 Status = STATUS_SUCCESS;
211 }
212 else
213 {
214 vbsfNtFreeNonPagedMem(pShflString);
215 Status = STATUS_INSUFFICIENT_RESOURCES;
216 }
217 }
218 else
219 Status = STATUS_INSUFFICIENT_RESOURCES;
220
221 return Status;
222}
223
224#if defined(DEBUG) || defined(LOG_ENABLED)
225
226/** Debug routine for translating a minor PNP function to a string. */
227static const char *vbsfNtMinorPnpFunctionName(LONG MinorFunction)
228{
229 switch (MinorFunction)
230 {
231 case IRP_MN_START_DEVICE:
232 return "IRP_MJ_PNP - IRP_MN_START_DEVICE";
233 case IRP_MN_QUERY_REMOVE_DEVICE:
234 return "IRP_MJ_PNP - IRP_MN_QUERY_REMOVE_DEVICE";
235 case IRP_MN_REMOVE_DEVICE:
236 return "IRP_MJ_PNP - IRP_MN_REMOVE_DEVICE";
237 case IRP_MN_CANCEL_REMOVE_DEVICE:
238 return "IRP_MJ_PNP - IRP_MN_CANCEL_REMOVE_DEVICE";
239 case IRP_MN_STOP_DEVICE:
240 return "IRP_MJ_PNP - IRP_MN_STOP_DEVICE";
241 case IRP_MN_QUERY_STOP_DEVICE:
242 return "IRP_MJ_PNP - IRP_MN_QUERY_STOP_DEVICE";
243 case IRP_MN_CANCEL_STOP_DEVICE:
244 return "IRP_MJ_PNP - IRP_MN_CANCEL_STOP_DEVICE";
245 case IRP_MN_QUERY_DEVICE_RELATIONS:
246 return "IRP_MJ_PNP - IRP_MN_QUERY_DEVICE_RELATIONS";
247 case IRP_MN_QUERY_INTERFACE:
248 return "IRP_MJ_PNP - IRP_MN_QUERY_INTERFACE";
249 case IRP_MN_QUERY_CAPABILITIES:
250 return "IRP_MJ_PNP - IRP_MN_QUERY_CAPABILITIES";
251 case IRP_MN_QUERY_RESOURCES:
252 return "IRP_MJ_PNP - IRP_MN_QUERY_RESOURCES";
253 case IRP_MN_QUERY_RESOURCE_REQUIREMENTS:
254 return "IRP_MJ_PNP - IRP_MN_QUERY_RESOURCE_REQUIREMENTS";
255 case IRP_MN_QUERY_DEVICE_TEXT:
256 return "IRP_MJ_PNP - IRP_MN_QUERY_DEVICE_TEXT";
257 case IRP_MN_FILTER_RESOURCE_REQUIREMENTS:
258 return "IRP_MJ_PNP - IRP_MN_FILTER_RESOURCE_REQUIREMENTS";
259 case IRP_MN_READ_CONFIG:
260 return "IRP_MJ_PNP - IRP_MN_READ_CONFIG";
261 case IRP_MN_WRITE_CONFIG:
262 return "IRP_MJ_PNP - IRP_MN_WRITE_CONFIG";
263 case IRP_MN_EJECT:
264 return "IRP_MJ_PNP - IRP_MN_EJECT";
265 case IRP_MN_SET_LOCK:
266 return "IRP_MJ_PNP - IRP_MN_SET_LOCK";
267 case IRP_MN_QUERY_ID:
268 return "IRP_MJ_PNP - IRP_MN_QUERY_ID";
269 case IRP_MN_QUERY_PNP_DEVICE_STATE:
270 return "IRP_MJ_PNP - IRP_MN_QUERY_PNP_DEVICE_STATE";
271 case IRP_MN_QUERY_BUS_INFORMATION:
272 return "IRP_MJ_PNP - IRP_MN_QUERY_BUS_INFORMATION";
273 case IRP_MN_DEVICE_USAGE_NOTIFICATION:
274 return "IRP_MJ_PNP - IRP_MN_DEVICE_USAGE_NOTIFICATION";
275 case IRP_MN_SURPRISE_REMOVAL:
276 return "IRP_MJ_PNP - IRP_MN_SURPRISE_REMOVAL";
277 default:
278 return "IRP_MJ_PNP - unknown_pnp_irp";
279 }
280}
281
282/** Debug routine for translating a major+minor IPR function to a string. */
283const char *vbsfNtMajorFunctionName(UCHAR MajorFunction, LONG MinorFunction)
284{
285 switch (MajorFunction)
286 {
287 RT_CASE_RET_STR(IRP_MJ_CREATE);
288 RT_CASE_RET_STR(IRP_MJ_CREATE_NAMED_PIPE);
289 RT_CASE_RET_STR(IRP_MJ_CLOSE);
290 RT_CASE_RET_STR(IRP_MJ_READ);
291 RT_CASE_RET_STR(IRP_MJ_WRITE);
292 RT_CASE_RET_STR(IRP_MJ_QUERY_INFORMATION);
293 RT_CASE_RET_STR(IRP_MJ_SET_INFORMATION);
294 RT_CASE_RET_STR(IRP_MJ_QUERY_EA);
295 RT_CASE_RET_STR(IRP_MJ_SET_EA);
296 RT_CASE_RET_STR(IRP_MJ_FLUSH_BUFFERS);
297 RT_CASE_RET_STR(IRP_MJ_QUERY_VOLUME_INFORMATION);
298 RT_CASE_RET_STR(IRP_MJ_SET_VOLUME_INFORMATION);
299 RT_CASE_RET_STR(IRP_MJ_DIRECTORY_CONTROL);
300 RT_CASE_RET_STR(IRP_MJ_FILE_SYSTEM_CONTROL);
301 RT_CASE_RET_STR(IRP_MJ_DEVICE_CONTROL);
302 RT_CASE_RET_STR(IRP_MJ_INTERNAL_DEVICE_CONTROL);
303 RT_CASE_RET_STR(IRP_MJ_SHUTDOWN);
304 RT_CASE_RET_STR(IRP_MJ_LOCK_CONTROL);
305 RT_CASE_RET_STR(IRP_MJ_CLEANUP);
306 RT_CASE_RET_STR(IRP_MJ_CREATE_MAILSLOT);
307 RT_CASE_RET_STR(IRP_MJ_QUERY_SECURITY);
308 RT_CASE_RET_STR(IRP_MJ_SET_SECURITY);
309 RT_CASE_RET_STR(IRP_MJ_POWER);
310 RT_CASE_RET_STR(IRP_MJ_SYSTEM_CONTROL);
311 RT_CASE_RET_STR(IRP_MJ_DEVICE_CHANGE);
312 RT_CASE_RET_STR(IRP_MJ_QUERY_QUOTA);
313 RT_CASE_RET_STR(IRP_MJ_SET_QUOTA);
314 case IRP_MJ_PNP:
315 return vbsfNtMinorPnpFunctionName(MinorFunction);
316 default:
317 return "IRP_MJ_UNKNOWN";
318 }
319}
320
321#endif /* DEBUG || LOG_ENABLED */
322
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