VirtualBox

source: vbox/trunk/src/VBox/Additions/solaris/SharedFolders/vboxfs_vfs.c@ 76748

Last change on this file since 76748 was 76553, checked in by vboxsync, 6 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 14.3 KB
Line 
1/* $Id: vboxfs_vfs.c 76553 2019-01-01 01:45:53Z vboxsync $ */
2/** @file
3 * VirtualBox File System for Solaris Guests, VFS implementation.
4 */
5
6/*
7 * Copyright (C) 2009-2019 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 <VBox/log.h>
28#include <VBox/version.h>
29
30#include <sys/types.h>
31#include <sys/mntent.h>
32#include <sys/param.h>
33#include <sys/modctl.h>
34#include <sys/mount.h>
35#include <sys/policy.h>
36#include <sys/atomic.h>
37#include <sys/sysmacros.h>
38#include <sys/ddi.h>
39#include <sys/sunddi.h>
40#include <sys/vfs.h>
41#if !defined(VBOX_VFS_SOLARIS_10U6)
42# include <sys/vfs_opreg.h>
43#endif
44#include <sys/pathname.h>
45#include <sys/cmn_err.h>
46#undef u /* /usr/include/sys/user.h:249:1 is where this is defined to (curproc->p_user). very cool. */
47
48#include "vboxfs_prov.h"
49#include "vboxfs_vnode.h"
50#include "vboxfs_vfs.h"
51#include "vboxfs.h"
52
53
54#define VBOXSOLQUOTE2(x) #x
55#define VBOXSOLQUOTE(x) VBOXSOLQUOTE2(x)
56/** The module name. */
57#define DEVICE_NAME "vboxfs"
58/** The module description as seen in 'modinfo'. */
59#define DEVICE_DESC "VirtualBox ShrdFS"
60
61
62/*
63 * Shared Folders filesystem implementation of the Solaris VFS interfaces.
64 * Much of this is cookie cutter code for Solaris filesystem implementation.
65 */
66
67/* forward declarations */
68static int sffs_init(int fstype, char *name);
69static int sffs_mount(vfs_t *, vnode_t *, struct mounta *, cred_t *);
70static int sffs_unmount(vfs_t *vfsp, int flag, cred_t *cr);
71static int sffs_root(vfs_t *vfsp, vnode_t **vpp);
72static int sffs_statvfs(vfs_t *vfsp, statvfs64_t *sbp);
73
74static mntopt_t sffs_options[] = {
75 /* Option Cancels Opt Arg Flags Data */
76 {"uid", NULL, NULL, MO_HASVALUE, NULL},
77 {"gid", NULL, NULL, MO_HASVALUE, NULL},
78 {"dmode", NULL, NULL, MO_HASVALUE, NULL},
79 {"fmode", NULL, NULL, MO_HASVALUE, NULL},
80 {"dmask", NULL, NULL, MO_HASVALUE, NULL},
81 {"fmask", NULL, NULL, MO_HASVALUE, NULL},
82 {"stat_ttl", NULL, NULL, MO_HASVALUE, NULL},
83 {"fsync", NULL, NULL, 0, NULL},
84 {"tag", NULL, NULL, MO_HASVALUE, NULL}
85};
86
87static mntopts_t sffs_options_table = {
88 sizeof (sffs_options) / sizeof (mntopt_t),
89 sffs_options
90};
91
92static vfsdef_t sffs_vfsdef = {
93 VFSDEF_VERSION,
94 DEVICE_NAME,
95 sffs_init,
96 VSW_HASPROTO,
97 &sffs_options_table
98};
99
100static int sffs_fstype;
101static int sffs_major; /* major number for device */
102
103kmutex_t sffs_minor_lock;
104int sffs_minor; /* minor number for device */
105
106/*
107 * Module linkage information
108 */
109static struct modlfs modlfs = {
110 &mod_fsops,
111 DEVICE_DESC " " VBOX_VERSION_STRING "r" VBOXSOLQUOTE(VBOX_SVN_REV),
112 &sffs_vfsdef
113};
114
115static struct modlinkage modlinkage = {
116 MODREV_1, &modlfs, NULL
117};
118
119static sfp_connection_t *sfprov = NULL;
120
121int
122_init()
123{
124 return (mod_install(&modlinkage));
125}
126
127int
128_info(struct modinfo *modinfop)
129{
130 return (mod_info(&modlinkage, modinfop));
131}
132
133
134int
135_fini()
136{
137 int error;
138
139 error = mod_remove(&modlinkage);
140 if (error)
141 return (error);
142
143 /*
144 * Tear down the operations vectors
145 */
146 sffs_vnode_fini();
147 (void) vfs_freevfsops_by_type(sffs_fstype);
148
149 /*
150 * close connection to the provider
151 */
152 sfprov_disconnect(sfprov);
153 return (0);
154}
155
156
157static int
158sffs_init(int fstype, char *name)
159{
160#if defined(VBOX_VFS_SOLARIS_10U6)
161 static const fs_operation_def_t sffs_vfsops_template[] = {
162 VFSNAME_MOUNT, sffs_mount,
163 VFSNAME_UNMOUNT, sffs_unmount,
164 VFSNAME_ROOT, sffs_root,
165 VFSNAME_STATVFS, sffs_statvfs,
166 NULL, NULL
167 };
168#else
169 static const fs_operation_def_t sffs_vfsops_template[] = {
170 VFSNAME_MOUNT, { .vfs_mount = sffs_mount },
171 VFSNAME_UNMOUNT, { .vfs_unmount = sffs_unmount },
172 VFSNAME_ROOT, { .vfs_root = sffs_root },
173 VFSNAME_STATVFS, { .vfs_statvfs = sffs_statvfs },
174 NULL, NULL
175 };
176#endif
177 int error;
178
179 ASSERT(fstype != 0);
180 sffs_fstype = fstype;
181 LogFlowFunc(("sffs_init() name=%s\n", name));
182
183 /*
184 * This may seem a silly way to do things for now. But the code
185 * is structured to easily allow it to be used on other hypervisors
186 * which would have a different implementation of the provider.
187 * Hopefully that'll never happen. :)
188 */
189 sfprov = sfprov_connect(SFPROV_VERSION);
190 if (sfprov == NULL) {
191 cmn_err(CE_WARN, "sffs_init: couldn't init sffs provider");
192 return (ENODEV);
193 }
194
195 error = sfprov_set_show_symlinks();
196 if (error != 0) {
197 cmn_err(CE_WARN, "sffs_init: host unable to show symlinks, "
198 "rc=%d\n", error);
199 }
200
201 error = vfs_setfsops(fstype, sffs_vfsops_template, NULL);
202 if (error != 0) {
203 cmn_err(CE_WARN, "sffs_init: bad vfs ops template");
204 return (error);
205 }
206
207 error = sffs_vnode_init();
208 if (error != 0) {
209 (void) vfs_freevfsops_by_type(fstype);
210 cmn_err(CE_WARN, "sffs_init: bad vnode ops template");
211 return (error);
212 }
213
214 if ((sffs_major = getudev()) == (major_t)-1) {
215 cmn_err(CE_WARN, "sffs_init: Can't get unique device number.");
216 sffs_major = 0;
217 }
218 mutex_init(&sffs_minor_lock, NULL, MUTEX_DEFAULT, NULL);
219 return (0);
220}
221
222/*
223 * wrapper for pn_get
224 */
225static int
226sf_pn_get(char *rawpath, struct mounta *uap, char **outpath)
227{
228 pathname_t path;
229 int error;
230
231 error = pn_get(rawpath, (uap->flags & MS_SYSSPACE) ? UIO_SYSSPACE :
232 UIO_USERSPACE, &path);
233 if (error) {
234 LogFlowFunc(("pn_get(%s) failed\n", rawpath));
235 return (error);
236 }
237 *outpath = kmem_alloc(path.pn_pathlen + 1, KM_SLEEP);
238 strcpy(*outpath, path.pn_path);
239 pn_free(&path);
240 return (0);
241}
242
243static void
244sffs_print(sffs_data_t *sffs)
245{
246 cmn_err(CE_NOTE, "sffs_data_t at 0x%p\n", sffs);
247 cmn_err(CE_NOTE, " vfs_t *sf_vfsp = 0x%p\n", sffs->sf_vfsp);
248 cmn_err(CE_NOTE, " vnode_t *sf_rootnode = 0x%p\n", sffs->sf_rootnode);
249 cmn_err(CE_NOTE, " uid_t sf_uid = 0x%lu\n", (ulong_t)sffs->sf_handle->sf_uid);
250 cmn_err(CE_NOTE, " gid_t sf_gid = 0x%lu\n", (ulong_t)sffs->sf_handle->sf_gid);
251 cmn_err(CE_NOTE, " mode_t sf_dmode = 0x%lu\n", (ulong_t)sffs->sf_handle->sf_dmode);
252 cmn_err(CE_NOTE, " mode_t sf_fmode = 0x%lu\n", (ulong_t)sffs->sf_handle->sf_fmode);
253 cmn_err(CE_NOTE, " mode_t sf_dmask = 0x%lu\n", (ulong_t)sffs->sf_handle->sf_dmask);
254 cmn_err(CE_NOTE, " mode_t sf_fmask = 0x%lu\n", (ulong_t)sffs->sf_handle->sf_fmask);
255 cmn_err(CE_NOTE, " char *sf_share_name = %s\n", sffs->sf_share_name);
256 cmn_err(CE_NOTE, " char *sf_mntpath = %s\n", sffs->sf_mntpath);
257 cmn_err(CE_NOTE, " sfp_mount_t *sf_handle = 0x%p\n", sffs->sf_handle);
258}
259
260static int
261sffs_mount(vfs_t *vfsp, vnode_t *mvp, struct mounta *uap, cred_t *cr)
262{
263 sffs_data_t *sffs;
264 char *mount_point = NULL;
265 char *share_name = NULL;
266 int error;
267 dev_t dev;
268 uid_t uid = 0;
269 gid_t gid = 0;
270 mode_t dmode = ~0U;
271 mode_t fmode = ~0U;
272 mode_t dmask = 0;
273 mode_t fmask = 0;
274 int stat_ttl = DEF_STAT_TTL_MS;
275 int fsync = 0;
276 char *optval;
277 long val;
278 char *path;
279 sfp_mount_t *handle;
280 sfnode_t *sfnode;
281
282 /*
283 * check we have permission to do the mount
284 */
285 LogFlowFunc(("sffs_mount() started\n"));
286 error = secpolicy_fs_mount(cr, mvp, vfsp);
287 if (error != 0)
288 return (error);
289
290 /*
291 * Mount point must be a directory
292 */
293 if (mvp->v_type != VDIR)
294 return (ENOTDIR);
295
296 /*
297 * no support for remount (what is it?)
298 */
299 if (uap->flags & MS_REMOUNT)
300 return (ENOTSUP);
301
302 /*
303 * Ensure that nothing else is actively in/under the mount point
304 */
305 mutex_enter(&mvp->v_lock);
306 if ((uap->flags & MS_OVERLAY) == 0 &&
307 (mvp->v_count != 1 || (mvp->v_flag & VROOT))) {
308 mutex_exit(&mvp->v_lock);
309 return (EBUSY);
310 }
311 mutex_exit(&mvp->v_lock);
312
313 /*
314 * check for read only has to be done early
315 */
316 if (uap->flags & MS_RDONLY) {
317 vfsp->vfs_flag |= VFS_RDONLY;
318 vfs_setmntopt(vfsp, MNTOPT_RO, NULL, 0);
319 }
320
321 /*
322 * UID to use for all files
323 */
324 if (vfs_optionisset(vfsp, "uid", &optval) &&
325 ddi_strtol(optval, NULL, 10, &val) == 0 &&
326 (uid_t)val == val)
327 uid = val;
328
329 /*
330 * GID to use for all files
331 */
332 if (vfs_optionisset(vfsp, "gid", &optval) &&
333 ddi_strtol(optval, NULL, 10, &val) == 0 &&
334 (gid_t)val == val)
335 gid = val;
336
337 /*
338 * dmode to use for all directories
339 */
340 if (vfs_optionisset(vfsp, "dmode", &optval) &&
341 ddi_strtol(optval, NULL, 8, &val) == 0 &&
342 (mode_t)val == val)
343 dmode = val;
344
345 /*
346 * fmode to use for all files
347 */
348 if (vfs_optionisset(vfsp, "fmode", &optval) &&
349 ddi_strtol(optval, NULL, 8, &val) == 0 &&
350 (mode_t)val == val)
351 fmode = val;
352
353 /*
354 * dmask to use for all directories
355 */
356 if (vfs_optionisset(vfsp, "dmask", &optval) &&
357 ddi_strtol(optval, NULL, 8, &val) == 0 &&
358 (mode_t)val == val)
359 dmask = val;
360
361 /*
362 * fmask to use for all files
363 */
364 if (vfs_optionisset(vfsp, "fmask", &optval) &&
365 ddi_strtol(optval, NULL, 8, &val) == 0 &&
366 (mode_t)val == val)
367 fmask = val;
368
369 /*
370 * umask to use for all directories & files
371 */
372 if (vfs_optionisset(vfsp, "umask", &optval) &&
373 ddi_strtol(optval, NULL, 8, &val) == 0 &&
374 (mode_t)val == val)
375 dmask = fmask = val;
376
377 /*
378 * ttl to use for stat caches
379 */
380 if (vfs_optionisset(vfsp, "stat_ttl", &optval) &&
381 ddi_strtol(optval, NULL, 10, &val) == 0 &&
382 (int)val == val)
383 {
384 stat_ttl = val;
385 }
386 else
387 vfs_setmntopt(vfsp, "stat_ttl", VBOXSOLQUOTE(DEF_STAT_TTL_MS), 0);
388
389 /*
390 * whether to honor fsync
391 */
392 if (vfs_optionisset(vfsp, "fsync", &optval))
393 fsync = 1;
394
395 /*
396 * Any unknown options are an error
397 */
398 if ((uap->flags & MS_DATA) && uap->datalen > 0) {
399 cmn_err(CE_WARN, "sffs: unknown mount options specified");
400 return (EINVAL);
401 }
402
403 /*
404 * get the mount point pathname
405 */
406 error = sf_pn_get(uap->dir, uap, &mount_point);
407 if (error)
408 return (error);
409
410 /*
411 * find what we are mounting
412 */
413 error = sf_pn_get(uap->spec, uap, &share_name);
414 if (error) {
415 kmem_free(mount_point, strlen(mount_point) + 1);
416 return (error);
417 }
418
419 /*
420 * Invoke Hypervisor mount interface before proceeding
421 */
422 error = sfprov_mount(sfprov, share_name, &handle);
423 if (error) {
424 kmem_free(share_name, strlen(share_name) + 1);
425 kmem_free(mount_point, strlen(mount_point) + 1);
426 return (error);
427 }
428
429 /*
430 * find an available minor device number for this mount
431 */
432 mutex_enter(&sffs_minor_lock);
433 do {
434 sffs_minor = (sffs_minor + 1) & L_MAXMIN32;
435 dev = makedevice(sffs_major, sffs_minor);
436 } while (vfs_devismounted(dev));
437 mutex_exit(&sffs_minor_lock);
438
439 /*
440 * allocate and fill in the sffs structure
441 */
442 sffs = kmem_alloc(sizeof (*sffs), KM_SLEEP);
443 sffs->sf_vfsp = vfsp;
444 sffs->sf_handle = handle;
445 sffs->sf_handle->sf_uid = uid;
446 sffs->sf_handle->sf_gid = gid;
447 sffs->sf_handle->sf_dmode = dmode;
448 sffs->sf_handle->sf_fmode = fmode;
449 sffs->sf_handle->sf_dmask = dmask;
450 sffs->sf_handle->sf_fmask = fmask;
451 sffs->sf_stat_ttl = stat_ttl;
452 sffs->sf_fsync = fsync;
453 sffs->sf_share_name = share_name;
454 sffs->sf_mntpath = mount_point;
455 sffs->sf_ino = 3; /* root mount point is always '3' */
456
457 /*
458 * fill in the vfs structure
459 */
460 vfsp->vfs_data = (caddr_t)sffs;
461 vfsp->vfs_fstype = sffs_fstype;
462 vfsp->vfs_dev = dev;
463 vfsp->vfs_bsize = PAGESIZE; /* HERE JOE ??? */
464 vfsp->vfs_flag |= VFS_NOTRUNC; /* HERE JOE ???? */
465 vfs_make_fsid(&vfsp->vfs_fsid, dev, sffs_fstype);
466
467 /*
468 * create the root vnode.
469 * XXX JOE What should the path be here? is "/" really right?
470 * other options?
471 */
472 path = kmem_alloc(2, KM_SLEEP);
473 strcpy(path, ".");
474 mutex_enter(&sffs_lock);
475 sfnode = sfnode_make(sffs, path, VDIR, NULL, NULL, NULL, 0);
476 sffs->sf_rootnode = sfnode_get_vnode(sfnode);
477 sffs->sf_rootnode->v_flag |= VROOT;
478 sffs->sf_rootnode->v_vfsp = vfsp;
479 mutex_exit(&sffs_lock);
480
481 LogFlowFunc(("sffs_mount() success sffs=0x%p\n", sffs));
482#ifdef DEBUG_ramshankar
483 sffs_print(sffs);
484#endif
485 return (error);
486}
487
488static int
489sffs_unmount(vfs_t *vfsp, int flag, cred_t *cr)
490{
491 sffs_data_t *sffs = (sffs_data_t *)vfsp->vfs_data;
492 int error;
493
494 /*
495 * generic security check
496 */
497 LogFlowFunc(("sffs_unmount() of sffs=0x%p\n", sffs));
498 if ((error = secpolicy_fs_unmount(cr, vfsp)) != 0)
499 return (error);
500
501 /*
502 * forced unmount is not supported by this file system
503 * and thus, ENOTSUP, is being returned.
504 */
505 if (flag & MS_FORCE) {
506 LogFlowFunc(("sffs_unmount(MS_FORCE) returns ENOSUP\n"));
507 return (ENOTSUP);
508 }
509
510 /*
511 * Mark the file system unmounted.
512 */
513 vfsp->vfs_flag |= VFS_UNMOUNTED;
514
515 /*
516 * Make sure nothing is still in use.
517 */
518 if (sffs_purge(sffs) != 0) {
519 vfsp->vfs_flag &= ~VFS_UNMOUNTED;
520 LogFlowFunc(("sffs_unmount() returns EBUSY\n"));
521 return (EBUSY);
522 }
523
524 /*
525 * Invoke Hypervisor unmount interface before proceeding
526 */
527 error = sfprov_unmount(sffs->sf_handle);
528 if (error != 0) {
529 /* TBD anything here? */
530 }
531
532 kmem_free(sffs->sf_share_name, strlen(sffs->sf_share_name) + 1);
533 kmem_free(sffs->sf_mntpath, strlen(sffs->sf_mntpath) + 1);
534 kmem_free(sffs, sizeof(*sffs));
535 LogFlowFunc(("sffs_unmount() done\n"));
536 return (0);
537}
538
539/*
540 * return the vnode for the root of the mounted file system
541 */
542static int
543sffs_root(vfs_t *vfsp, vnode_t **vpp)
544{
545 sffs_data_t *sffs = (sffs_data_t *)vfsp->vfs_data;
546 vnode_t *vp = sffs->sf_rootnode;
547
548 VN_HOLD(vp);
549 *vpp = vp;
550 return (0);
551}
552
553/*
554 * get some stats.. fake up the rest
555 */
556static int
557sffs_statvfs(vfs_t *vfsp, statvfs64_t *sbp)
558{
559 sffs_data_t *sffs = (sffs_data_t *)vfsp->vfs_data;
560 sffs_fsinfo_t fsinfo;
561 dev32_t d32;
562 int error;
563
564 bzero(sbp, sizeof(*sbp));
565 error = sfprov_get_fsinfo(sffs->sf_handle, &fsinfo);
566 if (error != 0)
567 return (error);
568
569 sbp->f_bsize = fsinfo.blksize;
570 sbp->f_frsize = fsinfo.blksize;
571
572 sbp->f_bfree = fsinfo.blksavail;
573 sbp->f_bavail = fsinfo.blksavail;
574 sbp->f_files = fsinfo.blksavail / 4; /* some kind of reasonable value */
575 sbp->f_ffree = fsinfo.blksavail / 4;
576 sbp->f_favail = fsinfo.blksavail / 4;
577
578 sbp->f_blocks = fsinfo.blksused + sbp->f_bavail;
579
580 (void) cmpldev(&d32, vfsp->vfs_dev);
581 sbp->f_fsid = d32;
582 strcpy(&sbp->f_basetype[0], "sffs");
583 sbp->f_flag |= ST_NOSUID;
584
585 if (fsinfo.readonly)
586 sbp->f_flag |= ST_RDONLY;
587
588 sbp->f_namemax = fsinfo.maxnamesize;
589 return (0);
590}
591
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