VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/SharedFolders/driver/vbsfhlp.c@ 74881

Last change on this file since 74881 was 69500, checked in by vboxsync, 7 years ago

*: scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.9 KB
Line 
1/* $Id: vbsfhlp.c 69500 2017-10-28 15:14:05Z vboxsync $ */
2/** @file
3 * VirtualBox Windows Guest Shared Folders - File System Driver system helpers
4 */
5
6/*
7 * Copyright (C) 2012-2017 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#include <iprt/nt/nt.h> /* includes ntifs.h and wdm.h */
19#include <iprt/win/ntdddisk.h>
20
21#include "vbsfhlp.h"
22
23#ifdef DEBUG
24static int s_iAllocRefCount = 0;
25#endif
26
27void vbsfHlpSleep(ULONG ulMillies)
28{
29 KEVENT event;
30 LARGE_INTEGER dueTime;
31
32 KeInitializeEvent(&event, NotificationEvent, FALSE);
33
34 dueTime.QuadPart = -10000 * (int)ulMillies;
35
36 KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, &dueTime);
37}
38
39/**
40 * Convert VBox IRT file attributes to NT file attributes
41 *
42 * @returns NT file attributes
43 * @param fMode IRT file attributes
44 *
45 */
46uint32_t VBoxToNTFileAttributes(uint32_t fMode)
47{
48 uint32_t FileAttributes = 0;
49
50 if (fMode & RTFS_DOS_READONLY)
51 FileAttributes |= FILE_ATTRIBUTE_READONLY;
52 if (fMode & RTFS_DOS_HIDDEN)
53 FileAttributes |= FILE_ATTRIBUTE_HIDDEN;
54 if (fMode & RTFS_DOS_SYSTEM)
55 FileAttributes |= FILE_ATTRIBUTE_SYSTEM;
56 if (fMode & RTFS_DOS_DIRECTORY)
57 FileAttributes |= FILE_ATTRIBUTE_DIRECTORY;
58 if (fMode & RTFS_DOS_ARCHIVED)
59 FileAttributes |= FILE_ATTRIBUTE_ARCHIVE;
60 if (fMode & RTFS_DOS_NT_TEMPORARY)
61 FileAttributes |= FILE_ATTRIBUTE_TEMPORARY;
62 if (fMode & RTFS_DOS_NT_SPARSE_FILE)
63 FileAttributes |= FILE_ATTRIBUTE_SPARSE_FILE;
64 if (fMode & RTFS_DOS_NT_REPARSE_POINT)
65 FileAttributes |= FILE_ATTRIBUTE_REPARSE_POINT;
66 if (fMode & RTFS_DOS_NT_COMPRESSED)
67 FileAttributes |= FILE_ATTRIBUTE_COMPRESSED;
68 /* if (fMode & RTFS_DOS_NT_OFFLINE) */
69 if (fMode & RTFS_DOS_NT_NOT_CONTENT_INDEXED)
70 FileAttributes |= FILE_ATTRIBUTE_NOT_CONTENT_INDEXED;
71 if (fMode & RTFS_DOS_NT_ENCRYPTED)
72 FileAttributes |= FILE_ATTRIBUTE_ENCRYPTED;
73 if (fMode & RTFS_DOS_NT_NORMAL)
74 {
75 FileAttributes |= FILE_ATTRIBUTE_NORMAL;
76 }
77 return FileAttributes;
78}
79
80/**
81 * Convert VBox IRT file attributes to NT file attributes
82 *
83 * @returns NT file attributes
84 * @param fMode IRT file attributes
85 *
86 */
87uint32_t NTToVBoxFileAttributes(uint32_t fMode)
88{
89 uint32_t FileAttributes = 0;
90
91 if (fMode & FILE_ATTRIBUTE_READONLY)
92 FileAttributes |= RTFS_DOS_READONLY;
93 if (fMode & FILE_ATTRIBUTE_HIDDEN)
94 FileAttributes |= RTFS_DOS_HIDDEN;
95 if (fMode & FILE_ATTRIBUTE_SYSTEM)
96 FileAttributes |= RTFS_DOS_SYSTEM;
97 if (fMode & FILE_ATTRIBUTE_DIRECTORY)
98 FileAttributes |= RTFS_DOS_DIRECTORY;
99 if (fMode & FILE_ATTRIBUTE_ARCHIVE)
100 FileAttributes |= RTFS_DOS_ARCHIVED;
101 if (fMode & FILE_ATTRIBUTE_TEMPORARY)
102 FileAttributes |= RTFS_DOS_NT_TEMPORARY;
103 if (fMode & FILE_ATTRIBUTE_SPARSE_FILE)
104 FileAttributes |= RTFS_DOS_NT_SPARSE_FILE;
105 if (fMode & FILE_ATTRIBUTE_REPARSE_POINT)
106 FileAttributes |= RTFS_DOS_NT_REPARSE_POINT;
107 if (fMode & FILE_ATTRIBUTE_COMPRESSED)
108 FileAttributes |= RTFS_DOS_NT_COMPRESSED;
109 if (fMode & FILE_ATTRIBUTE_NOT_CONTENT_INDEXED)
110 FileAttributes |= RTFS_DOS_NT_NOT_CONTENT_INDEXED;
111 if (fMode & FILE_ATTRIBUTE_ENCRYPTED)
112 FileAttributes |= RTFS_DOS_NT_ENCRYPTED;
113 if (fMode & FILE_ATTRIBUTE_NORMAL)
114 {
115 FileAttributes |= RTFS_DOS_NT_NORMAL;
116 }
117 return FileAttributes;
118}
119
120NTSTATUS vbsfHlpCreateDriveLetter(WCHAR Letter, UNICODE_STRING *pDeviceName)
121{
122 UNICODE_STRING driveName;
123 RtlInitUnicodeString(&driveName,L"\\??\\_:" );
124
125 /* Replace '_' with actual drive letter */
126 driveName.Buffer[driveName.Length/sizeof(WCHAR) - 2] = Letter;
127
128 return IoCreateSymbolicLink(&driveName, pDeviceName);
129}
130
131NTSTATUS vbsfHlpDeleteDriveLetter(WCHAR Letter)
132{
133 UNICODE_STRING driveName;
134 RtlInitUnicodeString(&driveName,L"\\??\\_:" );
135
136 /* Replace '_' with actual drive letter */
137 driveName.Buffer[driveName.Length/sizeof(WCHAR) - 2] = Letter;
138
139 return IoDeleteSymbolicLink(&driveName);
140}
141
142 /**
143 * Convert VBox error code to NT status code
144 *
145 * @returns NT status code
146 * @param vboxRC VBox error code
147 *
148 */
149NTSTATUS VBoxErrorToNTStatus(int vboxRC)
150{
151 NTSTATUS Status;
152
153 switch (vboxRC)
154 {
155 case VINF_SUCCESS:
156 Status = STATUS_SUCCESS;
157 break;
158
159 case VERR_ACCESS_DENIED:
160 Status = STATUS_ACCESS_DENIED;
161 break;
162
163 case VERR_NO_MORE_FILES:
164 Status = STATUS_NO_MORE_FILES;
165 break;
166
167 case VERR_PATH_NOT_FOUND:
168 Status = STATUS_OBJECT_PATH_NOT_FOUND;
169 break;
170
171 case VERR_FILE_NOT_FOUND:
172 Status = STATUS_OBJECT_NAME_NOT_FOUND;
173 break;
174
175 case VERR_DIR_NOT_EMPTY:
176 Status = STATUS_DIRECTORY_NOT_EMPTY;
177 break;
178
179 case VERR_SHARING_VIOLATION:
180 Status = STATUS_SHARING_VIOLATION;
181 break;
182
183 case VERR_FILE_LOCK_VIOLATION:
184 Status = STATUS_FILE_LOCK_CONFLICT;
185 break;
186
187 case VERR_FILE_LOCK_FAILED:
188 Status = STATUS_LOCK_NOT_GRANTED;
189 break;
190
191 case VINF_BUFFER_OVERFLOW:
192 Status = STATUS_BUFFER_OVERFLOW;
193 break;
194
195 case VERR_EOF:
196 case VINF_EOF:
197 Status = STATUS_END_OF_FILE;
198 break;
199
200 case VERR_READ_ERROR:
201 case VERR_WRITE_ERROR:
202 case VERR_FILE_IO_ERROR:
203 Status = STATUS_UNEXPECTED_IO_ERROR;
204 break;
205
206 case VERR_WRITE_PROTECT:
207 Status = STATUS_MEDIA_WRITE_PROTECTED;
208 break;
209
210 case VERR_ALREADY_EXISTS:
211 Status = STATUS_OBJECT_NAME_COLLISION;
212 break;
213
214 case VERR_NOT_A_DIRECTORY:
215 Status = STATUS_NOT_A_DIRECTORY;
216 break;
217
218 case VERR_SEEK:
219 Status = STATUS_INVALID_PARAMETER;
220 break;
221
222 case VERR_INVALID_PARAMETER:
223 Status = STATUS_INVALID_PARAMETER;
224 break;
225
226 case VERR_NOT_SUPPORTED:
227 Status = STATUS_NOT_SUPPORTED;
228 break;
229
230 default:
231 /** @todo error handling */
232 Status = STATUS_INVALID_PARAMETER;
233 Log(("Unexpected vbox error %Rrc\n",
234 vboxRC));
235 break;
236 }
237 return Status;
238}
239
240PVOID vbsfAllocNonPagedMem(ULONG ulSize)
241{
242 PVOID pMemory = NULL;
243
244#ifdef DEBUG
245 s_iAllocRefCount = s_iAllocRefCount + 1;
246 Log(("vbsfAllocNonPagedMem: RefCnt after incrementing: %d\n", s_iAllocRefCount));
247#endif
248
249 /* Tag is reversed (a.k.a "SHFL") to display correctly in debuggers, so search for "SHFL" */
250 pMemory = ExAllocatePoolWithTag(NonPagedPool, ulSize, 'LFHS');
251
252 if (NULL != pMemory)
253 {
254 RtlZeroMemory(pMemory, ulSize);
255#ifdef DEBUG
256 Log(("vbsfAllocNonPagedMem: Allocated %d bytes of memory at %p.\n", ulSize, pMemory));
257#endif
258 }
259 else
260 {
261#ifdef DEBUG
262 Log(("vbsfAllocNonPagedMem: ERROR: Could not allocate %d bytes of memory!\n", ulSize));
263#endif
264 }
265
266 return pMemory;
267}
268
269void vbsfFreeNonPagedMem(PVOID lpMem)
270{
271#ifdef DEBUG
272 s_iAllocRefCount = s_iAllocRefCount - 1;
273 Log(("vbsfFreeNonPagedMem: RefCnt after decrementing: %d\n", s_iAllocRefCount));
274#endif
275
276 Assert(lpMem);
277
278 /* MSDN: The ExFreePoolWithTag routine issues a bug check if the specified value for Tag does not match the tag value passed
279 to the routine that originally allocated the memory block. Otherwise, the behavior of this routine is identical to ExFreePool. */
280 ExFreePoolWithTag(lpMem, 'LFHS');
281 lpMem = NULL;
282}
283
284#if 0 //def DEBUG
285/**
286 * Callback for RTLogFormatV which writes to the backdoor.
287 * See PFNLOGOUTPUT() for details.
288 */
289static DECLCALLBACK(size_t) rtLogBackdoorOutput(void *pv, const char *pachChars, size_t cbChars)
290{
291 RTLogWriteUser(pachChars, cbChars);
292 return cbChars;
293}
294
295int RTLogBackdoorPrintf1(const char *pszFormat, ...)
296{
297 va_list args;
298
299 LARGE_INTEGER time;
300
301 KeQueryTickCount(&time);
302
303 RTLogBackdoorPrintf("T=%RX64 ", time.QuadPart);
304 va_start(args, pszFormat);
305 RTLogFormatV(rtLogBackdoorOutput, NULL, pszFormat, args);
306 va_end(args);
307
308 return 0;
309}
310#endif
311
312#if defined(DEBUG) || defined(LOG_ENABLED)
313
314static PCHAR PnPMinorFunctionString(LONG MinorFunction)
315{
316 switch (MinorFunction)
317 {
318 case IRP_MN_START_DEVICE:
319 return "IRP_MJ_PNP - IRP_MN_START_DEVICE";
320 case IRP_MN_QUERY_REMOVE_DEVICE:
321 return "IRP_MJ_PNP - IRP_MN_QUERY_REMOVE_DEVICE";
322 case IRP_MN_REMOVE_DEVICE:
323 return "IRP_MJ_PNP - IRP_MN_REMOVE_DEVICE";
324 case IRP_MN_CANCEL_REMOVE_DEVICE:
325 return "IRP_MJ_PNP - IRP_MN_CANCEL_REMOVE_DEVICE";
326 case IRP_MN_STOP_DEVICE:
327 return "IRP_MJ_PNP - IRP_MN_STOP_DEVICE";
328 case IRP_MN_QUERY_STOP_DEVICE:
329 return "IRP_MJ_PNP - IRP_MN_QUERY_STOP_DEVICE";
330 case IRP_MN_CANCEL_STOP_DEVICE:
331 return "IRP_MJ_PNP - IRP_MN_CANCEL_STOP_DEVICE";
332 case IRP_MN_QUERY_DEVICE_RELATIONS:
333 return "IRP_MJ_PNP - IRP_MN_QUERY_DEVICE_RELATIONS";
334 case IRP_MN_QUERY_INTERFACE:
335 return "IRP_MJ_PNP - IRP_MN_QUERY_INTERFACE";
336 case IRP_MN_QUERY_CAPABILITIES:
337 return "IRP_MJ_PNP - IRP_MN_QUERY_CAPABILITIES";
338 case IRP_MN_QUERY_RESOURCES:
339 return "IRP_MJ_PNP - IRP_MN_QUERY_RESOURCES";
340 case IRP_MN_QUERY_RESOURCE_REQUIREMENTS:
341 return "IRP_MJ_PNP - IRP_MN_QUERY_RESOURCE_REQUIREMENTS";
342 case IRP_MN_QUERY_DEVICE_TEXT:
343 return "IRP_MJ_PNP - IRP_MN_QUERY_DEVICE_TEXT";
344 case IRP_MN_FILTER_RESOURCE_REQUIREMENTS:
345 return "IRP_MJ_PNP - IRP_MN_FILTER_RESOURCE_REQUIREMENTS";
346 case IRP_MN_READ_CONFIG:
347 return "IRP_MJ_PNP - IRP_MN_READ_CONFIG";
348 case IRP_MN_WRITE_CONFIG:
349 return "IRP_MJ_PNP - IRP_MN_WRITE_CONFIG";
350 case IRP_MN_EJECT:
351 return "IRP_MJ_PNP - IRP_MN_EJECT";
352 case IRP_MN_SET_LOCK:
353 return "IRP_MJ_PNP - IRP_MN_SET_LOCK";
354 case IRP_MN_QUERY_ID:
355 return "IRP_MJ_PNP - IRP_MN_QUERY_ID";
356 case IRP_MN_QUERY_PNP_DEVICE_STATE:
357 return "IRP_MJ_PNP - IRP_MN_QUERY_PNP_DEVICE_STATE";
358 case IRP_MN_QUERY_BUS_INFORMATION:
359 return "IRP_MJ_PNP - IRP_MN_QUERY_BUS_INFORMATION";
360 case IRP_MN_DEVICE_USAGE_NOTIFICATION:
361 return "IRP_MJ_PNP - IRP_MN_DEVICE_USAGE_NOTIFICATION";
362 case IRP_MN_SURPRISE_REMOVAL:
363 return "IRP_MJ_PNP - IRP_MN_SURPRISE_REMOVAL";
364 default:
365 return "IRP_MJ_PNP - unknown_pnp_irp";
366 }
367}
368
369PCHAR MajorFunctionString(UCHAR MajorFunction, LONG MinorFunction)
370{
371 switch (MajorFunction)
372 {
373 case IRP_MJ_CREATE:
374 return "IRP_MJ_CREATE";
375 case IRP_MJ_CREATE_NAMED_PIPE:
376 return "IRP_MJ_CREATE_NAMED_PIPE";
377 case IRP_MJ_CLOSE:
378 return "IRP_MJ_CLOSE";
379 case IRP_MJ_READ:
380 return "IRP_MJ_READ";
381 case IRP_MJ_WRITE:
382 return "IRP_MJ_WRITE";
383 case IRP_MJ_QUERY_INFORMATION:
384 return "IRP_MJ_QUERY_INFORMATION";
385 case IRP_MJ_SET_INFORMATION:
386 return "IRP_MJ_SET_INFORMATION";
387 case IRP_MJ_QUERY_EA:
388 return "IRP_MJ_QUERY_EA";
389 case IRP_MJ_SET_EA:
390 return "IRP_MJ_SET_EA";
391 case IRP_MJ_FLUSH_BUFFERS:
392 return "IRP_MJ_FLUSH_BUFFERS";
393 case IRP_MJ_QUERY_VOLUME_INFORMATION:
394 return "IRP_MJ_QUERY_VOLUME_INFORMATION";
395 case IRP_MJ_SET_VOLUME_INFORMATION:
396 return "IRP_MJ_SET_VOLUME_INFORMATION";
397 case IRP_MJ_DIRECTORY_CONTROL:
398 return "IRP_MJ_DIRECTORY_CONTROL";
399 case IRP_MJ_FILE_SYSTEM_CONTROL:
400 return "IRP_MJ_FILE_SYSTEM_CONTROL";
401 case IRP_MJ_DEVICE_CONTROL:
402 return "IRP_MJ_DEVICE_CONTROL";
403 case IRP_MJ_INTERNAL_DEVICE_CONTROL:
404 return "IRP_MJ_INTERNAL_DEVICE_CONTROL";
405 case IRP_MJ_SHUTDOWN:
406 return "IRP_MJ_SHUTDOWN";
407 case IRP_MJ_LOCK_CONTROL:
408 return "IRP_MJ_LOCK_CONTROL";
409 case IRP_MJ_CLEANUP:
410 return "IRP_MJ_CLEANUP";
411 case IRP_MJ_CREATE_MAILSLOT:
412 return "IRP_MJ_CREATE_MAILSLOT";
413 case IRP_MJ_QUERY_SECURITY:
414 return "IRP_MJ_QUERY_SECURITY";
415 case IRP_MJ_SET_SECURITY:
416 return "IRP_MJ_SET_SECURITY";
417 case IRP_MJ_POWER:
418 return "IRP_MJ_POWER";
419 case IRP_MJ_SYSTEM_CONTROL:
420 return "IRP_MJ_SYSTEM_CONTROL";
421 case IRP_MJ_DEVICE_CHANGE:
422 return "IRP_MJ_DEVICE_CHANGE";
423 case IRP_MJ_QUERY_QUOTA:
424 return "IRP_MJ_QUERY_QUOTA";
425 case IRP_MJ_SET_QUOTA:
426 return "IRP_MJ_SET_QUOTA";
427 case IRP_MJ_PNP:
428 return PnPMinorFunctionString(MinorFunction);
429
430 default:
431 return "unknown_pnp_irp";
432 }
433}
434
435#endif /* DEBUG || LOG_ENABLED */
436
437/** Allocate and initialize a SHFLSTRING from a UNICODE string.
438 *
439 * @param ppShflString Where to store the pointer to the allocated SHFLSTRING structure.
440 * The structure must be deallocated with vbsfFreeNonPagedMem.
441 * @param pwc The UNICODE string. If NULL then SHFL is only allocated.
442 * @param cb Size of the UNICODE string in bytes without the trailing nul.
443 *
444 * @return Status code.
445 */
446NTSTATUS vbsfShflStringFromUnicodeAlloc(PSHFLSTRING *ppShflString, const WCHAR *pwc, uint16_t cb)
447{
448 NTSTATUS Status = STATUS_SUCCESS;
449
450 PSHFLSTRING pShflString;
451 ULONG ulShflStringSize;
452
453 /* Calculate length required for the SHFL structure: header + chars + nul. */
454 ulShflStringSize = SHFLSTRING_HEADER_SIZE + cb + sizeof(WCHAR);
455 pShflString = (PSHFLSTRING)vbsfAllocNonPagedMem(ulShflStringSize);
456 if (pShflString)
457 {
458 if (ShflStringInitBuffer(pShflString, ulShflStringSize))
459 {
460 if (pwc)
461 {
462 RtlCopyMemory(pShflString->String.ucs2, pwc, cb);
463 pShflString->String.ucs2[cb / sizeof(WCHAR)] = 0;
464 pShflString->u16Length = cb; /* without terminating null */
465 AssertMsg(pShflString->u16Length + sizeof(WCHAR) == pShflString->u16Size,
466 ("u16Length %d, u16Size %d\n", pShflString->u16Length, pShflString->u16Size));
467 }
468 else
469 {
470 RtlZeroMemory(pShflString->String.ucs2, cb + sizeof(WCHAR));
471 pShflString->u16Length = 0; /* without terminating null */
472 AssertMsg(pShflString->u16Size >= sizeof(WCHAR),
473 ("u16Size %d\n", pShflString->u16Size));
474 }
475
476 *ppShflString = pShflString;
477 }
478 else
479 {
480 vbsfFreeNonPagedMem(pShflString);
481 Status = STATUS_INSUFFICIENT_RESOURCES;
482 }
483 }
484 else
485 {
486 Status = STATUS_INSUFFICIENT_RESOURCES;
487 }
488
489 return Status;
490}
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