/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright (c) 2003, 2010, Oracle and/or its affiliates. All rights reserved. */ #include #include #include #include #include #include #include extern struct autofs_globals *autofs_zone_init(void); int autofssys(enum autofssys_op opcode, uintptr_t arg) { int error = 0; switch (opcode) { case AUTOFS_UNMOUNTALL: { /* attempt to remove all autofs mounts */ zone_t *zone; zoneid_t zoneid; struct autofs_globals *fngp; zoneid = (zoneid_t)arg; if (secpolicy_fs_unmount(CRED(), NULL) != 0 || crgetzoneid(CRED()) != GLOBAL_ZONEID) return (set_errno(EPERM)); if ((zone = zone_find_by_id(zoneid)) == NULL) return (set_errno(EINVAL)); mutex_enter(&autofs_minor_lock); fngp = zone_getspecific(autofs_key, zone); if (fngp == NULL) { mutex_exit(&autofs_minor_lock); zone_rele(zone); /* * There were no mounts, so no work to do. Success. */ return (0); } mutex_exit(&autofs_minor_lock); unmount_tree(fngp, B_TRUE); zone_rele(zone); break; } case AUTOFS_SETDOOR: { /* set door handle for zone */ uint_t did; struct autofs_globals *fngp; /* * We need to use the minor_lock to serialize setting this. */ mutex_enter(&autofs_minor_lock); fngp = zone_getspecific(autofs_key, curproc->p_zone); if (fngp == NULL) { fngp = autofs_zone_init(); (void) zone_setspecific(autofs_key, curproc->p_zone, fngp); } mutex_exit(&autofs_minor_lock); ASSERT(fngp != NULL); if (copyin((uint_t *)arg, &did, sizeof (uint_t))) return (set_errno(EFAULT)); mutex_enter(&fngp->fng_autofs_daemon_lock); if (fngp->fng_autofs_daemon_dh) door_ki_rele(fngp->fng_autofs_daemon_dh); fngp->fng_autofs_daemon_dh = door_ki_lookup(did); fngp->fng_autofs_pid = curproc->p_pid; mutex_exit(&fngp->fng_autofs_daemon_lock); break; } default: error = EINVAL; break; } return (error ? set_errno(error) : 0); }