/* * Copyright (c) 2012 Apple Inc. All rights reserved. * * @APPLE_OSREFERENCE_LICENSE_HEADER_START@ * * This file contains Original Code and/or Modifications of Original Code * as defined in and that are subject to the Apple Public Source License * Version 2.0 (the 'License'). You may not use this file except in * compliance with the License. The rights granted to you under the License * may not be used to create, or enable the creation or redistribution of, * unlawful or unlicensed copies of an Apple operating system, or to * circumvent, violate, or enable the circumvention or violation of, any * terms of an Apple operating system software license agreement. * * Please obtain a copy of the License at * http://www.opensource.apple.com/apsl/ and read it before using this file. * * The Original Code and all software distributed under the License are * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. * Please see the License for the specific language governing rights and * limitations under the License. * * @APPLE_OSREFERENCE_LICENSE_HEADER_END@ */ #include #include #include #include #include #include #include #include #include #include /* * For the moment, most operations that change the fsnode will be called only in the context of * mockfs_mountroot, so they should not need to use a mutex. The exceptions are mockfs_fsnode_vnode, * and mockfs_fsnode_drop_vnode, which will use a tree-wide mutex (that lives in the mockfs_mount_t * for the mount). * * mockfs_fsnode_child_by_type doesn't require locking right now (we're only looking at the structure of * the node tree, which should not change during VNOP operations. */ /* mockfs_fsnode_create: * Given a mount (mp) and mockfs node type (type), creates a new fsnode for that mountpoint (*fsnpp). * For the moment (while type == fileid) we should have at most one node of any given type. * * Returns 0 on success, or an error. */ int mockfs_fsnode_create(mount_t mp, uint8_t type, mockfs_fsnode_t * fsnpp) { int rvalue; uint64_t new_size; rvalue = 0; new_size = 0; if (!fsnpp || !mp) { rvalue = EINVAL; goto done; } switch (type) { case MOCKFS_ROOT: break; case MOCKFS_DEV: break; case MOCKFS_FILE: /* * For a regular file, size is meaningful, but it will always be equal to the * size of the backing device. */ new_size = mp->mnt_devvp->v_specinfo->si_devsize; break; default: rvalue = EINVAL; goto done; } *fsnpp = kalloc_type(mockfs_fsnode_t, Z_WAITOK | Z_ZERO | Z_NOFAIL); (*fsnpp)->size = new_size; (*fsnpp)->type = type; (*fsnpp)->mnt = mp; done: return rvalue; } /* * mockfs_fsnode_destroy: * Given a node (fsnp), tears down and deallocates that node and the entire subtree that it is the * root of (deallocates you, and your children, and your children's children! ...for three months). * * Returns 0 on success, or an error. */ int mockfs_fsnode_destroy(mockfs_fsnode_t fsnp) { int rvalue; rvalue = 0; /* * We will not destroy a root node that is actively pointed to by the mount structure; the * mount must drop the reference to the mockfs tree before we can deallocate it. */ if (!fsnp || (((mockfs_mount_t)fsnp->mnt->mnt_data)->mockfs_root == fsnp)) { rvalue = EINVAL; goto done; } /* * For now, panic in this case; I don't expect anyone to ask us to destroy a node with a live * vfs reference, but this will tell me if that assumption is untrue. */ if (fsnp->vp) { panic("mockfs_fsnode_destroy called on node with live vnode; fsnp = %p (in case gdb is screwing with you)", fsnp); } /* * If this node has children, we need to destroy them. * * At least for now, we aren't guaranteeing destroy will be clean; we may get partway through * and encounter an error, in which case we will panic (we may still have a sane tree, but * we've failed to destroy the subtree, which means someone called destroy when they should * not have done so). */ if (fsnp->child_a) { if ((rvalue = mockfs_fsnode_destroy(fsnp->child_a))) { panic("mockfs_fsnode_destroy failed on child_a; fsnp = %p (in case gdb is screwing with you), rvalue = %d", fsnp, rvalue); } } if (fsnp->child_b) { if ((rvalue = mockfs_fsnode_destroy(fsnp->child_b))) { panic("mockfs_fsnode_destroy failed on child_b; fsnp = %p (in case gdb is screwing with you), rvalue = %d", fsnp, rvalue); } } /* * We need to orphan this node before we destroy it. */ if (fsnp->parent) { if ((rvalue = mockfs_fsnode_orphan(fsnp))) { panic("mockfs_fsnode_orphan failed during destroy; fsnp = %p (in case gdb is screwing with you), rvalue = %d", fsnp, rvalue); } } kfree_type(mockfs_fsnode_t, fsnp); done: return rvalue; } /* * mockfs_fsnode_adopt: * Given two nodes (parent, child), makes one node the child of the other node. * * Returns 0 on success, or an error. */ int mockfs_fsnode_adopt(mockfs_fsnode_t parent, mockfs_fsnode_t child) { int rvalue; rvalue = 0; /* * The child must be an orphan, and the parent cannot be the child. */ if ((!parent || !child || child->parent) && (parent != child)) { rvalue = EINVAL; goto done; } /* * Nodes are actually tied to a specific mount, so assert that both nodes belong to the same mount. */ if (parent->mnt != child->mnt) { rvalue = EINVAL; goto done; } /* * TODO: Get rid of this check if I ever get around to making the tree non-binary. * TODO: Enforce that the parent cannot have two children of the same type (for the moment, this is * implicit in the structure of the tree constructed by mockfs_mountroot, so we don't need to * worry about it). * * Can the parent support another child (food, shelter, unused pointers)? */ if (!parent->child_a) { parent->child_a = child; child->parent = parent; } else if (!parent->child_b) { parent->child_b = child; child->parent = parent; } else { rvalue = ENOMEM; } done: return rvalue; } /* * mockfs_fsnode_orphan: * * Returns 0 on success, or an error. */ int mockfs_fsnode_orphan(mockfs_fsnode_t fsnp) { int rvalue; mockfs_fsnode_t parent; rvalue = 0; if (!fsnp || !fsnp->parent) { rvalue = EINVAL; goto done; } /* * Disallow orphaning a node with a live vnode for now. */ if (fsnp->vp) { panic("mockfs_fsnode_orphan called on node with live vnode; fsnp = %p (in case gdb is screwing with you)", fsnp); } parent = fsnp->parent; if (parent->child_a == fsnp) { parent->child_a = NULL; fsnp->parent = NULL; } else if (parent->child_b == fsnp) { parent->child_b = NULL; fsnp->parent = NULL; } else { panic("mockfs_fsnode_orphan insanity, fsnp->parent != parent->child; fsnp = %p (in case gdb is screwing with you)", fsnp); } done: return rvalue; } /* * mockfs_fsnode_child_by_type: * Given a node (parent) and a type (type), returns the first child (*child) found corresponding to the * requested type. This method exists to support lookup (which is responsible for mapping names, which * we have no conception of currently, onto vnodes). * * This should be safe, as we are walking the read-only parts of the filesystem structure (not touching * the vnode). * * Returns 0 on success, or an error. */ int mockfs_fsnode_child_by_type(mockfs_fsnode_t parent, uint8_t type, mockfs_fsnode_t * child) { int rvalue; rvalue = 0; if (!parent || !child) { rvalue = EINVAL; goto done; } if ((parent->child_a) && (parent->child_a->type == type)) { *child = parent->child_a; } else if ((parent->child_b) && (parent->child_b->type == type)) { *child = parent->child_b; } else { rvalue = ENOENT; } done: return rvalue; } /* * mockfs_fsnode_vnode: * Given a mockfs node (fsnp), returns a vnode (*vpp) corresponding to the mockfs node; the vnode will * have an iocount on it. * * Returns 0 on success, or an error. */ int mockfs_fsnode_vnode(mockfs_fsnode_t fsnp, vnode_t * vpp) { int rvalue; memory_object_control_t ubc_mem_object; mockfs_mount_t mockfs_mnt; struct vnode_fsparam vnfs_param; if ((!fsnp) || (!vpp)) { rvalue = EINVAL; goto done; } mockfs_mnt = ((mockfs_mount_t) fsnp->mnt->mnt_data); lck_mtx_lock(&mockfs_mnt->mockfs_mnt_mtx); if (fsnp->vp) { /* * The vnode already exists; this should be easy. */ rvalue = vnode_get(fsnp->vp); if (!rvalue) { *vpp = fsnp->vp; } } else { /* * We need to create the vnode; this will be unpleasant. */ vnfs_param.vnfs_mp = fsnp->mnt; vnfs_param.vnfs_vtype = (fsnp->type == MOCKFS_FILE) ? VREG : VDIR; vnfs_param.vnfs_str = "mockfs"; vnfs_param.vnfs_dvp = (fsnp->type == MOCKFS_ROOT) ? NULL : fsnp->parent->vp; vnfs_param.vnfs_fsnode = fsnp; vnfs_param.vnfs_vops = mockfs_vnodeop_p; vnfs_param.vnfs_markroot = (fsnp->type == MOCKFS_ROOT); vnfs_param.vnfs_marksystem = 0; vnfs_param.vnfs_rdev = 0; vnfs_param.vnfs_filesize = fsnp->size; vnfs_param.vnfs_cnp = NULL; vnfs_param.vnfs_flags = VNFS_CANTCACHE | VNFS_NOCACHE; rvalue = vnode_create(VNCREATE_FLAVOR, VCREATESIZE, &vnfs_param, &fsnp->vp); if ((!rvalue) && (fsnp->type == MOCKFS_FILE) && (mockfs_mnt->mockfs_memory_backed)) { /* * We're memory backed; point the pager towards the backing store of the device. */ ubc_mem_object = ubc_getobject(fsnp->vp, 0); if (!ubc_mem_object) { panic("mockfs_fsvnode failed to get ubc_mem_object for a new vnode"); } rvalue = pager_map_to_phys_contiguous(ubc_mem_object, 0, (mockfs_mnt->mockfs_memdev_base << PAGE_SHIFT), fsnp->size); if (rvalue) { panic("mockfs_fsnode_vnode failed to create fictitious pages for a memory-backed device; rvalue = %d", rvalue); } } if (!rvalue) { *vpp = fsnp->vp; } } lck_mtx_unlock(&mockfs_mnt->mockfs_mnt_mtx); done: return rvalue; } /* * mockfs_fsnode_vnode: * Given a mockfs node (fsnp) that has a vnode associated with it, causes them to drop their * references to each other. This exists to support mockfs_reclaim. This method will grab the tree * mutex, as this will mutate the tree. * * Returns 0 on success, or an error. */ int mockfs_fsnode_drop_vnode(mockfs_fsnode_t fsnp) { int rvalue; mockfs_mount_t mockfs_mnt; vnode_t vp; rvalue = 0; if (!fsnp) { rvalue = EINVAL; goto done; } mockfs_mnt = ((mockfs_mount_t) fsnp->mnt->mnt_data); lck_mtx_lock(&mockfs_mnt->mockfs_mnt_mtx); if (!(fsnp->vp)) { panic("mock_fsnode_drop_vnode: target fsnode does not have an associated vnode"); } vp = fsnp->vp; fsnp->vp = NULL; vnode_clearfsnode(vp); lck_mtx_unlock(&mockfs_mnt->mockfs_mnt_mtx); done: return rvalue; }