VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/SharedFolders/driver/path.c@ 71735

Last change on this file since 71735 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: 33.8 KB
Line 
1/* $Id: path.c 69500 2017-10-28 15:14:05Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox Windows Guest Shared Folders
5 *
6 * File System Driver path related routines
7 */
8
9/*
10 * Copyright (C) 2012-2017 Oracle Corporation
11 *
12 * This file is part of VirtualBox Open Source Edition (OSE), as
13 * available from http://www.virtualbox.org. This file is free software;
14 * you can redistribute it and/or modify it under the terms of the GNU
15 * General Public License (GPL) as published by the Free Software
16 * Foundation, in version 2 as it comes in the "COPYING" file of the
17 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19 */
20
21#include "vbsf.h"
22
23static UNICODE_STRING UnicodeBackslash = { 2, 4, L"\\" };
24
25static NTSTATUS vbsfProcessCreate(PRX_CONTEXT RxContext,
26 PUNICODE_STRING RemainingName,
27 FILE_BASIC_INFORMATION *pFileBasicInfo,
28 FILE_STANDARD_INFORMATION *pFileStandardInfo,
29 PVOID EaBuffer,
30 ULONG EaLength,
31 ULONG *pulCreateAction,
32 SHFLHANDLE *pHandle)
33{
34 NTSTATUS Status = STATUS_SUCCESS;
35
36 RxCaptureFcb;
37
38 PMRX_VBOX_DEVICE_EXTENSION pDeviceExtension = VBoxMRxGetDeviceExtension(RxContext);
39 PMRX_VBOX_NETROOT_EXTENSION pNetRootExtension = VBoxMRxGetNetRootExtension(capFcb->pNetRoot);
40
41 int vboxRC = VINF_SUCCESS;
42
43 /* Various boolean flags. */
44 struct
45 {
46 ULONG CreateDirectory :1;
47 ULONG OpenDirectory :1;
48 ULONG DirectoryFile :1;
49 ULONG NonDirectoryFile :1;
50 ULONG DeleteOnClose :1;
51 ULONG TemporaryFile :1;
52 } bf;
53
54 ACCESS_MASK DesiredAccess;
55 ULONG Options;
56 UCHAR FileAttributes;
57 ULONG ShareAccess;
58 ULONG CreateDisposition;
59 SHFLCREATEPARMS *pCreateParms = NULL;
60
61 RT_NOREF(EaBuffer);
62
63 if (EaLength)
64 {
65 Log(("VBOXSF: vbsfProcessCreate: Unsupported: extended attributes!\n"));
66 Status = STATUS_NOT_SUPPORTED;
67 goto failure;
68 }
69
70 if (BooleanFlagOn(capFcb->FcbState, FCB_STATE_PAGING_FILE))
71 {
72 Log(("VBOXSF: vbsfProcessCreate: Unsupported: paging file!\n"));
73 Status = STATUS_NOT_IMPLEMENTED;
74 goto failure;
75 }
76
77 Log(("VBOXSF: vbsfProcessCreate: FileAttributes = 0x%08x\n",
78 RxContext->Create.NtCreateParameters.FileAttributes));
79 Log(("VBOXSF: vbsfProcessCreate: CreateOptions = 0x%08x\n",
80 RxContext->Create.NtCreateParameters.CreateOptions));
81
82 RtlZeroMemory (&bf, sizeof (bf));
83
84 DesiredAccess = RxContext->Create.NtCreateParameters.DesiredAccess;
85 Options = RxContext->Create.NtCreateParameters.CreateOptions & FILE_VALID_OPTION_FLAGS;
86 FileAttributes = (UCHAR)(RxContext->Create.NtCreateParameters.FileAttributes & ~FILE_ATTRIBUTE_NORMAL);
87 ShareAccess = RxContext->Create.NtCreateParameters.ShareAccess;
88
89 /* We do not support opens by file ids. */
90 if (FlagOn(Options, FILE_OPEN_BY_FILE_ID))
91 {
92 Log(("VBOXSF: vbsfProcessCreate: Unsupported: file open by id!\n"));
93 Status = STATUS_NOT_IMPLEMENTED;
94 goto failure;
95 }
96
97 /* Mask out unsupported attribute bits. */
98 FileAttributes &= (FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_ARCHIVE);
99
100 bf.DirectoryFile = BooleanFlagOn(Options, FILE_DIRECTORY_FILE);
101 bf.NonDirectoryFile = BooleanFlagOn(Options, FILE_NON_DIRECTORY_FILE);
102 bf.DeleteOnClose = BooleanFlagOn(Options, FILE_DELETE_ON_CLOSE);
103 if (bf.DeleteOnClose)
104 Log(("VBOXSF: vbsfProcessCreate: Delete on close!\n"));
105
106 CreateDisposition = RxContext->Create.NtCreateParameters.Disposition;
107
108 bf.CreateDirectory = (BOOLEAN)(bf.DirectoryFile && ((CreateDisposition == FILE_CREATE) || (CreateDisposition == FILE_OPEN_IF)));
109 bf.OpenDirectory = (BOOLEAN)(bf.DirectoryFile && ((CreateDisposition == FILE_OPEN) || (CreateDisposition == FILE_OPEN_IF)));
110 bf.TemporaryFile = BooleanFlagOn(RxContext->Create.NtCreateParameters.FileAttributes, FILE_ATTRIBUTE_TEMPORARY);
111
112 if (FlagOn(capFcb->FcbState, FCB_STATE_TEMPORARY))
113 bf.TemporaryFile = TRUE;
114
115 Log(("VBOXSF: vbsfProcessCreate: bf.TemporaryFile %d, bf.CreateDirectory %d, bf.DirectoryFile = %d\n",
116 (ULONG)bf.TemporaryFile, (ULONG)bf.CreateDirectory, (ULONG)bf.DirectoryFile));
117
118 /* Check consistency in specified flags. */
119 if (bf.TemporaryFile && bf.CreateDirectory) /* Directories with temporary flag set are not allowed! */
120 {
121 Log(("VBOXSF: vbsfProcessCreate: Not allowed: Temporary directories!\n"));
122 Status = STATUS_INVALID_PARAMETER;
123 goto failure;
124 }
125
126 if (bf.DirectoryFile && bf.NonDirectoryFile)
127 {
128 Log(("VBOXSF: vbsfProcessCreate: Unsupported combination: dir && !dir\n"));
129 Status = STATUS_INVALID_PARAMETER;
130 goto failure;
131 }
132
133 /* Initialize create parameters. */
134 pCreateParms = (SHFLCREATEPARMS *)vbsfAllocNonPagedMem(sizeof(SHFLCREATEPARMS));
135 if (!pCreateParms)
136 {
137 Status = STATUS_INSUFFICIENT_RESOURCES;
138 goto failure;
139 }
140
141 RtlZeroMemory(pCreateParms, sizeof (SHFLCREATEPARMS));
142
143 pCreateParms->Handle = SHFL_HANDLE_NIL;
144 pCreateParms->Result = SHFL_NO_RESULT;
145
146 if (bf.DirectoryFile)
147 {
148 if (CreateDisposition != FILE_CREATE && CreateDisposition != FILE_OPEN && CreateDisposition != FILE_OPEN_IF)
149 {
150 Log(("VBOXSF: vbsfProcessCreate: Invalid disposition 0x%08X for directory!\n",
151 CreateDisposition));
152 Status = STATUS_INVALID_PARAMETER;
153 goto failure;
154 }
155
156 pCreateParms->CreateFlags |= SHFL_CF_DIRECTORY;
157 }
158
159 Log(("VBOXSF: vbsfProcessCreate: CreateDisposition = 0x%08X\n",
160 CreateDisposition));
161
162 switch (CreateDisposition)
163 {
164 case FILE_SUPERSEDE:
165 pCreateParms->CreateFlags |= SHFL_CF_ACT_REPLACE_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW;
166 Log(("VBOXSF: vbsfProcessCreate: CreateFlags |= SHFL_CF_ACT_REPLACE_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW\n"));
167 break;
168
169 case FILE_OPEN:
170 pCreateParms->CreateFlags |= SHFL_CF_ACT_OPEN_IF_EXISTS | SHFL_CF_ACT_FAIL_IF_NEW;
171 Log(("VBOXSF: vbsfProcessCreate: CreateFlags |= SHFL_CF_ACT_OPEN_IF_EXISTS | SHFL_CF_ACT_FAIL_IF_NEW\n"));
172 break;
173
174 case FILE_CREATE:
175 pCreateParms->CreateFlags |= SHFL_CF_ACT_FAIL_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW;
176 Log(("VBOXSF: vbsfProcessCreate: CreateFlags |= SHFL_CF_ACT_FAIL_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW\n"));
177 break;
178
179 case FILE_OPEN_IF:
180 pCreateParms->CreateFlags |= SHFL_CF_ACT_OPEN_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW;
181 Log(("VBOXSF: vbsfProcessCreate: CreateFlags |= SHFL_CF_ACT_OPEN_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW\n"));
182 break;
183
184 case FILE_OVERWRITE:
185 pCreateParms->CreateFlags |= SHFL_CF_ACT_OVERWRITE_IF_EXISTS | SHFL_CF_ACT_FAIL_IF_NEW;
186 Log(("VBOXSF: vbsfProcessCreate: CreateFlags |= SHFL_CF_ACT_OVERWRITE_IF_EXISTS | SHFL_CF_ACT_FAIL_IF_NEW\n"));
187 break;
188
189 case FILE_OVERWRITE_IF:
190 pCreateParms->CreateFlags |= SHFL_CF_ACT_OVERWRITE_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW;
191 Log(("VBOXSF: vbsfProcessCreate: CreateFlags |= SHFL_CF_ACT_OVERWRITE_IF_EXISTS | SHFL_CF_ACT_CREATE_IF_NEW\n"));
192 break;
193
194 default:
195 Log(("VBOXSF: vbsfProcessCreate: Unexpected create disposition: 0x%08X\n",
196 CreateDisposition));
197 Status = STATUS_INVALID_PARAMETER;
198 goto failure;
199 }
200
201 Log(("VBOXSF: vbsfProcessCreate: DesiredAccess = 0x%08X\n",
202 DesiredAccess));
203 Log(("VBOXSF: vbsfProcessCreate: ShareAccess = 0x%08X\n",
204 ShareAccess));
205
206 if (DesiredAccess & FILE_READ_DATA)
207 {
208 Log(("VBOXSF: vbsfProcessCreate: FILE_READ_DATA\n"));
209 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_READ;
210 }
211
212 if (DesiredAccess & FILE_WRITE_DATA)
213 {
214 Log(("VBOXSF: vbsfProcessCreate: FILE_WRITE_DATA\n"));
215 /* FILE_WRITE_DATA means write access regardless of FILE_APPEND_DATA bit.
216 */
217 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_WRITE;
218 }
219 else if (DesiredAccess & FILE_APPEND_DATA)
220 {
221 Log(("VBOXSF: vbsfProcessCreate: FILE_APPEND_DATA\n"));
222 /* FILE_APPEND_DATA without FILE_WRITE_DATA means append only mode.
223 *
224 * Both write and append access flags are required for shared folders,
225 * as on Windows FILE_APPEND_DATA implies write access.
226 */
227 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_WRITE | SHFL_CF_ACCESS_APPEND;
228 }
229
230 if (DesiredAccess & FILE_READ_ATTRIBUTES)
231 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_ATTR_READ;
232 if (DesiredAccess & FILE_WRITE_ATTRIBUTES)
233 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_ATTR_WRITE;
234
235 if (ShareAccess & (FILE_SHARE_READ | FILE_SHARE_WRITE))
236 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_DENYNONE;
237 else if (ShareAccess & FILE_SHARE_READ)
238 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_DENYWRITE;
239 else if (ShareAccess & FILE_SHARE_WRITE)
240 pCreateParms->CreateFlags |= SHFL_CF_ACCESS_DENYREAD;
241 else pCreateParms->CreateFlags |= SHFL_CF_ACCESS_DENYALL;
242
243 /* Set initial allocation size. */
244 pCreateParms->Info.cbObject = RxContext->Create.NtCreateParameters.AllocationSize.QuadPart;
245
246 if (FileAttributes == 0)
247 FileAttributes = FILE_ATTRIBUTE_NORMAL;
248
249 pCreateParms->Info.Attr.fMode = NTToVBoxFileAttributes(FileAttributes);
250
251 {
252 PSHFLSTRING ParsedPath;
253 Log(("VBOXSF: vbsfProcessCreate: RemainingName->Length = %d\n", RemainingName->Length));
254
255 Status = vbsfShflStringFromUnicodeAlloc(&ParsedPath, RemainingName->Buffer, RemainingName->Length);
256 if (Status != STATUS_SUCCESS)
257 {
258 goto failure;
259 }
260
261 Log(("VBOXSF: ParsedPath: %.*ls\n",
262 ParsedPath->u16Length / sizeof(WCHAR), ParsedPath->String.ucs2));
263
264 /* Call host. */
265 Log(("VBOXSF: vbsfProcessCreate: VbglR0SfCreate called.\n"));
266 vboxRC = VbglR0SfCreate(&pDeviceExtension->hgcmClient, &pNetRootExtension->map, ParsedPath, pCreateParms);
267
268 vbsfFreeNonPagedMem(ParsedPath);
269 }
270
271 Log(("VBOXSF: vbsfProcessCreate: VbglR0SfCreate returns vboxRC = %Rrc, Result = 0x%x\n",
272 vboxRC, pCreateParms->Result));
273
274 if (RT_FAILURE(vboxRC))
275 {
276 /* Map some VBoxRC to STATUS codes expected by the system. */
277 switch (vboxRC)
278 {
279 case VERR_ALREADY_EXISTS:
280 {
281 *pulCreateAction = FILE_EXISTS;
282 Status = STATUS_OBJECT_NAME_COLLISION;
283 goto failure;
284 }
285
286 /* On POSIX systems, the "mkdir" command returns VERR_FILE_NOT_FOUND when
287 doing a recursive directory create. Handle this case. */
288 case VERR_FILE_NOT_FOUND:
289 {
290 pCreateParms->Result = SHFL_PATH_NOT_FOUND;
291 break;
292 }
293
294 default:
295 {
296 *pulCreateAction = FILE_DOES_NOT_EXIST;
297 Status = VBoxErrorToNTStatus(vboxRC);
298 goto failure;
299 }
300 }
301 }
302
303 /*
304 * The request succeeded. Analyze host response,
305 */
306 switch (pCreateParms->Result)
307 {
308 case SHFL_PATH_NOT_FOUND:
309 {
310 /* Path to object does not exist. */
311 Log(("VBOXSF: vbsfProcessCreate: Path not found\n"));
312 *pulCreateAction = FILE_DOES_NOT_EXIST;
313 Status = STATUS_OBJECT_PATH_NOT_FOUND;
314 goto failure;
315 }
316
317 case SHFL_FILE_NOT_FOUND:
318 {
319 Log(("VBOXSF: vbsfProcessCreate: File not found\n"));
320 *pulCreateAction = FILE_DOES_NOT_EXIST;
321 if (pCreateParms->Handle == SHFL_HANDLE_NIL)
322 {
323 Status = STATUS_OBJECT_NAME_NOT_FOUND;
324 goto failure;
325 }
326
327 Log(("VBOXSF: vbsfProcessCreate: File not found but have a handle!\n"));
328 Status = STATUS_UNSUCCESSFUL;
329 goto failure;
330
331 break;
332 }
333
334 case SHFL_FILE_EXISTS:
335 {
336 Log(("VBOXSF: vbsfProcessCreate: File exists, Handle = 0x%RX64\n",
337 pCreateParms->Handle));
338 if (pCreateParms->Handle == SHFL_HANDLE_NIL)
339 {
340 *pulCreateAction = FILE_EXISTS;
341 if (CreateDisposition == FILE_CREATE)
342 {
343 /* File was not opened because we requested a create. */
344 Status = STATUS_OBJECT_NAME_COLLISION;
345 goto failure;
346 }
347
348 /* Actually we should not go here, unless we have no rights to open the object. */
349 Log(("VBOXSF: vbsfProcessCreate: Existing file was not opened!\n"));
350 Status = STATUS_ACCESS_DENIED;
351 goto failure;
352 }
353
354 *pulCreateAction = FILE_OPENED;
355
356 /* Existing file was opened. Go check flags and create FCB. */
357 break;
358 }
359
360 case SHFL_FILE_CREATED:
361 {
362 /* A new file was created. */
363 Assert(pCreateParms->Handle != SHFL_HANDLE_NIL);
364
365 *pulCreateAction = FILE_CREATED;
366
367 /* Go check flags and create FCB. */
368 break;
369 }
370
371 case SHFL_FILE_REPLACED:
372 {
373 /* Existing file was replaced or overwriting. */
374 Assert(pCreateParms->Handle != SHFL_HANDLE_NIL);
375
376 if (CreateDisposition == FILE_SUPERSEDE)
377 *pulCreateAction = FILE_SUPERSEDED;
378 else
379 *pulCreateAction = FILE_OVERWRITTEN;
380 /* Go check flags and create FCB. */
381 break;
382 }
383
384 default:
385 {
386 Log(("VBOXSF: vbsfProcessCreate: Invalid CreateResult from host (0x%08X)\n",
387 pCreateParms->Result));
388 *pulCreateAction = FILE_DOES_NOT_EXIST;
389 Status = STATUS_OBJECT_PATH_NOT_FOUND;
390 goto failure;
391 }
392 }
393
394 /* Check flags. */
395 if (bf.NonDirectoryFile && FlagOn(pCreateParms->Info.Attr.fMode, RTFS_DOS_DIRECTORY))
396 {
397 /* Caller wanted only a file, but the object is a directory. */
398 Log(("VBOXSF: vbsfProcessCreate: File is a directory!\n"));
399 Status = STATUS_FILE_IS_A_DIRECTORY;
400 goto failure;
401 }
402
403 if (bf.DirectoryFile && !FlagOn(pCreateParms->Info.Attr.fMode, RTFS_DOS_DIRECTORY))
404 {
405 /* Caller wanted only a directory, but the object is not a directory. */
406 Log(("VBOXSF: vbsfProcessCreate: File is not a directory!\n"));
407 Status = STATUS_NOT_A_DIRECTORY;
408 goto failure;
409 }
410
411 *pHandle = pCreateParms->Handle;
412
413 /* Translate attributes */
414 pFileBasicInfo->FileAttributes = VBoxToNTFileAttributes(pCreateParms->Info.Attr.fMode);
415
416 /* Translate file times */
417 pFileBasicInfo->CreationTime.QuadPart = RTTimeSpecGetNtTime(&pCreateParms->Info.BirthTime); /* ridiculous name */
418 pFileBasicInfo->LastAccessTime.QuadPart = RTTimeSpecGetNtTime(&pCreateParms->Info.AccessTime);
419 pFileBasicInfo->LastWriteTime.QuadPart = RTTimeSpecGetNtTime(&pCreateParms->Info.ModificationTime);
420 pFileBasicInfo->ChangeTime.QuadPart = RTTimeSpecGetNtTime(&pCreateParms->Info.ChangeTime);
421
422 if (!FlagOn(pCreateParms->Info.Attr.fMode, RTFS_DOS_DIRECTORY))
423 {
424 pFileStandardInfo->AllocationSize.QuadPart = pCreateParms->Info.cbAllocated;
425 pFileStandardInfo->EndOfFile.QuadPart = pCreateParms->Info.cbObject;
426 pFileStandardInfo->Directory = FALSE;
427
428 Log(("VBOXSF: vbsfProcessCreate: AllocationSize = 0x%RX64, EndOfFile = 0x%RX64\n",
429 pCreateParms->Info.cbAllocated, pCreateParms->Info.cbObject));
430 }
431 else
432 {
433 pFileStandardInfo->AllocationSize.QuadPart = 0;
434 pFileStandardInfo->EndOfFile.QuadPart = 0;
435 pFileStandardInfo->Directory = TRUE;
436 }
437 pFileStandardInfo->NumberOfLinks = 0;
438 pFileStandardInfo->DeletePending = FALSE;
439
440 vbsfFreeNonPagedMem(pCreateParms);
441
442 return Status;
443
444failure:
445
446 Log(("VBOXSF: vbsfProcessCreate: Returned with status = 0x%08X\n",
447 Status));
448
449 if (pCreateParms && pCreateParms->Handle != SHFL_HANDLE_NIL)
450 {
451 VbglR0SfClose(&pDeviceExtension->hgcmClient, &pNetRootExtension->map, pCreateParms->Handle);
452 *pHandle = SHFL_HANDLE_NIL;
453 }
454
455 if (pCreateParms)
456 vbsfFreeNonPagedMem(pCreateParms);
457
458 return Status;
459}
460
461NTSTATUS VBoxMRxCreate(IN OUT PRX_CONTEXT RxContext)
462{
463 NTSTATUS Status = STATUS_SUCCESS;
464
465 RxCaptureFcb;
466 RxCaptureFobx;
467
468 PMRX_NET_ROOT pNetRoot = capFcb->pNetRoot;
469 PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
470 PUNICODE_STRING RemainingName = GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext);
471
472 FILE_BASIC_INFORMATION FileBasicInfo;
473 FILE_STANDARD_INFORMATION FileStandardInfo;
474
475 ULONG CreateAction = FILE_CREATED;
476 SHFLHANDLE Handle = SHFL_HANDLE_NIL;
477 PMRX_VBOX_FOBX pVBoxFobx;
478
479 RT_NOREF(__C_Fobx); /* RxCaptureFobx */
480
481 Log(("VBOXSF: MRxCreate: name ptr %p length=%d, SrvOpen->Flags 0x%08X\n",
482 RemainingName, RemainingName->Length, SrvOpen->Flags));
483
484 /* Disable FastIO. It causes a verifier bugcheck. */
485#ifdef SRVOPEN_FLAG_DONTUSE_READ_CACHING
486 SetFlag(SrvOpen->Flags, SRVOPEN_FLAG_DONTUSE_READ_CACHING | SRVOPEN_FLAG_DONTUSE_WRITE_CACHING);
487#else
488 SetFlag(SrvOpen->Flags, SRVOPEN_FLAG_DONTUSE_READ_CACHEING | SRVOPEN_FLAG_DONTUSE_WRITE_CACHEING);
489#endif
490
491 if (RemainingName->Length)
492 {
493 Log(("VBOXSF: MRxCreate: Attempt to open %.*ls\n",
494 RemainingName->Length/sizeof(WCHAR), RemainingName->Buffer));
495 }
496 else
497 {
498 if (FlagOn(RxContext->Create.Flags, RX_CONTEXT_CREATE_FLAG_STRIPPED_TRAILING_BACKSLASH))
499 {
500 Log(("VBOXSF: MRxCreate: Empty name -> Only backslash used\n"));
501 RemainingName = &UnicodeBackslash;
502 }
503 }
504
505 if ( pNetRoot->Type != NET_ROOT_WILD
506 && pNetRoot->Type != NET_ROOT_DISK)
507 {
508 Log(("VBOXSF: MRxCreate: netroot type %d not supported\n",
509 pNetRoot->Type));
510 Status = STATUS_NOT_IMPLEMENTED;
511 goto Exit;
512 }
513
514 FileBasicInfo.FileAttributes = FILE_ATTRIBUTE_NORMAL;
515
516 Status = vbsfProcessCreate(RxContext,
517 RemainingName,
518 &FileBasicInfo,
519 &FileStandardInfo,
520 RxContext->Create.EaBuffer,
521 RxContext->Create.EaLength,
522 &CreateAction,
523 &Handle);
524
525 if (Status != STATUS_SUCCESS)
526 {
527 Log(("VBOXSF: MRxCreate: vbsfProcessCreate failed 0x%08X\n",
528 Status));
529 goto Exit;
530 }
531
532 Log(("VBOXSF: MRxCreate: EOF is 0x%RX64 AllocSize is 0x%RX64\n",
533 FileStandardInfo.EndOfFile.QuadPart, FileStandardInfo.AllocationSize.QuadPart));
534
535 RxContext->pFobx = RxCreateNetFobx(RxContext, SrvOpen);
536 if (!RxContext->pFobx)
537 {
538 Log(("VBOXSF: MRxCreate: RxCreateNetFobx failed\n"));
539 Status = STATUS_INSUFFICIENT_RESOURCES;
540 goto Exit;
541 }
542
543 Log(("VBOXSF: MRxCreate: CreateAction = 0x%08X\n",
544 CreateAction));
545
546 RxContext->Create.ReturnedCreateInformation = CreateAction;
547
548 if (capFcb->OpenCount == 0)
549 {
550 FCB_INIT_PACKET InitPacket;
551 RxFormInitPacket(InitPacket,
552 &FileBasicInfo.FileAttributes,
553 &FileStandardInfo.NumberOfLinks,
554 &FileBasicInfo.CreationTime,
555 &FileBasicInfo.LastAccessTime,
556 &FileBasicInfo.LastWriteTime,
557 &FileBasicInfo.ChangeTime,
558 &FileStandardInfo.AllocationSize,
559 &FileStandardInfo.EndOfFile,
560 &FileStandardInfo.EndOfFile);
561 RxFinishFcbInitialization(capFcb, RDBSS_STORAGE_NTC(FileTypeFile), &InitPacket);
562 }
563
564 SrvOpen->BufferingFlags = 0;
565
566 RxContext->pFobx->OffsetOfNextEaToReturn = 1;
567
568 pVBoxFobx = VBoxMRxGetFileObjectExtension(RxContext->pFobx);
569
570 Log(("VBOXSF: MRxCreate: VBoxFobx = %p\n",
571 pVBoxFobx));
572
573 if (!pVBoxFobx)
574 {
575 Log(("VBOXSF: MRxCreate: no VBoxFobx!\n"));
576 AssertFailed();
577 Status = STATUS_INSUFFICIENT_RESOURCES;
578 goto Exit;
579 }
580
581 Log(("VBOXSF: MRxCreate: FileBasicInformation: CreationTime %RX64\n", FileBasicInfo.CreationTime.QuadPart));
582 Log(("VBOXSF: MRxCreate: FileBasicInformation: LastAccessTime %RX64\n", FileBasicInfo.LastAccessTime.QuadPart));
583 Log(("VBOXSF: MRxCreate: FileBasicInformation: LastWriteTime %RX64\n", FileBasicInfo.LastWriteTime.QuadPart));
584 Log(("VBOXSF: MRxCreate: FileBasicInformation: ChangeTime %RX64\n", FileBasicInfo.ChangeTime.QuadPart));
585 Log(("VBOXSF: MRxCreate: FileBasicInformation: FileAttributes %RX32\n", FileBasicInfo.FileAttributes));
586
587 pVBoxFobx->hFile = Handle;
588 pVBoxFobx->pSrvCall = RxContext->Create.pSrvCall;
589 pVBoxFobx->FileStandardInfo = FileStandardInfo;
590 pVBoxFobx->FileBasicInfo = FileBasicInfo;
591 pVBoxFobx->fKeepCreationTime = FALSE;
592 pVBoxFobx->fKeepLastAccessTime = FALSE;
593 pVBoxFobx->fKeepLastWriteTime = FALSE;
594 pVBoxFobx->fKeepChangeTime = FALSE;
595 pVBoxFobx->SetFileInfoOnCloseFlags = 0;
596
597 if (!RxIsFcbAcquiredExclusive(capFcb))
598 {
599 RxAcquireExclusiveFcbResourceInMRx(capFcb);
600 }
601
602 Log(("VBOXSF: MRxCreate: NetRoot is %p, Fcb is %p, SrvOpen is %p, Fobx is %p\n",
603 pNetRoot, capFcb, SrvOpen, RxContext->pFobx));
604 Log(("VBOXSF: MRxCreate: return 0x%08X\n",
605 Status));
606
607Exit:
608 return Status;
609}
610
611NTSTATUS VBoxMRxComputeNewBufferingState(IN OUT PMRX_SRV_OPEN pMRxSrvOpen, IN PVOID pMRxContext, OUT PULONG pNewBufferingState)
612{
613 RT_NOREF(pMRxSrvOpen, pMRxContext, pNewBufferingState);
614 Log(("VBOXSF: MRxComputeNewBufferingState\n"));
615 return STATUS_NOT_SUPPORTED;
616}
617
618NTSTATUS VBoxMRxDeallocateForFcb(IN OUT PMRX_FCB pFcb)
619{
620 RT_NOREF(pFcb);
621 Log(("VBOXSF: MRxDeallocateForFcb\n"));
622 return STATUS_SUCCESS;
623}
624
625NTSTATUS VBoxMRxDeallocateForFobx(IN OUT PMRX_FOBX pFobx)
626{
627 RT_NOREF(pFobx);
628 Log(("VBOXSF: MRxDeallocateForFobx\n"));
629 return STATUS_SUCCESS;
630}
631
632NTSTATUS VBoxMRxTruncate(IN PRX_CONTEXT RxContext)
633{
634 RT_NOREF(RxContext);
635 Log(("VBOXSF: MRxTruncate\n"));
636 return STATUS_NOT_IMPLEMENTED;
637}
638
639NTSTATUS VBoxMRxCleanupFobx(IN PRX_CONTEXT RxContext)
640{
641 PMRX_VBOX_FOBX pVBoxFobx = VBoxMRxGetFileObjectExtension(RxContext->pFobx);
642
643 Log(("VBOXSF: MRxCleanupFobx: pVBoxFobx = %p, Handle = 0x%RX64\n", pVBoxFobx, pVBoxFobx? pVBoxFobx->hFile: 0));
644
645 if (!pVBoxFobx)
646 return STATUS_INVALID_PARAMETER;
647
648 return STATUS_SUCCESS;
649}
650
651NTSTATUS VBoxMRxForceClosed(IN PMRX_SRV_OPEN pSrvOpen)
652{
653 RT_NOREF(pSrvOpen);
654 Log(("VBOXSF: MRxForceClosed\n"));
655 return STATUS_NOT_IMPLEMENTED;
656}
657
658NTSTATUS vbsfSetFileInfo(PMRX_VBOX_DEVICE_EXTENSION pDeviceExtension,
659 PMRX_VBOX_NETROOT_EXTENSION pNetRootExtension,
660 PMRX_VBOX_FOBX pVBoxFobx,
661 PFILE_BASIC_INFORMATION pInfo,
662 BYTE SetAttrFlags)
663{
664 NTSTATUS Status = STATUS_SUCCESS;
665
666 int vboxRC;
667 PSHFLFSOBJINFO pSHFLFileInfo;
668
669 uint8_t *pHGCMBuffer = NULL;
670 uint32_t cbBuffer = 0;
671
672 Log(("VBOXSF: vbsfSetFileInfo: SetAttrFlags 0x%02X\n", SetAttrFlags));
673 Log(("VBOXSF: vbsfSetFileInfo: FileBasicInformation: CreationTime %RX64\n", pInfo->CreationTime.QuadPart));
674 Log(("VBOXSF: vbsfSetFileInfo: FileBasicInformation: LastAccessTime %RX64\n", pInfo->LastAccessTime.QuadPart));
675 Log(("VBOXSF: vbsfSetFileInfo: FileBasicInformation: LastWriteTime %RX64\n", pInfo->LastWriteTime.QuadPart));
676 Log(("VBOXSF: vbsfSetFileInfo: FileBasicInformation: ChangeTime %RX64\n", pInfo->ChangeTime.QuadPart));
677 Log(("VBOXSF: vbsfSetFileInfo: FileBasicInformation: FileAttributes %RX32\n", pInfo->FileAttributes));
678
679 if (SetAttrFlags == 0)
680 {
681 Log(("VBOXSF: vbsfSetFileInfo: nothing to set\n"));
682 return STATUS_SUCCESS;
683 }
684
685 cbBuffer = sizeof(SHFLFSOBJINFO);
686 pHGCMBuffer = (uint8_t *)vbsfAllocNonPagedMem(cbBuffer);
687 if (!pHGCMBuffer)
688 {
689 AssertFailed();
690 return STATUS_INSUFFICIENT_RESOURCES;
691 }
692 RtlZeroMemory(pHGCMBuffer, cbBuffer);
693 pSHFLFileInfo = (PSHFLFSOBJINFO)pHGCMBuffer;
694
695 /* The properties, that need to be changed, are set to something other than zero */
696 if (pInfo->CreationTime.QuadPart && (SetAttrFlags & VBOX_FOBX_F_INFO_CREATION_TIME) != 0)
697 RTTimeSpecSetNtTime(&pSHFLFileInfo->BirthTime, pInfo->CreationTime.QuadPart);
698 if (pInfo->LastAccessTime.QuadPart && (SetAttrFlags & VBOX_FOBX_F_INFO_LASTACCESS_TIME) != 0)
699 RTTimeSpecSetNtTime(&pSHFLFileInfo->AccessTime, pInfo->LastAccessTime.QuadPart);
700 if (pInfo->LastWriteTime.QuadPart && (SetAttrFlags & VBOX_FOBX_F_INFO_LASTWRITE_TIME) != 0)
701 RTTimeSpecSetNtTime(&pSHFLFileInfo->ModificationTime, pInfo->LastWriteTime.QuadPart);
702 if (pInfo->ChangeTime.QuadPart && (SetAttrFlags & VBOX_FOBX_F_INFO_CHANGE_TIME) != 0)
703 RTTimeSpecSetNtTime(&pSHFLFileInfo->ChangeTime, pInfo->ChangeTime.QuadPart);
704 if (pInfo->FileAttributes && (SetAttrFlags & VBOX_FOBX_F_INFO_ATTRIBUTES) != 0)
705 pSHFLFileInfo->Attr.fMode = NTToVBoxFileAttributes(pInfo->FileAttributes);
706
707 vboxRC = VbglR0SfFsInfo(&pDeviceExtension->hgcmClient, &pNetRootExtension->map, pVBoxFobx->hFile,
708 SHFL_INFO_SET | SHFL_INFO_FILE, &cbBuffer, (PSHFLDIRINFO)pSHFLFileInfo);
709
710 if (vboxRC != VINF_SUCCESS)
711 Status = VBoxErrorToNTStatus(vboxRC);
712
713 if (pHGCMBuffer)
714 vbsfFreeNonPagedMem(pHGCMBuffer);
715
716 Log(("VBOXSF: vbsfSetFileInfo: Returned 0x%08X\n", Status));
717 return Status;
718}
719
720/*
721 * Closes an opened file handle of a MRX_VBOX_FOBX.
722 * Updates file attributes if necessary.
723 */
724NTSTATUS vbsfCloseFileHandle(PMRX_VBOX_DEVICE_EXTENSION pDeviceExtension,
725 PMRX_VBOX_NETROOT_EXTENSION pNetRootExtension,
726 PMRX_VBOX_FOBX pVBoxFobx)
727{
728 NTSTATUS Status = STATUS_SUCCESS;
729
730 int vboxRC;
731
732 if (pVBoxFobx->hFile == SHFL_HANDLE_NIL)
733 {
734 Log(("VBOXSF: vbsfCloseFileHandle: SHFL_HANDLE_NIL\n"));
735 return STATUS_SUCCESS;
736 }
737
738 Log(("VBOXSF: vbsfCloseFileHandle: 0x%RX64, on close info 0x%02X\n",
739 pVBoxFobx->hFile, pVBoxFobx->SetFileInfoOnCloseFlags));
740
741 if (pVBoxFobx->SetFileInfoOnCloseFlags)
742 {
743 /* If the file timestamps were set by the user, then update them before closing the handle,
744 * to cancel any effect of the file read/write operations on the host.
745 */
746 Status = vbsfSetFileInfo(pDeviceExtension,
747 pNetRootExtension,
748 pVBoxFobx,
749 &pVBoxFobx->FileBasicInfo,
750 pVBoxFobx->SetFileInfoOnCloseFlags);
751 }
752
753 vboxRC = VbglR0SfClose(&pDeviceExtension->hgcmClient,
754 &pNetRootExtension->map,
755 pVBoxFobx->hFile);
756
757 pVBoxFobx->hFile = SHFL_HANDLE_NIL;
758
759 if (vboxRC != VINF_SUCCESS)
760 Status = VBoxErrorToNTStatus(vboxRC);
761
762 Log(("VBOXSF: vbsfCloseFileHandle: Returned 0x%08X\n", Status));
763 return Status;
764}
765
766NTSTATUS VBoxMRxCloseSrvOpen(IN PRX_CONTEXT RxContext)
767{
768 NTSTATUS Status = STATUS_SUCCESS;
769
770 RxCaptureFcb;
771 RxCaptureFobx;
772
773 PMRX_VBOX_DEVICE_EXTENSION pDeviceExtension = VBoxMRxGetDeviceExtension(RxContext);
774 PMRX_VBOX_NETROOT_EXTENSION pNetRootExtension = VBoxMRxGetNetRootExtension(capFcb->pNetRoot);
775 PMRX_VBOX_FOBX pVBoxFobx = VBoxMRxGetFileObjectExtension(capFobx);
776 PMRX_SRV_OPEN pSrvOpen = capFobx->pSrvOpen;
777
778 PUNICODE_STRING RemainingName = NULL;
779
780 Log(("VBOXSF: MRxCloseSrvOpen: capFcb = %p, capFobx = %p, pVBoxFobx = %p, pSrvOpen = %p\n",
781 capFcb, capFobx, pVBoxFobx, pSrvOpen));
782
783 RemainingName = pSrvOpen->pAlreadyPrefixedName;
784
785 Log(("VBOXSF: MRxCloseSrvOpen: Remaining name = %.*ls, Len = %d\n",
786 RemainingName->Length / sizeof(WCHAR), RemainingName->Buffer, RemainingName->Length));
787
788 if (!pVBoxFobx)
789 return STATUS_INVALID_PARAMETER;
790
791 if (FlagOn(pSrvOpen->Flags, (SRVOPEN_FLAG_FILE_RENAMED | SRVOPEN_FLAG_FILE_DELETED)))
792 {
793 /* If we renamed or delete the file/dir, then it's already closed */
794 Assert(pVBoxFobx->hFile == SHFL_HANDLE_NIL);
795 Log(("VBOXSF: MRxCloseSrvOpen: File was renamed, handle 0x%RX64 ignore close.\n",
796 pVBoxFobx->hFile));
797 return STATUS_SUCCESS;
798 }
799
800 /* Close file */
801 if (pVBoxFobx->hFile != SHFL_HANDLE_NIL)
802 vbsfCloseFileHandle(pDeviceExtension, pNetRootExtension, pVBoxFobx);
803
804 if (capFcb->FcbState & FCB_STATE_DELETE_ON_CLOSE)
805 {
806 Log(("VBOXSF: MRxCloseSrvOpen: Delete on close. Open count = %d\n",
807 capFcb->OpenCount));
808
809 /* Remove file or directory if delete action is pending. */
810 if (capFcb->OpenCount == 0)
811 Status = vbsfRemove(RxContext);
812 }
813
814 return Status;
815}
816
817NTSTATUS vbsfRemove(IN PRX_CONTEXT RxContext)
818{
819 NTSTATUS Status = STATUS_SUCCESS;
820
821 RxCaptureFcb;
822 RxCaptureFobx;
823
824 PMRX_VBOX_DEVICE_EXTENSION pDeviceExtension = VBoxMRxGetDeviceExtension(RxContext);
825 PMRX_VBOX_NETROOT_EXTENSION pNetRootExtension = VBoxMRxGetNetRootExtension(capFcb->pNetRoot);
826 PMRX_VBOX_FOBX pVBoxFobx = VBoxMRxGetFileObjectExtension(capFobx);
827
828 PUNICODE_STRING RemainingName = GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext);
829
830 int vboxRC;
831 PSHFLSTRING ParsedPath = NULL;
832
833 Log(("VBOXSF: vbsfRemove: Delete %.*ls. open count = %d\n",
834 RemainingName->Length / sizeof(WCHAR), RemainingName->Buffer, capFcb->OpenCount));
835
836 /* Close file first if not already done. */
837 if (pVBoxFobx->hFile != SHFL_HANDLE_NIL)
838 vbsfCloseFileHandle(pDeviceExtension, pNetRootExtension, pVBoxFobx);
839
840 Log(("VBOXSF: vbsfRemove: RemainingName->Length %d\n", RemainingName->Length));
841 Status = vbsfShflStringFromUnicodeAlloc(&ParsedPath, RemainingName->Buffer, RemainingName->Length);
842 if (Status != STATUS_SUCCESS)
843 return Status;
844
845 /* Call host. */
846 vboxRC = VbglR0SfRemove(&pDeviceExtension->hgcmClient, &pNetRootExtension->map,
847 ParsedPath,
848 (pVBoxFobx->FileStandardInfo.Directory) ? SHFL_REMOVE_DIR : SHFL_REMOVE_FILE);
849
850 if (ParsedPath)
851 vbsfFreeNonPagedMem(ParsedPath);
852
853 if (vboxRC == VINF_SUCCESS)
854 SetFlag(capFobx->pSrvOpen->Flags, SRVOPEN_FLAG_FILE_DELETED);
855
856 Status = VBoxErrorToNTStatus(vboxRC);
857 if (vboxRC != VINF_SUCCESS)
858 Log(("VBOXSF: vbsfRemove: VbglR0SfRemove failed with %Rrc\n", vboxRC));
859
860 Log(("VBOXSF: vbsfRemove: Returned 0x%08X\n", Status));
861 return Status;
862}
863
864NTSTATUS vbsfRename(IN PRX_CONTEXT RxContext,
865 IN FILE_INFORMATION_CLASS FileInformationClass,
866 IN PVOID pBuffer,
867 IN ULONG BufferLength)
868{
869 NTSTATUS Status = STATUS_SUCCESS;
870
871 RxCaptureFcb;
872 RxCaptureFobx;
873
874 PMRX_VBOX_DEVICE_EXTENSION pDeviceExtension = VBoxMRxGetDeviceExtension(RxContext);
875 PMRX_VBOX_NETROOT_EXTENSION pNetRootExtension = VBoxMRxGetNetRootExtension(capFcb->pNetRoot);
876 PMRX_VBOX_FOBX pVBoxFobx = VBoxMRxGetFileObjectExtension(capFobx);
877 PMRX_SRV_OPEN pSrvOpen = capFobx->pSrvOpen;
878
879 PFILE_RENAME_INFORMATION RenameInformation = (PFILE_RENAME_INFORMATION)RxContext->Info.Buffer;
880 PUNICODE_STRING RemainingName = GET_ALREADY_PREFIXED_NAME(pSrvOpen, capFcb);
881
882 int vboxRC;
883 PSHFLSTRING SrcPath = 0, DestPath = 0;
884 ULONG flags;
885
886 RT_NOREF(FileInformationClass, pBuffer, BufferLength);
887
888 Assert(FileInformationClass == FileRenameInformation);
889
890 Log(("VBOXSF: vbsfRename: FileName = %.*ls\n",
891 RenameInformation->FileNameLength / sizeof(WCHAR), &RenameInformation->FileName[0]));
892
893 /* Must close the file before renaming it! */
894 if (pVBoxFobx->hFile != SHFL_HANDLE_NIL)
895 vbsfCloseFileHandle(pDeviceExtension, pNetRootExtension, pVBoxFobx);
896
897 /* Mark it as renamed, so we do nothing during close */
898 SetFlag(pSrvOpen->Flags, SRVOPEN_FLAG_FILE_RENAMED);
899
900 Log(("VBOXSF: vbsfRename: RenameInformation->FileNameLength = %d\n", RenameInformation->FileNameLength));
901 Status = vbsfShflStringFromUnicodeAlloc(&DestPath, RenameInformation->FileName, (uint16_t)RenameInformation->FileNameLength);
902 if (Status != STATUS_SUCCESS)
903 return Status;
904
905 Log(("VBOXSF: vbsfRename: Destination path = %.*ls\n",
906 DestPath->u16Length / sizeof(WCHAR), &DestPath->String.ucs2[0]));
907
908 Log(("VBOXSF: vbsfRename: RemainingName->Length = %d\n", RemainingName->Length));
909 Status = vbsfShflStringFromUnicodeAlloc(&SrcPath, RemainingName->Buffer, RemainingName->Length);
910 if (Status != STATUS_SUCCESS)
911 {
912 vbsfFreeNonPagedMem(DestPath);
913 return Status;
914 }
915
916 Log(("VBOXSF: vbsfRename: Source path = %.*ls\n",
917 SrcPath->u16Length / sizeof(WCHAR), &SrcPath->String.ucs2[0]));
918
919 /* Call host. */
920 flags = pVBoxFobx->FileStandardInfo.Directory? SHFL_RENAME_DIR : SHFL_RENAME_FILE;
921 if (RenameInformation->ReplaceIfExists)
922 flags |= SHFL_RENAME_REPLACE_IF_EXISTS;
923
924 Log(("VBOXSF: vbsfRename: Calling VbglR0SfRename\n"));
925 vboxRC = VbglR0SfRename(&pDeviceExtension->hgcmClient, &pNetRootExtension->map, SrcPath, DestPath, flags);
926
927 vbsfFreeNonPagedMem(SrcPath);
928 vbsfFreeNonPagedMem(DestPath);
929
930 Status = VBoxErrorToNTStatus(vboxRC);
931 if (vboxRC != VINF_SUCCESS)
932 Log(("VBOXSF: vbsfRename: VbglR0SfRename failed with %Rrc\n", vboxRC));
933
934 Log(("VBOXSF: vbsfRename: Returned 0x%08X\n", Status));
935 return Status;
936}
937
938NTSTATUS VBoxMRxShouldTryToCollapseThisOpen(IN PRX_CONTEXT RxContext)
939{
940 RT_NOREF(RxContext);
941 Log(("VBOXSF: MRxShouldTryToCollapseThisOpen\n"));
942 return STATUS_MORE_PROCESSING_REQUIRED;
943}
944
945NTSTATUS VBoxMRxCollapseOpen(IN OUT PRX_CONTEXT RxContext)
946{
947 RT_NOREF(RxContext);
948 Log(("VBOXSF: MRxCollapseOpen\n"));
949 return STATUS_MORE_PROCESSING_REQUIRED;
950}
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