VirtualBox

source: vbox/trunk/src/VBox/Additions/solaris/SharedFolders/vboxfs_vnode.c@ 50040

Last change on this file since 50040 was 48943, checked in by vboxsync, 11 years ago

Additions/solaris: Whitespace and svn:keyword cleanups by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 55.0 KB
Line 
1/** @file
2 * VirtualBox File System for Solaris Guests, vnode implementation.
3 * Portions contributed by: Ronald.
4 */
5
6/*
7 * Copyright (C) 2009-2013 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/*
28 * Shared Folder File System is used from Solaris when run as a guest operating
29 * system on VirtualBox, though is meant to be usable with any hypervisor that
30 * can provide similar functionality. The sffs code handles all the Solaris
31 * specific semantics and relies on a provider module to actually access
32 * directories, files, etc. The provider interfaces are described in
33 * "vboxfs_prov.h" and the module implementing them is shipped as part of the
34 * VirtualBox Guest Additions for Solaris.
35 *
36 * The shared folder file system is similar to a networked file system,
37 * but with some caveats. The sffs code caches minimal information and proxies
38 * out to the provider whenever possible. Here are some things that are
39 * handled in this code and not by the proxy:
40 *
41 * - a way to open ".." from any already open directory
42 * - st_ino numbers
43 * - detecting directory changes that happened on the host.
44 *
45 * The implementation builds a cache of information for every file/directory
46 * ever accessed in all mounted sffs filesystems using sf_node structures.
47 *
48 * This information for both open or closed files can become invalid if
49 * asynchronous changes are made on the host. Solaris should not panic() in
50 * this event, but some file system operations may return unexpected errors.
51 * Information for such directories or files while they have active vnodes
52 * is removed from the regular cache and stored in a "stale" bucket until
53 * the vnode becomes completely inactive.
54 *
55 * We suppport only read-only mmap (VBOXVFS_WITH_MMAP) i.e. MAP_SHARED,
56 * MAP_PRIVATE in PROT_READ, this data caching would not be coherent with
57 * normal simultaneous read()/write() operations, nor will it be coherent
58 * with data access on the host. Writable mmap(MAP_SHARED) access is not
59 * implemented, as guaranteeing any kind of coherency with concurrent
60 * activity on the host would be near impossible with the existing
61 * interfaces.
62 *
63 * A note about locking. sffs is not a high performance file system.
64 * No fine grained locking is done. The one sffs_lock protects just about
65 * everything.
66 */
67
68#include <VBox/log.h>
69#include <iprt/asm.h>
70
71#include <unistd.h>
72#include <sys/types.h>
73#include <sys/stat.h>
74#include <sys/mntent.h>
75#include <sys/param.h>
76#include <sys/modctl.h>
77#include <sys/mount.h>
78#include <sys/policy.h>
79#include <sys/atomic.h>
80#include <sys/sysmacros.h>
81#include <sys/ddi.h>
82#include <sys/sunddi.h>
83#include <sys/vfs.h>
84#include <sys/vmsystm.h>
85#include <vm/seg_kpm.h>
86#include <vm/pvn.h>
87#if !defined(VBOX_VFS_SOLARIS_10U6)
88#include <sys/vfs_opreg.h>
89#endif
90#include <sys/pathname.h>
91#include <sys/dirent.h>
92#include <sys/fs_subr.h>
93#include <sys/time.h>
94#include "vboxfs_prov.h"
95#include "vboxfs_vnode.h"
96#include "vboxfs_vfs.h"
97
98/*
99 * Solaris 11u1b10 Extended Policy putback CR 7121445 removes secpolicy_vnode_access from sys/policy.h
100 */
101#ifdef VBOX_VFS_EXTENDED_POLICY
102int secpolicy_vnode_access(const cred_t *, vnode_t *, uid_t, mode_t);
103#endif
104
105#define VBOXVFS_WITH_MMAP
106
107static struct vnodeops *sffs_ops = NULL;
108
109kmutex_t sffs_lock;
110static avl_tree_t sfnodes;
111static avl_tree_t stale_sfnodes;
112
113/*
114 * For now we'll use an I/O buffer that doesn't page fault for VirtualBox
115 * to transfer data into.
116 */
117char *sffs_buffer;
118
119/*
120 * sfnode_compare() is needed for AVL tree functionality.
121 * The nodes are sorted by mounted filesystem, then path. If the
122 * nodes are stale, the node pointer itself is used to force uniqueness.
123 */
124static int
125sfnode_compare(const void *a, const void *b)
126{
127 sfnode_t *x = (sfnode_t *)a;
128 sfnode_t *y = (sfnode_t *)b;
129 int diff;
130
131 if (x->sf_is_stale) {
132 ASSERT(y->sf_is_stale);
133 diff = strcmp(x->sf_path, y->sf_path);
134 if (diff == 0)
135 diff = (uintptr_t)y - (uintptr_t)x;
136 } else {
137 ASSERT(!y->sf_is_stale);
138 diff = (uintptr_t)y->sf_sffs - (uintptr_t)x->sf_sffs;
139 if (diff == 0)
140 diff = strcmp(x->sf_path, y->sf_path);
141 }
142 if (diff < 0)
143 return (-1);
144 if (diff > 0)
145 return (1);
146 return (0);
147}
148
149/*
150 * Construct a new pathname given an sfnode plus an optional tail component.
151 * This handles ".." and "."
152 */
153static char *
154sfnode_construct_path(sfnode_t *node, char *tail)
155{
156 char *p;
157
158 if (strcmp(tail, ".") == 0 || strcmp(tail, "..") == 0)
159 panic("construct path for %s", tail);
160 p = kmem_alloc(strlen(node->sf_path) + 1 + strlen(tail) + 1, KM_SLEEP);
161 strcpy(p, node->sf_path);
162 strcat(p, "/");
163 strcat(p, tail);
164 return (p);
165}
166
167/*
168 * Clears the (cached) directory listing for the node.
169 */
170static void
171sfnode_clear_dir_list(sfnode_t *node)
172{
173 ASSERT(MUTEX_HELD(&sffs_lock));
174
175 while (node->sf_dir_list != NULL) {
176 sffs_dirents_t *next = node->sf_dir_list->sf_next;
177 kmem_free(node->sf_dir_list, SFFS_DIRENTS_SIZE);
178 node->sf_dir_list = next;
179 }
180}
181
182/*
183 * Open the provider file associated with a vnode. Holding the file open is
184 * the only way we have of trying to have a vnode continue to refer to the
185 * same host file in the host in light of the possibility of host side renames.
186 */
187static void
188sfnode_open(sfnode_t *node, int flag)
189{
190 int error;
191 sfp_file_t *fp;
192
193 if (node->sf_file != NULL)
194 return;
195 error = sfprov_open(node->sf_sffs->sf_handle, node->sf_path, &fp, flag);
196 if (error == 0)
197 {
198 node->sf_file = fp;
199 node->sf_flag = flag;
200 }
201 else
202 node->sf_flag = ~0;
203}
204
205/*
206 * get a new vnode reference for an sfnode
207 */
208vnode_t *
209sfnode_get_vnode(sfnode_t *node)
210{
211 vnode_t *vp;
212
213 if (node->sf_vnode != NULL) {
214 VN_HOLD(node->sf_vnode);
215 } else {
216 vp = vn_alloc(KM_SLEEP);
217 LogFlowFunc((" %s gets vnode 0x%p\n", node->sf_path, vp));
218 vp->v_type = node->sf_type;
219 vp->v_vfsp = node->sf_sffs->sf_vfsp;
220 vn_setops(vp, sffs_ops);
221 vp->v_flag = VNOSWAP;
222#ifndef VBOXVFS_WITH_MMAP
223 vp->v_flag |= VNOMAP;
224#endif
225 vn_exists(vp);
226 vp->v_data = node;
227 node->sf_vnode = vp;
228 }
229 return (node->sf_vnode);
230}
231
232/*
233 * Allocate and initialize a new sfnode and assign it a vnode
234 */
235sfnode_t *
236sfnode_make(
237 sffs_data_t *sffs,
238 char *path,
239 vtype_t type,
240 sfp_file_t *fp,
241 sfnode_t *parent, /* can be NULL for root */
242 sffs_stat_t *stat,
243 uint64_t stat_time)
244{
245 sfnode_t *node;
246 avl_index_t where;
247
248 ASSERT(MUTEX_HELD(&sffs_lock));
249 ASSERT(path != NULL);
250
251 /*
252 * build the sfnode
253 */
254 LogFlowFunc(("sffs_make(%s)\n", path));
255 node = kmem_alloc(sizeof (*node), KM_SLEEP);
256 node->sf_sffs = sffs;
257 VFS_HOLD(node->sf_sffs->sf_vfsp);
258 node->sf_path = path;
259 node->sf_ino = sffs->sf_ino++;
260 node->sf_type = type;
261 node->sf_is_stale = 0; /* never stale at creation */
262 node->sf_file = fp;
263 node->sf_flag = ~0;
264 node->sf_vnode = NULL; /* do this before any sfnode_get_vnode() */
265 node->sf_children = 0;
266 node->sf_parent = parent;
267 if (parent)
268 ++parent->sf_children;
269 node->sf_dir_list = NULL;
270 if (stat != NULL) {
271 node->sf_stat = *stat;
272 node->sf_stat_time = stat_time;
273 } else {
274 node->sf_stat_time = 0;
275 }
276
277 /*
278 * add the new node to our cache
279 */
280 if (avl_find(&sfnodes, node, &where) != NULL)
281 panic("sffs_create_sfnode(%s): duplicate sfnode_t", path);
282 avl_insert(&sfnodes, node, where);
283 return (node);
284}
285
286/*
287 * destroy an sfnode
288 */
289static void
290sfnode_destroy(sfnode_t *node)
291{
292 avl_index_t where;
293 avl_tree_t *tree;
294 sfnode_t *parent;
295top:
296 parent = node->sf_parent;
297 ASSERT(MUTEX_HELD(&sffs_lock));
298 ASSERT(node->sf_path != NULL);
299 LogFlowFunc(("sffs_destroy(%s)%s\n", node->sf_path, node->sf_is_stale ? " stale": ""));
300 if (node->sf_children != 0)
301 panic("sfnode_destroy(%s) has %d children", node->sf_path, node->sf_children);
302 if (node->sf_vnode != NULL)
303 panic("sfnode_destroy(%s) has active vnode", node->sf_path);
304
305 if (node->sf_is_stale)
306 tree = &stale_sfnodes;
307 else
308 tree = &sfnodes;
309 if (avl_find(tree, node, &where) == NULL)
310 panic("sfnode_destroy(%s) not found", node->sf_path);
311 avl_remove(tree, node);
312
313 VFS_RELE(node->sf_sffs->sf_vfsp);
314 sfnode_clear_dir_list(node);
315 kmem_free(node->sf_path, strlen(node->sf_path) + 1);
316 kmem_free(node, sizeof (*node));
317 if (parent != NULL) {
318 sfnode_clear_dir_list(parent);
319 if (parent->sf_children == 0)
320 panic("sfnode_destroy parent (%s) has no child", parent->sf_path);
321 --parent->sf_children;
322 if (parent->sf_children == 0 &&
323 parent->sf_is_stale &&
324 parent->sf_vnode == NULL) {
325 node = parent;
326 goto top;
327 }
328 }
329}
330
331/*
332 * Some sort of host operation on an sfnode has failed or it has been
333 * deleted. Mark this node and any children as stale, deleting knowledge
334 * about any which do not have active vnodes or children
335 * This also handle deleting an inactive node that was already stale.
336 */
337static void
338sfnode_make_stale(sfnode_t *node)
339{
340 sfnode_t *n;
341 int len;
342 ASSERT(MUTEX_HELD(&sffs_lock));
343 avl_index_t where;
344
345 /*
346 * First deal with any children of a directory node.
347 * If a directory becomes stale, anything below it becomes stale too.
348 */
349 if (!node->sf_is_stale && node->sf_type == VDIR) {
350 len = strlen(node->sf_path);
351
352 n = node;
353 while ((n = AVL_NEXT(&sfnodes, node)) != NULL) {
354 ASSERT(!n->sf_is_stale);
355
356 /*
357 * quit when no longer seeing children of node
358 */
359 if (n->sf_sffs != node->sf_sffs ||
360 strncmp(node->sf_path, n->sf_path, len) != 0 ||
361 n->sf_path[len] != '/')
362 break;
363
364 /*
365 * Either mark the child as stale or destroy it
366 */
367 if (n->sf_vnode == NULL && n->sf_children == 0) {
368 sfnode_destroy(n);
369 } else {
370 LogFlowFunc(("sffs_make_stale(%s) sub\n", n->sf_path));
371 sfnode_clear_dir_list(n);
372 if (avl_find(&sfnodes, n, &where) == NULL)
373 panic("sfnode_make_stale(%s)"
374 " not in sfnodes", n->sf_path);
375 avl_remove(&sfnodes, n);
376 n->sf_is_stale = 1;
377 if (avl_find(&stale_sfnodes, n, &where) != NULL)
378 panic("sffs_make_stale(%s) duplicates",
379 n->sf_path);
380 avl_insert(&stale_sfnodes, n, where);
381 }
382 }
383 }
384
385 /*
386 * Now deal with the given node.
387 */
388 if (node->sf_vnode == NULL && node->sf_children == 0) {
389 sfnode_destroy(node);
390 } else if (!node->sf_is_stale) {
391 LogFlowFunc(("sffs_make_stale(%s)\n", node->sf_path));
392 sfnode_clear_dir_list(node);
393 if (node->sf_parent)
394 sfnode_clear_dir_list(node->sf_parent);
395 if (avl_find(&sfnodes, node, &where) == NULL)
396 panic("sfnode_make_stale(%s) not in sfnodes",
397 node->sf_path);
398 avl_remove(&sfnodes, node);
399 node->sf_is_stale = 1;
400 if (avl_find(&stale_sfnodes, node, &where) != NULL)
401 panic("sffs_make_stale(%s) duplicates", node->sf_path);
402 avl_insert(&stale_sfnodes, node, where);
403 }
404}
405
406static uint64_t
407sfnode_cur_time_usec(void)
408{
409 clock_t now = drv_hztousec(ddi_get_lbolt());
410 return now;
411}
412
413static int
414sfnode_stat_cached(sfnode_t *node)
415{
416 return (sfnode_cur_time_usec() - node->sf_stat_time) <
417 node->sf_sffs->sf_stat_ttl * 1000L;
418}
419
420static void
421sfnode_invalidate_stat_cache(sfnode_t *node)
422{
423 node->sf_stat_time = 0;
424}
425
426static int
427sfnode_update_stat_cache(sfnode_t *node)
428{
429 int error;
430
431 error = sfprov_get_attr(node->sf_sffs->sf_handle, node->sf_path,
432 &node->sf_stat);
433 if (error == ENOENT)
434 sfnode_make_stale(node);
435 if (error == 0)
436 node->sf_stat_time = sfnode_cur_time_usec();
437
438 return (error);
439}
440
441/*
442 * Rename a file or a directory
443 */
444static void
445sfnode_rename(sfnode_t *node, sfnode_t *newparent, char *path)
446{
447 sfnode_t *n;
448 sfnode_t template;
449 avl_index_t where;
450 int len = strlen(path);
451 int old_len;
452 char *new_path;
453 char *tail;
454 ASSERT(MUTEX_HELD(&sffs_lock));
455
456 ASSERT(!node->sf_is_stale);
457
458 /*
459 * Have to remove anything existing that had the new name.
460 */
461 template.sf_sffs = node->sf_sffs;
462 template.sf_path = path;
463 template.sf_is_stale = 0;
464 n = avl_find(&sfnodes, &template, &where);
465 if (n != NULL)
466 sfnode_make_stale(n);
467
468 /*
469 * Do the renaming, deal with any children of this node first.
470 */
471 if (node->sf_type == VDIR) {
472 old_len = strlen(node->sf_path);
473 while ((n = AVL_NEXT(&sfnodes, node)) != NULL) {
474
475 /*
476 * quit when no longer seeing children of node
477 */
478 if (n->sf_sffs != node->sf_sffs ||
479 strncmp(node->sf_path, n->sf_path, old_len) != 0 ||
480 n->sf_path[old_len] != '/')
481 break;
482
483 /*
484 * Rename the child:
485 * - build the new path name
486 * - unlink the AVL node
487 * - assign the new name
488 * - re-insert the AVL name
489 */
490 ASSERT(strlen(n->sf_path) > old_len);
491 tail = n->sf_path + old_len; /* includes initial "/" */
492 new_path = kmem_alloc(len + strlen(tail) + 1,
493 KM_SLEEP);
494 strcpy(new_path, path);
495 strcat(new_path, tail);
496 if (avl_find(&sfnodes, n, &where) == NULL)
497 panic("sfnode_rename(%s) not in sfnodes",
498 n->sf_path);
499 avl_remove(&sfnodes, n);
500 LogFlowFunc(("sfnode_rname(%s to %s) sub\n", n->sf_path, new_path));
501 kmem_free(n->sf_path, strlen(n->sf_path) + 1);
502 n->sf_path = new_path;
503 if (avl_find(&sfnodes, n, &where) != NULL)
504 panic("sfnode_rename(%s) duplicates",
505 n->sf_path);
506 avl_insert(&sfnodes, n, where);
507 }
508 }
509
510 /*
511 * Deal with the given node.
512 */
513 if (avl_find(&sfnodes, node, &where) == NULL)
514 panic("sfnode_rename(%s) not in sfnodes", node->sf_path);
515 avl_remove(&sfnodes, node);
516 LogFlowFunc(("sfnode_rname(%s to %s)\n", node->sf_path, path));
517 kmem_free(node->sf_path, strlen(node->sf_path) + 1);
518 node->sf_path = path;
519 if (avl_find(&sfnodes, node, &where) != NULL)
520 panic("sfnode_rename(%s) duplicates", node->sf_path);
521 avl_insert(&sfnodes, node, where);
522
523 /*
524 * change the parent
525 */
526 if (node->sf_parent == NULL)
527 panic("sfnode_rename(%s) no parent", node->sf_path);
528 if (node->sf_parent->sf_children == 0)
529 panic("sfnode_rename(%s) parent has no child", node->sf_path);
530 sfnode_clear_dir_list(node->sf_parent);
531 sfnode_clear_dir_list(newparent);
532 --node->sf_parent->sf_children;
533 node->sf_parent = newparent;
534 ++newparent->sf_children;
535}
536
537/*
538 * Look for a cached node, if not found either handle ".." or try looking
539 * via the provider. Create an entry in sfnodes if found but not cached yet.
540 * If the create flag is set, a file or directory is created. If the file
541 * already existed, an error is returned.
542 * Nodes returned from this routine always have a vnode with its ref count
543 * bumped by 1.
544 */
545static sfnode_t *
546sfnode_lookup(
547 sfnode_t *dir,
548 char *name,
549 vtype_t create,
550 mode_t c_mode,
551 sffs_stat_t *stat,
552 uint64_t stat_time,
553 int *err)
554{
555 avl_index_t where;
556 sfnode_t template;
557 sfnode_t *node;
558 int error = 0;
559 int type;
560 char *fullpath;
561 sfp_file_t *fp;
562 sffs_stat_t tmp_stat;
563
564 ASSERT(MUTEX_HELD(&sffs_lock));
565
566 if (err)
567 *err = error;
568
569 /*
570 * handle referencing myself
571 */
572 if (strcmp(name, "") == 0 || strcmp(name, ".") == 0)
573 return (dir);
574
575 /*
576 * deal with parent
577 */
578 if (strcmp(name, "..") == 0)
579 return (dir->sf_parent);
580
581 /*
582 * Look for an existing node.
583 */
584 fullpath = sfnode_construct_path(dir, name);
585 template.sf_sffs = dir->sf_sffs;
586 template.sf_path = fullpath;
587 template.sf_is_stale = 0;
588 node = avl_find(&sfnodes, &template, &where);
589 if (node != NULL) {
590 kmem_free(fullpath, strlen(fullpath) + 1);
591 if (create != VNON)
592 return (NULL);
593 return (node);
594 }
595
596 /*
597 * No entry for this path currently.
598 * Check if the file exists with the provider and get the type from
599 * there.
600 */
601 if (create == VREG) {
602 type = VREG;
603 stat = &tmp_stat;
604 error = sfprov_create(dir->sf_sffs->sf_handle, fullpath, c_mode,
605 &fp, stat);
606 stat_time = sfnode_cur_time_usec();
607 } else if (create == VDIR) {
608 type = VDIR;
609 stat = &tmp_stat;
610 error = sfprov_mkdir(dir->sf_sffs->sf_handle, fullpath, c_mode,
611 &fp, stat);
612 stat_time = sfnode_cur_time_usec();
613 } else {
614 mode_t m;
615 fp = NULL;
616 type = VNON;
617 if (stat == NULL) {
618 stat = &tmp_stat;
619 error = sfprov_get_attr(dir->sf_sffs->sf_handle,
620 fullpath, stat);
621 stat_time = sfnode_cur_time_usec();
622 } else {
623 error = 0;
624 }
625 m = stat->sf_mode;
626 if (error != 0)
627 error = ENOENT;
628 else if (S_ISDIR(m))
629 type = VDIR;
630 else if (S_ISREG(m))
631 type = VREG;
632 else if (S_ISLNK(m))
633 type = VLNK;
634 }
635
636 if (err)
637 *err = error;
638
639 /*
640 * If no errors, make a new node and return it.
641 */
642 if (error) {
643 kmem_free(fullpath, strlen(fullpath) + 1);
644 return (NULL);
645 }
646 node = sfnode_make(dir->sf_sffs, fullpath, type, fp, dir, stat,
647 stat_time);
648 return (node);
649}
650
651
652/*
653 * uid and gid in sffs determine owner and group for all files.
654 */
655static int
656sfnode_access(sfnode_t *node, mode_t mode, cred_t *cr)
657{
658 sffs_data_t *sffs = node->sf_sffs;
659 mode_t m;
660 int shift = 0;
661 int error;
662 vnode_t *vp;
663
664 ASSERT(MUTEX_HELD(&sffs_lock));
665
666 /*
667 * get the mode from the cache or provider
668 */
669 if (sfnode_stat_cached(node))
670 error = 0;
671 else
672 error = sfnode_update_stat_cache(node);
673 m = (error == 0) ? node->sf_stat.sf_mode : 0;
674
675 /*
676 * mask off the permissions based on uid/gid
677 */
678 if (crgetuid(cr) != sffs->sf_uid) {
679 shift += 3;
680 if (groupmember(sffs->sf_gid, cr) == 0)
681 shift += 3;
682 }
683 mode &= ~(m << shift);
684
685 if (mode == 0) {
686 error = 0;
687 } else {
688 vp = sfnode_get_vnode(node);
689 error = secpolicy_vnode_access(cr, vp, sffs->sf_uid, mode);
690 VN_RELE(vp);
691 }
692 return (error);
693}
694
695
696/*
697 *
698 * Everything below this point are the vnode operations used by Solaris VFS
699 */
700static int
701sffs_readdir(
702 vnode_t *vp,
703 uio_t *uiop,
704 cred_t *cred,
705 int *eofp,
706 caller_context_t *ct,
707 int flag)
708{
709 sfnode_t *dir = VN2SFN(vp);
710 sfnode_t *node;
711 struct sffs_dirent *dirent = NULL;
712 sffs_dirents_t *cur_buf;
713 offset_t offset = 0;
714 offset_t orig_off = uiop->uio_loffset;
715 int dummy_eof;
716 int error = 0;
717
718 if (uiop->uio_iovcnt != 1)
719 return (EINVAL);
720
721 if (vp->v_type != VDIR)
722 return (ENOTDIR);
723
724 if (eofp == NULL)
725 eofp = &dummy_eof;
726 *eofp = 0;
727
728 if (uiop->uio_loffset >= MAXOFFSET_T) {
729 *eofp = 1;
730 return (0);
731 }
732
733 /*
734 * Get the directory entry names from the host. This gets all
735 * entries. These are stored in a linked list of sffs_dirents_t
736 * buffers, each of which contains a list of dirent64_t's.
737 */
738 mutex_enter(&sffs_lock);
739
740 if (dir->sf_dir_list == NULL) {
741 error = sfprov_readdir(dir->sf_sffs->sf_handle, dir->sf_path,
742 &dir->sf_dir_list, flag);
743 if (error != 0)
744 goto done;
745 }
746
747 /*
748 * Validate and skip to the desired offset.
749 */
750 cur_buf = dir->sf_dir_list;
751 offset = 0;
752
753 while (cur_buf != NULL &&
754 offset + cur_buf->sf_len <= uiop->uio_loffset) {
755 offset += cur_buf->sf_len;
756 cur_buf = cur_buf->sf_next;
757 }
758
759 if (cur_buf == NULL && offset != uiop->uio_loffset) {
760 error = EINVAL;
761 goto done;
762 }
763 if (cur_buf != NULL && offset != uiop->uio_loffset) {
764 offset_t off = offset;
765 int step;
766 dirent = &cur_buf->sf_entries[0];
767
768 while (off < uiop->uio_loffset) {
769 if (dirent->sf_entry.d_off == uiop->uio_loffset)
770 break;
771 step = sizeof(sffs_stat_t) + dirent->sf_entry.d_reclen;
772 dirent = (struct sffs_dirent *) (((char *) dirent) + step);
773 off += step;
774 }
775
776 if (off >= uiop->uio_loffset) {
777 error = EINVAL;
778 goto done;
779 }
780 }
781
782 offset = uiop->uio_loffset - offset;
783
784 /*
785 * Lookup each of the names, so that we have ino's, and copy to
786 * result buffer.
787 */
788 while (cur_buf != NULL) {
789 if (offset >= cur_buf->sf_len) {
790 cur_buf = cur_buf->sf_next;
791 offset = 0;
792 continue;
793 }
794
795 dirent = (struct sffs_dirent *)
796 (((char *) &cur_buf->sf_entries[0]) + offset);
797 if (dirent->sf_entry.d_reclen > uiop->uio_resid)
798 break;
799
800 if (strcmp(dirent->sf_entry.d_name, ".") == 0) {
801 node = dir;
802 } else if (strcmp(dirent->sf_entry.d_name, "..") == 0) {
803 node = dir->sf_parent;
804 if (node == NULL)
805 node = dir;
806 } else {
807 node = sfnode_lookup(dir, dirent->sf_entry.d_name, VNON,
808 0, &dirent->sf_stat, sfnode_cur_time_usec(), NULL);
809 if (node == NULL)
810 panic("sffs_readdir() lookup failed");
811 }
812 dirent->sf_entry.d_ino = node->sf_ino;
813
814 error = uiomove(&dirent->sf_entry, dirent->sf_entry.d_reclen, UIO_READ, uiop);
815 if (error != 0)
816 break;
817
818 uiop->uio_loffset= dirent->sf_entry.d_off;
819 offset += sizeof(sffs_stat_t) + dirent->sf_entry.d_reclen;
820 }
821 if (error == 0 && cur_buf == NULL)
822 *eofp = 1;
823done:
824 mutex_exit(&sffs_lock);
825 if (error != 0)
826 uiop->uio_loffset = orig_off;
827 return (error);
828}
829
830
831#if defined(VBOX_VFS_SOLARIS_10U6)
832/*
833 * HERE JOE.. this may need more logic, need to look at other file systems
834 */
835static int
836sffs_pathconf(
837 vnode_t *vp,
838 int cmd,
839 ulong_t *valp,
840 cred_t *cr)
841{
842 return (fs_pathconf(vp, cmd, valp, cr));
843}
844#else
845/*
846 * HERE JOE.. this may need more logic, need to look at other file systems
847 */
848static int
849sffs_pathconf(
850 vnode_t *vp,
851 int cmd,
852 ulong_t *valp,
853 cred_t *cr,
854 caller_context_t *ct)
855{
856 return (fs_pathconf(vp, cmd, valp, cr, ct));
857}
858#endif
859
860static int
861sffs_getattr(
862 vnode_t *vp,
863 vattr_t *vap,
864 int flags,
865 cred_t *cred,
866 caller_context_t *ct)
867{
868 sfnode_t *node = VN2SFN(vp);
869 sffs_data_t *sffs = node->sf_sffs;
870 mode_t mode;
871 int error = 0;
872
873 mutex_enter(&sffs_lock);
874 vap->va_type = vp->v_type;
875 vap->va_uid = sffs->sf_uid;
876 vap->va_gid = sffs->sf_gid;
877 vap->va_fsid = sffs->sf_vfsp->vfs_dev;
878 vap->va_nodeid = node->sf_ino;
879 vap->va_nlink = 1;
880 vap->va_rdev = sffs->sf_vfsp->vfs_dev;
881 vap->va_seq = 0;
882
883 if (!sfnode_stat_cached(node)) {
884 error = sfnode_update_stat_cache(node);
885 if (error != 0)
886 goto done;
887 }
888
889 vap->va_atime = node->sf_stat.sf_atime;
890 vap->va_mtime = node->sf_stat.sf_mtime;
891 vap->va_ctime = node->sf_stat.sf_ctime;
892
893 mode = node->sf_stat.sf_mode;
894 vap->va_mode = mode & MODEMASK;
895 if (S_ISDIR(mode))
896 {
897 vap->va_type = VDIR;
898 vap->va_mode = sffs->sf_dmode != ~0 ? (sffs->sf_dmode & 0777) : vap->va_mode;
899 vap->va_mode &= ~sffs->sf_dmask;
900 vap->va_mode |= S_IFDIR;
901 }
902 else if (S_ISREG(mode))
903 {
904 vap->va_type = VREG;
905 vap->va_mode = sffs->sf_fmode != ~0 ? (sffs->sf_fmode & 0777) : vap->va_mode;
906 vap->va_mode &= ~sffs->sf_fmask;
907 vap->va_mode |= S_IFREG;
908 }
909 else if (S_ISFIFO(mode))
910 vap->va_type = VFIFO;
911 else if (S_ISCHR(mode))
912 vap->va_type = VCHR;
913 else if (S_ISBLK(mode))
914 vap->va_type = VBLK;
915 else if (S_ISLNK(mode))
916 {
917 vap->va_type = VLNK;
918 vap->va_mode = sffs->sf_fmode != ~0 ? (sffs->sf_fmode & 0777) : vap->va_mode;
919 vap->va_mode &= ~sffs->sf_fmask;
920 vap->va_mode |= S_IFLNK;
921 }
922 else if (S_ISSOCK(mode))
923 vap->va_type = VSOCK;
924
925 vap->va_size = node->sf_stat.sf_size;
926 vap->va_blksize = 512;
927 vap->va_nblocks = (node->sf_stat.sf_alloc + 511) / 512;
928
929done:
930 mutex_exit(&sffs_lock);
931 return (error);
932}
933
934static int
935sffs_setattr(
936 vnode_t *vp,
937 vattr_t *vap,
938 int flags,
939 cred_t *cred,
940 caller_context_t *ct)
941{
942 sfnode_t *node = VN2SFN(vp);
943 int error;
944 mode_t mode;
945
946 mode = vap->va_mode;
947 if (vp->v_type == VREG)
948 mode |= S_IFREG;
949 else if (vp->v_type == VDIR)
950 mode |= S_IFDIR;
951 else if (vp->v_type == VBLK)
952 mode |= S_IFBLK;
953 else if (vp->v_type == VCHR)
954 mode |= S_IFCHR;
955 else if (vp->v_type == VLNK)
956 mode |= S_IFLNK;
957 else if (vp->v_type == VFIFO)
958 mode |= S_IFIFO;
959 else if (vp->v_type == VSOCK)
960 mode |= S_IFSOCK;
961
962 mutex_enter(&sffs_lock);
963
964 sfnode_invalidate_stat_cache(node);
965 error = sfprov_set_attr(node->sf_sffs->sf_handle, node->sf_path,
966 vap->va_mask, mode, vap->va_atime, vap->va_mtime, vap->va_ctime);
967 if (error == ENOENT)
968 sfnode_make_stale(node);
969
970 mutex_exit(&sffs_lock);
971 return (error);
972}
973
974static int
975sffs_space(
976 vnode_t *vp,
977 int cmd,
978 struct flock64 *bfp,
979 int flags,
980 offset_t off,
981 cred_t *cred,
982 caller_context_t *ct)
983{
984 sfnode_t *node = VN2SFN(vp);
985 int error;
986
987 /* we only support changing the length of the file */
988 if (bfp->l_whence != SEEK_SET || bfp->l_len != 0)
989 return ENOSYS;
990
991 mutex_enter(&sffs_lock);
992
993 sfnode_invalidate_stat_cache(node);
994
995 error = sfprov_set_size(node->sf_sffs->sf_handle, node->sf_path,
996 bfp->l_start);
997 if (error == ENOENT)
998 sfnode_make_stale(node);
999
1000 mutex_exit(&sffs_lock);
1001 return (error);
1002}
1003
1004/*ARGSUSED*/
1005static int
1006sffs_read(
1007 vnode_t *vp,
1008 struct uio *uio,
1009 int ioflag,
1010 cred_t *cred,
1011 caller_context_t *ct)
1012{
1013 sfnode_t *node = VN2SFN(vp);
1014 int error = 0;
1015 uint32_t bytes;
1016 uint32_t done;
1017 ulong_t offset;
1018 ssize_t total;
1019
1020 if (vp->v_type == VDIR)
1021 return (EISDIR);
1022 if (vp->v_type != VREG)
1023 return (EINVAL);
1024 if (uio->uio_loffset >= MAXOFFSET_T)
1025 {
1026 proc_t *p = ttoproc(curthread);
1027 mutex_enter(&p->p_lock);
1028 (void) rctl_action(rctlproc_legacy[RLIMIT_FSIZE], p->p_rctls,
1029 p, RCA_UNSAFE_SIGINFO);
1030 mutex_exit(&p->p_lock);
1031 return (EFBIG);
1032 }
1033 if (uio->uio_loffset < 0)
1034 return (EINVAL);
1035 total = uio->uio_resid;
1036 if (total == 0)
1037 return (0);
1038
1039 mutex_enter(&sffs_lock);
1040 if (node->sf_file == NULL) {
1041 ASSERT(node->sf_flag != ~0);
1042 sfnode_open(node, node->sf_flag);
1043 if (node->sf_file == NULL)
1044 return (EBADF);
1045 }
1046
1047 do {
1048 offset = uio->uio_offset;
1049 done = bytes = MIN(PAGESIZE, uio->uio_resid);
1050 error = sfprov_read(node->sf_file, sffs_buffer, offset, &done);
1051 if (error == 0 && done > 0)
1052 error = uiomove(sffs_buffer, done, UIO_READ, uio);
1053 } while (error == 0 && uio->uio_resid > 0 && done > 0);
1054
1055 mutex_exit(&sffs_lock);
1056
1057 /*
1058 * a partial read is never an error
1059 */
1060 if (total != uio->uio_resid)
1061 error = 0;
1062 return (error);
1063}
1064
1065/*ARGSUSED*/
1066static int
1067sffs_write(
1068 vnode_t *vp,
1069 struct uio *uiop,
1070 int ioflag,
1071 cred_t *cred,
1072 caller_context_t *ct)
1073{
1074 sfnode_t *node = VN2SFN(vp);
1075 int error = 0;
1076 uint32_t bytes;
1077 uint32_t done;
1078 ulong_t offset;
1079 ssize_t total;
1080 rlim64_t limit = uiop->uio_llimit;
1081
1082 if (vp->v_type == VDIR)
1083 return (EISDIR);
1084 if (vp->v_type != VREG)
1085 return (EINVAL);
1086
1087 /*
1088 * We have to hold this lock for a long time to keep
1089 * multiple FAPPEND writes from intermixing
1090 */
1091 mutex_enter(&sffs_lock);
1092 if (node->sf_file == NULL) {
1093 ASSERT(node->sf_flag != ~0);
1094 sfnode_open(node, node->sf_flag);
1095 if (node->sf_file == NULL)
1096 return (EBADF);
1097 }
1098
1099 sfnode_invalidate_stat_cache(node);
1100
1101 if (ioflag & FAPPEND) {
1102 uint64_t endoffile;
1103
1104 error = sfprov_get_size(node->sf_sffs->sf_handle,
1105 node->sf_path, &endoffile);
1106 if (error == ENOENT)
1107 sfnode_make_stale(node);
1108 if (error != 0) {
1109 mutex_exit(&sffs_lock);
1110 return (error);
1111 }
1112 uiop->uio_loffset = endoffile;
1113 }
1114
1115 if (vp->v_type != VREG || uiop->uio_loffset < 0) {
1116 mutex_exit(&sffs_lock);
1117 return (EINVAL);
1118 }
1119 if (limit == RLIM64_INFINITY || limit > MAXOFFSET_T)
1120 limit = MAXOFFSET_T;
1121
1122 if (uiop->uio_loffset >= limit) {
1123 proc_t *p = ttoproc(curthread);
1124 mutex_enter(&p->p_lock);
1125 (void) rctl_action(rctlproc_legacy[RLIMIT_FSIZE], p->p_rctls,
1126 p, RCA_UNSAFE_SIGINFO);
1127 mutex_exit(&p->p_lock);
1128 mutex_exit(&sffs_lock);
1129 return (EFBIG);
1130 }
1131
1132 if (uiop->uio_loffset >= MAXOFFSET_T) {
1133 mutex_exit(&sffs_lock);
1134 return (EFBIG);
1135 }
1136
1137 total = uiop->uio_resid;
1138 if (total == 0) {
1139 mutex_exit(&sffs_lock);
1140 return (0);
1141 }
1142
1143 do {
1144 offset = uiop->uio_offset;
1145 bytes = MIN(PAGESIZE, uiop->uio_resid);
1146 if (offset + bytes >= limit) {
1147 if (offset >= limit) {
1148 error = EFBIG;
1149 break;
1150 }
1151 bytes = limit - offset;
1152 }
1153 error = uiomove(sffs_buffer, bytes, UIO_WRITE, uiop);
1154 if (error != 0)
1155 break;
1156 done = bytes;
1157 if (error == 0)
1158 error = sfprov_write(node->sf_file, sffs_buffer,
1159 offset, &done);
1160 total -= done;
1161 if (done != bytes) {
1162 uiop->uio_resid += bytes - done;
1163 break;
1164 }
1165 } while (error == 0 && uiop->uio_resid > 0 && done > 0);
1166
1167 mutex_exit(&sffs_lock);
1168
1169 /*
1170 * A short write is never really an error.
1171 */
1172 if (total != uiop->uio_resid)
1173 error = 0;
1174 return (error);
1175}
1176
1177/*ARGSUSED*/
1178static int
1179sffs_access(vnode_t *vp, int mode, int flags, cred_t *cr, caller_context_t *ct)
1180{
1181 sfnode_t *node = VN2SFN(vp);
1182 int error;
1183
1184 mutex_enter(&sffs_lock);
1185 error = sfnode_access(node, mode, cr);
1186 mutex_exit(&sffs_lock);
1187 return (error);
1188}
1189
1190/*
1191 * Lookup an entry in a directory and create a new vnode if found.
1192 */
1193/* ARGSUSED3 */
1194static int
1195sffs_lookup(
1196 vnode_t *dvp, /* the directory vnode */
1197 char *name, /* the name of the file or directory */
1198 vnode_t **vpp, /* the vnode we found or NULL */
1199 struct pathname *pnp,
1200 int flags,
1201 vnode_t *rdir,
1202 cred_t *cred,
1203 caller_context_t *ct,
1204 int *direntflags,
1205 struct pathname *realpnp)
1206{
1207 int error;
1208 sfnode_t *node;
1209
1210 /*
1211 * dvp must be a directory
1212 */
1213 if (dvp->v_type != VDIR)
1214 return (ENOTDIR);
1215
1216 /*
1217 * An empty component name or just "." means the directory itself.
1218 * Don't do any further lookup or checking.
1219 */
1220 if (strcmp(name, "") == 0 || strcmp(name, ".") == 0) {
1221 VN_HOLD(dvp);
1222 *vpp = dvp;
1223 return (0);
1224 }
1225
1226 /*
1227 * Check permission to look at this directory. We always allow "..".
1228 */
1229 mutex_enter(&sffs_lock);
1230 if (strcmp(name, "..") != 0) {
1231 error = sfnode_access(VN2SFN(dvp), VEXEC, cred);
1232 if (error) {
1233 mutex_exit(&sffs_lock);
1234 return (error);
1235 }
1236 }
1237
1238 /*
1239 * Lookup the node.
1240 */
1241 node = sfnode_lookup(VN2SFN(dvp), name, VNON, 0, NULL, 0, NULL);
1242 if (node != NULL)
1243 *vpp = sfnode_get_vnode(node);
1244 mutex_exit(&sffs_lock);
1245 return ((node == NULL) ? ENOENT : 0);
1246}
1247
1248/*ARGSUSED*/
1249static int
1250sffs_create(
1251 vnode_t *dvp,
1252 char *name,
1253 struct vattr *vap,
1254 vcexcl_t exclusive,
1255 int mode,
1256 vnode_t **vpp,
1257 cred_t *cr,
1258 int flag,
1259 caller_context_t *ct,
1260 vsecattr_t *vsecp)
1261{
1262 vnode_t *vp;
1263 sfnode_t *node;
1264 int error;
1265
1266 ASSERT(name != NULL);
1267
1268 /*
1269 * this is used for regular files, not mkdir
1270 */
1271 if (vap->va_type == VDIR)
1272 return (EISDIR);
1273 if (vap->va_type != VREG)
1274 return (EINVAL);
1275
1276 /*
1277 * is this a pre-existing file?
1278 */
1279 error = sffs_lookup(dvp, name, &vp,
1280 NULL, 0, NULL, cr, ct, NULL, NULL);
1281 if (error == ENOENT)
1282 vp = NULL;
1283 else if (error != 0)
1284 return (error);
1285
1286 /*
1287 * Operation on a pre-existing file.
1288 */
1289 if (vp != NULL) {
1290 if (exclusive == EXCL) {
1291 VN_RELE(vp);
1292 return (EEXIST);
1293 }
1294 if (vp->v_type == VDIR && (mode & VWRITE) == VWRITE) {
1295 VN_RELE(vp);
1296 return (EISDIR);
1297 }
1298
1299 mutex_enter(&sffs_lock);
1300 node = VN2SFN(vp);
1301 error = sfnode_access(node, mode, cr);
1302 if (error != 0) {
1303 mutex_exit(&sffs_lock);
1304 VN_RELE(vp);
1305 return (error);
1306 }
1307
1308 sfnode_invalidate_stat_cache(VN2SFN(dvp));
1309
1310 /*
1311 * handle truncating an existing file
1312 */
1313 if (vp->v_type == VREG && (vap->va_mask & AT_SIZE) &&
1314 vap->va_size == 0) {
1315 sfnode_open(node, flag | FTRUNC);
1316 if (node->sf_path == NULL) {
1317 mutex_exit(&sffs_lock);
1318 VN_RELE(vp);
1319 return (ENOENT);
1320 }
1321 }
1322 mutex_exit(&sffs_lock);
1323 *vpp = vp;
1324 return (0);
1325 }
1326
1327 /*
1328 * Create a new node. First check for a race creating it.
1329 */
1330 mutex_enter(&sffs_lock);
1331 node = sfnode_lookup(VN2SFN(dvp), name, VNON, 0, NULL, 0, NULL);
1332 if (node != NULL) {
1333 mutex_exit(&sffs_lock);
1334 return (EEXIST);
1335 }
1336
1337 /*
1338 * Doesn't exist yet and we have the lock, so create it.
1339 */
1340 sfnode_invalidate_stat_cache(VN2SFN(dvp));
1341 int lookuperr;
1342 node = sfnode_lookup(VN2SFN(dvp), name, VREG,
1343 (vap->va_mask & AT_MODE) ? vap->va_mode : 0, NULL, 0, &lookuperr);
1344
1345 if (node && node->sf_parent)
1346 sfnode_clear_dir_list(node->sf_parent);
1347
1348 mutex_exit(&sffs_lock);
1349 if (node == NULL)
1350 return (lookuperr);
1351 *vpp = sfnode_get_vnode(node);
1352 return (0);
1353}
1354
1355/*ARGSUSED*/
1356static int
1357sffs_mkdir(
1358 vnode_t *dvp,
1359 char *nm,
1360 vattr_t *va,
1361 vnode_t **vpp,
1362 cred_t *cred,
1363 caller_context_t *ct,
1364 int flags,
1365 vsecattr_t *vsecp)
1366{
1367 sfnode_t *node;
1368 vnode_t *vp;
1369 int error;
1370
1371 /*
1372 * These should never happen
1373 */
1374 ASSERT(nm != NULL);
1375 ASSERT(strcmp(nm, "") != 0);
1376 ASSERT(strcmp(nm, ".") != 0);
1377 ASSERT(strcmp(nm, "..") != 0);
1378
1379 /*
1380 * Do an unlocked look up first
1381 */
1382 error = sffs_lookup(dvp, nm, &vp, NULL, 0, NULL, cred, ct, NULL, NULL);
1383 if (error == 0) {
1384 VN_RELE(vp);
1385 return (EEXIST);
1386 }
1387 if (error != ENOENT)
1388 return (error);
1389
1390 /*
1391 * Must be able to write in current directory
1392 */
1393 mutex_enter(&sffs_lock);
1394 error = sfnode_access(VN2SFN(dvp), VWRITE, cred);
1395 if (error) {
1396 mutex_exit(&sffs_lock);
1397 return (error);
1398 }
1399
1400 sfnode_invalidate_stat_cache(VN2SFN(dvp));
1401 int lookuperr = EACCES;
1402 node = sfnode_lookup(VN2SFN(dvp), nm, VDIR,
1403 (va->va_mode & AT_MODE) ? va->va_mode : 0, NULL, 0, &lookuperr);
1404
1405 if (node && node->sf_parent)
1406 sfnode_clear_dir_list(node->sf_parent);
1407
1408 mutex_exit(&sffs_lock);
1409 if (node == NULL)
1410 return (lookuperr);
1411 *vpp = sfnode_get_vnode(node);
1412 return (0);
1413}
1414
1415/*ARGSUSED*/
1416static int
1417sffs_rmdir(
1418 struct vnode *dvp,
1419 char *nm,
1420 vnode_t *cdir,
1421 cred_t *cred,
1422 caller_context_t *ct,
1423 int flags)
1424{
1425 sfnode_t *node;
1426 vnode_t *vp;
1427 int error;
1428
1429 /*
1430 * Return error when removing . and ..
1431 */
1432 if (strcmp(nm, ".") == 0 || strcmp(nm, "") == 0)
1433 return (EINVAL);
1434 if (strcmp(nm, "..") == 0)
1435 return (EEXIST);
1436
1437 error = sffs_lookup(dvp, nm, &vp, NULL, 0, NULL, cred, ct, NULL, NULL);
1438 if (error)
1439 return (error);
1440 if (vp->v_type != VDIR) {
1441 VN_RELE(vp);
1442 return (ENOTDIR);
1443 }
1444
1445#ifdef VBOXVFS_WITH_MMAP
1446 if (vn_vfswlock(vp)) {
1447 VN_RELE(vp);
1448 return (EBUSY);
1449 }
1450#endif
1451
1452 if (vn_mountedvfs(vp)) {
1453 VN_RELE(vp);
1454 return (EBUSY);
1455 }
1456
1457 node = VN2SFN(vp);
1458
1459 mutex_enter(&sffs_lock);
1460 error = sfnode_access(VN2SFN(dvp), VEXEC | VWRITE, cred);
1461 if (error)
1462 goto done;
1463
1464 /*
1465 * If anything else is using this vnode, then fail the remove.
1466 * Why? Windows hosts can't remove something that is open,
1467 * so we have to sfprov_close() it first.
1468 * There is no errno for this - since it's not a problem on UNIX,
1469 * but EINVAL is the closest.
1470 */
1471 if (node->sf_file != NULL) {
1472 if (vp->v_count > 1) {
1473 error = EINVAL;
1474 goto done;
1475 }
1476 (void)sfprov_close(node->sf_file);
1477 node->sf_file = NULL;
1478 }
1479
1480 /*
1481 * Remove the directory on the host and mark the node as stale.
1482 */
1483 sfnode_invalidate_stat_cache(VN2SFN(dvp));
1484 error = sfprov_rmdir(node->sf_sffs->sf_handle, node->sf_path);
1485 if (error == ENOENT || error == 0)
1486 sfnode_make_stale(node);
1487
1488 if (node->sf_parent)
1489 sfnode_clear_dir_list(node->sf_parent);
1490done:
1491 mutex_exit(&sffs_lock);
1492#ifdef VBOXVFS_WITH_MMAP
1493 vn_vfsunlock(vp);
1494#endif
1495 VN_RELE(vp);
1496 return (error);
1497}
1498
1499
1500#ifdef VBOXVFS_WITH_MMAP
1501static caddr_t
1502sffs_page_map(
1503 page_t *ppage,
1504 enum seg_rw segaccess)
1505{
1506 /* Use seg_kpm driver if possible (64-bit) */
1507 if (kpm_enable)
1508 return (hat_kpm_mapin(ppage, NULL));
1509 ASSERT(segaccess == S_READ || segaccess == S_WRITE);
1510 return (ppmapin(ppage, PROT_READ | ((segaccess == S_WRITE) ? PROT_WRITE : 0), (caddr_t)-1));
1511}
1512
1513
1514static void
1515sffs_page_unmap(
1516 page_t *ppage,
1517 caddr_t addr)
1518{
1519 if (kpm_enable)
1520 hat_kpm_mapout(ppage, NULL, addr);
1521 else
1522 ppmapout(addr);
1523}
1524
1525
1526/*
1527 * Called when there's no page in the cache. This will create new page(s) and read
1528 * the file data into it.
1529 */
1530static int
1531sffs_readpages(
1532 vnode_t *dvp,
1533 offset_t off,
1534 page_t *pagelist[],
1535 size_t pagelistsize,
1536 struct seg *segp,
1537 caddr_t addr,
1538 enum seg_rw segaccess)
1539{
1540 ASSERT(MUTEX_HELD(&sffs_lock));
1541
1542 int error = 0;
1543 u_offset_t io_off, total;
1544 size_t io_len;
1545 page_t *ppages;
1546 page_t *pcur;
1547
1548 sfnode_t *node = VN2SFN(dvp);
1549 ASSERT(node);
1550 ASSERT(node->sf_file);
1551
1552 if (pagelistsize == PAGESIZE)
1553 {
1554 io_off = off;
1555 io_len = PAGESIZE;
1556 ppages = page_create_va(dvp, io_off, io_len, PG_WAIT | PG_EXCL, segp, addr);
1557 }
1558 else
1559 ppages = pvn_read_kluster(dvp, off, segp, addr, &io_off, &io_len, off, pagelistsize, 0);
1560
1561 /* If page already exists return success */
1562 if (!ppages)
1563 {
1564 *pagelist = NULL;
1565 return (0);
1566 }
1567
1568 /*
1569 * Map & read page-by-page.
1570 */
1571 total = io_off + io_len;
1572 pcur = ppages;
1573 while (io_off < total)
1574 {
1575 ASSERT3U(io_off, ==, pcur->p_offset);
1576
1577 caddr_t virtaddr = sffs_page_map(pcur, segaccess);
1578 uint32_t bytes = PAGESIZE;
1579 error = sfprov_read(node->sf_file, virtaddr, io_off, &bytes);
1580 /*
1581 * If we reuse pages without zero'ing them, one process can mmap() and read-past the length
1582 * to read previously mmap'd contents (from possibly other processes).
1583 */
1584 if (error == 0 && bytes < PAGESIZE)
1585 memset(virtaddr + bytes, 0, PAGESIZE - bytes);
1586 sffs_page_unmap(pcur, virtaddr);
1587 if (error != 0)
1588 {
1589 cmn_err(CE_WARN, "sffs_readpages: sfprov_read() failed. error=%d bytes=%u\n", error, bytes);
1590 /* Get rid of all kluster pages read & bail. */
1591 pvn_read_done(ppages, B_ERROR);
1592 return (error);
1593 }
1594 pcur = pcur->p_next;
1595 io_off += PAGESIZE;
1596 }
1597
1598 /*
1599 * Fill in the pagelist from kluster at the requested offset.
1600 */
1601 pvn_plist_init(ppages, pagelist, pagelistsize, off, io_len, segaccess);
1602 ASSERT(pagelist == NULL || (*pagelist)->p_offset == off);
1603 return (0);
1604}
1605
1606
1607/*ARGSUSED*/
1608static int
1609sffs_getpage(
1610 vnode_t *dvp,
1611 offset_t off,
1612 size_t len,
1613 uint_t *protp,
1614 page_t *pagelist[],
1615 size_t pagelistsize,
1616 struct seg *segp,
1617 caddr_t addr,
1618 enum seg_rw segaccess,
1619 cred_t *credp
1620#if !defined(VBOX_VFS_SOLARIS_10U6)
1621 , caller_context_t *ct
1622#endif
1623 )
1624{
1625 int error = 0;
1626 int is_recursive = 0;
1627 page_t **pageliststart = pagelist;
1628 sfnode_t *node = VN2SFN(dvp);
1629 ASSERT(node);
1630 ASSERT(node->sf_file);
1631
1632 if (segaccess == S_WRITE)
1633 return (ENOSYS); /* Will this ever happen? */
1634
1635 /* Don't bother about faultahead for now. */
1636 if (pagelist == NULL)
1637 return (0);
1638
1639 if (len > pagelistsize)
1640 len = pagelistsize;
1641 else
1642 len = P2ROUNDUP(len, PAGESIZE);
1643 ASSERT(pagelistsize >= len);
1644
1645 if (protp)
1646 *protp = PROT_ALL;
1647
1648 /*
1649 * The buffer passed to sffs_write may be mmap'd so we may get a
1650 * pagefault there, in which case we'll end up here with this thread
1651 * already owning the mutex. Mutexes aren't recursive.
1652 */
1653 if (mutex_owner(&sffs_lock) == curthread)
1654 is_recursive = 1;
1655 else
1656 mutex_enter(&sffs_lock);
1657
1658 /* Don't map pages past end of the file. */
1659 if (off + len > node->sf_stat.sf_size + PAGEOFFSET)
1660 {
1661 if (!is_recursive)
1662 mutex_exit(&sffs_lock);
1663 return (EFAULT);
1664 }
1665
1666 while (len > 0)
1667 {
1668 /*
1669 * Look for pages in the requested offset range, or create them if we can't find any.
1670 */
1671 if ((*pagelist = page_lookup(dvp, off, SE_SHARED)) != NULL)
1672 *(pagelist + 1) = NULL;
1673 else if ((error = sffs_readpages(dvp, off, pagelist, pagelistsize, segp, addr, segaccess)) != 0)
1674 {
1675 while (pagelist > pageliststart)
1676 page_unlock(*--pagelist);
1677
1678 *pagelist = NULL;
1679 if (!is_recursive)
1680 mutex_exit(&sffs_lock);
1681 return (error);
1682 }
1683
1684 while (*pagelist)
1685 {
1686 ASSERT3U((*pagelist)->p_offset, ==, off);
1687 off += PAGESIZE;
1688 addr += PAGESIZE;
1689 if (len > 0)
1690 {
1691 ASSERT3U(len, >=, PAGESIZE);
1692 len -= PAGESIZE;
1693 }
1694
1695 ASSERT3U(pagelistsize, >=, PAGESIZE);
1696 pagelistsize -= PAGESIZE;
1697 pagelist++;
1698 }
1699 }
1700
1701 /*
1702 * Fill the page list array with any pages left in the cache.
1703 */
1704 while ( pagelistsize > 0
1705 && (*pagelist++ = page_lookup_nowait(dvp, off, SE_SHARED)))
1706 {
1707 off += PAGESIZE;
1708 pagelistsize -= PAGESIZE;
1709 }
1710
1711 *pagelist = NULL;
1712 if (!is_recursive)
1713 mutex_exit(&sffs_lock);
1714 return (error);
1715}
1716
1717
1718/*ARGSUSED*/
1719static int
1720sffs_putpage(
1721 vnode_t *dvp,
1722 offset_t off,
1723 size_t len,
1724 int flags,
1725 cred_t *credp
1726#if !defined(VBOX_VFS_SOLARIS_10U6)
1727 , caller_context_t *ct
1728#endif
1729 )
1730{
1731 /*
1732 * We don't support PROT_WRITE mmaps.
1733 */
1734 return (ENOSYS);
1735}
1736
1737
1738/*ARGSUSED*/
1739static int
1740sffs_discardpage(
1741 vnode_t *dvp,
1742 page_t *ppage,
1743 u_offset_t *poff,
1744 size_t *plen,
1745 int flags,
1746 cred_t *pcred)
1747{
1748 /*
1749 * This would not get invoked i.e. via pvn_vplist_dirty() since we don't support
1750 * PROT_WRITE mmaps and therefore will not have dirty pages.
1751 */
1752 pvn_write_done(ppage, B_INVAL | B_ERROR | B_FORCE);
1753 return (0);
1754}
1755
1756
1757/*ARGSUSED*/
1758static int
1759sffs_map(
1760 vnode_t *dvp,
1761 offset_t off,
1762 struct as *asp,
1763 caddr_t *addrp,
1764 size_t len,
1765 uchar_t prot,
1766 uchar_t maxprot,
1767 uint_t flags,
1768 cred_t *credp
1769#if !defined(VBOX_VFS_SOLARIS_10U6)
1770 , caller_context_t *ct
1771#endif
1772 )
1773{
1774 /*
1775 * Invocation: mmap()->smmap_common()->VOP_MAP()->sffs_map(). Once the
1776 * segment driver creates the new segment via segvn_create(), it'll
1777 * invoke down the line VOP_ADDMAP()->sffs_addmap()
1778 */
1779 int error = 0;
1780 sfnode_t *node = VN2SFN(dvp);
1781 ASSERT(node);
1782 if ((flags & MAP_SHARED) && (prot & PROT_WRITE))
1783 return (ENOTSUP);
1784
1785 if (off < 0 || len > MAXOFFSET_T - off)
1786 return (ENXIO);
1787
1788 if (dvp->v_type != VREG)
1789 return (ENODEV);
1790
1791 if (dvp->v_flag & VNOMAP)
1792 return (ENOSYS);
1793
1794 if (vn_has_mandatory_locks(dvp, node->sf_stat.sf_mode))
1795 return (EAGAIN);
1796
1797 mutex_enter(&sffs_lock);
1798 as_rangelock(asp);
1799
1800#if defined(VBOX_VFS_SOLARIS_10U6)
1801 if ((flags & MAP_FIXED) == 0)
1802 {
1803 map_addr(addrp, len, off, 1, flags);
1804 if (*addrp == NULL)
1805 error = ENOMEM;
1806 }
1807 else
1808 as_unmap(asp, *addrp, len); /* User specified address, remove any previous mappings */
1809#else
1810 error = choose_addr(asp, addrp, len, off, ADDR_VACALIGN, flags);
1811#endif
1812
1813 if (error)
1814 {
1815 as_rangeunlock(asp);
1816 mutex_exit(&sffs_lock);
1817 return (error);
1818 }
1819
1820 segvn_crargs_t vnodeargs;
1821 memset(&vnodeargs, 0, sizeof(vnodeargs));
1822 vnodeargs.vp = dvp;
1823 vnodeargs.cred = credp;
1824 vnodeargs.offset = off;
1825 vnodeargs.type = flags & MAP_TYPE;
1826 vnodeargs.prot = prot;
1827 vnodeargs.maxprot = maxprot;
1828 vnodeargs.flags = flags & ~MAP_TYPE;
1829 vnodeargs.amp = NULL; /* anon. mapping */
1830 vnodeargs.szc = 0; /* preferred page size code */
1831 vnodeargs.lgrp_mem_policy_flags = 0;
1832
1833 error = as_map(asp, *addrp, len, segvn_create, &vnodeargs);
1834
1835 as_rangeunlock(asp);
1836 mutex_exit(&sffs_lock);
1837 return (error);
1838}
1839
1840
1841/*ARGSUSED*/
1842static int
1843sffs_addmap(
1844 vnode_t *dvp,
1845 offset_t off,
1846 struct as *asp,
1847 caddr_t addr,
1848 size_t len,
1849 uchar_t prot,
1850 uchar_t maxprot,
1851 uint_t flags,
1852 cred_t *credp
1853#if !defined(VBOX_VFS_SOLARIS_10U6)
1854 , caller_context_t *ct
1855#endif
1856 )
1857{
1858 if (dvp->v_flag & VNOMAP)
1859 return (ENOSYS);
1860 return (0);
1861}
1862
1863
1864/*ARGSUSED*/
1865static int
1866sffs_delmap(
1867 vnode_t *dvp,
1868 offset_t off,
1869 struct as *asp,
1870 caddr_t addr,
1871 size_t len,
1872 uint_t prot,
1873 uint_t maxprot,
1874 uint_t flags,
1875 cred_t *credp
1876#if !defined(VBOX_VFS_SOLARIS_10U6)
1877 , caller_context_t *ct
1878#endif
1879 )
1880{
1881 if (dvp->v_flag & VNOMAP)
1882 return (ENOSYS);
1883
1884 return (0);
1885}
1886#endif /* VBOXVFS_WITH_MMAP */
1887
1888
1889/*ARGSUSED*/
1890static int
1891sffs_readlink(
1892 vnode_t *vp,
1893 uio_t *uiop,
1894 cred_t *cred
1895#if !defined(VBOX_VFS_SOLARIS_10U6)
1896 ,
1897 caller_context_t *ct
1898#endif
1899 )
1900{
1901 sfnode_t *node;
1902 int error = 0;
1903 char *target = NULL;
1904
1905 if (uiop->uio_iovcnt != 1)
1906 return (EINVAL);
1907
1908 if (vp->v_type != VLNK)
1909 return (EINVAL);
1910
1911 mutex_enter(&sffs_lock);
1912 node = VN2SFN(vp);
1913
1914 target = kmem_alloc(MAXPATHLEN, KM_SLEEP);
1915
1916 error = sfprov_readlink(node->sf_sffs->sf_handle, node->sf_path, target,
1917 MAXPATHLEN);
1918 if (error)
1919 goto done;
1920
1921 error = uiomove(target, strlen(target), UIO_READ, uiop);
1922
1923done:
1924 mutex_exit(&sffs_lock);
1925 if (target)
1926 kmem_free(target, MAXPATHLEN);
1927 return (error);
1928}
1929
1930
1931/*ARGSUSED*/
1932static int
1933sffs_symlink(
1934 vnode_t *dvp,
1935 char *linkname,
1936 vattr_t *vap,
1937 char *target,
1938 cred_t *cred
1939#if !defined(VBOX_VFS_SOLARIS_10U6)
1940 ,
1941 caller_context_t *ct,
1942 int flags
1943#endif
1944 )
1945{
1946 sfnode_t *dir;
1947 sfnode_t *node;
1948 sffs_stat_t stat;
1949 int error = 0;
1950 char *fullpath;
1951
1952 /*
1953 * These should never happen
1954 */
1955 ASSERT(linkname != NULL);
1956 ASSERT(strcmp(linkname, "") != 0);
1957 ASSERT(strcmp(linkname, ".") != 0);
1958 ASSERT(strcmp(linkname, "..") != 0);
1959
1960 /*
1961 * Basic checks.
1962 */
1963 if (vap->va_type != VLNK)
1964 return (EINVAL);
1965
1966 mutex_enter(&sffs_lock);
1967
1968 if (sfnode_lookup(VN2SFN(dvp), linkname, VNON, 0, NULL, 0, NULL) !=
1969 NULL) {
1970 error = EEXIST;
1971 goto done;
1972 }
1973
1974 dir = VN2SFN(dvp);
1975 error = sfnode_access(dir, VWRITE, cred);
1976 if (error)
1977 goto done;
1978
1979 /*
1980 * Create symlink. Note that we ignore vap->va_mode because generally
1981 * we can't change the attributes of the symlink itself.
1982 */
1983 fullpath = sfnode_construct_path(dir, linkname);
1984 error = sfprov_symlink(dir->sf_sffs->sf_handle, fullpath, target,
1985 &stat);
1986 kmem_free(fullpath, strlen(fullpath) + 1);
1987 if (error)
1988 goto done;
1989
1990 node = sfnode_lookup(dir, linkname, VLNK, 0, &stat,
1991 sfnode_cur_time_usec(), NULL);
1992
1993 sfnode_invalidate_stat_cache(dir);
1994 sfnode_clear_dir_list(dir);
1995
1996done:
1997 mutex_exit(&sffs_lock);
1998 return (error);
1999}
2000
2001
2002/*ARGSUSED*/
2003static int
2004sffs_remove(
2005 vnode_t *dvp,
2006 char *name,
2007 cred_t *cred,
2008 caller_context_t *ct,
2009 int flags)
2010{
2011 vnode_t *vp;
2012 sfnode_t *node;
2013 int error;
2014
2015 /*
2016 * These should never happen
2017 */
2018 ASSERT(name != NULL);
2019 ASSERT(strcmp(name, "..") != 0);
2020
2021 error = sffs_lookup(dvp, name, &vp,
2022 NULL, 0, NULL, cred, ct, NULL, NULL);
2023 if (error)
2024 return (error);
2025 node = VN2SFN(vp);
2026
2027 mutex_enter(&sffs_lock);
2028 error = sfnode_access(VN2SFN(dvp), VEXEC | VWRITE, cred);
2029 if (error)
2030 goto done;
2031
2032 /*
2033 * If anything else is using this vnode, then fail the remove.
2034 * Why? Windows hosts can't sfprov_remove() a file that is open,
2035 * so we have to sfprov_close() it first.
2036 * There is no errno for this - since it's not a problem on UNIX,
2037 * but ETXTBSY is the closest.
2038 */
2039 if (node->sf_file != NULL) {
2040 if (vp->v_count > 1) {
2041 error = ETXTBSY;
2042 goto done;
2043 }
2044 (void)sfprov_close(node->sf_file);
2045 node->sf_file = NULL;
2046 }
2047
2048 /*
2049 * Remove the file on the host and mark the node as stale.
2050 */
2051 sfnode_invalidate_stat_cache(VN2SFN(dvp));
2052
2053 error = sfprov_remove(node->sf_sffs->sf_handle, node->sf_path,
2054 node->sf_type == VLNK);
2055 if (error == ENOENT || error == 0)
2056 sfnode_make_stale(node);
2057
2058 if (node->sf_parent)
2059 sfnode_clear_dir_list(node->sf_parent);
2060done:
2061 mutex_exit(&sffs_lock);
2062 VN_RELE(vp);
2063 return (error);
2064}
2065
2066/*ARGSUSED*/
2067static int
2068sffs_rename(
2069 vnode_t *old_dir,
2070 char *old_nm,
2071 vnode_t *new_dir,
2072 char *new_nm,
2073 cred_t *cred,
2074 caller_context_t *ct,
2075 int flags)
2076{
2077 char *newpath;
2078 int error;
2079 sfnode_t *node;
2080
2081 if (strcmp(new_nm, "") == 0 ||
2082 strcmp(new_nm, ".") == 0 ||
2083 strcmp(new_nm, "..") == 0 ||
2084 strcmp(old_nm, "") == 0 ||
2085 strcmp(old_nm, ".") == 0 ||
2086 strcmp(old_nm, "..") == 0)
2087 return (EINVAL);
2088
2089 /*
2090 * make sure we have permission to do the rename
2091 */
2092 mutex_enter(&sffs_lock);
2093 error = sfnode_access(VN2SFN(old_dir), VEXEC | VWRITE, cred);
2094 if (error == 0 && new_dir != old_dir)
2095 error = sfnode_access(VN2SFN(new_dir), VEXEC | VWRITE, cred);
2096 if (error)
2097 goto done;
2098
2099 node = sfnode_lookup(VN2SFN(old_dir), old_nm, VNON, 0, NULL, 0, NULL);
2100 if (node == NULL) {
2101 error = ENOENT;
2102 goto done;
2103 }
2104
2105 /*
2106 * Rename the file on the host and in our caches.
2107 */
2108 sfnode_invalidate_stat_cache(node);
2109 sfnode_invalidate_stat_cache(VN2SFN(old_dir));
2110 sfnode_invalidate_stat_cache(VN2SFN(new_dir));
2111
2112 newpath = sfnode_construct_path(VN2SFN(new_dir), new_nm);
2113 error = sfprov_rename(node->sf_sffs->sf_handle, node->sf_path, newpath,
2114 node->sf_type == VDIR);
2115 if (error == 0)
2116 sfnode_rename(node, VN2SFN(new_dir), newpath);
2117 else {
2118 kmem_free(newpath, strlen(newpath) + 1);
2119 if (error == ENOENT)
2120 sfnode_make_stale(node);
2121 }
2122done:
2123 mutex_exit(&sffs_lock);
2124 return (error);
2125}
2126
2127
2128/*ARGSUSED*/
2129static int
2130sffs_fsync(vnode_t *vp, int flag, cred_t *cr, caller_context_t *ct)
2131{
2132 sfnode_t *node;
2133 int error;
2134
2135 /*
2136 * Ask the host to sync any data it may have cached for open files.
2137 */
2138 mutex_enter(&sffs_lock);
2139 node = VN2SFN(vp);
2140 if (node->sf_file == NULL)
2141 error = EBADF;
2142 else if (node->sf_sffs->sf_fsync)
2143 error = sfprov_fsync(node->sf_file);
2144 else
2145 error = 0;
2146 mutex_exit(&sffs_lock);
2147 return (error);
2148}
2149
2150/*
2151 * This may be the last reference, possibly time to close the file and
2152 * destroy the vnode. If the sfnode is stale, we'll destroy that too.
2153 */
2154/*ARGSUSED*/
2155static void
2156#if defined(VBOX_VFS_SOLARIS_10U6)
2157sffs_inactive(vnode_t *vp, cred_t *cr)
2158#else
2159sffs_inactive(vnode_t *vp, cred_t *cr, caller_context_t *ct)
2160#endif
2161{
2162 sfnode_t *node;
2163
2164 /*
2165 * nothing to do if this isn't the last use
2166 */
2167 mutex_enter(&sffs_lock);
2168 node = VN2SFN(vp);
2169 mutex_enter(&vp->v_lock);
2170 if (vp->v_count > 1) {
2171 --vp->v_count;
2172 mutex_exit(&vp->v_lock);
2173 mutex_exit(&sffs_lock);
2174 return;
2175 }
2176
2177 if (vn_has_cached_data(vp)) {
2178#ifdef VBOXVFS_WITH_MMAP
2179 /* We're fine with releasing the vnode lock here as we should be covered by the sffs_lock */
2180 mutex_exit(&vp->v_lock);
2181 /* We won't have any dirty pages, this will just invalidate (destroy) the pages and move it to the cachelist. */
2182 pvn_vplist_dirty(vp, 0 /* offset */, sffs_discardpage, B_INVAL, cr);
2183 mutex_enter(&vp->v_lock);
2184#else
2185 panic("sffs_inactive() found cached data");
2186#endif
2187 }
2188
2189 /*
2190 * destroy the vnode
2191 */
2192 node->sf_vnode = NULL;
2193 mutex_exit(&vp->v_lock);
2194 vn_invalid(vp);
2195 vn_free(vp);
2196 LogFlowFunc((" %s vnode cleared\n", node->sf_path));
2197
2198 /*
2199 * Close the sf_file for the node.
2200 */
2201 if (node->sf_file != NULL) {
2202 (void)sfprov_close(node->sf_file);
2203 node->sf_file = NULL;
2204 }
2205
2206 /*
2207 * Free the directory entries for the node. This should normally
2208 * have been taken care of in sffs_close(), but better safe than
2209 * sorry.
2210 */
2211 sfnode_clear_dir_list(node);
2212
2213 /*
2214 * If the node is stale, we can also destroy it.
2215 */
2216 if (node->sf_is_stale && node->sf_children == 0)
2217 sfnode_destroy(node);
2218
2219 mutex_exit(&sffs_lock);
2220 return;
2221}
2222
2223/*
2224 * All the work for this is really done in sffs_lookup().
2225 */
2226/*ARGSUSED*/
2227static int
2228sffs_open(vnode_t **vpp, int flag, cred_t *cr, caller_context_t *ct)
2229{
2230 sfnode_t *node;
2231 int error = 0;
2232
2233 mutex_enter(&sffs_lock);
2234
2235 node = VN2SFN(*vpp);
2236 sfnode_open(node, flag);
2237 if (node->sf_file == NULL)
2238 error = EINVAL;
2239 mutex_exit(&sffs_lock);
2240
2241 return (error);
2242}
2243
2244/*
2245 * All the work for this is really done in inactive.
2246 */
2247/*ARGSUSED*/
2248static int
2249sffs_close(
2250 vnode_t *vp,
2251 int flag,
2252 int count,
2253 offset_t offset,
2254 cred_t *cr,
2255 caller_context_t *ct)
2256{
2257 sfnode_t *node;
2258
2259 mutex_enter(&sffs_lock);
2260 node = VN2SFN(vp);
2261
2262 /*
2263 * Free the directory entries for the node. We do this on this call
2264 * here because the directory node may not become inactive for a long
2265 * time after the readdir is over. Case in point, if somebody cd's into
2266 * the directory then it won't become inactive until they cd away again.
2267 * In such a case we would end up with the directory listing not getting
2268 * updated (i.e. the result of 'ls' always being the same) until they
2269 * change the working directory.
2270 */
2271 sfnode_clear_dir_list(node);
2272
2273 sfnode_invalidate_stat_cache(node);
2274
2275 if (node->sf_file != NULL && vp->v_count <= 1)
2276 {
2277 (void)sfprov_close(node->sf_file);
2278 node->sf_file = NULL;
2279 }
2280
2281 mutex_exit(&sffs_lock);
2282 return (0);
2283}
2284
2285/* ARGSUSED */
2286static int
2287sffs_seek(vnode_t *v, offset_t o, offset_t *no, caller_context_t *ct)
2288{
2289 if (*no < 0 || *no > MAXOFFSET_T)
2290 return (EINVAL);
2291
2292 if (v->v_type == VDIR)
2293 {
2294 sffs_dirents_t *cur_buf = VN2SFN(v)->sf_dir_list;
2295 off_t offset = 0;
2296
2297 if (cur_buf == NULL)
2298 return (0);
2299
2300 while (cur_buf != NULL) {
2301 if (*no >= offset && *no <= offset + cur_buf->sf_len)
2302 return (0);
2303 offset += cur_buf->sf_len;
2304 cur_buf = cur_buf->sf_next;
2305 }
2306 return (EINVAL);
2307 }
2308 return (0);
2309}
2310
2311
2312
2313/*
2314 * By returning an error for this, we prevent anything in sffs from
2315 * being re-exported by NFS
2316 */
2317/* ARGSUSED */
2318static int
2319sffs_fid(vnode_t *vp, fid_t *fidp, caller_context_t *ct)
2320{
2321 return (ENOTSUP);
2322}
2323
2324/*
2325 * vnode operations for regular files
2326 */
2327const fs_operation_def_t sffs_ops_template[] = {
2328#if defined(VBOX_VFS_SOLARIS_10U6)
2329 VOPNAME_ACCESS, sffs_access,
2330 VOPNAME_CLOSE, sffs_close,
2331 VOPNAME_CREATE, sffs_create,
2332 VOPNAME_FID, sffs_fid,
2333 VOPNAME_FSYNC, sffs_fsync,
2334 VOPNAME_GETATTR, sffs_getattr,
2335 VOPNAME_INACTIVE, sffs_inactive,
2336 VOPNAME_LOOKUP, sffs_lookup,
2337 VOPNAME_MKDIR, sffs_mkdir,
2338 VOPNAME_OPEN, sffs_open,
2339 VOPNAME_PATHCONF, sffs_pathconf,
2340 VOPNAME_READ, sffs_read,
2341 VOPNAME_READDIR, sffs_readdir,
2342 VOPNAME_READLINK, sffs_readlink,
2343 VOPNAME_REMOVE, sffs_remove,
2344 VOPNAME_RENAME, sffs_rename,
2345 VOPNAME_RMDIR, sffs_rmdir,
2346 VOPNAME_SEEK, sffs_seek,
2347 VOPNAME_SETATTR, sffs_setattr,
2348 VOPNAME_SPACE, sffs_space,
2349 VOPNAME_SYMLINK, sffs_symlink,
2350 VOPNAME_WRITE, sffs_write,
2351
2352# ifdef VBOXVFS_WITH_MMAP
2353 VOPNAME_MAP, sffs_map,
2354 VOPNAME_ADDMAP, sffs_addmap,
2355 VOPNAME_DELMAP, sffs_delmap,
2356 VOPNAME_GETPAGE, sffs_getpage,
2357 VOPNAME_PUTPAGE, sffs_putpage,
2358# endif
2359
2360 NULL, NULL
2361#else
2362 VOPNAME_ACCESS, { .vop_access = sffs_access },
2363 VOPNAME_CLOSE, { .vop_close = sffs_close },
2364 VOPNAME_CREATE, { .vop_create = sffs_create },
2365 VOPNAME_FID, { .vop_fid = sffs_fid },
2366 VOPNAME_FSYNC, { .vop_fsync = sffs_fsync },
2367 VOPNAME_GETATTR, { .vop_getattr = sffs_getattr },
2368 VOPNAME_INACTIVE, { .vop_inactive = sffs_inactive },
2369 VOPNAME_LOOKUP, { .vop_lookup = sffs_lookup },
2370 VOPNAME_MKDIR, { .vop_mkdir = sffs_mkdir },
2371 VOPNAME_OPEN, { .vop_open = sffs_open },
2372 VOPNAME_PATHCONF, { .vop_pathconf = sffs_pathconf },
2373 VOPNAME_READ, { .vop_read = sffs_read },
2374 VOPNAME_READDIR, { .vop_readdir = sffs_readdir },
2375 VOPNAME_READLINK, { .vop_readlink = sffs_readlink },
2376 VOPNAME_REMOVE, { .vop_remove = sffs_remove },
2377 VOPNAME_RENAME, { .vop_rename = sffs_rename },
2378 VOPNAME_RMDIR, { .vop_rmdir = sffs_rmdir },
2379 VOPNAME_SEEK, { .vop_seek = sffs_seek },
2380 VOPNAME_SETATTR, { .vop_setattr = sffs_setattr },
2381 VOPNAME_SPACE, { .vop_space = sffs_space },
2382 VOPNAME_SYMLINK, { .vop_symlink = sffs_symlink },
2383 VOPNAME_WRITE, { .vop_write = sffs_write },
2384
2385# ifdef VBOXVFS_WITH_MMAP
2386 VOPNAME_MAP, { .vop_map = sffs_map },
2387 VOPNAME_ADDMAP, { .vop_addmap = sffs_addmap },
2388 VOPNAME_DELMAP, { .vop_delmap = sffs_delmap },
2389 VOPNAME_GETPAGE, { .vop_getpage = sffs_getpage },
2390 VOPNAME_PUTPAGE, { .vop_putpage = sffs_putpage },
2391# endif
2392
2393 NULL, NULL
2394#endif
2395};
2396
2397/*
2398 * Also, init and fini functions...
2399 */
2400int
2401sffs_vnode_init(void)
2402{
2403 int err;
2404
2405 err = vn_make_ops("sffs", sffs_ops_template, &sffs_ops);
2406 if (err)
2407 return (err);
2408
2409 avl_create(&sfnodes, sfnode_compare, sizeof (sfnode_t),
2410 offsetof(sfnode_t, sf_linkage));
2411 avl_create(&stale_sfnodes, sfnode_compare, sizeof (sfnode_t),
2412 offsetof(sfnode_t, sf_linkage));
2413
2414 sffs_buffer = kmem_alloc(PAGESIZE, KM_SLEEP);
2415
2416 return (0);
2417}
2418
2419void
2420sffs_vnode_fini(void)
2421{
2422 if (sffs_ops)
2423 vn_freevnodeops(sffs_ops);
2424 ASSERT(avl_first(&sfnodes) == NULL);
2425 avl_destroy(&sfnodes);
2426 if (sffs_buffer != NULL) {
2427 kmem_free(sffs_buffer, PAGESIZE);
2428 sffs_buffer = NULL;
2429 }
2430}
2431
2432/*
2433 * Utility at unmount to get all nodes in that mounted filesystem removed.
2434 */
2435int
2436sffs_purge(struct sffs_data *sffs)
2437{
2438 sfnode_t *node;
2439 sfnode_t *prev;
2440
2441 /*
2442 * Check that no vnodes are active.
2443 */
2444 if (sffs->sf_rootnode->v_count > 1)
2445 return (-1);
2446 for (node = avl_first(&sfnodes); node;
2447 node = AVL_NEXT(&sfnodes, node)) {
2448 if (node->sf_sffs == sffs && node->sf_vnode &&
2449 node->sf_vnode != sffs->sf_rootnode)
2450 return (-1);
2451 }
2452 for (node = avl_first(&stale_sfnodes); node;
2453 node = AVL_NEXT(&stale_sfnodes, node)) {
2454 if (node->sf_sffs == sffs && node->sf_vnode &&
2455 node->sf_vnode != sffs->sf_rootnode)
2456 return (-1);
2457 }
2458
2459 /*
2460 * All clear to destroy all node information. Since there are no
2461 * vnodes, the make stale will cause deletion.
2462 */
2463 VN_RELE(sffs->sf_rootnode);
2464 mutex_enter(&sffs_lock);
2465 for (prev = NULL;;) {
2466 if (prev == NULL)
2467 node = avl_first(&sfnodes);
2468 else
2469 node = AVL_NEXT(&sfnodes, prev);
2470
2471 if (node == NULL)
2472 break;
2473
2474 if (node->sf_sffs == sffs) {
2475 if (node->sf_vnode != NULL)
2476 panic("vboxfs: purge hit active vnode");
2477 sfnode_make_stale(node);
2478 } else {
2479 prev = node;
2480 }
2481 }
2482 mutex_exit(&sffs_lock);
2483 return (0);
2484}
2485
2486#if 0
2487/* Debug helper functions */
2488static void
2489sfnode_print(sfnode_t *node)
2490{
2491 Log(("0x%p", node));
2492 Log((" type=%s (%d)",
2493 node->sf_type == VDIR ? "VDIR" :
2494 node->sf_type == VNON ? "VNON" :
2495 node->sf_type == VLNK ? "VLNK" :
2496 node->sf_type == VREG ? "VREG" : "other", node->sf_type));
2497 Log((" ino=%d", (uint_t)node->sf_ino));
2498 Log((" path=%s", node->sf_path));
2499 Log((" parent=0x%p", node->sf_parent));
2500 if (node->sf_children)
2501 Log((" children=%d", node->sf_children));
2502 if (node->sf_vnode)
2503 Log((" vnode=0x%p", node->sf_vnode));
2504 Log(("%s\n", node->sf_is_stale ? " STALE" : ""));
2505}
2506
2507static void
2508sfnode_list(void)
2509{
2510 sfnode_t *n;
2511 for (n = avl_first(&sfnodes); n != NULL; n = AVL_NEXT(&sfnodes, n))
2512 sfnode_print(n);
2513 for (n = avl_first(&stale_sfnodes); n != NULL;
2514 n = AVL_NEXT(&stale_sfnodes, n))
2515 sfnode_print(n);
2516}
2517#endif
2518
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