root/usr/src/uts/common/fs/autofs/auto_sys.c
/*
 * 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 <sys/types.h>
#include <sys/systm.h>
#include <sys/zone.h>
#include <sys/errno.h>
#include <sys/cred.h>
#include <sys/policy.h>

#include <sys/fs/autofs.h>

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);
}