VirtualBox

source: vbox/trunk/src/VBox/Additions/darwin/vboxfs/VBoxVFS-VFSOps.cpp@ 58154

Last change on this file since 58154 was 57063, checked in by vboxsync, 9 years ago

Mac OS X GAs: shared folders: add headers and export to OSE.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.4 KB
Line 
1/* $Id: VBoxVFS-VFSOps.cpp 57063 2015-07-23 15:50:11Z vboxsync $ */
2/** @file
3 * VBoxVFS - Filesystem operations.
4 */
5
6/*
7 * Copyright (C) 2013-2015 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#include <mach/kmod.h>
28#include <libkern/libkern.h>
29#include <mach/mach_types.h>
30
31#include <sys/mount.h>
32#include <sys/vnode.h>
33
34#include <iprt/assert.h>
35#include <iprt/mem.h>
36#include <iprt/asm.h>
37
38#include "vboxvfs.h"
39
40/* States of VBoxVFS object used in atomic variables
41 * in order to reach sync between several concurrently running threads. */
42#define VBOXVFS_OBJECT_UNINITIALIZED (0)
43#define VBOXVFS_OBJECT_INITIALIZING (1)
44#define VBOXVFS_OBJECT_INITIALIZED (2)
45#define VBOXVFS_OBJECT_INVALID (3)
46
47
48/**
49 * Mount helper: Get mounting parameters from user space and validate them.
50 *
51 * @param pUserData Mounting parameters provided by user space mount tool.
52 * @param pKernelData Buffer to store pUserData in kernel space.
53 *
54 * @return 0 on success or BSD error code otherwise.
55 */
56static int
57vboxvfs_get_mount_info(user_addr_t pUserData, struct vboxvfs_mount_info *pKernelData)
58{
59 AssertReturn(pKernelData, EINVAL);
60 AssertReturn(pUserData, EINVAL);
61
62 /* Get mount parameters from user space */
63 if (copyin(pUserData, pKernelData, sizeof(struct vboxvfs_mount_info)) != 0)
64 {
65 PERROR("Unable to copy mount parameters from user space");
66 return EINVAL;
67 }
68
69 /* Validate data magic */
70 if (pKernelData->magic != VBOXVFS_MOUNTINFO_MAGIC)
71 {
72 PERROR("Mount parameter magic mismatch");
73 return EINVAL;
74 }
75
76 return 0;
77}
78
79
80/**
81 * Mount helper: Provide VFS layer with a VBox share name (stored as mounted device).
82 *
83 * @param mp Mount data provided by VFS layer.
84 * @param szShareName VBox share name.
85 * @param cbShareName Returning parameter which contains VBox share name string length.
86 *
87 * @return 0 on success or BSD error code otherwise.
88 */
89static int
90vboxvfs_set_share_name(struct mount *mp, char *szShareName, size_t *cbShareName)
91{
92 struct vfsstatfs *pVfsInfo;
93
94 AssertReturn(mp, EINVAL);
95 AssertReturn(szShareName, EINVAL);
96 AssertReturn(cbShareName, EINVAL);
97
98 pVfsInfo = vfs_statfs(mp);
99 if (!pVfsInfo)
100 {
101 PERROR("Unable to get VFS data for the mount structure");
102 return EINVAL;
103 }
104
105 return copystr(szShareName, pVfsInfo->f_mntfromname, MAXPATHLEN, cbShareName);
106}
107
108
109/**
110 * Mount helper: allocate locking group attribute and locking group itself.
111 * Store allocated data into VBoxVFS private data.
112 *
113 * @param pMount VBoxVFS global data which will be updated with
114 * locking group and its attribute in case of success;
115 * otherwise pMount unchanged.
116 *
117 * @return 0 on success or BSD error code otherwise.
118 *
119 */
120static int
121vboxvfs_prepare_locking(vboxvfs_mount_t *pMount)
122{
123 lck_grp_attr_t *pGrpAttr;
124 lck_grp_t *pGrp;
125
126 AssertReturn(pMount, EINVAL);
127
128 pGrpAttr = lck_grp_attr_alloc_init();
129 if (pGrpAttr)
130 {
131 pGrp = lck_grp_alloc_init("VBoxVFS", pGrpAttr);
132 if (pGrp)
133 {
134 pMount->pLockGroupAttr = pGrpAttr;
135 pMount->pLockGroup = pGrp;
136
137 return 0;
138 }
139 else
140 PERROR("Unable to allocate locking group");
141
142 lck_grp_attr_free(pGrpAttr);
143 }
144 else
145 PERROR("Unable to allocate locking group attribute");
146
147 return ENOMEM;
148}
149
150
151/**
152 * Mount and unmount helper: destroy locking group attribute and locking group itself.
153 *
154 * @param pMount VBoxVFS global data for which locking
155 * group and attribute will be deallocated and set to NULL.
156 */
157static void
158vboxvfs_destroy_locking(vboxvfs_mount_t *pMount)
159{
160 AssertReturnVoid(pMount);
161
162 if (pMount->pLockGroup)
163 {
164 lck_grp_free(pMount->pLockGroup);
165 pMount->pLockGroup = NULL;
166 }
167
168 if (pMount->pLockGroupAttr)
169 {
170 lck_grp_attr_free(pMount->pLockGroupAttr);
171 pMount->pLockGroupAttr = NULL;
172 }
173}
174
175/**
176 * Mount helper: Allocate and init VBoxVFS global data.
177 *
178 * @param mp Mount data provided by VFS layer.
179 * @param pUserData Mounting parameters provided by user space mount tool.
180 *
181 * @return VBoxVFS global data or NULL.
182 */
183static vboxvfs_mount_t *
184vboxvfs_alloc_internal_data(struct mount *mp, user_addr_t pUserData)
185{
186 vboxvfs_mount_t *pMount;
187 struct vboxvfs_mount_info mountInfo;
188 struct vfsstatfs *pVfsInfo;
189 size_t cbShareName;
190
191 int rc;
192
193 AssertReturn(mp, NULL);
194 AssertReturn(pUserData, NULL);
195
196 pVfsInfo = vfs_statfs(mp);
197 AssertReturn(pVfsInfo, NULL);
198
199 /* Allocate memory for VBoxVFS internal data */
200 pMount = (vboxvfs_mount_t *)RTMemAllocZ(sizeof(vboxvfs_mount_t));
201 if (pMount)
202 {
203 rc = vboxvfs_get_mount_info(pUserData, &mountInfo);
204 if (rc == 0)
205 {
206 PDEBUG("Mounting shared folder '%s'", mountInfo.name);
207
208 /* Prepare for locking. We prepare locking group and attr data here,
209 * but allocate and initialize real lock in vboxvfs_create_vnode_internal().
210 * We use the same pLockGroup and pLockAttr for all vnodes related to this mount point. */
211 rc = vboxvfs_prepare_locking(pMount);
212 if (rc == 0)
213 {
214 rc = vboxvfs_set_share_name(mp, (char *)&mountInfo.name, &cbShareName);
215 if (rc == 0)
216 {
217 pMount->pShareName = vboxvfs_construct_shflstring((char *)&mountInfo.name, cbShareName);
218 if (pMount->pShareName)
219 {
220 /* Remember user who mounted this share */
221 pMount->owner = pVfsInfo->f_owner;
222
223 /* Mark root vnode as uninitialized */
224 ASMAtomicWriteU8(&pMount->fRootVnodeState, VBOXVFS_OBJECT_UNINITIALIZED);
225
226 return pMount;
227 }
228 }
229 }
230
231 vboxvfs_destroy_locking(pMount);
232 }
233
234 RTMemFree(pMount);
235 }
236
237 return NULL;
238}
239
240
241/**
242 * Mount and unmount helper: Release VBoxVFS internal resources.
243 * Deallocates ppMount as well.
244 *
245 * @param ppMount Pointer to reference of VBoxVFS internal data.
246 */
247static void
248vboxvfs_destroy_internal_data(vboxvfs_mount_t **ppMount)
249{
250 AssertReturnVoid(ppMount);
251 AssertReturnVoid(*ppMount);
252 AssertReturnVoid((*ppMount)->pShareName);
253
254 RTMemFree((*ppMount)->pShareName);
255 (*ppMount)->pShareName = NULL;
256
257 vboxvfs_destroy_locking(*ppMount);
258 RTMemFree(*ppMount);
259 *ppMount = NULL;
260}
261
262
263/**
264 * Mount VBoxVFS.
265 *
266 * @param mp Mount data provided by VFS layer.
267 * @param pDev Device structure provided by VFS layer.
268 * @param pUserData Mounting parameters provided by user space mount tool.
269 * @param pContext kAuth context needed in order to authentificate mount operation.
270 *
271 * @return 0 on success or BSD error code otherwise.
272 */
273static int
274vboxvfs_mount(struct mount *mp, vnode_t pDev, user_addr_t pUserData, vfs_context_t pContext)
275{
276 NOREF(pDev);
277 NOREF(pContext);
278
279 vboxvfs_mount_t *pMount;
280
281 int rc = ENOMEM;
282
283 PDEBUG("Mounting...");
284
285 pMount = vboxvfs_alloc_internal_data(mp, pUserData);
286 if (pMount)
287 {
288 rc = vboxCallMapFolder(&g_vboxSFClient, pMount->pShareName, &pMount->pMap);
289 if (RT_SUCCESS(rc))
290 {
291 /* Private data should be set before vboxvfs_create_vnode_internal() call
292 * because we need it in order to create vnode. */
293 vfs_setfsprivate(mp, pMount);
294
295 /* Reset root vnode */
296 pMount->pRootVnode = NULL;
297
298 vfs_setflags(mp, MNT_RDONLY | MNT_SYNCHRONOUS | MNT_NOEXEC | MNT_NOSUID | MNT_NODEV);
299
300 PDEBUG("VirtualBox shared folder successfully mounted");
301
302 return 0;
303 }
304 else
305 PDEBUG("Unable to map shared folder");
306
307 vboxvfs_destroy_internal_data(&pMount);
308 }
309 else
310 PDEBUG("Unable to allocate internal data");
311
312 return rc;
313}
314
315
316/**
317 * Unmount VBoxVFS.
318 *
319 * @param mp Mount data provided by VFS layer.
320 * @param fFlags Unmounting flags.
321 * @param pContext kAuth context needed in order to authentificate mount operation.
322 *
323 * @return 0 on success or BSD error code otherwise.
324 */
325static int
326vboxvfs_unmount(struct mount *mp, int fFlags, vfs_context_t pContext)
327{
328 NOREF(pContext);
329
330 vboxvfs_mount_t *pMount;
331 int rc = EBUSY;
332 int fFlush = (fFlags & MNT_FORCE) ? FORCECLOSE : 0;
333
334 PDEBUG("Attempting to %s unmount a shared folder", (fFlags & MNT_FORCE) ? "forcibly" : "normally");
335
336 AssertReturn(mp, EINVAL);
337
338 pMount = (vboxvfs_mount_t *)vfs_fsprivate(mp);
339
340 AssertReturn(pMount, EINVAL);
341 AssertReturn(pMount->pRootVnode, EINVAL);
342
343 /* Check if we can do unmount at the moment */
344 if (!vnode_isinuse(pMount->pRootVnode, 1))
345 {
346 /* Flush child vnodes first */
347 rc = vflush(mp, pMount->pRootVnode, fFlush);
348 if (rc == 0)
349 {
350 /* Flush root vnode */
351 rc = vflush(mp, NULL, fFlush);
352 if (rc == 0)
353 {
354 vfs_setfsprivate(mp, NULL);
355
356 rc = vboxCallUnmapFolder(&g_vboxSFClient, &pMount->pMap);
357 if (RT_SUCCESS(rc))
358 {
359 vboxvfs_destroy_internal_data(&pMount);
360 PDEBUG("A shared folder has been successfully unmounted");
361 return 0;
362 }
363 else
364 {
365 PDEBUG("Unable to unmount shared folder");
366 rc = EPROTO;
367 }
368 }
369 else
370 PDEBUG("Unable to flush filesystem before unmount, some data might be lost");
371 }
372 else
373 PDEBUG("Unable to flush child vnodes");
374
375 }
376 else
377 PDEBUG("Root vnode is in use, can't unmount");
378
379 vnode_put(pMount->pRootVnode);
380
381 return rc;
382}
383
384
385/**
386 * Get VBoxVFS root vnode.
387 *
388 * Handle three cases here:
389 * - vnode does not exist yet: create a new one
390 * - currently creating vnode: wait till the end, increment usage count and return existing one
391 * - vnode already created: increment usage count and return existing one
392 * - vnode was failed to create: give a chance to try to re-create it later
393 *
394 * @param mp Mount data provided by VFS layer.
395 * @param ppVnode vnode to return.
396 * @param pContext kAuth context needed in order to authentificate mount operation.
397 *
398 * @return 0 on success or BSD error code otherwise.
399 */
400static int
401vboxvfs_root(struct mount *mp, struct vnode **ppVnode, vfs_context_t pContext)
402{
403 NOREF(pContext);
404
405 vboxvfs_mount_t *pMount;
406 int rc = 0;
407 uint32_t vid;
408
409 PDEBUG("Getting root vnode...");
410
411 AssertReturn(mp, EINVAL);
412 AssertReturn(ppVnode, EINVAL);
413
414 pMount = (vboxvfs_mount_t *)vfs_fsprivate(mp);
415 AssertReturn(pMount, EINVAL);
416
417 /* Check case when vnode does not exist yet */
418 if (ASMAtomicCmpXchgU8(&pMount->fRootVnodeState, VBOXVFS_OBJECT_INITIALIZING, VBOXVFS_OBJECT_UNINITIALIZED))
419 {
420 PDEBUG("Create new root vnode");
421
422 /* Allocate empty SHFLSTRING to indicate path to root vnode within Shared Folder */
423 char szEmpty[1];
424 SHFLSTRING *pSFVnodePath;
425
426 pSFVnodePath = vboxvfs_construct_shflstring((char *)szEmpty, 0);
427 if (pSFVnodePath)
428 {
429 int rc2;
430 rc2 = vboxvfs_create_vnode_internal(mp, VDIR, NULL, TRUE, pSFVnodePath, &pMount->pRootVnode);
431 if (rc2 != 0)
432 {
433 RTMemFree(pSFVnodePath);
434 rc = ENOTSUP;
435 }
436 }
437 else
438 rc = ENOMEM;
439
440 /* Notify other threads about result */
441 if (rc == 0)
442 ASMAtomicWriteU8(&pMount->fRootVnodeState, VBOXVFS_OBJECT_INITIALIZED);
443 else
444 ASMAtomicWriteU8(&pMount->fRootVnodeState, VBOXVFS_OBJECT_INVALID);
445 }
446 else
447 {
448 /* Check case if we are currently creating vnode. Wait while other thread to finish allocation. */
449 uint8_t fRootVnodeState = VBOXVFS_OBJECT_UNINITIALIZED;
450 while (fRootVnodeState != VBOXVFS_OBJECT_INITIALIZED
451 && fRootVnodeState != VBOXVFS_OBJECT_INVALID)
452 {
453 /* @todo: Currently, we are burning CPU cycles while waiting. This is for a short
454 * time but we should relax here! */
455 fRootVnodeState = ASMAtomicReadU8(&pMount->fRootVnodeState);
456
457 }
458
459 /* Check if the other thread initialized root vnode and it is ready to be returned */
460 if (fRootVnodeState == VBOXVFS_OBJECT_INITIALIZED)
461 {
462 /* Take care about iocount */
463 vid = vnode_vid(pMount->pRootVnode);
464 rc = vnode_getwithvid(pMount->pRootVnode, vid);
465 }
466 else
467 {
468 /* Other thread reported initialization failure.
469 * Set vnode state VBOXVFS_OBJECT_UNINITIALIZED in order to try recreate root
470 * vnode in other attempt */
471 ASMAtomicWriteU8(&pMount->fRootVnodeState, VBOXVFS_OBJECT_UNINITIALIZED);
472 }
473
474 }
475
476 /* Only return vnode if we got success */
477 if (rc == 0)
478 {
479 PDEBUG("Root vnode can be returned");
480 *ppVnode = pMount->pRootVnode;
481 }
482 else
483 PDEBUG("Root vnode cannot be returned: 0x%X", rc);
484
485 return rc;
486}
487
488/**
489 * VBoxVFS get VFS layer object attribute callback.
490 *
491 * @param mp Mount data provided by VFS layer.
492 * @param pAttr Output buffer to return attributes.
493 * @param pContext kAuth context needed in order to authentificate mount operation.
494 *
495 * @returns 0 for success, else an error code.
496 */
497static int
498vboxvfs_getattr(struct mount *mp, struct vfs_attr *pAttr, vfs_context_t pContext)
499{
500 NOREF(pContext);
501
502 vboxvfs_mount_t *pMount;
503 SHFLVOLINFO SHFLVolumeInfo;
504
505 int rc;
506 uint32_t cbBuffer = sizeof(SHFLVolumeInfo);
507
508 uint32_t u32bsize;
509 uint64_t u64blocks;
510 uint64_t u64bfree;
511
512 PDEBUG("Getting attribute...\n");
513
514 AssertReturn(mp, EINVAL);
515
516 pMount = (vboxvfs_mount_t *)vfs_fsprivate(mp);
517 AssertReturn(pMount, EINVAL);
518 AssertReturn(pMount->pShareName, EINVAL);
519
520 rc = vboxCallFSInfo(&g_vboxSFClient, &pMount->pMap, 0, SHFL_INFO_GET | SHFL_INFO_VOLUME,
521 &cbBuffer, (PSHFLDIRINFO)&SHFLVolumeInfo);
522 AssertReturn(rc == 0, EPROTO);
523
524 u32bsize = (uint32_t)SHFLVolumeInfo.ulBytesPerAllocationUnit;
525 AssertReturn(u32bsize > 0, ENOTSUP);
526
527 u64blocks = (uint64_t)SHFLVolumeInfo.ullTotalAllocationBytes / (uint64_t)u32bsize;
528 u64bfree = (uint64_t)SHFLVolumeInfo.ullAvailableAllocationBytes / (uint64_t)u32bsize;
529
530 VFSATTR_RETURN(pAttr, f_bsize, u32bsize);
531 VFSATTR_RETURN(pAttr, f_blocks, u64blocks);
532 VFSATTR_RETURN(pAttr, f_bfree, u64bfree);
533 VFSATTR_RETURN(pAttr, f_bavail, u64bfree);
534 VFSATTR_RETURN(pAttr, f_bused, u64blocks - u64bfree);
535
536 VFSATTR_RETURN(pAttr, f_owner, pMount->owner);
537
538 VFSATTR_CLEAR_ACTIVE(pAttr, f_iosize);
539 VFSATTR_CLEAR_ACTIVE(pAttr, f_files);
540 VFSATTR_CLEAR_ACTIVE(pAttr, f_ffree);
541 VFSATTR_CLEAR_ACTIVE(pAttr, f_fssubtype);
542
543 /* todo: take care about f_capabilities and f_attributes, f_fsid */
544 VFSATTR_CLEAR_ACTIVE(pAttr, f_capabilities);
545 VFSATTR_CLEAR_ACTIVE(pAttr, f_attributes);
546 VFSATTR_CLEAR_ACTIVE(pAttr, f_fsid);
547
548 /* todo: take care about f_create_time, f_modify_time, f_access_time, f_backup_time */
549 VFSATTR_CLEAR_ACTIVE(pAttr, f_create_time);
550 VFSATTR_CLEAR_ACTIVE(pAttr, f_modify_time);
551 VFSATTR_CLEAR_ACTIVE(pAttr, f_access_time);
552 VFSATTR_CLEAR_ACTIVE(pAttr, f_backup_time);
553
554 VFSATTR_CLEAR_ACTIVE(pAttr, f_signature);
555 VFSATTR_CLEAR_ACTIVE(pAttr, f_carbon_fsid);
556 VFSATTR_CLEAR_ACTIVE(pAttr, f_uuid);
557
558 if (VFSATTR_IS_ACTIVE(pAttr, f_vol_name))
559 {
560 strlcpy(pAttr->f_vol_name, (char*)pMount->pShareName->String.utf8, MAXPATHLEN);
561 VFSATTR_SET_SUPPORTED(pAttr, f_vol_name);
562 }
563
564 VFSATTR_ALL_SUPPORTED(pAttr);
565
566 return 0;
567}
568
569/* VFS options */
570struct vfsops g_oVBoxVFSOpts = {
571 /* Standard operations */
572 &vboxvfs_mount,
573 NULL, /* Skipped: vfs_start() */
574 &vboxvfs_unmount,
575 &vboxvfs_root,
576 NULL, /* Skipped: vfs_quotactl */
577 &vboxvfs_getattr,
578 NULL, /* Skipped: vfs_sync */
579 NULL, /* Skipped: vfs_vget */
580 NULL, /* Skipped: vfs_fhtovp */
581 NULL, /* Skipped: vfs_vptofh */
582 NULL, /* Skipped: vfs_init */
583 NULL, /* Skipped: vfs_sysctl */
584 NULL, /* Skipped: vfs_setattr */
585 /* Reserved */
586 { NULL, NULL, NULL, NULL, NULL, NULL, NULL, },
587};
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