10Sstevel@tonic-gate /* 20Sstevel@tonic-gate * CDDL HEADER START 30Sstevel@tonic-gate * 40Sstevel@tonic-gate * The contents of this file are subject to the terms of the 52170Sevanl * Common Development and Distribution License (the "License"). 62170Sevanl * You may not use this file except in compliance with the License. 70Sstevel@tonic-gate * 80Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 90Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing. 100Sstevel@tonic-gate * See the License for the specific language governing permissions 110Sstevel@tonic-gate * and limitations under the License. 120Sstevel@tonic-gate * 130Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each 140Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 150Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the 160Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying 170Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner] 180Sstevel@tonic-gate * 190Sstevel@tonic-gate * CDDL HEADER END 200Sstevel@tonic-gate */ 215891Sraf 220Sstevel@tonic-gate /* 23*12184SJan.Kryl@Sun.COM * Copyright (c) 1992, 2010, Oracle and/or its affiliates. All rights reserved. 240Sstevel@tonic-gate */ 250Sstevel@tonic-gate 260Sstevel@tonic-gate #include <sys/param.h> 270Sstevel@tonic-gate #include <sys/kmem.h> 280Sstevel@tonic-gate #include <sys/errno.h> 290Sstevel@tonic-gate #include <sys/proc.h> 300Sstevel@tonic-gate #include <sys/disp.h> 310Sstevel@tonic-gate #include <sys/vfs.h> 320Sstevel@tonic-gate #include <sys/vnode.h> 330Sstevel@tonic-gate #include <sys/pathname.h> 340Sstevel@tonic-gate #include <sys/cred.h> 350Sstevel@tonic-gate #include <sys/mount.h> 360Sstevel@tonic-gate #include <sys/cmn_err.h> 370Sstevel@tonic-gate #include <sys/debug.h> 380Sstevel@tonic-gate #include <sys/systm.h> 390Sstevel@tonic-gate #include <sys/dirent.h> 400Sstevel@tonic-gate #include <fs/fs_subr.h> 410Sstevel@tonic-gate #include <sys/fs/autofs.h> 420Sstevel@tonic-gate #include <sys/callb.h> 430Sstevel@tonic-gate #include <sys/sysmacros.h> 440Sstevel@tonic-gate #include <sys/zone.h> 452170Sevanl #include <sys/door.h> 460Sstevel@tonic-gate #include <sys/fs/mntdata.h> 472170Sevanl #include <nfs/mount.h> 482170Sevanl #include <rpc/clnt.h> 492170Sevanl #include <rpcsvc/autofs_prot.h> 502170Sevanl #include <nfs/rnode.h> 512170Sevanl #include <sys/utsname.h> 525891Sraf #include <sys/schedctl.h> 530Sstevel@tonic-gate 540Sstevel@tonic-gate /* 550Sstevel@tonic-gate * Autofs and Zones: 560Sstevel@tonic-gate * 570Sstevel@tonic-gate * Zones are delegated the responsibility of managing their own autofs mounts 580Sstevel@tonic-gate * and maps. Each zone runs its own copy of automountd, with its own timeouts, 590Sstevel@tonic-gate * and other logically "global" parameters. kRPC and virtualization in the 600Sstevel@tonic-gate * loopback transport (tl) will prevent a zone from communicating with another 610Sstevel@tonic-gate * zone's automountd. 620Sstevel@tonic-gate * 630Sstevel@tonic-gate * Each zone has its own "rootfnnode" and associated tree of auto nodes. 640Sstevel@tonic-gate * 650Sstevel@tonic-gate * Each zone also has its own set of "unmounter" kernel threads; these are 660Sstevel@tonic-gate * created and run within the zone's context (ie, they are created via 670Sstevel@tonic-gate * zthread_create()). 680Sstevel@tonic-gate * 690Sstevel@tonic-gate * Cross-zone mount triggers are disallowed. There is a check in 700Sstevel@tonic-gate * auto_trigger_mount() to this effect; EPERM is returned to indicate that the 710Sstevel@tonic-gate * mount is not owned by the caller. 720Sstevel@tonic-gate * 730Sstevel@tonic-gate * autofssys() enables a caller in the global zone to clean up in-kernel (as 740Sstevel@tonic-gate * well as regular) autofs mounts via the unmount_tree() mechanism. This is 750Sstevel@tonic-gate * routinely done when all mounts are removed as part of zone shutdown. 760Sstevel@tonic-gate */ 770Sstevel@tonic-gate #define TYPICALMAXPATHLEN 64 780Sstevel@tonic-gate 790Sstevel@tonic-gate static kmutex_t autofs_nodeid_lock; 800Sstevel@tonic-gate 81*12184SJan.Kryl@Sun.COM /* max number of unmount threads running */ 82*12184SJan.Kryl@Sun.COM static int autofs_unmount_threads = 5; 83*12184SJan.Kryl@Sun.COM static int autofs_unmount_thread_timer = 120; /* in seconds */ 84*12184SJan.Kryl@Sun.COM 850Sstevel@tonic-gate static int auto_perform_link(fnnode_t *, struct linka *, cred_t *); 860Sstevel@tonic-gate static int auto_perform_actions(fninfo_t *, fnnode_t *, 870Sstevel@tonic-gate action_list *, cred_t *); 880Sstevel@tonic-gate static int auto_getmntpnt(vnode_t *, char *, vnode_t **, cred_t *); 890Sstevel@tonic-gate static int auto_lookup_request(fninfo_t *, char *, struct linka *, 903391Ssemery bool_t, bool_t *, cred_t *); 913391Ssemery static int auto_mount_request(fninfo_t *, char *, action_list **, cred_t *, 922170Sevanl bool_t); 932170Sevanl 940Sstevel@tonic-gate /* 950Sstevel@tonic-gate * Clears the MF_INPROG flag, and wakes up those threads sleeping on 960Sstevel@tonic-gate * fn_cv_mount if MF_WAITING is set. 970Sstevel@tonic-gate */ 980Sstevel@tonic-gate void 990Sstevel@tonic-gate auto_unblock_others( 1000Sstevel@tonic-gate fnnode_t *fnp, 1010Sstevel@tonic-gate uint_t operation) /* either MF_INPROG or MF_LOOKUP */ 1020Sstevel@tonic-gate { 1030Sstevel@tonic-gate ASSERT(operation & (MF_INPROG | MF_LOOKUP)); 1040Sstevel@tonic-gate fnp->fn_flags &= ~operation; 1050Sstevel@tonic-gate if (fnp->fn_flags & MF_WAITING) { 1060Sstevel@tonic-gate fnp->fn_flags &= ~MF_WAITING; 1070Sstevel@tonic-gate cv_broadcast(&fnp->fn_cv_mount); 1080Sstevel@tonic-gate } 1090Sstevel@tonic-gate } 1100Sstevel@tonic-gate 1110Sstevel@tonic-gate int 1120Sstevel@tonic-gate auto_wait4mount(fnnode_t *fnp) 1130Sstevel@tonic-gate { 1140Sstevel@tonic-gate int error; 1150Sstevel@tonic-gate k_sigset_t smask; 1160Sstevel@tonic-gate 1170Sstevel@tonic-gate AUTOFS_DPRINT((4, "auto_wait4mount: fnp=%p\n", (void *)fnp)); 1180Sstevel@tonic-gate 1190Sstevel@tonic-gate mutex_enter(&fnp->fn_lock); 1200Sstevel@tonic-gate while (fnp->fn_flags & (MF_INPROG | MF_LOOKUP)) { 1210Sstevel@tonic-gate /* 1220Sstevel@tonic-gate * There is a mount or a lookup in progress. 1230Sstevel@tonic-gate */ 1240Sstevel@tonic-gate fnp->fn_flags |= MF_WAITING; 1250Sstevel@tonic-gate sigintr(&smask, 1); 1260Sstevel@tonic-gate if (!cv_wait_sig(&fnp->fn_cv_mount, &fnp->fn_lock)) { 1270Sstevel@tonic-gate /* 1280Sstevel@tonic-gate * Decided not to wait for operation to 1290Sstevel@tonic-gate * finish after all. 1300Sstevel@tonic-gate */ 1310Sstevel@tonic-gate sigunintr(&smask); 1320Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 1330Sstevel@tonic-gate return (EINTR); 1340Sstevel@tonic-gate } 1350Sstevel@tonic-gate sigunintr(&smask); 1360Sstevel@tonic-gate } 1370Sstevel@tonic-gate error = fnp->fn_error; 1380Sstevel@tonic-gate 1390Sstevel@tonic-gate if (error == EINTR) { 1400Sstevel@tonic-gate /* 1410Sstevel@tonic-gate * The thread doing the mount got interrupted, we need to 1420Sstevel@tonic-gate * try again, by returning EAGAIN. 1430Sstevel@tonic-gate */ 1440Sstevel@tonic-gate error = EAGAIN; 1450Sstevel@tonic-gate } 1460Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 1470Sstevel@tonic-gate 1480Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_wait4mount: fnp=%p error=%d\n", (void *)fnp, 1490Sstevel@tonic-gate error)); 1500Sstevel@tonic-gate return (error); 1510Sstevel@tonic-gate } 1520Sstevel@tonic-gate 1530Sstevel@tonic-gate int 1540Sstevel@tonic-gate auto_lookup_aux(fnnode_t *fnp, char *name, cred_t *cred) 1550Sstevel@tonic-gate { 1560Sstevel@tonic-gate struct fninfo *fnip; 1570Sstevel@tonic-gate struct linka link; 1580Sstevel@tonic-gate bool_t mountreq = FALSE; 1590Sstevel@tonic-gate int error = 0; 1600Sstevel@tonic-gate 1610Sstevel@tonic-gate fnip = vfstofni(fntovn(fnp)->v_vfsp); 1620Sstevel@tonic-gate bzero(&link, sizeof (link)); 1633391Ssemery error = auto_lookup_request(fnip, name, &link, TRUE, &mountreq, cred); 1640Sstevel@tonic-gate if (!error) { 1652170Sevanl if (link.link != NULL || link.link != '\0') { 1660Sstevel@tonic-gate /* 1670Sstevel@tonic-gate * This node should be a symlink 1680Sstevel@tonic-gate */ 1690Sstevel@tonic-gate error = auto_perform_link(fnp, &link, cred); 1700Sstevel@tonic-gate } else if (mountreq) { 1710Sstevel@tonic-gate /* 1720Sstevel@tonic-gate * The automount daemon is requesting a mount, 1730Sstevel@tonic-gate * implying this entry must be a wildcard match and 1740Sstevel@tonic-gate * therefore in need of verification that the entry 1750Sstevel@tonic-gate * exists on the server. 1760Sstevel@tonic-gate */ 1770Sstevel@tonic-gate mutex_enter(&fnp->fn_lock); 1780Sstevel@tonic-gate AUTOFS_BLOCK_OTHERS(fnp, MF_INPROG); 1790Sstevel@tonic-gate fnp->fn_error = 0; 1800Sstevel@tonic-gate 1810Sstevel@tonic-gate /* 1820Sstevel@tonic-gate * Unblock other lookup requests on this node, 1830Sstevel@tonic-gate * this is needed to let the lookup generated by 1840Sstevel@tonic-gate * the mount call to complete. The caveat is 1850Sstevel@tonic-gate * other lookups on this node can also get by, 1860Sstevel@tonic-gate * i.e., another lookup on this node that occurs 1870Sstevel@tonic-gate * while this lookup is attempting the mount 1880Sstevel@tonic-gate * would return a positive result no matter what. 1890Sstevel@tonic-gate * Therefore two lookups on the this node could 1900Sstevel@tonic-gate * potentially get disparate results. 1910Sstevel@tonic-gate */ 1920Sstevel@tonic-gate AUTOFS_UNBLOCK_OTHERS(fnp, MF_LOOKUP); 1930Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 1940Sstevel@tonic-gate /* 1950Sstevel@tonic-gate * auto_new_mount_thread fires up a new thread which 1960Sstevel@tonic-gate * calls automountd finishing up the work 1970Sstevel@tonic-gate */ 1980Sstevel@tonic-gate auto_new_mount_thread(fnp, name, cred); 1990Sstevel@tonic-gate 2000Sstevel@tonic-gate /* 2010Sstevel@tonic-gate * At this point, we are simply another thread 2020Sstevel@tonic-gate * waiting for the mount to complete 2030Sstevel@tonic-gate */ 2040Sstevel@tonic-gate error = auto_wait4mount(fnp); 2050Sstevel@tonic-gate if (error == AUTOFS_SHUTDOWN) 2060Sstevel@tonic-gate error = ENOENT; 2070Sstevel@tonic-gate } 2080Sstevel@tonic-gate } 2090Sstevel@tonic-gate 2102170Sevanl if (link.link) 2112170Sevanl kmem_free(link.link, strlen(link.link) + 1); 2122170Sevanl if (link.dir) 2132170Sevanl kmem_free(link.dir, strlen(link.dir) + 1); 2140Sstevel@tonic-gate mutex_enter(&fnp->fn_lock); 2150Sstevel@tonic-gate fnp->fn_error = error; 2160Sstevel@tonic-gate 2170Sstevel@tonic-gate /* 2180Sstevel@tonic-gate * Notify threads waiting for lookup/mount that 2190Sstevel@tonic-gate * it's done. 2200Sstevel@tonic-gate */ 2210Sstevel@tonic-gate if (mountreq) { 2220Sstevel@tonic-gate AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2230Sstevel@tonic-gate } else { 2240Sstevel@tonic-gate AUTOFS_UNBLOCK_OTHERS(fnp, MF_LOOKUP); 2250Sstevel@tonic-gate } 2260Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 2270Sstevel@tonic-gate return (error); 2280Sstevel@tonic-gate } 2290Sstevel@tonic-gate 2300Sstevel@tonic-gate /* 2310Sstevel@tonic-gate * Starting point for thread to handle mount requests with automountd. 2320Sstevel@tonic-gate * XXX auto_mount_thread() is not suspend-safe within the scope of 2330Sstevel@tonic-gate * the present model defined for cpr to suspend the system. Calls 2340Sstevel@tonic-gate * made by the auto_mount_thread() that have been identified to be unsafe 2350Sstevel@tonic-gate * are (1) RPC client handle setup and client calls to automountd which 2360Sstevel@tonic-gate * can block deep down in the RPC library, (2) kmem_alloc() calls with the 2370Sstevel@tonic-gate * KM_SLEEP flag which can block if memory is low, and (3) VFS_*(), and 2380Sstevel@tonic-gate * lookuppnvp() calls which can result in over the wire calls to servers. 2390Sstevel@tonic-gate * The thread should be completely reevaluated to make it suspend-safe in 2400Sstevel@tonic-gate * case of future updates to the cpr model. 2410Sstevel@tonic-gate */ 2420Sstevel@tonic-gate static void 2430Sstevel@tonic-gate auto_mount_thread(struct autofs_callargs *argsp) 2440Sstevel@tonic-gate { 2452170Sevanl struct fninfo *fnip; 2462170Sevanl fnnode_t *fnp; 2472170Sevanl vnode_t *vp; 2482170Sevanl char *name; 2492170Sevanl size_t namelen; 2502170Sevanl cred_t *cred; 2512170Sevanl action_list *alp = NULL; 2522170Sevanl int error; 2532170Sevanl callb_cpr_t cprinfo; 2542170Sevanl kmutex_t auto_mount_thread_cpr_lock; 2550Sstevel@tonic-gate 2560Sstevel@tonic-gate mutex_init(&auto_mount_thread_cpr_lock, NULL, MUTEX_DEFAULT, NULL); 2572170Sevanl CALLB_CPR_INIT(&cprinfo, &auto_mount_thread_cpr_lock, 2586046Srmesta callb_generic_cpr, "auto_mount_thread"); 2590Sstevel@tonic-gate 2600Sstevel@tonic-gate fnp = argsp->fnc_fnp; 2610Sstevel@tonic-gate vp = fntovn(fnp); 2620Sstevel@tonic-gate fnip = vfstofni(vp->v_vfsp); 2630Sstevel@tonic-gate name = argsp->fnc_name; 2640Sstevel@tonic-gate cred = argsp->fnc_cred; 2650Sstevel@tonic-gate ASSERT(crgetzoneid(argsp->fnc_cred) == fnip->fi_zoneid); 2660Sstevel@tonic-gate 2673391Ssemery error = auto_mount_request(fnip, name, &alp, cred, TRUE); 2680Sstevel@tonic-gate if (!error) 2690Sstevel@tonic-gate error = auto_perform_actions(fnip, fnp, alp, cred); 2700Sstevel@tonic-gate mutex_enter(&fnp->fn_lock); 2710Sstevel@tonic-gate fnp->fn_error = error; 2720Sstevel@tonic-gate 2730Sstevel@tonic-gate /* 2740Sstevel@tonic-gate * Notify threads waiting for mount that 2750Sstevel@tonic-gate * it's done. 2760Sstevel@tonic-gate */ 2770Sstevel@tonic-gate AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2780Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 2790Sstevel@tonic-gate 2800Sstevel@tonic-gate VN_RELE(vp); 2810Sstevel@tonic-gate crfree(argsp->fnc_cred); 2820Sstevel@tonic-gate namelen = strlen(argsp->fnc_name) + 1; 2830Sstevel@tonic-gate kmem_free(argsp->fnc_name, namelen); 2840Sstevel@tonic-gate kmem_free(argsp, sizeof (*argsp)); 2850Sstevel@tonic-gate 2860Sstevel@tonic-gate mutex_enter(&auto_mount_thread_cpr_lock); 2870Sstevel@tonic-gate CALLB_CPR_EXIT(&cprinfo); 2880Sstevel@tonic-gate mutex_destroy(&auto_mount_thread_cpr_lock); 2890Sstevel@tonic-gate zthread_exit(); 2900Sstevel@tonic-gate /* NOTREACHED */ 2910Sstevel@tonic-gate } 2920Sstevel@tonic-gate 2930Sstevel@tonic-gate static int autofs_thr_success = 0; 2940Sstevel@tonic-gate 2950Sstevel@tonic-gate /* 2960Sstevel@tonic-gate * Creates new thread which calls auto_mount_thread which does 2970Sstevel@tonic-gate * the bulk of the work calling automountd, via 'auto_perform_actions'. 2980Sstevel@tonic-gate */ 2990Sstevel@tonic-gate void 3000Sstevel@tonic-gate auto_new_mount_thread(fnnode_t *fnp, char *name, cred_t *cred) 3010Sstevel@tonic-gate { 3020Sstevel@tonic-gate struct autofs_callargs *argsp; 3030Sstevel@tonic-gate 3040Sstevel@tonic-gate argsp = kmem_alloc(sizeof (*argsp), KM_SLEEP); 3050Sstevel@tonic-gate VN_HOLD(fntovn(fnp)); 3060Sstevel@tonic-gate argsp->fnc_fnp = fnp; 3070Sstevel@tonic-gate argsp->fnc_name = kmem_alloc(strlen(name) + 1, KM_SLEEP); 3080Sstevel@tonic-gate (void) strcpy(argsp->fnc_name, name); 3090Sstevel@tonic-gate argsp->fnc_origin = curthread; 3100Sstevel@tonic-gate crhold(cred); 3110Sstevel@tonic-gate argsp->fnc_cred = cred; 3120Sstevel@tonic-gate 3130Sstevel@tonic-gate (void) zthread_create(NULL, 0, auto_mount_thread, argsp, 0, 3140Sstevel@tonic-gate minclsyspri); 3150Sstevel@tonic-gate autofs_thr_success++; 3160Sstevel@tonic-gate } 3170Sstevel@tonic-gate 3186046Srmesta #define DOOR_BUF_ALIGN (1024*1024) 3196046Srmesta #define DOOR_BUF_MULTIPLIER 3 3206046Srmesta #define DOOR_BUF_DEFAULT_SZ (DOOR_BUF_MULTIPLIER * DOOR_BUF_ALIGN) 3216046Srmesta int doorbuf_defsz = DOOR_BUF_DEFAULT_SZ; 3226046Srmesta 3236046Srmesta /*ARGSUSED*/ 3240Sstevel@tonic-gate int 3250Sstevel@tonic-gate auto_calldaemon( 3262170Sevanl zoneid_t zoneid, 3272170Sevanl int which, 3282170Sevanl xdrproc_t xarg_func, 3292170Sevanl void *argsp, 3302170Sevanl xdrproc_t xresp_func, 3312170Sevanl void *resp, 3322170Sevanl int reslen, 3332170Sevanl bool_t hard) /* retry forever? */ 3340Sstevel@tonic-gate { 3356046Srmesta int retry; 3366046Srmesta int error = 0; 3376046Srmesta k_sigset_t smask; 3386046Srmesta door_arg_t door_args; 3396046Srmesta door_handle_t dh; 3406046Srmesta XDR xdrarg; 3416046Srmesta XDR xdrres; 3422170Sevanl struct autofs_globals *fngp = NULL; 3436046Srmesta void *orp = NULL; 3446046Srmesta int orl; 3456060Srmesta int rlen = 0; /* MUST be initialized */ 3462170Sevanl autofs_door_args_t *xdr_argsp; 3476046Srmesta int xdr_len = 0; 3486046Srmesta int printed_not_running_msg = 0; 3494092Sevanl klwp_t *lwp = ttolwp(curthread); 3502170Sevanl 3512170Sevanl /* 3522170Sevanl * We know that the current thread is doing work on 3532170Sevanl * behalf of its own zone, so it's ok to use 3542170Sevanl * curproc->p_zone. 3552170Sevanl */ 3562170Sevanl ASSERT(zoneid == getzoneid()); 3576046Srmesta if (zone_status_get(curproc->p_zone) >= ZONE_IS_SHUTTING_DOWN) { 3582170Sevanl /* 3592170Sevanl * There's no point in trying to talk to 3602170Sevanl * automountd. Plus, zone_shutdown() is 3612170Sevanl * waiting for us. 3622170Sevanl */ 3632170Sevanl return (ECONNREFUSED); 3642170Sevanl } 3650Sstevel@tonic-gate 3664092Sevanl do { 3674092Sevanl retry = 0; 3684092Sevanl mutex_enter(&autofs_minor_lock); 3694092Sevanl fngp = zone_getspecific(autofs_key, curproc->p_zone); 3704092Sevanl mutex_exit(&autofs_minor_lock); 3714092Sevanl if (fngp == NULL) { 3724092Sevanl if (hard) { 3734092Sevanl AUTOFS_DPRINT((5, 3744092Sevanl "auto_calldaemon: "\ 3754092Sevanl "failed to get door handle\n")); 3764092Sevanl if (!printed_not_running_msg) { 3774092Sevanl printed_not_running_msg = 1; 3784092Sevanl zprintf(zoneid, "automountd not "\ 3794092Sevanl "running, retrying\n"); 3804092Sevanl } 3814092Sevanl delay(hz); 3824092Sevanl retry = 1; 3834092Sevanl } else { 3844092Sevanl /* 3854092Sevanl * There is no global data so no door. 3864092Sevanl * There's no point in attempting to talk 3874092Sevanl * to automountd if we can't get the door 3884092Sevanl * handle. 3894092Sevanl */ 3904092Sevanl return (ECONNREFUSED); 3914092Sevanl } 3924092Sevanl } 3934092Sevanl } while (retry); 3944092Sevanl 3954092Sevanl if (printed_not_running_msg) { 3964092Sevanl fngp->fng_printed_not_running_msg = printed_not_running_msg; 3972170Sevanl } 3982170Sevanl 3992170Sevanl ASSERT(fngp != NULL); 4002170Sevanl 4012170Sevanl if (argsp != NULL && (xdr_len = xdr_sizeof(xarg_func, argsp)) == 0) 4022170Sevanl return (EINVAL); 4032170Sevanl xdr_argsp = kmem_zalloc(xdr_len + sizeof (*xdr_argsp), KM_SLEEP); 4042170Sevanl xdr_argsp->xdr_len = xdr_len; 4052170Sevanl xdr_argsp->cmd = which; 4062170Sevanl 4072170Sevanl if (argsp) { 4082170Sevanl xdrmem_create(&xdrarg, (char *)&xdr_argsp->xdr_arg, 4096046Srmesta xdr_argsp->xdr_len, XDR_ENCODE); 4102170Sevanl 4112170Sevanl if (!(*xarg_func)(&xdrarg, argsp)) { 4122170Sevanl kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 4132170Sevanl return (EINVAL); 4142170Sevanl } 4150Sstevel@tonic-gate } 4160Sstevel@tonic-gate 4170Sstevel@tonic-gate /* 4182170Sevanl * We're saving off the original pointer and length due to the 4192170Sevanl * possibility that the results buffer returned by the door 4202170Sevanl * upcall can be different then what we passed in. This is because 4212170Sevanl * the door will allocate new memory if the results buffer passed 4222170Sevanl * in isn't large enough to hold what we need to send back. 4232170Sevanl * In this case we need to free the memory originally allocated 4242170Sevanl * for that buffer. 4250Sstevel@tonic-gate */ 4266046Srmesta if (resp) 4276046Srmesta rlen = xdr_sizeof(xresp_func, resp); 4286046Srmesta orl = (rlen == 0) ? doorbuf_defsz : MAX(rlen, doorbuf_defsz); 4296046Srmesta orp = kmem_zalloc(orl, KM_SLEEP); 4300Sstevel@tonic-gate 4310Sstevel@tonic-gate do { 4322170Sevanl retry = 0; 4332170Sevanl mutex_enter(&fngp->fng_autofs_daemon_lock); 4342170Sevanl dh = fngp->fng_autofs_daemon_dh; 4352170Sevanl if (dh) 4362170Sevanl door_ki_hold(dh); 4372170Sevanl mutex_exit(&fngp->fng_autofs_daemon_lock); 4380Sstevel@tonic-gate 4392170Sevanl if (dh == NULL) { 4406046Srmesta if (orp) 4416046Srmesta kmem_free(orp, orl); 4422170Sevanl kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 4432170Sevanl return (ENOENT); 4442170Sevanl } 4452170Sevanl door_args.data_ptr = (char *)xdr_argsp; 4462170Sevanl door_args.data_size = sizeof (*xdr_argsp) + xdr_argsp->xdr_len; 4472170Sevanl door_args.desc_ptr = NULL; 4482170Sevanl door_args.desc_num = 0; 4496046Srmesta door_args.rbuf = orp ? (char *)orp : NULL; 4506046Srmesta door_args.rsize = orl; 4510Sstevel@tonic-gate 4522170Sevanl sigintr(&smask, 1); 4536997Sjwadams error = 4546997Sjwadams door_ki_upcall_limited(dh, &door_args, NULL, SIZE_MAX, 0); 4550Sstevel@tonic-gate sigunintr(&smask); 4560Sstevel@tonic-gate 4572170Sevanl door_ki_rele(dh); 4580Sstevel@tonic-gate 4596046Srmesta /* 4606046Srmesta * Handle daemon errors 4616046Srmesta */ 4622170Sevanl if (!error) { 4636046Srmesta /* 4646046Srmesta * Upcall successful. Let's check for soft errors 4656046Srmesta * from the daemon. We only recover from overflow 4666046Srmesta * type scenarios. Any other errors, we return to 4676046Srmesta * the caller. 4686046Srmesta */ 4692170Sevanl autofs_door_res_t *adr = 4706046Srmesta (autofs_door_res_t *)door_args.rbuf; 4716046Srmesta 4726046Srmesta if (door_args.rbuf != NULL) { 4736046Srmesta int nl; 4746046Srmesta 4756046Srmesta switch (error = adr->res_status) { 4766046Srmesta case 0: /* no error; continue */ 4776046Srmesta break; 4786046Srmesta 4796046Srmesta case EOVERFLOW: 4806046Srmesta /* 4816046Srmesta * orig landing buf not big enough. 4826046Srmesta * xdr_len in XDR_BYTES_PER_UNIT 4836046Srmesta */ 4846046Srmesta if ((nl = adr->xdr_len) > 0 && 4856046Srmesta (btopr(nl) < freemem/64)) { 4866046Srmesta if (orp) 4876046Srmesta kmem_free(orp, orl); 4886046Srmesta orp = kmem_zalloc(nl, KM_SLEEP); 4896046Srmesta orl = nl; 4906046Srmesta retry = 1; 4916046Srmesta break; 4926046Srmesta } 4936046Srmesta /*FALLTHROUGH*/ 4946046Srmesta 4956046Srmesta default: 4966046Srmesta kmem_free(xdr_argsp, 4976046Srmesta xdr_len + sizeof (*xdr_argsp)); 4986046Srmesta if (orp) 4996046Srmesta kmem_free(orp, orl); 5006046Srmesta return (error); 5016046Srmesta } 5022170Sevanl } 5032170Sevanl continue; 5042170Sevanl } 5056046Srmesta 5066046Srmesta /* 5076046Srmesta * no daemon errors; now process door/comm errors (if any) 5086046Srmesta */ 5092170Sevanl switch (error) { 5102170Sevanl case EINTR: 5110Sstevel@tonic-gate /* 5122170Sevanl * interrupts should be handled properly by the 5134092Sevanl * door upcall. If the door doesn't handle the 5144092Sevanl * interupt completely then we need to bail out. 5154092Sevanl */ 5164092Sevanl if (lwp && (ISSIG(curthread, 5174092Sevanl JUSTLOOKING) || MUSTRETURN(curproc, curthread))) { 5184092Sevanl if (ISSIG(curthread, FORREAL) || 5194092Sevanl lwp->lwp_sysabort || 5204092Sevanl MUSTRETURN(curproc, curthread)) { 5214092Sevanl lwp->lwp_sysabort = 0; 5224092Sevanl return (EINTR); 5234092Sevanl } 5244092Sevanl } 5254092Sevanl /* 5262170Sevanl * We may have gotten EINTR for other reasons 5272170Sevanl * like the door being revoked on us. Instead 5282170Sevanl * of trying to extract this out of the door 5292170Sevanl * handle, sleep and try again, if still 5302170Sevanl * revoked we will get EBADF next time 5312170Sevanl * through. 5325891Sraf * 5335891Sraf * If we have a pending cancellation and we don't 5345891Sraf * have cancellation disabled, we will get EINTR 5355891Sraf * forever, no matter how many times we retry, 5365891Sraf * so just get out now if this is the case. 5370Sstevel@tonic-gate */ 5385891Sraf if (schedctl_cancel_pending()) 5395891Sraf break; 5404092Sevanl /* FALLTHROUGH */ 5412170Sevanl case EAGAIN: /* process may be forking */ 5422170Sevanl /* 5432170Sevanl * Back off for a bit 5442170Sevanl */ 5452170Sevanl delay(hz); 5462170Sevanl retry = 1; 5472170Sevanl break; 5482170Sevanl case EBADF: /* Invalid door */ 5492170Sevanl case EINVAL: /* Not a door, wrong target */ 5500Sstevel@tonic-gate /* 5512170Sevanl * A fatal door error, if our failing door 5522170Sevanl * handle is the current door handle, clean 5532170Sevanl * up our state. 5540Sstevel@tonic-gate */ 5552170Sevanl mutex_enter(&fngp->fng_autofs_daemon_lock); 5562170Sevanl if (dh == fngp->fng_autofs_daemon_dh) { 5572170Sevanl door_ki_rele(fngp->fng_autofs_daemon_dh); 5582170Sevanl fngp->fng_autofs_daemon_dh = NULL; 5590Sstevel@tonic-gate } 5602170Sevanl mutex_exit(&fngp->fng_autofs_daemon_lock); 5616046Srmesta AUTOFS_DPRINT((5, "auto_calldaemon error=%d\n", error)); 5622170Sevanl if (hard) { 5632170Sevanl if (!fngp->fng_printed_not_running_msg) { 5646046Srmesta fngp->fng_printed_not_running_msg = 1; 5656046Srmesta zprintf(zoneid, "automountd not " 5666046Srmesta "running, retrying\n"); 5672170Sevanl } 5682170Sevanl delay(hz); 5692170Sevanl retry = 1; 5702170Sevanl break; 5712170Sevanl } else { 5722170Sevanl error = ECONNREFUSED; 5732170Sevanl kmem_free(xdr_argsp, 5746046Srmesta xdr_len + sizeof (*xdr_argsp)); 5756046Srmesta if (orp) 5766046Srmesta kmem_free(orp, orl); 5772170Sevanl return (error); 5782170Sevanl } 5792170Sevanl default: /* Unknown must be fatal */ 5800Sstevel@tonic-gate error = ENOENT; 5812170Sevanl kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 5826046Srmesta if (orp) 5836046Srmesta kmem_free(orp, orl); 5842170Sevanl return (error); 5850Sstevel@tonic-gate } 5862170Sevanl } while (retry); 5870Sstevel@tonic-gate 5882170Sevanl if (fngp->fng_printed_not_running_msg == 1) { 5892170Sevanl fngp->fng_printed_not_running_msg = 0; 5902170Sevanl zprintf(zoneid, "automountd OK\n"); 5910Sstevel@tonic-gate } 5920Sstevel@tonic-gate 5936046Srmesta if (orp && orl) { 5942170Sevanl autofs_door_res_t *door_resp; 5956046Srmesta door_resp = (autofs_door_res_t *)door_args.rbuf; 5966046Srmesta 5976046Srmesta if ((void *)door_args.rbuf != orp) 5986046Srmesta kmem_free(orp, orl); 5996046Srmesta 6002170Sevanl xdrmem_create(&xdrres, (char *)&door_resp->xdr_res, 6016046Srmesta door_resp->xdr_len, XDR_DECODE); 6026046Srmesta 6032170Sevanl if (!((*xresp_func)(&xdrres, resp))) 6042170Sevanl error = EINVAL; 6052170Sevanl kmem_free(door_args.rbuf, door_args.rsize); 6062170Sevanl } 6072170Sevanl kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 6080Sstevel@tonic-gate return (error); 6090Sstevel@tonic-gate } 6100Sstevel@tonic-gate 6110Sstevel@tonic-gate static int 612*12184SJan.Kryl@Sun.COM auto_null_request(zoneid_t zoneid, bool_t hard) 6130Sstevel@tonic-gate { 6140Sstevel@tonic-gate int error; 6150Sstevel@tonic-gate 6160Sstevel@tonic-gate AUTOFS_DPRINT((4, "\tauto_null_request\n")); 6170Sstevel@tonic-gate 618*12184SJan.Kryl@Sun.COM error = auto_calldaemon(zoneid, NULLPROC, 6196046Srmesta xdr_void, NULL, xdr_void, NULL, 0, hard); 6200Sstevel@tonic-gate 6210Sstevel@tonic-gate AUTOFS_DPRINT((5, "\tauto_null_request: error=%d\n", error)); 6220Sstevel@tonic-gate return (error); 6230Sstevel@tonic-gate } 6240Sstevel@tonic-gate 6250Sstevel@tonic-gate static int 6260Sstevel@tonic-gate auto_lookup_request( 6270Sstevel@tonic-gate fninfo_t *fnip, 6280Sstevel@tonic-gate char *key, 6290Sstevel@tonic-gate struct linka *lnp, 6300Sstevel@tonic-gate bool_t hard, 6313391Ssemery bool_t *mountreq, 6323391Ssemery cred_t *cred) 6330Sstevel@tonic-gate { 6342170Sevanl int error; 6352170Sevanl struct autofs_globals *fngp; 6366046Srmesta struct autofs_lookupargs reqst; 6372170Sevanl autofs_lookupres *resp; 6382170Sevanl struct linka *p; 6390Sstevel@tonic-gate 6402170Sevanl 6412170Sevanl AUTOFS_DPRINT((4, "auto_lookup_equest: path=%s name=%s\n", 6420Sstevel@tonic-gate fnip->fi_path, key)); 6430Sstevel@tonic-gate 6440Sstevel@tonic-gate fngp = vntofn(fnip->fi_rootvp)->fn_globals; 6452170Sevanl 6462170Sevanl reqst.map = fnip->fi_map; 6472170Sevanl reqst.path = fnip->fi_path; 6480Sstevel@tonic-gate 6490Sstevel@tonic-gate if (fnip->fi_flags & MF_DIRECT) 6502170Sevanl reqst.name = fnip->fi_key; 6510Sstevel@tonic-gate else 6522170Sevanl reqst.name = key; 6532170Sevanl AUTOFS_DPRINT((4, "auto_lookup_request: using key=%s\n", reqst.name)); 6540Sstevel@tonic-gate 6552170Sevanl reqst.subdir = fnip->fi_subdir; 6562170Sevanl reqst.opts = fnip->fi_opts; 6572170Sevanl reqst.isdirect = fnip->fi_flags & MF_DIRECT ? TRUE : FALSE; 6583391Ssemery reqst.uid = crgetuid(cred); 6592170Sevanl 6602170Sevanl resp = kmem_zalloc(sizeof (*resp), KM_SLEEP); 6610Sstevel@tonic-gate 6626046Srmesta error = auto_calldaemon(fngp->fng_zoneid, AUTOFS_LOOKUP, 6636046Srmesta xdr_autofs_lookupargs, &reqst, xdr_autofs_lookupres, 6646046Srmesta (void *)resp, sizeof (autofs_lookupres), hard); 6652170Sevanl 6662170Sevanl if (error) { 6672170Sevanl xdr_free(xdr_autofs_lookupres, (char *)resp); 6682170Sevanl kmem_free(resp, sizeof (*resp)); 6692170Sevanl return (error); 6702170Sevanl } 6712170Sevanl 6720Sstevel@tonic-gate if (!error) { 6732170Sevanl fngp->fng_verbose = resp->lu_verbose; 6742170Sevanl switch (resp->lu_res) { 6750Sstevel@tonic-gate case AUTOFS_OK: 6762170Sevanl switch (resp->lu_type.action) { 6770Sstevel@tonic-gate case AUTOFS_MOUNT_RQ: 6780Sstevel@tonic-gate lnp->link = NULL; 6790Sstevel@tonic-gate lnp->dir = NULL; 6800Sstevel@tonic-gate *mountreq = TRUE; 6810Sstevel@tonic-gate break; 6826046Srmesta 6830Sstevel@tonic-gate case AUTOFS_LINK_RQ: 6846046Srmesta p = &resp->lu_type.lookup_result_type_u.lt_linka; 6850Sstevel@tonic-gate lnp->dir = kmem_alloc(strlen(p->dir) + 1, 6866046Srmesta KM_SLEEP); 6870Sstevel@tonic-gate (void) strcpy(lnp->dir, p->dir); 6880Sstevel@tonic-gate lnp->link = kmem_alloc(strlen(p->link) + 1, 6896046Srmesta KM_SLEEP); 6900Sstevel@tonic-gate (void) strcpy(lnp->link, p->link); 6910Sstevel@tonic-gate break; 6926046Srmesta 6930Sstevel@tonic-gate case AUTOFS_NONE: 6940Sstevel@tonic-gate lnp->link = NULL; 6950Sstevel@tonic-gate lnp->dir = NULL; 6960Sstevel@tonic-gate break; 6976046Srmesta 6980Sstevel@tonic-gate default: 6996046Srmesta auto_log(fngp->fng_verbose, fngp->fng_zoneid, 7006046Srmesta CE_WARN, "auto_lookup_request: bad action " 7016046Srmesta "type %d", resp->lu_res); 7020Sstevel@tonic-gate error = ENOENT; 7030Sstevel@tonic-gate } 7040Sstevel@tonic-gate break; 7056046Srmesta 7060Sstevel@tonic-gate case AUTOFS_NOENT: 7070Sstevel@tonic-gate error = ENOENT; 7080Sstevel@tonic-gate break; 7096046Srmesta 7100Sstevel@tonic-gate default: 7110Sstevel@tonic-gate error = ENOENT; 7122170Sevanl auto_log(fngp->fng_verbose, fngp->fng_zoneid, CE_WARN, 7130Sstevel@tonic-gate "auto_lookup_request: unknown result: %d", 7142170Sevanl resp->lu_res); 7150Sstevel@tonic-gate break; 7160Sstevel@tonic-gate } 7170Sstevel@tonic-gate } 7180Sstevel@tonic-gate done: 7192170Sevanl xdr_free(xdr_autofs_lookupres, (char *)resp); 7202170Sevanl kmem_free(resp, sizeof (*resp)); 7210Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_lookup_request: path=%s name=%s error=%d\n", 7220Sstevel@tonic-gate fnip->fi_path, key, error)); 7230Sstevel@tonic-gate return (error); 7240Sstevel@tonic-gate } 7250Sstevel@tonic-gate 7260Sstevel@tonic-gate static int 7270Sstevel@tonic-gate auto_mount_request( 7280Sstevel@tonic-gate fninfo_t *fnip, 7290Sstevel@tonic-gate char *key, 7300Sstevel@tonic-gate action_list **alpp, 7313391Ssemery cred_t *cred, 7320Sstevel@tonic-gate bool_t hard) 7330Sstevel@tonic-gate { 7342170Sevanl int error; 7352170Sevanl struct autofs_globals *fngp; 7362170Sevanl autofs_lookupargs reqst; 7372170Sevanl autofs_mountres *xdrres = NULL; 7380Sstevel@tonic-gate 7390Sstevel@tonic-gate AUTOFS_DPRINT((4, "auto_mount_request: path=%s name=%s\n", 7400Sstevel@tonic-gate fnip->fi_path, key)); 7410Sstevel@tonic-gate 7420Sstevel@tonic-gate fngp = vntofn(fnip->fi_rootvp)->fn_globals; 7432170Sevanl reqst.map = fnip->fi_map; 7442170Sevanl reqst.path = fnip->fi_path; 7450Sstevel@tonic-gate 7460Sstevel@tonic-gate if (fnip->fi_flags & MF_DIRECT) 7472170Sevanl reqst.name = fnip->fi_key; 7480Sstevel@tonic-gate else 7492170Sevanl reqst.name = key; 7502170Sevanl 7512170Sevanl AUTOFS_DPRINT((4, "auto_mount_request: using key=%s\n", reqst.name)); 7520Sstevel@tonic-gate 7532170Sevanl reqst.subdir = fnip->fi_subdir; 7542170Sevanl reqst.opts = fnip->fi_opts; 7552170Sevanl reqst.isdirect = fnip->fi_flags & MF_DIRECT ? TRUE : FALSE; 7563391Ssemery reqst.uid = crgetuid(cred); 7572170Sevanl 7582170Sevanl xdrres = kmem_zalloc(sizeof (*xdrres), KM_SLEEP); 7590Sstevel@tonic-gate 7606046Srmesta error = auto_calldaemon(fngp->fng_zoneid, AUTOFS_MNTINFO, 7616046Srmesta xdr_autofs_lookupargs, &reqst, xdr_autofs_mountres, 7626046Srmesta (void *)xdrres, sizeof (autofs_mountres), hard); 7632170Sevanl 7640Sstevel@tonic-gate if (!error) { 7652170Sevanl fngp->fng_verbose = xdrres->mr_verbose; 7662170Sevanl switch (xdrres->mr_type.status) { 7670Sstevel@tonic-gate case AUTOFS_ACTION: 7680Sstevel@tonic-gate error = 0; 7690Sstevel@tonic-gate /* 7700Sstevel@tonic-gate * Save the action list since it is used by 7710Sstevel@tonic-gate * the caller. We NULL the action list pointer 7720Sstevel@tonic-gate * in 'result' so that xdr_free() will not free 7730Sstevel@tonic-gate * the list. 7740Sstevel@tonic-gate */ 7752170Sevanl *alpp = xdrres->mr_type.mount_result_type_u.list; 7762170Sevanl xdrres->mr_type.mount_result_type_u.list = NULL; 7770Sstevel@tonic-gate break; 7780Sstevel@tonic-gate case AUTOFS_DONE: 7792170Sevanl error = xdrres->mr_type.mount_result_type_u.error; 7800Sstevel@tonic-gate break; 7810Sstevel@tonic-gate default: 7820Sstevel@tonic-gate error = ENOENT; 7832170Sevanl auto_log(fngp->fng_verbose, fngp->fng_zoneid, CE_WARN, 7840Sstevel@tonic-gate "auto_mount_request: unknown status %d", 7852170Sevanl xdrres->mr_type.status); 7860Sstevel@tonic-gate break; 7870Sstevel@tonic-gate } 7880Sstevel@tonic-gate } 7890Sstevel@tonic-gate 7902170Sevanl xdr_free(xdr_autofs_mountres, (char *)xdrres); 7912170Sevanl kmem_free(xdrres, sizeof (*xdrres)); 7922170Sevanl 7930Sstevel@tonic-gate 7940Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_mount_request: path=%s name=%s error=%d\n", 7950Sstevel@tonic-gate fnip->fi_path, key, error)); 7960Sstevel@tonic-gate return (error); 7970Sstevel@tonic-gate } 7980Sstevel@tonic-gate 7990Sstevel@tonic-gate 8000Sstevel@tonic-gate static int 8010Sstevel@tonic-gate auto_send_unmount_request( 8020Sstevel@tonic-gate fninfo_t *fnip, 8030Sstevel@tonic-gate umntrequest *ul, 8040Sstevel@tonic-gate bool_t hard) 8050Sstevel@tonic-gate { 8062170Sevanl int error; 8072170Sevanl umntres xdrres; 8082170Sevanl 8092170Sevanl struct autofs_globals *fngp = vntofn(fnip->fi_rootvp)->fn_globals; 8100Sstevel@tonic-gate 8110Sstevel@tonic-gate AUTOFS_DPRINT((4, "\tauto_send_unmount_request: fstype=%s " 8126046Srmesta " mntpnt=%s\n", ul->fstype, ul->mntpnt)); 8130Sstevel@tonic-gate 8142170Sevanl bzero(&xdrres, sizeof (umntres)); 8156046Srmesta error = auto_calldaemon(fngp->fng_zoneid, AUTOFS_UNMOUNT, 8166046Srmesta xdr_umntrequest, (void *)ul, xdr_umntres, (void *)&xdrres, 8176046Srmesta sizeof (umntres), hard); 8180Sstevel@tonic-gate 8192395Sevanl if (!error) 8202395Sevanl error = xdrres.status; 8212395Sevanl 8220Sstevel@tonic-gate AUTOFS_DPRINT((5, "\tauto_send_unmount_request: error=%d\n", error)); 8230Sstevel@tonic-gate 8240Sstevel@tonic-gate return (error); 8250Sstevel@tonic-gate } 8260Sstevel@tonic-gate 8270Sstevel@tonic-gate static int 8280Sstevel@tonic-gate auto_perform_link(fnnode_t *fnp, struct linka *linkp, cred_t *cred) 8290Sstevel@tonic-gate { 8300Sstevel@tonic-gate vnode_t *vp; 8310Sstevel@tonic-gate size_t len; 8320Sstevel@tonic-gate char *tmp; 8330Sstevel@tonic-gate 8340Sstevel@tonic-gate AUTOFS_DPRINT((3, "auto_perform_link: fnp=%p dir=%s link=%s\n", 8350Sstevel@tonic-gate (void *)fnp, linkp->dir, linkp->link)); 8360Sstevel@tonic-gate 8370Sstevel@tonic-gate len = strlen(linkp->link) + 1; /* include '\0' */ 8380Sstevel@tonic-gate tmp = kmem_zalloc(len, KM_SLEEP); 8390Sstevel@tonic-gate (void) kcopy(linkp->link, tmp, len); 8400Sstevel@tonic-gate mutex_enter(&fnp->fn_lock); 8410Sstevel@tonic-gate fnp->fn_symlink = tmp; 8420Sstevel@tonic-gate fnp->fn_symlinklen = (uint_t)len; 8430Sstevel@tonic-gate fnp->fn_flags |= MF_THISUID_MATCH_RQD; 8440Sstevel@tonic-gate crhold(cred); 8450Sstevel@tonic-gate fnp->fn_cred = cred; 8460Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 8470Sstevel@tonic-gate 8480Sstevel@tonic-gate vp = fntovn(fnp); 8490Sstevel@tonic-gate vp->v_type = VLNK; 8500Sstevel@tonic-gate 8510Sstevel@tonic-gate return (0); 8520Sstevel@tonic-gate } 8530Sstevel@tonic-gate 8544068Sdm120769 static void 8554068Sdm120769 auto_free_autofs_args(struct mounta *m) 8564068Sdm120769 { 8574068Sdm120769 autofs_args *aargs = (autofs_args *)m->dataptr; 8584068Sdm120769 8594068Sdm120769 if (aargs->addr.buf) 8604068Sdm120769 kmem_free(aargs->addr.buf, aargs->addr.len); 8614068Sdm120769 if (aargs->path) 8624068Sdm120769 kmem_free(aargs->path, strlen(aargs->path) + 1); 8634068Sdm120769 if (aargs->opts) 8644068Sdm120769 kmem_free(aargs->opts, strlen(aargs->opts) + 1); 8654068Sdm120769 if (aargs->map) 8664068Sdm120769 kmem_free(aargs->map, strlen(aargs->map) + 1); 8674068Sdm120769 if (aargs->subdir) 8684068Sdm120769 kmem_free(aargs->subdir, strlen(aargs->subdir) + 1); 8694068Sdm120769 if (aargs->key) 8704068Sdm120769 kmem_free(aargs->key, strlen(aargs->key) + 1); 8714068Sdm120769 kmem_free(aargs, sizeof (*aargs)); 8724068Sdm120769 } 8734068Sdm120769 8744068Sdm120769 static void 8754068Sdm120769 auto_free_action_list(action_list *alp) 8764068Sdm120769 { 8774068Sdm120769 struct mounta *m; 8784068Sdm120769 action_list *lastalp; 8794068Sdm120769 char *fstype; 8804068Sdm120769 8814068Sdm120769 m = &alp->action.action_list_entry_u.mounta; 8824068Sdm120769 while (alp != NULL) { 8834068Sdm120769 fstype = alp->action.action_list_entry_u.mounta.fstype; 8844068Sdm120769 m = &alp->action.action_list_entry_u.mounta; 8854068Sdm120769 if (m->dataptr) { 8864068Sdm120769 if (strcmp(fstype, "autofs") == 0) { 8874068Sdm120769 auto_free_autofs_args(m); 8884068Sdm120769 } 8894068Sdm120769 } 8904068Sdm120769 if (m->spec) 8914068Sdm120769 kmem_free(m->spec, strlen(m->spec) + 1); 8924068Sdm120769 if (m->dir) 8934068Sdm120769 kmem_free(m->dir, strlen(m->dir) + 1); 8944068Sdm120769 if (m->fstype) 8954068Sdm120769 kmem_free(m->fstype, strlen(m->fstype) + 1); 8964068Sdm120769 if (m->optptr) 8974068Sdm120769 kmem_free(m->optptr, m->optlen); 8984068Sdm120769 lastalp = alp; 8994068Sdm120769 alp = alp->next; 9004068Sdm120769 kmem_free(lastalp, sizeof (*lastalp)); 9014068Sdm120769 } 9024068Sdm120769 } 9034068Sdm120769 9040Sstevel@tonic-gate static boolean_t 9052170Sevanl auto_invalid_autofs(fninfo_t *dfnip, fnnode_t *dfnp, action_list *p) 9060Sstevel@tonic-gate { 9070Sstevel@tonic-gate struct mounta *m; 9080Sstevel@tonic-gate struct autofs_args *argsp; 9090Sstevel@tonic-gate vnode_t *dvp; 9100Sstevel@tonic-gate char buff[AUTOFS_MAXPATHLEN]; 9110Sstevel@tonic-gate size_t len; 9120Sstevel@tonic-gate struct autofs_globals *fngp; 9134068Sdm120769 9140Sstevel@tonic-gate fngp = dfnp->fn_globals; 9150Sstevel@tonic-gate dvp = fntovn(dfnp); 9162170Sevanl 9170Sstevel@tonic-gate m = &p->action.action_list_entry_u.mounta; 9180Sstevel@tonic-gate /* 9190Sstevel@tonic-gate * Make sure we aren't geting passed NULL values or a "dir" that 9200Sstevel@tonic-gate * isn't "." and doesn't begin with "./". 9210Sstevel@tonic-gate * 9220Sstevel@tonic-gate * We also only want to perform autofs mounts, so make sure 9230Sstevel@tonic-gate * no-one is trying to trick us into doing anything else. 9240Sstevel@tonic-gate */ 9254068Sdm120769 if (m->spec == NULL || m->dir == NULL || m->dir[0] != '.' || 9260Sstevel@tonic-gate (m->dir[1] != '/' && m->dir[1] != '\0') || 9274068Sdm120769 m->fstype == NULL || strcmp(m->fstype, "autofs") != 0 || 9284068Sdm120769 m->dataptr == NULL || m->datalen != sizeof (struct autofs_args) || 9290Sstevel@tonic-gate m->optptr == NULL) 9300Sstevel@tonic-gate return (B_TRUE); 9310Sstevel@tonic-gate /* 9320Sstevel@tonic-gate * We also don't like ".."s in the pathname. Symlinks are 9330Sstevel@tonic-gate * handled by the fact that we'll use NOFOLLOW when we do 9340Sstevel@tonic-gate * lookup()s. 9350Sstevel@tonic-gate */ 9360Sstevel@tonic-gate if (strstr(m->dir, "/../") != NULL || 9370Sstevel@tonic-gate (len = strlen(m->dir)) > sizeof ("/..") - 1 && 9380Sstevel@tonic-gate m->dir[len] == '.' && m->dir[len - 1] == '.' && 9390Sstevel@tonic-gate m->dir[len - 2] == '/') 9400Sstevel@tonic-gate return (B_TRUE); 9410Sstevel@tonic-gate argsp = (struct autofs_args *)m->dataptr; 9420Sstevel@tonic-gate /* 9430Sstevel@tonic-gate * We don't want NULL values here either. 9440Sstevel@tonic-gate */ 9450Sstevel@tonic-gate if (argsp->addr.buf == NULL || argsp->path == NULL || 9460Sstevel@tonic-gate argsp->opts == NULL || argsp->map == NULL || argsp->subdir == NULL) 9470Sstevel@tonic-gate return (B_TRUE); 9480Sstevel@tonic-gate /* 9490Sstevel@tonic-gate * We know what the claimed pathname *should* look like: 9500Sstevel@tonic-gate * 9510Sstevel@tonic-gate * If the parent (dfnp) is a mount point (VROOT), then 9520Sstevel@tonic-gate * the path should be (dfnip->fi_path + m->dir). 9530Sstevel@tonic-gate * 9540Sstevel@tonic-gate * Else, we know we're only two levels deep, so we use 9550Sstevel@tonic-gate * (dfnip->fi_path + dfnp->fn_name + m->dir). 9560Sstevel@tonic-gate * 9570Sstevel@tonic-gate * Furthermore, "." only makes sense if dfnp is a 9580Sstevel@tonic-gate * trigger node. 9590Sstevel@tonic-gate * 9600Sstevel@tonic-gate * At this point it seems like the passed-in path is 9610Sstevel@tonic-gate * redundant. 9620Sstevel@tonic-gate */ 9630Sstevel@tonic-gate if (dvp->v_flag & VROOT) { 9640Sstevel@tonic-gate if (m->dir[1] == '\0' && !(dfnp->fn_flags & MF_TRIGGER)) 9650Sstevel@tonic-gate return (B_TRUE); 9660Sstevel@tonic-gate (void) snprintf(buff, sizeof (buff), "%s%s", 9670Sstevel@tonic-gate dfnip->fi_path, m->dir + 1); 9680Sstevel@tonic-gate } else { 9690Sstevel@tonic-gate (void) snprintf(buff, sizeof (buff), "%s/%s%s", 9700Sstevel@tonic-gate dfnip->fi_path, dfnp->fn_name, m->dir + 1); 9710Sstevel@tonic-gate } 9720Sstevel@tonic-gate if (strcmp(argsp->path, buff) != 0) { 9732170Sevanl auto_log(fngp->fng_verbose, fngp->fng_zoneid, 9742170Sevanl CE_WARN, "autofs: expected path of '%s', " 9750Sstevel@tonic-gate "got '%s' instead.", buff, argsp->path); 9760Sstevel@tonic-gate return (B_TRUE); 9770Sstevel@tonic-gate } 9780Sstevel@tonic-gate return (B_FALSE); /* looks OK */ 9790Sstevel@tonic-gate } 9800Sstevel@tonic-gate 9812170Sevanl /* 9822170Sevanl * auto_invalid_action will validate the action_list received. If all is good 9834068Sdm120769 * this function returns FALSE, if there is a problem it returns TRUE. 9842170Sevanl */ 9852170Sevanl static boolean_t 9862170Sevanl auto_invalid_action(fninfo_t *dfnip, fnnode_t *dfnp, action_list *alistpp) 9872170Sevanl { 9882170Sevanl 9892170Sevanl /* 9902170Sevanl * Before we go any further, this better be a mount request. 9912170Sevanl */ 9922170Sevanl if (alistpp->action.action != AUTOFS_MOUNT_RQ) 9932170Sevanl return (B_TRUE); 9944068Sdm120769 return (auto_invalid_autofs(dfnip, dfnp, alistpp)); 9952170Sevanl 9962170Sevanl } 9972170Sevanl 9980Sstevel@tonic-gate static int 9990Sstevel@tonic-gate auto_perform_actions( 10000Sstevel@tonic-gate fninfo_t *dfnip, 10010Sstevel@tonic-gate fnnode_t *dfnp, 10020Sstevel@tonic-gate action_list *alp, 10030Sstevel@tonic-gate cred_t *cred) /* Credentials of the caller */ 10040Sstevel@tonic-gate { 10052170Sevanl 10060Sstevel@tonic-gate action_list *p; 10072170Sevanl struct mounta *m, margs; 10082170Sevanl struct autofs_args *argsp; 10092170Sevanl int error, success = 0; 10102170Sevanl vnode_t *mvp, *dvp, *newvp; 10112170Sevanl fnnode_t *newfnp, *mfnp; 10122170Sevanl int auto_mount = 0; 10132170Sevanl int save_triggers = 0; 10142170Sevanl int update_times = 0; 10152170Sevanl char *mntpnt; 10162170Sevanl char buff[AUTOFS_MAXPATHLEN]; 10172170Sevanl timestruc_t now; 10182170Sevanl struct autofs_globals *fngp; 10192170Sevanl cred_t *zcred; 10200Sstevel@tonic-gate 10216046Srmesta AUTOFS_DPRINT((4, "auto_perform_actions: alp=%p\n", (void *)alp)); 10220Sstevel@tonic-gate 10230Sstevel@tonic-gate fngp = dfnp->fn_globals; 10240Sstevel@tonic-gate dvp = fntovn(dfnp); 10250Sstevel@tonic-gate 10260Sstevel@tonic-gate /* 10270Sstevel@tonic-gate * As automountd running in a zone may be compromised, and this may be 10280Sstevel@tonic-gate * an attack, we can't trust everything passed in by automountd, and we 10290Sstevel@tonic-gate * need to do argument verification. We'll issue a warning and drop 10300Sstevel@tonic-gate * the request if it doesn't seem right. 10310Sstevel@tonic-gate */ 10322170Sevanl 10330Sstevel@tonic-gate for (p = alp; p != NULL; p = p->next) { 10340Sstevel@tonic-gate if (auto_invalid_action(dfnip, dfnp, p)) { 10350Sstevel@tonic-gate /* 10360Sstevel@tonic-gate * This warning should be sent to the global zone, 10370Sstevel@tonic-gate * since presumably the zone administrator is the same 10380Sstevel@tonic-gate * as the attacker. 10390Sstevel@tonic-gate */ 10400Sstevel@tonic-gate cmn_err(CE_WARN, "autofs: invalid action list received " 10410Sstevel@tonic-gate "by automountd in zone %s.", 10420Sstevel@tonic-gate curproc->p_zone->zone_name); 10430Sstevel@tonic-gate /* 10440Sstevel@tonic-gate * This conversation is over. 10450Sstevel@tonic-gate */ 10460Sstevel@tonic-gate xdr_free(xdr_action_list, (char *)alp); 10470Sstevel@tonic-gate return (EINVAL); 10480Sstevel@tonic-gate } 10490Sstevel@tonic-gate } 10500Sstevel@tonic-gate 10510Sstevel@tonic-gate zcred = zone_get_kcred(getzoneid()); 10520Sstevel@tonic-gate ASSERT(zcred != NULL); 10530Sstevel@tonic-gate 10540Sstevel@tonic-gate if (vn_mountedvfs(dvp) != NULL) { 10550Sstevel@tonic-gate /* 10560Sstevel@tonic-gate * The daemon successfully mounted a filesystem 10570Sstevel@tonic-gate * on the AUTOFS root node. 10580Sstevel@tonic-gate */ 10590Sstevel@tonic-gate mutex_enter(&dfnp->fn_lock); 10600Sstevel@tonic-gate dfnp->fn_flags |= MF_MOUNTPOINT; 10610Sstevel@tonic-gate ASSERT(dfnp->fn_dirents == NULL); 10620Sstevel@tonic-gate mutex_exit(&dfnp->fn_lock); 10630Sstevel@tonic-gate success++; 10640Sstevel@tonic-gate } else { 10650Sstevel@tonic-gate /* 10660Sstevel@tonic-gate * Clear MF_MOUNTPOINT. 10670Sstevel@tonic-gate */ 10680Sstevel@tonic-gate mutex_enter(&dfnp->fn_lock); 10690Sstevel@tonic-gate if (dfnp->fn_flags & MF_MOUNTPOINT) { 10700Sstevel@tonic-gate AUTOFS_DPRINT((10, "autofs: clearing mountpoint " 10716046Srmesta "flag on %s.", dfnp->fn_name)); 10720Sstevel@tonic-gate ASSERT(dfnp->fn_dirents == NULL); 10730Sstevel@tonic-gate ASSERT(dfnp->fn_trigger == NULL); 10740Sstevel@tonic-gate } 10750Sstevel@tonic-gate dfnp->fn_flags &= ~MF_MOUNTPOINT; 10760Sstevel@tonic-gate mutex_exit(&dfnp->fn_lock); 10770Sstevel@tonic-gate } 10780Sstevel@tonic-gate 10790Sstevel@tonic-gate for (p = alp; p != NULL; p = p->next) { 10802170Sevanl 10810Sstevel@tonic-gate vfs_t *vfsp; /* dummy argument */ 10820Sstevel@tonic-gate vfs_t *mvfsp; 10830Sstevel@tonic-gate 10840Sstevel@tonic-gate auto_mount = 0; 10850Sstevel@tonic-gate 10860Sstevel@tonic-gate m = &p->action.action_list_entry_u.mounta; 10870Sstevel@tonic-gate argsp = (struct autofs_args *)m->dataptr; 10884068Sdm120769 ASSERT(strcmp(m->fstype, "autofs") == 0); 10894068Sdm120769 /* 10904068Sdm120769 * use the parent directory's timeout since it's the 10914068Sdm120769 * one specified/inherited by automount. 10924068Sdm120769 */ 10934068Sdm120769 argsp->mount_to = dfnip->fi_mount_to; 10944068Sdm120769 /* 10954068Sdm120769 * The mountpoint is relative, and it is guaranteed to 10964068Sdm120769 * begin with "." 10974068Sdm120769 * 10984068Sdm120769 */ 10994068Sdm120769 ASSERT(m->dir[0] == '.'); 11004068Sdm120769 if (m->dir[0] == '.' && m->dir[1] == '\0') { 11010Sstevel@tonic-gate /* 11024068Sdm120769 * mounting on the trigger node 11030Sstevel@tonic-gate */ 11044068Sdm120769 mvp = dvp; 11054068Sdm120769 VN_HOLD(mvp); 11064068Sdm120769 goto mount; 11074068Sdm120769 } 11084068Sdm120769 /* 11094068Sdm120769 * ignore "./" in front of mountpoint 11104068Sdm120769 */ 11114068Sdm120769 ASSERT(m->dir[1] == '/'); 11124068Sdm120769 mntpnt = m->dir + 2; 11130Sstevel@tonic-gate 11144068Sdm120769 AUTOFS_DPRINT((10, "\tdfnip->fi_path=%s\n", dfnip->fi_path)); 11154068Sdm120769 AUTOFS_DPRINT((10, "\tdfnip->fi_flags=%x\n", dfnip->fi_flags)); 11164068Sdm120769 AUTOFS_DPRINT((10, "\tmntpnt=%s\n", mntpnt)); 11170Sstevel@tonic-gate 11184068Sdm120769 if (dfnip->fi_flags & MF_DIRECT) { 11194068Sdm120769 AUTOFS_DPRINT((10, "\tDIRECT\n")); 11206046Srmesta (void) sprintf(buff, "%s/%s", dfnip->fi_path, mntpnt); 11214068Sdm120769 } else { 11224068Sdm120769 AUTOFS_DPRINT((10, "\tINDIRECT\n")); 11234068Sdm120769 (void) sprintf(buff, "%s/%s/%s", 11246046Srmesta dfnip->fi_path, dfnp->fn_name, mntpnt); 11254068Sdm120769 } 11260Sstevel@tonic-gate 11274068Sdm120769 if (vn_mountedvfs(dvp) == NULL) { 11284068Sdm120769 /* 11294068Sdm120769 * Daemon didn't mount anything on the root 11304068Sdm120769 * We have to create the mountpoint if it 11314068Sdm120769 * doesn't exist already 11324068Sdm120769 * 11334068Sdm120769 * We use the caller's credentials in case a 11344068Sdm120769 * UID-match is required 11354068Sdm120769 * (MF_THISUID_MATCH_RQD). 11364068Sdm120769 */ 11374068Sdm120769 rw_enter(&dfnp->fn_rwlock, RW_WRITER); 11384068Sdm120769 error = auto_search(dfnp, mntpnt, &mfnp, cred); 11394068Sdm120769 if (error == 0) { 11400Sstevel@tonic-gate /* 11414068Sdm120769 * AUTOFS mountpoint exists 11420Sstevel@tonic-gate */ 11434068Sdm120769 if (vn_mountedvfs(fntovn(mfnp)) != NULL) { 11444068Sdm120769 cmn_err(CE_PANIC, 11456046Srmesta "auto_perform_actions:" 11466046Srmesta " mfnp=%p covered", (void *)mfnp); 11470Sstevel@tonic-gate } 11480Sstevel@tonic-gate } else { 11490Sstevel@tonic-gate /* 11504068Sdm120769 * Create AUTOFS mountpoint 11510Sstevel@tonic-gate */ 11524068Sdm120769 ASSERT((dfnp->fn_flags & MF_MOUNTPOINT) == 0); 11534068Sdm120769 error = auto_enter(dfnp, mntpnt, &mfnp, cred); 11544068Sdm120769 ASSERT(mfnp->fn_linkcnt == 1); 11554068Sdm120769 mfnp->fn_linkcnt++; 11564068Sdm120769 } 11574068Sdm120769 if (!error) 11584068Sdm120769 update_times = 1; 11594068Sdm120769 rw_exit(&dfnp->fn_rwlock); 11604068Sdm120769 ASSERT(error != EEXIST); 11614068Sdm120769 if (!error) { 11624068Sdm120769 /* 11634068Sdm120769 * mfnp is already held. 11644068Sdm120769 */ 11654068Sdm120769 mvp = fntovn(mfnp); 11664068Sdm120769 } else { 11674068Sdm120769 auto_log(fngp->fng_verbose, fngp->fng_zoneid, 11686046Srmesta CE_WARN, "autofs: mount of %s " 11696046Srmesta "failed - can't create" 11706046Srmesta " mountpoint.", buff); 11716046Srmesta continue; 11720Sstevel@tonic-gate } 11734068Sdm120769 } else { 11744068Sdm120769 /* 11754068Sdm120769 * Find mountpoint in VFS mounted here. If not 11764068Sdm120769 * found, fail the submount, though the overall 11774068Sdm120769 * mount has succeeded since the root is 11784068Sdm120769 * mounted. 11794068Sdm120769 */ 11806046Srmesta if (error = auto_getmntpnt(dvp, mntpnt, &mvp, kcred)) { 11816046Srmesta auto_log(fngp->fng_verbose, fngp->fng_zoneid, 11826046Srmesta CE_WARN, "autofs: mount of %s " 11836046Srmesta "failed - mountpoint doesn't" 11846046Srmesta " exist.", buff); 11854068Sdm120769 continue; 11864068Sdm120769 } 11874068Sdm120769 if (mvp->v_type == VLNK) { 11886046Srmesta auto_log(fngp->fng_verbose, fngp->fng_zoneid, 11896046Srmesta CE_WARN, "autofs: %s symbolic " 11906046Srmesta "link: not a valid mountpoint " 11916046Srmesta "- mount failed", buff); 11924068Sdm120769 VN_RELE(mvp); 11934068Sdm120769 error = ENOENT; 11944068Sdm120769 continue; 11954068Sdm120769 } 11964068Sdm120769 } 11970Sstevel@tonic-gate mount: 11984068Sdm120769 m->flags |= MS_SYSSPACE | MS_OPTIONSTR; 11994068Sdm120769 12004068Sdm120769 /* 12014068Sdm120769 * Copy mounta struct here so we can substitute a 12024068Sdm120769 * buffer that is large enough to hold the returned 12034068Sdm120769 * option string, if that string is longer than the 12044068Sdm120769 * input option string. 12054068Sdm120769 * This can happen if there are default options enabled 12064068Sdm120769 * that were not in the input option string. 12074068Sdm120769 */ 12084068Sdm120769 bcopy(m, &margs, sizeof (*m)); 12094068Sdm120769 margs.optptr = kmem_alloc(MAX_MNTOPT_STR, KM_SLEEP); 12104068Sdm120769 margs.optlen = MAX_MNTOPT_STR; 12114068Sdm120769 (void) strcpy(margs.optptr, m->optptr); 12124068Sdm120769 margs.dir = argsp->path; 12134068Sdm120769 12144068Sdm120769 /* 12154068Sdm120769 * We use the zone's kcred because we don't want the 12164068Sdm120769 * zone to be able to thus do something it wouldn't 12174068Sdm120769 * normally be able to. 12184068Sdm120769 */ 12194068Sdm120769 error = domount(NULL, &margs, mvp, zcred, &vfsp); 12204068Sdm120769 kmem_free(margs.optptr, MAX_MNTOPT_STR); 12214068Sdm120769 if (error != 0) { 12224068Sdm120769 auto_log(fngp->fng_verbose, fngp->fng_zoneid, 12236046Srmesta CE_WARN, "autofs: domount of %s failed " 12246046Srmesta "error=%d", buff, error); 12254068Sdm120769 VN_RELE(mvp); 12264068Sdm120769 continue; 12274068Sdm120769 } 12284068Sdm120769 VFS_RELE(vfsp); 12292170Sevanl 12304068Sdm120769 /* 12314068Sdm120769 * If mountpoint is an AUTOFS node, then I'm going to 12324068Sdm120769 * flag it that the Filesystem mounted on top was 12334068Sdm120769 * mounted in the kernel so that the unmount can be 12344068Sdm120769 * done inside the kernel as well. 12354068Sdm120769 * I don't care to flag non-AUTOFS mountpoints when an 12364068Sdm120769 * AUTOFS in-kernel mount was done on top, because the 12374068Sdm120769 * unmount routine already knows that such case was 12384068Sdm120769 * done in the kernel. 12394068Sdm120769 */ 12406046Srmesta if (vfs_matchops(dvp->v_vfsp, vfs_getops(mvp->v_vfsp))) { 12414068Sdm120769 mfnp = vntofn(mvp); 12424068Sdm120769 mutex_enter(&mfnp->fn_lock); 12434068Sdm120769 mfnp->fn_flags |= MF_IK_MOUNT; 12444068Sdm120769 mutex_exit(&mfnp->fn_lock); 12454068Sdm120769 } 12460Sstevel@tonic-gate 12474068Sdm120769 (void) vn_vfswlock_wait(mvp); 12484068Sdm120769 mvfsp = vn_mountedvfs(mvp); 12494068Sdm120769 if (mvfsp != NULL) { 12504068Sdm120769 vfs_lock_wait(mvfsp); 12514068Sdm120769 vn_vfsunlock(mvp); 12524068Sdm120769 error = VFS_ROOT(mvfsp, &newvp); 12534068Sdm120769 vfs_unlock(mvfsp); 12544068Sdm120769 if (error) { 12554068Sdm120769 /* 12564068Sdm120769 * We've dropped the locks, so let's 12574068Sdm120769 * get the mounted vfs again in case 12584068Sdm120769 * it changed. 12594068Sdm120769 */ 12604068Sdm120769 (void) vn_vfswlock_wait(mvp); 12614068Sdm120769 mvfsp = vn_mountedvfs(mvp); 12624068Sdm120769 if (mvfsp != NULL) { 12634068Sdm120769 error = dounmount(mvfsp, 0, CRED()); 12644068Sdm120769 if (error) { 12654068Sdm120769 cmn_err(CE_WARN, 12666046Srmesta "autofs: could not unmount" 12676046Srmesta " vfs=%p", (void *)mvfsp); 12684068Sdm120769 } 12694068Sdm120769 } else 12704068Sdm120769 vn_vfsunlock(mvp); 12710Sstevel@tonic-gate VN_RELE(mvp); 12720Sstevel@tonic-gate continue; 12730Sstevel@tonic-gate } 12744068Sdm120769 } else { 12754068Sdm120769 vn_vfsunlock(mvp); 12764068Sdm120769 VN_RELE(mvp); 12774068Sdm120769 continue; 12784068Sdm120769 } 12793901Skr143551 12804068Sdm120769 auto_mount = vfs_matchops(dvp->v_vfsp, 12816046Srmesta vfs_getops(newvp->v_vfsp)); 12824068Sdm120769 newfnp = vntofn(newvp); 12834068Sdm120769 newfnp->fn_parent = dfnp; 12843901Skr143551 12854068Sdm120769 /* 12864068Sdm120769 * At this time we want to save the AUTOFS filesystem 12874068Sdm120769 * as a trigger node. (We only do this if the mount 12884068Sdm120769 * occurred on a node different from the root. 12894068Sdm120769 * We look at the trigger nodes during 12904068Sdm120769 * the automatic unmounting to make sure we remove them 12914068Sdm120769 * as a unit and remount them as a unit if the 12924068Sdm120769 * filesystem mounted at the root could not be 12934068Sdm120769 * unmounted. 12944068Sdm120769 */ 12954068Sdm120769 if (auto_mount && (error == 0) && (mvp != dvp)) { 12964068Sdm120769 save_triggers++; 12970Sstevel@tonic-gate /* 12984068Sdm120769 * Add AUTOFS mount to hierarchy 12990Sstevel@tonic-gate */ 13004068Sdm120769 newfnp->fn_flags |= MF_TRIGGER; 13014068Sdm120769 rw_enter(&newfnp->fn_rwlock, RW_WRITER); 13024068Sdm120769 newfnp->fn_next = dfnp->fn_trigger; 13034068Sdm120769 rw_exit(&newfnp->fn_rwlock); 13044068Sdm120769 rw_enter(&dfnp->fn_rwlock, RW_WRITER); 13054068Sdm120769 dfnp->fn_trigger = newfnp; 13064068Sdm120769 rw_exit(&dfnp->fn_rwlock); 13074068Sdm120769 /* 13084068Sdm120769 * Don't VN_RELE(newvp) here since dfnp now 13094068Sdm120769 * holds reference to it as its trigger node. 13104068Sdm120769 */ 13114068Sdm120769 AUTOFS_DPRINT((10, "\tadding trigger %s to %s\n", 13126046Srmesta newfnp->fn_name, dfnp->fn_name)); 13134068Sdm120769 AUTOFS_DPRINT((10, "\tfirst trigger is %s\n", 13146046Srmesta dfnp->fn_trigger->fn_name)); 13154068Sdm120769 if (newfnp->fn_next != NULL) 13166046Srmesta AUTOFS_DPRINT((10, "\tnext trigger is %s\n", 13176046Srmesta newfnp->fn_next->fn_name)); 13184068Sdm120769 else 13196046Srmesta AUTOFS_DPRINT((10, "\tno next trigger\n")); 13204068Sdm120769 } else 13214068Sdm120769 VN_RELE(newvp); 13220Sstevel@tonic-gate 13234068Sdm120769 if (!error) 13244068Sdm120769 success++; 13250Sstevel@tonic-gate 13264068Sdm120769 if (update_times) { 13274068Sdm120769 gethrestime(&now); 13284068Sdm120769 dfnp->fn_atime = dfnp->fn_mtime = now; 13294068Sdm120769 } 13303901Skr143551 13314068Sdm120769 VN_RELE(mvp); 13320Sstevel@tonic-gate } 13330Sstevel@tonic-gate 13340Sstevel@tonic-gate if (save_triggers) { 13350Sstevel@tonic-gate /* 13360Sstevel@tonic-gate * Make sure the parent can't be freed while it has triggers. 13370Sstevel@tonic-gate */ 13380Sstevel@tonic-gate VN_HOLD(dvp); 13390Sstevel@tonic-gate } 13400Sstevel@tonic-gate 13410Sstevel@tonic-gate crfree(zcred); 13420Sstevel@tonic-gate 13430Sstevel@tonic-gate done: 13440Sstevel@tonic-gate /* 13450Sstevel@tonic-gate * Return failure if daemon didn't mount anything, and all 13460Sstevel@tonic-gate * kernel mounts attempted failed. 13470Sstevel@tonic-gate */ 13480Sstevel@tonic-gate error = success ? 0 : ENOENT; 13490Sstevel@tonic-gate 13500Sstevel@tonic-gate if (alp != NULL) { 13510Sstevel@tonic-gate if ((error == 0) && save_triggers) { 13520Sstevel@tonic-gate /* 13530Sstevel@tonic-gate * Save action_list information, so that we can use it 13540Sstevel@tonic-gate * when it comes time to remount the trigger nodes 13550Sstevel@tonic-gate * The action list is freed when the directory node 13560Sstevel@tonic-gate * containing the reference to it is unmounted in 13570Sstevel@tonic-gate * unmount_tree(). 13580Sstevel@tonic-gate */ 13590Sstevel@tonic-gate mutex_enter(&dfnp->fn_lock); 13600Sstevel@tonic-gate ASSERT(dfnp->fn_alp == NULL); 13610Sstevel@tonic-gate dfnp->fn_alp = alp; 13620Sstevel@tonic-gate mutex_exit(&dfnp->fn_lock); 13630Sstevel@tonic-gate } else { 13640Sstevel@tonic-gate /* 13650Sstevel@tonic-gate * free the action list now, 13660Sstevel@tonic-gate */ 13670Sstevel@tonic-gate xdr_free(xdr_action_list, (char *)alp); 13680Sstevel@tonic-gate } 13690Sstevel@tonic-gate } 13700Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_perform_actions: error=%d\n", error)); 13710Sstevel@tonic-gate return (error); 13720Sstevel@tonic-gate } 13730Sstevel@tonic-gate 13740Sstevel@tonic-gate fnnode_t * 13750Sstevel@tonic-gate auto_makefnnode( 13760Sstevel@tonic-gate vtype_t type, 13770Sstevel@tonic-gate vfs_t *vfsp, 13780Sstevel@tonic-gate char *name, 13790Sstevel@tonic-gate cred_t *cred, 13800Sstevel@tonic-gate struct autofs_globals *fngp) 13810Sstevel@tonic-gate { 13820Sstevel@tonic-gate fnnode_t *fnp; 13830Sstevel@tonic-gate vnode_t *vp; 13840Sstevel@tonic-gate char *tmpname; 13850Sstevel@tonic-gate timestruc_t now; 13860Sstevel@tonic-gate /* 13870Sstevel@tonic-gate * autofs uses odd inode numbers 13880Sstevel@tonic-gate * automountd uses even inode numbers 13890Sstevel@tonic-gate * 13900Sstevel@tonic-gate * To preserve the age-old semantics that inum+devid is unique across 13910Sstevel@tonic-gate * the system, this variable must be global across zones. 13920Sstevel@tonic-gate */ 13930Sstevel@tonic-gate static ino_t nodeid = 3; 13940Sstevel@tonic-gate 13950Sstevel@tonic-gate fnp = kmem_zalloc(sizeof (*fnp), KM_SLEEP); 13960Sstevel@tonic-gate fnp->fn_vnode = vn_alloc(KM_SLEEP); 13970Sstevel@tonic-gate 13980Sstevel@tonic-gate vp = fntovn(fnp); 13990Sstevel@tonic-gate tmpname = kmem_alloc(strlen(name) + 1, KM_SLEEP); 14000Sstevel@tonic-gate (void) strcpy(tmpname, name); 14010Sstevel@tonic-gate fnp->fn_name = &tmpname[0]; 14020Sstevel@tonic-gate fnp->fn_namelen = (int)strlen(tmpname) + 1; /* include '\0' */ 14030Sstevel@tonic-gate fnp->fn_uid = crgetuid(cred); 14040Sstevel@tonic-gate fnp->fn_gid = crgetgid(cred); 14050Sstevel@tonic-gate /* 14060Sstevel@tonic-gate * ".." is added in auto_enter and auto_mount. 14070Sstevel@tonic-gate * "." is added in auto_mkdir and auto_mount. 14080Sstevel@tonic-gate */ 14090Sstevel@tonic-gate /* 14100Sstevel@tonic-gate * Note that fn_size and fn_linkcnt are already 0 since 14110Sstevel@tonic-gate * we used kmem_zalloc to allocated fnp 14120Sstevel@tonic-gate */ 14130Sstevel@tonic-gate fnp->fn_mode = AUTOFS_MODE; 14140Sstevel@tonic-gate gethrestime(&now); 14150Sstevel@tonic-gate fnp->fn_atime = fnp->fn_mtime = fnp->fn_ctime = now; 14160Sstevel@tonic-gate fnp->fn_ref_time = now.tv_sec; 14170Sstevel@tonic-gate mutex_enter(&autofs_nodeid_lock); 14180Sstevel@tonic-gate fnp->fn_nodeid = nodeid; 14190Sstevel@tonic-gate nodeid += 2; 14200Sstevel@tonic-gate fnp->fn_globals = fngp; 14210Sstevel@tonic-gate fngp->fng_fnnode_count++; 14220Sstevel@tonic-gate mutex_exit(&autofs_nodeid_lock); 14230Sstevel@tonic-gate vn_setops(vp, auto_vnodeops); 14240Sstevel@tonic-gate vp->v_type = type; 14250Sstevel@tonic-gate vp->v_data = (void *)fnp; 14260Sstevel@tonic-gate vp->v_vfsp = vfsp; 14270Sstevel@tonic-gate mutex_init(&fnp->fn_lock, NULL, MUTEX_DEFAULT, NULL); 14280Sstevel@tonic-gate rw_init(&fnp->fn_rwlock, NULL, RW_DEFAULT, NULL); 14290Sstevel@tonic-gate cv_init(&fnp->fn_cv_mount, NULL, CV_DEFAULT, NULL); 14300Sstevel@tonic-gate vn_exists(vp); 14310Sstevel@tonic-gate return (fnp); 14320Sstevel@tonic-gate } 14330Sstevel@tonic-gate 14340Sstevel@tonic-gate 14350Sstevel@tonic-gate void 14360Sstevel@tonic-gate auto_freefnnode(fnnode_t *fnp) 14370Sstevel@tonic-gate { 14380Sstevel@tonic-gate vnode_t *vp = fntovn(fnp); 14390Sstevel@tonic-gate 14400Sstevel@tonic-gate AUTOFS_DPRINT((4, "auto_freefnnode: fnp=%p\n", (void *)fnp)); 14410Sstevel@tonic-gate 14420Sstevel@tonic-gate ASSERT(fnp->fn_linkcnt == 0); 14430Sstevel@tonic-gate ASSERT(vp->v_count == 0); 14440Sstevel@tonic-gate ASSERT(fnp->fn_dirents == NULL); 14450Sstevel@tonic-gate ASSERT(fnp->fn_parent == NULL); 14460Sstevel@tonic-gate 14470Sstevel@tonic-gate vn_invalid(vp); 14480Sstevel@tonic-gate kmem_free(fnp->fn_name, fnp->fn_namelen); 14490Sstevel@tonic-gate if (fnp->fn_symlink) { 14500Sstevel@tonic-gate ASSERT(fnp->fn_flags & MF_THISUID_MATCH_RQD); 14510Sstevel@tonic-gate kmem_free(fnp->fn_symlink, fnp->fn_symlinklen); 14520Sstevel@tonic-gate } 14530Sstevel@tonic-gate if (fnp->fn_cred) 14540Sstevel@tonic-gate crfree(fnp->fn_cred); 14550Sstevel@tonic-gate mutex_destroy(&fnp->fn_lock); 14560Sstevel@tonic-gate rw_destroy(&fnp->fn_rwlock); 14570Sstevel@tonic-gate cv_destroy(&fnp->fn_cv_mount); 14580Sstevel@tonic-gate vn_free(vp); 14590Sstevel@tonic-gate 14600Sstevel@tonic-gate mutex_enter(&autofs_nodeid_lock); 14610Sstevel@tonic-gate fnp->fn_globals->fng_fnnode_count--; 14620Sstevel@tonic-gate mutex_exit(&autofs_nodeid_lock); 14630Sstevel@tonic-gate kmem_free(fnp, sizeof (*fnp)); 14640Sstevel@tonic-gate } 14650Sstevel@tonic-gate 14660Sstevel@tonic-gate void 14670Sstevel@tonic-gate auto_disconnect( 14680Sstevel@tonic-gate fnnode_t *dfnp, 14690Sstevel@tonic-gate fnnode_t *fnp) 14700Sstevel@tonic-gate { 14710Sstevel@tonic-gate fnnode_t *tmp, **fnpp; 14720Sstevel@tonic-gate vnode_t *vp = fntovn(fnp); 14730Sstevel@tonic-gate timestruc_t now; 14740Sstevel@tonic-gate 14750Sstevel@tonic-gate AUTOFS_DPRINT((4, 14760Sstevel@tonic-gate "auto_disconnect: dfnp=%p fnp=%p linkcnt=%d\n v_count=%d", 14770Sstevel@tonic-gate (void *)dfnp, (void *)fnp, fnp->fn_linkcnt, vp->v_count)); 14780Sstevel@tonic-gate 14790Sstevel@tonic-gate ASSERT(RW_WRITE_HELD(&dfnp->fn_rwlock)); 14800Sstevel@tonic-gate ASSERT(fnp->fn_linkcnt == 1); 14810Sstevel@tonic-gate 14820Sstevel@tonic-gate if (vn_mountedvfs(vp) != NULL) { 14830Sstevel@tonic-gate cmn_err(CE_PANIC, "auto_disconnect: vp %p mounted on", 14840Sstevel@tonic-gate (void *)vp); 14850Sstevel@tonic-gate } 14860Sstevel@tonic-gate 14870Sstevel@tonic-gate /* 14880Sstevel@tonic-gate * Decrement by 1 because we're removing the entry in dfnp. 14890Sstevel@tonic-gate */ 14900Sstevel@tonic-gate fnp->fn_linkcnt--; 14910Sstevel@tonic-gate fnp->fn_size--; 14920Sstevel@tonic-gate 14930Sstevel@tonic-gate /* 14940Sstevel@tonic-gate * only changed while holding parent's (dfnp) rw_lock 14950Sstevel@tonic-gate */ 14960Sstevel@tonic-gate fnp->fn_parent = NULL; 14970Sstevel@tonic-gate 14980Sstevel@tonic-gate fnpp = &dfnp->fn_dirents; 14990Sstevel@tonic-gate for (;;) { 15000Sstevel@tonic-gate tmp = *fnpp; 15010Sstevel@tonic-gate if (tmp == NULL) { 15020Sstevel@tonic-gate cmn_err(CE_PANIC, 15030Sstevel@tonic-gate "auto_disconnect: %p not in %p dirent list", 15040Sstevel@tonic-gate (void *)fnp, (void *)dfnp); 15050Sstevel@tonic-gate } 15060Sstevel@tonic-gate if (tmp == fnp) { 15070Sstevel@tonic-gate *fnpp = tmp->fn_next; /* remove it from the list */ 15080Sstevel@tonic-gate ASSERT(vp->v_count == 0); 15090Sstevel@tonic-gate /* child had a pointer to parent ".." */ 15100Sstevel@tonic-gate dfnp->fn_linkcnt--; 15110Sstevel@tonic-gate dfnp->fn_size--; 15120Sstevel@tonic-gate break; 15130Sstevel@tonic-gate } 15140Sstevel@tonic-gate fnpp = &tmp->fn_next; 15150Sstevel@tonic-gate } 15160Sstevel@tonic-gate 15170Sstevel@tonic-gate mutex_enter(&fnp->fn_lock); 15180Sstevel@tonic-gate gethrestime(&now); 15190Sstevel@tonic-gate fnp->fn_atime = fnp->fn_mtime = now; 15200Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 15210Sstevel@tonic-gate 15220Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_disconnect: done\n")); 15230Sstevel@tonic-gate } 15240Sstevel@tonic-gate 15250Sstevel@tonic-gate int 15260Sstevel@tonic-gate auto_enter(fnnode_t *dfnp, char *name, fnnode_t **fnpp, cred_t *cred) 15270Sstevel@tonic-gate { 15280Sstevel@tonic-gate struct fnnode *cfnp, **spp; 15290Sstevel@tonic-gate vnode_t *dvp = fntovn(dfnp); 15300Sstevel@tonic-gate ushort_t offset = 0; 15310Sstevel@tonic-gate ushort_t diff; 15320Sstevel@tonic-gate 15330Sstevel@tonic-gate AUTOFS_DPRINT((4, "auto_enter: dfnp=%p, name=%s ", (void *)dfnp, name)); 15340Sstevel@tonic-gate 15350Sstevel@tonic-gate ASSERT(RW_WRITE_HELD(&dfnp->fn_rwlock)); 15360Sstevel@tonic-gate 15370Sstevel@tonic-gate cfnp = dfnp->fn_dirents; 15380Sstevel@tonic-gate if (cfnp == NULL) { 15390Sstevel@tonic-gate /* 15400Sstevel@tonic-gate * offset = 0 for '.' and offset = 1 for '..' 15410Sstevel@tonic-gate */ 15420Sstevel@tonic-gate spp = &dfnp->fn_dirents; 15430Sstevel@tonic-gate offset = 2; 15440Sstevel@tonic-gate } 15450Sstevel@tonic-gate 15460Sstevel@tonic-gate for (; cfnp; cfnp = cfnp->fn_next) { 15470Sstevel@tonic-gate if (strcmp(cfnp->fn_name, name) == 0) { 15480Sstevel@tonic-gate mutex_enter(&cfnp->fn_lock); 15490Sstevel@tonic-gate if (cfnp->fn_flags & MF_THISUID_MATCH_RQD) { 15500Sstevel@tonic-gate /* 15510Sstevel@tonic-gate * "thisuser" kind of node, need to 15520Sstevel@tonic-gate * match CREDs as well 15530Sstevel@tonic-gate */ 15540Sstevel@tonic-gate mutex_exit(&cfnp->fn_lock); 15550Sstevel@tonic-gate if (crcmp(cfnp->fn_cred, cred) == 0) 15560Sstevel@tonic-gate return (EEXIST); 15570Sstevel@tonic-gate } else { 15580Sstevel@tonic-gate mutex_exit(&cfnp->fn_lock); 15590Sstevel@tonic-gate return (EEXIST); 15600Sstevel@tonic-gate } 15610Sstevel@tonic-gate } 15620Sstevel@tonic-gate 15630Sstevel@tonic-gate if (cfnp->fn_next != NULL) { 15640Sstevel@tonic-gate diff = (ushort_t) 15650Sstevel@tonic-gate (cfnp->fn_next->fn_offset - cfnp->fn_offset); 15660Sstevel@tonic-gate ASSERT(diff != 0); 15670Sstevel@tonic-gate if (diff > 1 && offset == 0) { 15680Sstevel@tonic-gate offset = (ushort_t)cfnp->fn_offset + 1; 15690Sstevel@tonic-gate spp = &cfnp->fn_next; 15700Sstevel@tonic-gate } 15710Sstevel@tonic-gate } else if (offset == 0) { 15720Sstevel@tonic-gate offset = (ushort_t)cfnp->fn_offset + 1; 15730Sstevel@tonic-gate spp = &cfnp->fn_next; 15740Sstevel@tonic-gate } 15750Sstevel@tonic-gate } 15760Sstevel@tonic-gate 15770Sstevel@tonic-gate *fnpp = auto_makefnnode(VDIR, dvp->v_vfsp, name, cred, 15780Sstevel@tonic-gate dfnp->fn_globals); 15790Sstevel@tonic-gate if (*fnpp == NULL) 15800Sstevel@tonic-gate return (ENOMEM); 15810Sstevel@tonic-gate 15820Sstevel@tonic-gate /* 15830Sstevel@tonic-gate * I don't hold the mutex on fnpp because I created it, and 15840Sstevel@tonic-gate * I'm already holding the writers lock for it's parent 15850Sstevel@tonic-gate * directory, therefore nobody can reference it without me first 15860Sstevel@tonic-gate * releasing the writers lock. 15870Sstevel@tonic-gate */ 15880Sstevel@tonic-gate (*fnpp)->fn_offset = offset; 15890Sstevel@tonic-gate (*fnpp)->fn_next = *spp; 15900Sstevel@tonic-gate *spp = *fnpp; 15910Sstevel@tonic-gate (*fnpp)->fn_parent = dfnp; 15920Sstevel@tonic-gate (*fnpp)->fn_linkcnt++; /* parent now holds reference to entry */ 15930Sstevel@tonic-gate (*fnpp)->fn_size++; 15940Sstevel@tonic-gate 15950Sstevel@tonic-gate /* 15960Sstevel@tonic-gate * dfnp->fn_linkcnt and dfnp->fn_size protected by dfnp->rw_lock 15970Sstevel@tonic-gate */ 15980Sstevel@tonic-gate dfnp->fn_linkcnt++; /* child now holds reference to parent '..' */ 15990Sstevel@tonic-gate dfnp->fn_size++; 16000Sstevel@tonic-gate 16010Sstevel@tonic-gate dfnp->fn_ref_time = gethrestime_sec(); 16020Sstevel@tonic-gate 16030Sstevel@tonic-gate AUTOFS_DPRINT((5, "*fnpp=%p\n", (void *)*fnpp)); 16040Sstevel@tonic-gate return (0); 16050Sstevel@tonic-gate } 16060Sstevel@tonic-gate 16070Sstevel@tonic-gate int 16080Sstevel@tonic-gate auto_search(fnnode_t *dfnp, char *name, fnnode_t **fnpp, cred_t *cred) 16090Sstevel@tonic-gate { 16100Sstevel@tonic-gate vnode_t *dvp; 16110Sstevel@tonic-gate fnnode_t *p; 16120Sstevel@tonic-gate int error = ENOENT, match = 0; 16130Sstevel@tonic-gate 16140Sstevel@tonic-gate AUTOFS_DPRINT((4, "auto_search: dfnp=%p, name=%s...\n", 16150Sstevel@tonic-gate (void *)dfnp, name)); 16160Sstevel@tonic-gate 16170Sstevel@tonic-gate dvp = fntovn(dfnp); 16180Sstevel@tonic-gate if (dvp->v_type != VDIR) { 16190Sstevel@tonic-gate cmn_err(CE_PANIC, "auto_search: dvp=%p not a directory", 16200Sstevel@tonic-gate (void *)dvp); 16210Sstevel@tonic-gate } 16220Sstevel@tonic-gate 16230Sstevel@tonic-gate ASSERT(RW_LOCK_HELD(&dfnp->fn_rwlock)); 16240Sstevel@tonic-gate for (p = dfnp->fn_dirents; p != NULL; p = p->fn_next) { 16250Sstevel@tonic-gate if (strcmp(p->fn_name, name) == 0) { 16260Sstevel@tonic-gate mutex_enter(&p->fn_lock); 16270Sstevel@tonic-gate if (p->fn_flags & MF_THISUID_MATCH_RQD) { 16280Sstevel@tonic-gate /* 16290Sstevel@tonic-gate * "thisuser" kind of node 16300Sstevel@tonic-gate * Need to match CREDs as well 16310Sstevel@tonic-gate */ 16320Sstevel@tonic-gate mutex_exit(&p->fn_lock); 16330Sstevel@tonic-gate match = crcmp(p->fn_cred, cred) == 0; 16340Sstevel@tonic-gate } else { 16350Sstevel@tonic-gate /* 16360Sstevel@tonic-gate * No need to check CRED 16370Sstevel@tonic-gate */ 16380Sstevel@tonic-gate mutex_exit(&p->fn_lock); 16390Sstevel@tonic-gate match = 1; 16400Sstevel@tonic-gate } 16410Sstevel@tonic-gate } 16420Sstevel@tonic-gate if (match) { 16430Sstevel@tonic-gate error = 0; 16440Sstevel@tonic-gate if (fnpp) { 16450Sstevel@tonic-gate *fnpp = p; 16460Sstevel@tonic-gate VN_HOLD(fntovn(*fnpp)); 16470Sstevel@tonic-gate } 16480Sstevel@tonic-gate break; 16490Sstevel@tonic-gate } 16500Sstevel@tonic-gate } 16510Sstevel@tonic-gate 16520Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_search: error=%d\n", error)); 16530Sstevel@tonic-gate return (error); 16540Sstevel@tonic-gate } 16550Sstevel@tonic-gate 16560Sstevel@tonic-gate /* 16570Sstevel@tonic-gate * If dvp is mounted on, get path's vnode in the mounted on 16580Sstevel@tonic-gate * filesystem. Path is relative to dvp, ie "./path". 16590Sstevel@tonic-gate * If successful, *mvp points to a the held mountpoint vnode. 16600Sstevel@tonic-gate */ 16610Sstevel@tonic-gate /* ARGSUSED */ 16620Sstevel@tonic-gate static int 16630Sstevel@tonic-gate auto_getmntpnt( 16640Sstevel@tonic-gate vnode_t *dvp, 16650Sstevel@tonic-gate char *path, 16660Sstevel@tonic-gate vnode_t **mvpp, /* vnode for mountpoint */ 16670Sstevel@tonic-gate cred_t *cred) 16680Sstevel@tonic-gate { 16690Sstevel@tonic-gate int error = 0; 16700Sstevel@tonic-gate vnode_t *newvp; 16710Sstevel@tonic-gate char namebuf[TYPICALMAXPATHLEN]; 16720Sstevel@tonic-gate struct pathname lookpn; 16730Sstevel@tonic-gate vfs_t *vfsp; 16740Sstevel@tonic-gate 16750Sstevel@tonic-gate AUTOFS_DPRINT((4, "auto_getmntpnt: path=%s\n", path)); 16760Sstevel@tonic-gate 16771153Snr123932 if (error = vn_vfsrlock_wait(dvp)) 16780Sstevel@tonic-gate return (error); 16790Sstevel@tonic-gate 16800Sstevel@tonic-gate /* 16810Sstevel@tonic-gate * Now that we have the vfswlock, check to see if dvp 16820Sstevel@tonic-gate * is still mounted on. If not, then just bail out as 16830Sstevel@tonic-gate * there is no need to remount the triggers since the 16840Sstevel@tonic-gate * higher level mount point has gotten unmounted. 16850Sstevel@tonic-gate */ 16860Sstevel@tonic-gate vfsp = vn_mountedvfs(dvp); 16870Sstevel@tonic-gate if (vfsp == NULL) { 16880Sstevel@tonic-gate vn_vfsunlock(dvp); 16890Sstevel@tonic-gate error = EBUSY; 16900Sstevel@tonic-gate goto done; 16910Sstevel@tonic-gate } 16920Sstevel@tonic-gate /* 16930Sstevel@tonic-gate * Since mounted on, lookup "path" in the new filesystem, 16940Sstevel@tonic-gate * it is important that we do the filesystem jump here to 16950Sstevel@tonic-gate * avoid lookuppn() calling auto_lookup on dvp and deadlock. 16960Sstevel@tonic-gate */ 16971153Snr123932 error = VFS_ROOT(vfsp, &newvp); 16980Sstevel@tonic-gate vn_vfsunlock(dvp); 16990Sstevel@tonic-gate if (error) 17000Sstevel@tonic-gate goto done; 17010Sstevel@tonic-gate 17020Sstevel@tonic-gate /* 17030Sstevel@tonic-gate * We do a VN_HOLD on newvp just in case the first call to 17040Sstevel@tonic-gate * lookuppnvp() fails with ENAMETOOLONG. We should still have a 17050Sstevel@tonic-gate * reference to this vnode for the second call to lookuppnvp(). 17060Sstevel@tonic-gate */ 17070Sstevel@tonic-gate VN_HOLD(newvp); 17080Sstevel@tonic-gate 17090Sstevel@tonic-gate /* 17100Sstevel@tonic-gate * Now create the pathname struct so we can make use of lookuppnvp, 17110Sstevel@tonic-gate * and pn_getcomponent. 17120Sstevel@tonic-gate * This code is similar to lookupname() in fs/lookup.c. 17130Sstevel@tonic-gate */ 17140Sstevel@tonic-gate error = pn_get_buf(path, UIO_SYSSPACE, &lookpn, 17156046Srmesta namebuf, sizeof (namebuf)); 17160Sstevel@tonic-gate if (error == 0) { 17170Sstevel@tonic-gate error = lookuppnvp(&lookpn, NULL, NO_FOLLOW, NULLVPP, 17180Sstevel@tonic-gate mvpp, rootdir, newvp, cred); 17190Sstevel@tonic-gate } else 17200Sstevel@tonic-gate VN_RELE(newvp); 17210Sstevel@tonic-gate if (error == ENAMETOOLONG) { 17220Sstevel@tonic-gate /* 17230Sstevel@tonic-gate * This thread used a pathname > TYPICALMAXPATHLEN bytes long. 17240Sstevel@tonic-gate * newvp is VN_RELE'd by this call to lookuppnvp. 17250Sstevel@tonic-gate * 17260Sstevel@tonic-gate * Using 'rootdir' in a zone's context is OK here: we already 17270Sstevel@tonic-gate * ascertained that there are no '..'s in the path, and we're 17280Sstevel@tonic-gate * not following symlinks. 17290Sstevel@tonic-gate */ 17300Sstevel@tonic-gate if ((error = pn_get(path, UIO_SYSSPACE, &lookpn)) == 0) { 17310Sstevel@tonic-gate error = lookuppnvp(&lookpn, NULL, NO_FOLLOW, NULLVPP, 17320Sstevel@tonic-gate mvpp, rootdir, newvp, cred); 17330Sstevel@tonic-gate pn_free(&lookpn); 17340Sstevel@tonic-gate } else 17350Sstevel@tonic-gate VN_RELE(newvp); 17360Sstevel@tonic-gate } else { 17370Sstevel@tonic-gate /* 17380Sstevel@tonic-gate * Need to release newvp here since we held it. 17390Sstevel@tonic-gate */ 17400Sstevel@tonic-gate VN_RELE(newvp); 17410Sstevel@tonic-gate } 17420Sstevel@tonic-gate 17430Sstevel@tonic-gate done: 17440Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_getmntpnt: path=%s *mvpp=%p error=%d\n", 17450Sstevel@tonic-gate path, (void *)*mvpp, error)); 17460Sstevel@tonic-gate return (error); 17470Sstevel@tonic-gate } 17480Sstevel@tonic-gate 17490Sstevel@tonic-gate #define DEEPER(x) (((x)->fn_dirents != NULL) || \ 17500Sstevel@tonic-gate (vn_mountedvfs(fntovn((x)))) != NULL) 17510Sstevel@tonic-gate 17520Sstevel@tonic-gate /* 17530Sstevel@tonic-gate * The caller, should have already VN_RELE'd its reference to the 17540Sstevel@tonic-gate * root vnode of this filesystem. 17550Sstevel@tonic-gate */ 17560Sstevel@tonic-gate static int 17570Sstevel@tonic-gate auto_inkernel_unmount(vfs_t *vfsp) 17580Sstevel@tonic-gate { 17590Sstevel@tonic-gate vnode_t *cvp = vfsp->vfs_vnodecovered; 17600Sstevel@tonic-gate int error; 17610Sstevel@tonic-gate 17620Sstevel@tonic-gate AUTOFS_DPRINT((4, 17630Sstevel@tonic-gate "auto_inkernel_unmount: devid=%lx mntpnt(%p) count %u\n", 17640Sstevel@tonic-gate vfsp->vfs_dev, (void *)cvp, cvp->v_count)); 17650Sstevel@tonic-gate 17660Sstevel@tonic-gate ASSERT(vn_vfswlock_held(cvp)); 17670Sstevel@tonic-gate 17680Sstevel@tonic-gate /* 17690Sstevel@tonic-gate * Perform the unmount 17700Sstevel@tonic-gate * The mountpoint has already been locked by the caller. 17710Sstevel@tonic-gate */ 17720Sstevel@tonic-gate error = dounmount(vfsp, 0, kcred); 17730Sstevel@tonic-gate 17740Sstevel@tonic-gate AUTOFS_DPRINT((5, "auto_inkernel_unmount: exit count %u\n", 17750Sstevel@tonic-gate cvp->v_count)); 17760Sstevel@tonic-gate return (error); 17770Sstevel@tonic-gate } 17780Sstevel@tonic-gate 17790Sstevel@tonic-gate /* 17800Sstevel@tonic-gate * unmounts trigger nodes in the kernel. 17810Sstevel@tonic-gate */ 17820Sstevel@tonic-gate static void 17830Sstevel@tonic-gate unmount_triggers(fnnode_t *fnp, action_list **alp) 17840Sstevel@tonic-gate { 17850Sstevel@tonic-gate fnnode_t *tp, *next; 17860Sstevel@tonic-gate int error = 0; 17870Sstevel@tonic-gate vfs_t *vfsp; 17880Sstevel@tonic-gate vnode_t *tvp; 17890Sstevel@tonic-gate 17900Sstevel@tonic-gate AUTOFS_DPRINT((4, "unmount_triggers: fnp=%p\n", (void *)fnp)); 17910Sstevel@tonic-gate ASSERT(RW_WRITE_HELD(&fnp->fn_rwlock)); 17920Sstevel@tonic-gate 17930Sstevel@tonic-gate *alp = fnp->fn_alp; 17940Sstevel@tonic-gate next = fnp->fn_trigger; 17950Sstevel@tonic-gate while ((tp = next) != NULL) { 17960Sstevel@tonic-gate tvp = fntovn(tp); 17970Sstevel@tonic-gate ASSERT(tvp->v_count >= 2); 17980Sstevel@tonic-gate next = tp->fn_next; 17990Sstevel@tonic-gate /* 18000Sstevel@tonic-gate * drop writer's lock since the unmount will end up 18010Sstevel@tonic-gate * disconnecting this node from fnp and needs to acquire 18020Sstevel@tonic-gate * the writer's lock again. 18030Sstevel@tonic-gate * next has at least a reference count >= 2 since it's 18040Sstevel@tonic-gate * a trigger node, therefore can not be accidentally freed 18050Sstevel@tonic-gate * by a VN_RELE 18060Sstevel@tonic-gate */ 18070Sstevel@tonic-gate rw_exit(&fnp->fn_rwlock); 18080Sstevel@tonic-gate 18090Sstevel@tonic-gate vfsp = tvp->v_vfsp; 18100Sstevel@tonic-gate 18110Sstevel@tonic-gate /* 18120Sstevel@tonic-gate * Its parent was holding a reference to it, since this 18130Sstevel@tonic-gate * is a trigger vnode. 18140Sstevel@tonic-gate */ 18150Sstevel@tonic-gate VN_RELE(tvp); 18160Sstevel@tonic-gate if (error = auto_inkernel_unmount(vfsp)) { 18170Sstevel@tonic-gate cmn_err(CE_PANIC, "unmount_triggers: " 18180Sstevel@tonic-gate "unmount of vp=%p failed error=%d", 18190Sstevel@tonic-gate (void *)tvp, error); 18200Sstevel@tonic-gate } 18210Sstevel@tonic-gate /* 18220Sstevel@tonic-gate * reacquire writer's lock 18230Sstevel@tonic-gate */ 18240Sstevel@tonic-gate rw_enter(&fnp->fn_rwlock, RW_WRITER); 18250Sstevel@tonic-gate } 18260Sstevel@tonic-gate 18270Sstevel@tonic-gate /* 18280Sstevel@tonic-gate * We were holding a reference to our parent. Drop that. 18290Sstevel@tonic-gate */ 18300Sstevel@tonic-gate VN_RELE(fntovn(fnp)); 18310Sstevel@tonic-gate fnp->fn_trigger = NULL; 18320Sstevel@tonic-gate fnp->fn_alp = NULL; 18330Sstevel@tonic-gate 18340Sstevel@tonic-gate AUTOFS_DPRINT((5, "unmount_triggers: finished\n")); 18350Sstevel@tonic-gate } 18360Sstevel@tonic-gate 18370Sstevel@tonic-gate /* 18380Sstevel@tonic-gate * This routine locks the mountpoint of every trigger node if they're 1839*12184SJan.Kryl@Sun.COM * not busy, or returns EBUSY if any node is busy. 18400Sstevel@tonic-gate */ 1841*12184SJan.Kryl@Sun.COM static boolean_t 1842*12184SJan.Kryl@Sun.COM triggers_busy(fnnode_t *fnp) 18430Sstevel@tonic-gate { 1844*12184SJan.Kryl@Sun.COM int done; 18450Sstevel@tonic-gate int lck_error = 0; 18460Sstevel@tonic-gate fnnode_t *tp, *t1p; 18470Sstevel@tonic-gate vfs_t *vfsp; 18480Sstevel@tonic-gate 18490Sstevel@tonic-gate ASSERT(RW_WRITE_HELD(&fnp->fn_rwlock)); 18500Sstevel@tonic-gate 18510Sstevel@tonic-gate for (tp = fnp->fn_trigger; tp != NULL; tp = tp->fn_next) { 18520Sstevel@tonic-gate AUTOFS_DPRINT((10, "\ttrigger: %s\n", tp->fn_name)); 1853*12184SJan.Kryl@Sun.COM /* MF_LOOKUP should never be set on trigger nodes */ 1854*12184SJan.Kryl@Sun.COM ASSERT((tp->fn_flags & MF_LOOKUP) == 0); 18550Sstevel@tonic-gate vfsp = fntovn(tp)->v_vfsp; 1856*12184SJan.Kryl@Sun.COM 18570Sstevel@tonic-gate /* 18580Sstevel@tonic-gate * The vn_vfsunlock will be done in auto_inkernel_unmount. 18590Sstevel@tonic-gate */ 18600Sstevel@tonic-gate lck_error = vn_vfswlock(vfsp->vfs_vnodecovered); 1861*12184SJan.Kryl@Sun.COM 1862*12184SJan.Kryl@Sun.COM if (lck_error != 0 || (tp->fn_flags & MF_INPROG) || 1863*12184SJan.Kryl@Sun.COM DEEPER(tp) || ((fntovn(tp))->v_count) > 2) { 18640Sstevel@tonic-gate /* 18650Sstevel@tonic-gate * couldn't lock it because it's busy, 18660Sstevel@tonic-gate * It is mounted on or has dirents? 18670Sstevel@tonic-gate * If reference count is greater than two, then 18680Sstevel@tonic-gate * somebody else is holding a reference to this vnode. 18690Sstevel@tonic-gate * One reference is for the mountpoint, and the second 18700Sstevel@tonic-gate * is for the trigger node. 18710Sstevel@tonic-gate */ 18720Sstevel@tonic-gate AUTOFS_DPRINT((10, "\ttrigger busy\n")); 1873*12184SJan.Kryl@Sun.COM 18740Sstevel@tonic-gate /* 18750Sstevel@tonic-gate * Unlock previously locked mountpoints 18760Sstevel@tonic-gate */ 18770Sstevel@tonic-gate for (done = 0, t1p = fnp->fn_trigger; !done; 18780Sstevel@tonic-gate t1p = t1p->fn_next) { 18790Sstevel@tonic-gate /* 18800Sstevel@tonic-gate * Unlock all nodes previously 18810Sstevel@tonic-gate * locked. All nodes up to 'tp' 18820Sstevel@tonic-gate * were successfully locked. If 'lck_err' is 18830Sstevel@tonic-gate * set, then 'tp' was not locked, and thus 18840Sstevel@tonic-gate * should not be unlocked. If 18850Sstevel@tonic-gate * 'lck_err' is not set, then 'tp' was 18860Sstevel@tonic-gate * successfully locked, and it should 18870Sstevel@tonic-gate * be unlocked. 18880Sstevel@tonic-gate */ 18890Sstevel@tonic-gate if (t1p != tp || !lck_error) { 18900Sstevel@tonic-gate vfsp = fntovn(t1p)->v_vfsp; 18910Sstevel@tonic-gate vn_vfsunlock(vfsp->vfs_vnodecovered); 18920Sstevel@tonic-gate } 18930Sstevel@tonic-gate done = (t1p == tp); 18940Sstevel@tonic-gate } 1895*12184SJan.Kryl@Sun.COM return (B_TRUE); 18960Sstevel@tonic-gate } 18970Sstevel@tonic-gate } 18980Sstevel@tonic-gate 1899*12184SJan.Kryl@Sun.COM return (B_FALSE); 19000Sstevel@tonic-gate } 19010Sstevel@tonic-gate 19020Sstevel@tonic-gate /* 19030Sstevel@tonic-gate * It is the caller's responsibility to grab the VVFSLOCK. 19040Sstevel@tonic-gate * Releases the VVFSLOCK upon return. 19050Sstevel@tonic-gate */ 19060Sstevel@tonic-gate static int 19070Sstevel@tonic-gate unmount_node(vnode_t *cvp, int force) 19080Sstevel@tonic-gate { 19090Sstevel@tonic-gate int error = 0; 19100Sstevel@tonic-gate fnnode_t *cfnp; 19110Sstevel@tonic-gate vfs_t *vfsp; 19120Sstevel@tonic-gate umntrequest ul; 19130Sstevel@tonic-gate fninfo_t *fnip; 19140Sstevel@tonic-gate 19150Sstevel@tonic-gate AUTOFS_DPRINT((4, "\tunmount_node cvp=%p\n", (void *)cvp)); 19160Sstevel@tonic-gate 19170Sstevel@tonic-gate ASSERT(vn_vfswlock_held(cvp)); 19180Sstevel@tonic-gate cfnp = vntofn(cvp); 19190Sstevel@tonic-gate vfsp = vn_mountedvfs(cvp); 19200Sstevel@tonic-gate 19210Sstevel@tonic-gate if (force || cfnp->fn_flags & MF_IK_MOUNT) { 19220Sstevel@tonic-gate /* 19230Sstevel@tonic-gate * Mount was performed in the kernel, so 19240Sstevel@tonic-gate * do an in-kernel unmount. auto_inkernel_unmount() 19250Sstevel@tonic-gate * will vn_vfsunlock(cvp). 19260Sstevel@tonic-gate */ 19270Sstevel@tonic-gate error = auto_inkernel_unmount(vfsp); 19280Sstevel@tonic-gate } else { 19290Sstevel@tonic-gate zone_t *zone = NULL; 19300Sstevel@tonic-gate refstr_t *mntpt, *resource; 19310Sstevel@tonic-gate size_t mntoptslen; 19320Sstevel@tonic-gate 19330Sstevel@tonic-gate /* 19340Sstevel@tonic-gate * Get the mnttab information of the node 19350Sstevel@tonic-gate * and ask the daemon to unmount it. 19360Sstevel@tonic-gate */ 19370Sstevel@tonic-gate bzero(&ul, sizeof (ul)); 19380Sstevel@tonic-gate mntfs_getmntopts(vfsp, &ul.mntopts, &mntoptslen); 19390Sstevel@tonic-gate if (ul.mntopts == NULL) { 19402170Sevanl auto_log(cfnp->fn_globals->fng_verbose, 19416046Srmesta cfnp->fn_globals->fng_zoneid, CE_WARN, 19426046Srmesta "unmount_node: no memory"); 19430Sstevel@tonic-gate vn_vfsunlock(cvp); 19440Sstevel@tonic-gate error = ENOMEM; 19450Sstevel@tonic-gate goto done; 19460Sstevel@tonic-gate } 19470Sstevel@tonic-gate if (mntoptslen > AUTOFS_MAXOPTSLEN) 19480Sstevel@tonic-gate ul.mntopts[AUTOFS_MAXOPTSLEN - 1] = '\0'; 19490Sstevel@tonic-gate 19500Sstevel@tonic-gate mntpt = vfs_getmntpoint(vfsp); 19510Sstevel@tonic-gate ul.mntpnt = (char *)refstr_value(mntpt); 19520Sstevel@tonic-gate resource = vfs_getresource(vfsp); 19530Sstevel@tonic-gate ul.mntresource = (char *)refstr_value(resource); 19540Sstevel@tonic-gate 19550Sstevel@tonic-gate fnip = vfstofni(cvp->v_vfsp); 19560Sstevel@tonic-gate ul.isdirect = fnip->fi_flags & MF_DIRECT ? TRUE : FALSE; 19570Sstevel@tonic-gate 19580Sstevel@tonic-gate /* 19590Sstevel@tonic-gate * Since a zone'd automountd's view of the autofs mount points 19600Sstevel@tonic-gate * differs from those in the kernel, we need to make sure we 19610Sstevel@tonic-gate * give it consistent mount points. 19620Sstevel@tonic-gate */ 19630Sstevel@tonic-gate ASSERT(fnip->fi_zoneid == getzoneid()); 19640Sstevel@tonic-gate zone = curproc->p_zone; 19650Sstevel@tonic-gate 19660Sstevel@tonic-gate if (fnip->fi_zoneid != GLOBAL_ZONEID) { 19670Sstevel@tonic-gate if (ZONE_PATH_VISIBLE(ul.mntpnt, zone)) { 19680Sstevel@tonic-gate ul.mntpnt = 19690Sstevel@tonic-gate ZONE_PATH_TRANSLATE(ul.mntpnt, zone); 19700Sstevel@tonic-gate } 19710Sstevel@tonic-gate if (ZONE_PATH_VISIBLE(ul.mntresource, zone)) { 19720Sstevel@tonic-gate ul.mntresource = 19730Sstevel@tonic-gate ZONE_PATH_TRANSLATE(ul.mntresource, zone); 19740Sstevel@tonic-gate } 19750Sstevel@tonic-gate } 19762170Sevanl 19770Sstevel@tonic-gate ul.fstype = vfssw[vfsp->vfs_fstype].vsw_name; 19780Sstevel@tonic-gate vn_vfsunlock(cvp); 19790Sstevel@tonic-gate 19802170Sevanl error = auto_send_unmount_request(fnip, &ul, FALSE); 19810Sstevel@tonic-gate kmem_free(ul.mntopts, mntoptslen); 19820Sstevel@tonic-gate refstr_rele(mntpt); 19830Sstevel@tonic-gate refstr_rele(resource); 19840Sstevel@tonic-gate } 19850Sstevel@tonic-gate 19860Sstevel@tonic-gate done: 19870Sstevel@tonic-gate AUTOFS_DPRINT((5, "\tunmount_node cvp=%p error=%d\n", (void *)cvp, 19880Sstevel@tonic-gate error)); 19890Sstevel@tonic-gate return (error); 19900Sstevel@tonic-gate } 19910Sstevel@tonic-gate 19920Sstevel@tonic-gate /* 19930Sstevel@tonic-gate * return EBUSY if any thread is holding a reference to this vnode 1994*12184SJan.Kryl@Sun.COM * other than us. Result of this function cannot be relied on, since 1995*12184SJan.Kryl@Sun.COM * it doesn't follow proper locking rules (i.e. vp->v_vfsmountedhere 1996*12184SJan.Kryl@Sun.COM * and fnp->fn_trigger can change throughout this function). However 1997*12184SJan.Kryl@Sun.COM * it's good enough for rough estimation. 19980Sstevel@tonic-gate */ 19990Sstevel@tonic-gate static int 20000Sstevel@tonic-gate check_auto_node(vnode_t *vp) 20010Sstevel@tonic-gate { 20020Sstevel@tonic-gate fnnode_t *fnp; 20030Sstevel@tonic-gate int error = 0; 20040Sstevel@tonic-gate /* 20050Sstevel@tonic-gate * number of references to expect for 20060Sstevel@tonic-gate * a non-busy vnode. 20070Sstevel@tonic-gate */ 20080Sstevel@tonic-gate uint_t count; 20090Sstevel@tonic-gate 20100Sstevel@tonic-gate AUTOFS_DPRINT((4, "\tcheck_auto_node vp=%p ", (void *)vp)); 20110Sstevel@tonic-gate fnp = vntofn(vp); 20120Sstevel@tonic-gate 20130Sstevel@tonic-gate count = 1; /* we are holding a reference to vp */ 20140Sstevel@tonic-gate if (fnp->fn_flags & MF_TRIGGER) { 20150Sstevel@tonic-gate /* 20160Sstevel@tonic-gate * parent holds a pointer to us (trigger) 20170Sstevel@tonic-gate */ 20180Sstevel@tonic-gate count++; 20190Sstevel@tonic-gate } 20200Sstevel@tonic-gate if (fnp->fn_trigger != NULL) { 20210Sstevel@tonic-gate /* 20220Sstevel@tonic-gate * The trigger nodes have a hold on us. 20230Sstevel@tonic-gate */ 20240Sstevel@tonic-gate count++; 20250Sstevel@tonic-gate } 2026*12184SJan.Kryl@Sun.COM if (vn_ismntpt(vp)) { 2027*12184SJan.Kryl@Sun.COM /* 2028*12184SJan.Kryl@Sun.COM * File system is mounted on us. 2029*12184SJan.Kryl@Sun.COM */ 2030*12184SJan.Kryl@Sun.COM count++; 2031*12184SJan.Kryl@Sun.COM } 20320Sstevel@tonic-gate mutex_enter(&vp->v_lock); 2033*12184SJan.Kryl@Sun.COM ASSERT(vp->v_count > 0); 20340Sstevel@tonic-gate if (vp->v_flag & VROOT) 20350Sstevel@tonic-gate count++; 20360Sstevel@tonic-gate AUTOFS_DPRINT((10, "\tcount=%u ", vp->v_count)); 20370Sstevel@tonic-gate if (vp->v_count > count) 20380Sstevel@tonic-gate error = EBUSY; 20390Sstevel@tonic-gate mutex_exit(&vp->v_lock); 20400Sstevel@tonic-gate 20410Sstevel@tonic-gate AUTOFS_DPRINT((5, "\tcheck_auto_node error=%d ", error)); 20420Sstevel@tonic-gate return (error); 20430Sstevel@tonic-gate } 20440Sstevel@tonic-gate 20450Sstevel@tonic-gate /* 20460Sstevel@tonic-gate * rootvp is the root of the AUTOFS filesystem. 20470Sstevel@tonic-gate * If rootvp is busy (v_count > 1) returns EBUSY. 20480Sstevel@tonic-gate * else removes every vnode under this tree. 20490Sstevel@tonic-gate * ASSUMPTION: Assumes that the only node which can be busy is 20500Sstevel@tonic-gate * the root vnode. This filesystem better be two levels deep only, 20510Sstevel@tonic-gate * the root and its immediate subdirs. 20520Sstevel@tonic-gate * The daemon will "AUTOFS direct-mount" only one level below the root. 20530Sstevel@tonic-gate */ 2054*12184SJan.Kryl@Sun.COM static void 20550Sstevel@tonic-gate unmount_autofs(vnode_t *rootvp) 20560Sstevel@tonic-gate { 20570Sstevel@tonic-gate fnnode_t *fnp, *rootfnp, *nfnp; 20580Sstevel@tonic-gate 20590Sstevel@tonic-gate AUTOFS_DPRINT((4, "\tunmount_autofs rootvp=%p ", (void *)rootvp)); 20600Sstevel@tonic-gate 2061*12184SJan.Kryl@Sun.COM /* 2062*12184SJan.Kryl@Sun.COM * Remove all its immediate subdirectories. 2063*12184SJan.Kryl@Sun.COM */ 2064*12184SJan.Kryl@Sun.COM rootfnp = vntofn(rootvp); 2065*12184SJan.Kryl@Sun.COM rw_enter(&rootfnp->fn_rwlock, RW_WRITER); 2066*12184SJan.Kryl@Sun.COM for (fnp = rootfnp->fn_dirents; fnp != NULL; fnp = nfnp) { 2067*12184SJan.Kryl@Sun.COM ASSERT(fntovn(fnp)->v_count == 0); 2068*12184SJan.Kryl@Sun.COM ASSERT(fnp->fn_dirents == NULL); 2069*12184SJan.Kryl@Sun.COM ASSERT(fnp->fn_linkcnt == 2); 2070*12184SJan.Kryl@Sun.COM fnp->fn_linkcnt--; 2071*12184SJan.Kryl@Sun.COM auto_disconnect(rootfnp, fnp); 2072*12184SJan.Kryl@Sun.COM nfnp = fnp->fn_next; 2073*12184SJan.Kryl@Sun.COM auto_freefnnode(fnp); 20740Sstevel@tonic-gate } 2075*12184SJan.Kryl@Sun.COM rw_exit(&rootfnp->fn_rwlock); 20760Sstevel@tonic-gate } 20770Sstevel@tonic-gate 20780Sstevel@tonic-gate /* 2079*12184SJan.Kryl@Sun.COM * If a node matches all unmount criteria, do: 2080*12184SJan.Kryl@Sun.COM * destroy subordinate trigger node(s) if there is any 2081*12184SJan.Kryl@Sun.COM * unmount filesystem mounted on top of the node if there is any 2082*12184SJan.Kryl@Sun.COM * 2083*12184SJan.Kryl@Sun.COM * Function should be called with locked fnp's mutex. The mutex is 2084*12184SJan.Kryl@Sun.COM * unlocked before return from function. 20850Sstevel@tonic-gate */ 2086*12184SJan.Kryl@Sun.COM static int 2087*12184SJan.Kryl@Sun.COM try_unmount_node(fnnode_t *fnp, boolean_t force) 20880Sstevel@tonic-gate { 2089*12184SJan.Kryl@Sun.COM boolean_t trigger_unmount = B_FALSE; 2090*12184SJan.Kryl@Sun.COM action_list *alp = NULL; 2091*12184SJan.Kryl@Sun.COM vnode_t *vp; 2092*12184SJan.Kryl@Sun.COM int error = 0; 2093*12184SJan.Kryl@Sun.COM fninfo_t *fnip; 2094*12184SJan.Kryl@Sun.COM vfs_t *vfsp; 2095*12184SJan.Kryl@Sun.COM struct autofs_globals *fngp; 20960Sstevel@tonic-gate 2097*12184SJan.Kryl@Sun.COM AUTOFS_DPRINT((10, "\ttry_unmount_node: processing node %p\n", 2098*12184SJan.Kryl@Sun.COM (void *)fnp)); 20990Sstevel@tonic-gate 2100*12184SJan.Kryl@Sun.COM ASSERT(MUTEX_HELD(&fnp->fn_lock)); 2101*12184SJan.Kryl@Sun.COM 2102*12184SJan.Kryl@Sun.COM fngp = fnp->fn_globals; 21030Sstevel@tonic-gate vp = fntovn(fnp); 21040Sstevel@tonic-gate fnip = vfstofni(vp->v_vfsp); 2105*12184SJan.Kryl@Sun.COM 21060Sstevel@tonic-gate /* 2107*12184SJan.Kryl@Sun.COM * If either a mount, lookup or another unmount of this subtree is in 2108*12184SJan.Kryl@Sun.COM * progress, don't attempt to unmount at this time. 21090Sstevel@tonic-gate */ 2110*12184SJan.Kryl@Sun.COM if (fnp->fn_flags & (MF_INPROG | MF_LOOKUP)) { 2111*12184SJan.Kryl@Sun.COM mutex_exit(&fnp->fn_lock); 2112*12184SJan.Kryl@Sun.COM return (EBUSY); 21130Sstevel@tonic-gate } 21140Sstevel@tonic-gate 21150Sstevel@tonic-gate /* 2116*12184SJan.Kryl@Sun.COM * Bail out if someone else is holding reference to this vnode. 2117*12184SJan.Kryl@Sun.COM * This check isn't just an optimization (someone is probably 2118*12184SJan.Kryl@Sun.COM * just about to trigger mount). It is necessary to prevent a deadlock 2119*12184SJan.Kryl@Sun.COM * in domount() called from auto_perform_actions() if unmount of 2120*12184SJan.Kryl@Sun.COM * trigger parent fails. domount() calls lookupname() to resolve 2121*12184SJan.Kryl@Sun.COM * special in mount arguments. Special is set to a map name in case 2122*12184SJan.Kryl@Sun.COM * of autofs triggers (i.e. auto_ws.sun.com). Thus if current 2123*12184SJan.Kryl@Sun.COM * working directory is set to currently processed node, lookupname() 2124*12184SJan.Kryl@Sun.COM * calls into autofs vnops in order to resolve special, which deadlocks 2125*12184SJan.Kryl@Sun.COM * the process. 2126*12184SJan.Kryl@Sun.COM * 2127*12184SJan.Kryl@Sun.COM * Note: This should be fixed. Autofs shouldn't pass the map name 2128*12184SJan.Kryl@Sun.COM * in special and avoid useless lookup with potentially disasterous 2129*12184SJan.Kryl@Sun.COM * consequence. 21300Sstevel@tonic-gate */ 2131*12184SJan.Kryl@Sun.COM if (check_auto_node(vp) == EBUSY) { 21320Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 2133*12184SJan.Kryl@Sun.COM return (EBUSY); 21340Sstevel@tonic-gate } 21350Sstevel@tonic-gate 2136*12184SJan.Kryl@Sun.COM /* 2137*12184SJan.Kryl@Sun.COM * If not forced operation, back out if node has been referenced 2138*12184SJan.Kryl@Sun.COM * recently. 2139*12184SJan.Kryl@Sun.COM */ 2140*12184SJan.Kryl@Sun.COM if (!force && 2141*12184SJan.Kryl@Sun.COM fnp->fn_ref_time + fnip->fi_mount_to > gethrestime_sec()) { 2142*12184SJan.Kryl@Sun.COM mutex_exit(&fnp->fn_lock); 2143*12184SJan.Kryl@Sun.COM return (EBUSY); 2144*12184SJan.Kryl@Sun.COM } 2145*12184SJan.Kryl@Sun.COM 2146*12184SJan.Kryl@Sun.COM /* block mounts/unmounts on the node */ 21470Sstevel@tonic-gate AUTOFS_BLOCK_OTHERS(fnp, MF_INPROG); 21480Sstevel@tonic-gate fnp->fn_error = 0; 21490Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 21500Sstevel@tonic-gate 2151*12184SJan.Kryl@Sun.COM /* unmount next level triggers if there are any */ 21520Sstevel@tonic-gate rw_enter(&fnp->fn_rwlock, RW_WRITER); 21530Sstevel@tonic-gate if (fnp->fn_trigger != NULL) { 2154*12184SJan.Kryl@Sun.COM trigger_unmount = B_TRUE; 21550Sstevel@tonic-gate 2156*12184SJan.Kryl@Sun.COM if (triggers_busy(fnp)) { 21570Sstevel@tonic-gate rw_exit(&fnp->fn_rwlock); 21580Sstevel@tonic-gate mutex_enter(&fnp->fn_lock); 21590Sstevel@tonic-gate AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 21600Sstevel@tonic-gate mutex_exit(&fnp->fn_lock); 2161*12184SJan.Kryl@Sun.COM return (EBUSY); 21620Sstevel@tonic-gate } 21630Sstevel@tonic-gate 21640Sstevel@tonic-gate /* 21650Sstevel@tonic-gate * At this point, we know all trigger nodes are locked, 21660Sstevel@tonic-gate * and they're not busy or mounted on. 2167*12184SJan.Kryl@Sun.COM * 2168*12184SJan.Kryl@Sun.COM * Attempt to unmount all trigger nodes, save the 2169*12184SJan.Kryl@Sun.COM * action_list in case we need to remount them later. 2170*12184SJan.Kryl@Sun.COM * The action_list will be freed later if there was no 2171*12184SJan.Kryl@Sun.COM * need to remount the trigger nodes. 21720Sstevel@tonic-gate */ 2173*12184SJan.Kryl@Sun.COM unmount_triggers(fnp, &alp); 21740Sstevel@tonic-gate } 21750Sstevel@tonic-gate rw_exit(&fnp->fn_rwlock); 21760Sstevel@tonic-gate 21770Sstevel@tonic-gate (void) vn_vfswlock_wait(vp); 21780Sstevel@tonic-gate 21790Sstevel@tonic-gate vfsp = vn_mountedvfs(vp); 21800Sstevel@tonic-gate if (vfsp != NULL) { 2181*12184SJan.Kryl@Sun.COM /* vn_vfsunlock(vp) is done inside unmount_node() */ 21820Sstevel@tonic-gate error = unmount_node(vp, force); 21830Sstevel@tonic-gate if (error == ECONNRESET) { 21840Sstevel@tonic-gate if (vn_mountedvfs(vp) == NULL) { 21850Sstevel@tonic-gate /* 21860Sstevel@tonic-gate * The filesystem was unmounted before the 21870Sstevel@tonic-gate * daemon died. Unfortunately we can not 21880Sstevel@tonic-gate * determine whether all the cleanup work was 21890Sstevel@tonic-gate * successfully finished (i.e. update mnttab, 21900Sstevel@tonic-gate * or notify NFS server of the unmount). 21910Sstevel@tonic-gate * We should not retry the operation since the 21920Sstevel@tonic-gate * filesystem has already been unmounted, and 21930Sstevel@tonic-gate * may have already been removed from mnttab, 21940Sstevel@tonic-gate * in such case the devid/rdevid we send to 21950Sstevel@tonic-gate * the daemon will not be matched. So we have 21962170Sevanl * to be content with the partial unmount. 21970Sstevel@tonic-gate * Since the mountpoint is no longer covered, we 21980Sstevel@tonic-gate * clear the error condition. 21990Sstevel@tonic-gate */ 22000Sstevel@tonic-gate error = 0; 22012170Sevanl auto_log(fngp->fng_verbose, fngp->fng_zoneid, 2202*12184SJan.Kryl@Sun.COM CE_WARN, "autofs: automountd " 2203*12184SJan.Kryl@Sun.COM "connection dropped when unmounting %s/%s", 2204*12184SJan.Kryl@Sun.COM fnip->fi_path, (fnip->fi_flags & MF_DIRECT) 2205*12184SJan.Kryl@Sun.COM ? "" : fnp->fn_name); 22060Sstevel@tonic-gate } 22070Sstevel@tonic-gate } 22080Sstevel@tonic-gate } else { 22090Sstevel@tonic-gate vn_vfsunlock(vp); 2210*12184SJan.Kryl@Sun.COM /* Destroy all dirents of fnp if we unmounted its triggers */ 2211*12184SJan.Kryl@Sun.COM if (trigger_unmount) 2212*12184SJan.Kryl@Sun.COM unmount_autofs(vp); 2213*12184SJan.Kryl@Sun.COM } 2214*12184SJan.Kryl@Sun.COM 2215*12184SJan.Kryl@Sun.COM /* If unmount failed, we got to remount triggers */ 2216*12184SJan.Kryl@Sun.COM if (error != 0) { 2217*12184SJan.Kryl@Sun.COM if (trigger_unmount) { 2218*12184SJan.Kryl@Sun.COM int ret; 2219*12184SJan.Kryl@Sun.COM 2220*12184SJan.Kryl@Sun.COM ASSERT((fnp->fn_flags & MF_THISUID_MATCH_RQD) == 0); 22210Sstevel@tonic-gate 2222*12184SJan.Kryl@Sun.COM /* 2223*12184SJan.Kryl@Sun.COM * The action list was free'd by auto_perform_actions 2224*12184SJan.Kryl@Sun.COM */ 2225*12184SJan.Kryl@Sun.COM ret = auto_perform_actions(fnip, fnp, alp, CRED()); 2226*12184SJan.Kryl@Sun.COM if (ret != 0) { 2227*12184SJan.Kryl@Sun.COM auto_log(fngp->fng_verbose, fngp->fng_zoneid, 2228*12184SJan.Kryl@Sun.COM CE_WARN, "autofs: can't remount triggers " 2229*12184SJan.Kryl@Sun.COM "fnp=%p error=%d", (void *)fnp, ret); 22300Sstevel@tonic-gate } 22310Sstevel@tonic-gate } 2232*12184SJan.Kryl@Sun.COM mutex_enter(&fnp->fn_lock); 2233*12184SJan.Kryl@Sun.COM AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2234*12184SJan.Kryl@Sun.COM mutex_exit(&fnp->fn_lock); 2235*12184SJan.Kryl@Sun.COM } else { 2236*12184SJan.Kryl@Sun.COM /* Free the action list here */ 2237*12184SJan.Kryl@Sun.COM if (trigger_unmount) 2238*12184SJan.Kryl@Sun.COM xdr_free(xdr_action_list, (char *)alp); 2239*12184SJan.Kryl@Sun.COM 2240*12184SJan.Kryl@Sun.COM /* 2241*12184SJan.Kryl@Sun.COM * Other threads may be waiting for this unmount to 2242*12184SJan.Kryl@Sun.COM * finish. We must let it know that in order to 2243*12184SJan.Kryl@Sun.COM * proceed, it must trigger the mount itself. 2244*12184SJan.Kryl@Sun.COM */ 2245*12184SJan.Kryl@Sun.COM mutex_enter(&fnp->fn_lock); 2246*12184SJan.Kryl@Sun.COM fnp->fn_flags &= ~MF_IK_MOUNT; 2247*12184SJan.Kryl@Sun.COM if (fnp->fn_flags & MF_WAITING) 2248*12184SJan.Kryl@Sun.COM fnp->fn_error = EAGAIN; 2249*12184SJan.Kryl@Sun.COM AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2250*12184SJan.Kryl@Sun.COM mutex_exit(&fnp->fn_lock); 22510Sstevel@tonic-gate } 22520Sstevel@tonic-gate 2253*12184SJan.Kryl@Sun.COM return (error); 2254*12184SJan.Kryl@Sun.COM } 2255*12184SJan.Kryl@Sun.COM 2256*12184SJan.Kryl@Sun.COM /* 2257*12184SJan.Kryl@Sun.COM * This is an implementation of depth-first search in a tree rooted by 2258*12184SJan.Kryl@Sun.COM * start_fnp and composed from fnnodes. Links between tree levels are 2259*12184SJan.Kryl@Sun.COM * fn_dirents, fn_trigger in fnnode_t and v_mountedvfs in vnode_t (if 2260*12184SJan.Kryl@Sun.COM * mounted vfs is autofs). The algorithm keeps track of visited nodes 2261*12184SJan.Kryl@Sun.COM * by means of a timestamp (fn_unmount_ref_time). 2262*12184SJan.Kryl@Sun.COM * 2263*12184SJan.Kryl@Sun.COM * Upon top-down traversal of the tree we apply following locking scheme: 2264*12184SJan.Kryl@Sun.COM * lock fn_rwlock of current node 2265*12184SJan.Kryl@Sun.COM * grab reference to child's vnode (VN_HOLD) 2266*12184SJan.Kryl@Sun.COM * unlock fn_rwlock 2267*12184SJan.Kryl@Sun.COM * free reference to current vnode (VN_RELE) 2268*12184SJan.Kryl@Sun.COM * Similar locking scheme is used for down-top and left-right traversal. 2269*12184SJan.Kryl@Sun.COM * 2270*12184SJan.Kryl@Sun.COM * Algorithm examines the most down-left node in tree, which hasn't been 2271*12184SJan.Kryl@Sun.COM * visited yet. From this follows that nodes are processed in bottom-up 2272*12184SJan.Kryl@Sun.COM * fashion. 2273*12184SJan.Kryl@Sun.COM * 2274*12184SJan.Kryl@Sun.COM * Function returns either zero if unmount of root node was successful 2275*12184SJan.Kryl@Sun.COM * or error code (mostly EBUSY). 2276*12184SJan.Kryl@Sun.COM */ 2277*12184SJan.Kryl@Sun.COM int 2278*12184SJan.Kryl@Sun.COM unmount_subtree(fnnode_t *rootfnp, boolean_t force) 2279*12184SJan.Kryl@Sun.COM { 2280*12184SJan.Kryl@Sun.COM fnnode_t *currfnp; /* currently examined node in the tree */ 2281*12184SJan.Kryl@Sun.COM fnnode_t *lastfnp; /* previously processed node */ 2282*12184SJan.Kryl@Sun.COM fnnode_t *nextfnp; /* next examined node in the tree */ 2283*12184SJan.Kryl@Sun.COM vnode_t *curvp; 2284*12184SJan.Kryl@Sun.COM vnode_t *newvp; 2285*12184SJan.Kryl@Sun.COM vfs_t *vfsp; 2286*12184SJan.Kryl@Sun.COM time_t timestamp; 2287*12184SJan.Kryl@Sun.COM 2288*12184SJan.Kryl@Sun.COM ASSERT(fntovn(rootfnp)->v_type != VLNK); 2289*12184SJan.Kryl@Sun.COM AUTOFS_DPRINT((10, "unmount_subtree: root=%p (%s)\n", (void *)rootfnp, 2290*12184SJan.Kryl@Sun.COM rootfnp->fn_name)); 2291*12184SJan.Kryl@Sun.COM 2292*12184SJan.Kryl@Sun.COM /* 2293*12184SJan.Kryl@Sun.COM * Timestamp, which visited nodes are marked with, to distinguish them 2294*12184SJan.Kryl@Sun.COM * from unvisited nodes. 2295*12184SJan.Kryl@Sun.COM */ 2296*12184SJan.Kryl@Sun.COM timestamp = gethrestime_sec(); 2297*12184SJan.Kryl@Sun.COM currfnp = lastfnp = rootfnp; 2298*12184SJan.Kryl@Sun.COM 2299*12184SJan.Kryl@Sun.COM /* Loop until we examine all nodes in the tree */ 2300*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 2301*12184SJan.Kryl@Sun.COM while (currfnp != rootfnp || rootfnp->fn_unmount_ref_time < timestamp) { 2302*12184SJan.Kryl@Sun.COM curvp = fntovn(currfnp); 2303*12184SJan.Kryl@Sun.COM AUTOFS_DPRINT((10, "\tunmount_subtree: entering node %p (%s)\n", 2304*12184SJan.Kryl@Sun.COM (void *)currfnp, currfnp->fn_name)); 2305*12184SJan.Kryl@Sun.COM 2306*12184SJan.Kryl@Sun.COM /* 2307*12184SJan.Kryl@Sun.COM * New candidate for processing must have been already visited, 2308*12184SJan.Kryl@Sun.COM * by us because we want to process tree nodes in bottom-up 2309*12184SJan.Kryl@Sun.COM * order. 2310*12184SJan.Kryl@Sun.COM */ 2311*12184SJan.Kryl@Sun.COM if (currfnp->fn_unmount_ref_time == timestamp && 2312*12184SJan.Kryl@Sun.COM currfnp != lastfnp) { 2313*12184SJan.Kryl@Sun.COM (void) try_unmount_node(currfnp, force); 2314*12184SJan.Kryl@Sun.COM lastfnp = currfnp; 2315*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 23160Sstevel@tonic-gate /* 2317*12184SJan.Kryl@Sun.COM * Fall through to next if-branch to pick 2318*12184SJan.Kryl@Sun.COM * sibling or parent of this node. 23190Sstevel@tonic-gate */ 23200Sstevel@tonic-gate } 23210Sstevel@tonic-gate 23220Sstevel@tonic-gate /* 2323*12184SJan.Kryl@Sun.COM * If this node has been already visited, it means that it's 2324*12184SJan.Kryl@Sun.COM * dead end and we need to pick sibling or parent as next node. 23250Sstevel@tonic-gate */ 2326*12184SJan.Kryl@Sun.COM if (currfnp->fn_unmount_ref_time >= timestamp || 2327*12184SJan.Kryl@Sun.COM curvp->v_type == VLNK) { 2328*12184SJan.Kryl@Sun.COM mutex_exit(&currfnp->fn_lock); 2329*12184SJan.Kryl@Sun.COM /* 2330*12184SJan.Kryl@Sun.COM * Obtain parent's readers lock before grabbing 2331*12184SJan.Kryl@Sun.COM * reference to sibling. 2332*12184SJan.Kryl@Sun.COM */ 2333*12184SJan.Kryl@Sun.COM rw_enter(&currfnp->fn_parent->fn_rwlock, RW_READER); 2334*12184SJan.Kryl@Sun.COM if ((nextfnp = currfnp->fn_next) != NULL) { 2335*12184SJan.Kryl@Sun.COM VN_HOLD(fntovn(nextfnp)); 2336*12184SJan.Kryl@Sun.COM rw_exit(&currfnp->fn_parent->fn_rwlock); 2337*12184SJan.Kryl@Sun.COM VN_RELE(curvp); 2338*12184SJan.Kryl@Sun.COM currfnp = nextfnp; 2339*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 2340*12184SJan.Kryl@Sun.COM continue; 2341*12184SJan.Kryl@Sun.COM } 2342*12184SJan.Kryl@Sun.COM rw_exit(&currfnp->fn_parent->fn_rwlock); 23430Sstevel@tonic-gate 2344*12184SJan.Kryl@Sun.COM /* 2345*12184SJan.Kryl@Sun.COM * All descendants and siblings were visited. Perform 2346*12184SJan.Kryl@Sun.COM * bottom-up move. 2347*12184SJan.Kryl@Sun.COM */ 2348*12184SJan.Kryl@Sun.COM nextfnp = currfnp->fn_parent; 2349*12184SJan.Kryl@Sun.COM VN_HOLD(fntovn(nextfnp)); 2350*12184SJan.Kryl@Sun.COM VN_RELE(curvp); 2351*12184SJan.Kryl@Sun.COM currfnp = nextfnp; 2352*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 2353*12184SJan.Kryl@Sun.COM continue; 2354*12184SJan.Kryl@Sun.COM } 2355*12184SJan.Kryl@Sun.COM 2356*12184SJan.Kryl@Sun.COM /* 2357*12184SJan.Kryl@Sun.COM * Mark node as visited. Note that the timestamp could have 2358*12184SJan.Kryl@Sun.COM * been updated by somebody else in the meantime. 2359*12184SJan.Kryl@Sun.COM */ 2360*12184SJan.Kryl@Sun.COM if (currfnp->fn_unmount_ref_time < timestamp) 2361*12184SJan.Kryl@Sun.COM currfnp->fn_unmount_ref_time = timestamp; 2362*12184SJan.Kryl@Sun.COM mutex_exit(&currfnp->fn_lock); 23630Sstevel@tonic-gate 2364*12184SJan.Kryl@Sun.COM /* 2365*12184SJan.Kryl@Sun.COM * Examine descendants in this order: triggers, dirents, autofs 2366*12184SJan.Kryl@Sun.COM * mounts. 2367*12184SJan.Kryl@Sun.COM */ 2368*12184SJan.Kryl@Sun.COM 2369*12184SJan.Kryl@Sun.COM rw_enter(&currfnp->fn_rwlock, RW_READER); 2370*12184SJan.Kryl@Sun.COM if ((nextfnp = currfnp->fn_trigger) != NULL) { 2371*12184SJan.Kryl@Sun.COM VN_HOLD(fntovn(nextfnp)); 2372*12184SJan.Kryl@Sun.COM rw_exit(&currfnp->fn_rwlock); 2373*12184SJan.Kryl@Sun.COM VN_RELE(curvp); 2374*12184SJan.Kryl@Sun.COM currfnp = nextfnp; 2375*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 2376*12184SJan.Kryl@Sun.COM continue; 2377*12184SJan.Kryl@Sun.COM } 2378*12184SJan.Kryl@Sun.COM 2379*12184SJan.Kryl@Sun.COM if ((nextfnp = currfnp->fn_dirents) != NULL) { 2380*12184SJan.Kryl@Sun.COM VN_HOLD(fntovn(nextfnp)); 2381*12184SJan.Kryl@Sun.COM rw_exit(&currfnp->fn_rwlock); 2382*12184SJan.Kryl@Sun.COM VN_RELE(curvp); 2383*12184SJan.Kryl@Sun.COM currfnp = nextfnp; 2384*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 2385*12184SJan.Kryl@Sun.COM continue; 23860Sstevel@tonic-gate } 2387*12184SJan.Kryl@Sun.COM rw_exit(&currfnp->fn_rwlock); 23880Sstevel@tonic-gate 2389*12184SJan.Kryl@Sun.COM (void) vn_vfswlock_wait(curvp); 2390*12184SJan.Kryl@Sun.COM vfsp = vn_mountedvfs(curvp); 2391*12184SJan.Kryl@Sun.COM if (vfsp != NULL && 2392*12184SJan.Kryl@Sun.COM vfs_matchops(vfsp, vfs_getops(curvp->v_vfsp))) { 2393*12184SJan.Kryl@Sun.COM /* 2394*12184SJan.Kryl@Sun.COM * Deal with /xfn/host/jurassic alikes here... 2395*12184SJan.Kryl@Sun.COM * 2396*12184SJan.Kryl@Sun.COM * We know this call to VFS_ROOT is safe to call while 2397*12184SJan.Kryl@Sun.COM * holding VVFSLOCK, since it resolves to a call to 2398*12184SJan.Kryl@Sun.COM * auto_root(). 2399*12184SJan.Kryl@Sun.COM */ 2400*12184SJan.Kryl@Sun.COM if (VFS_ROOT(vfsp, &newvp)) { 2401*12184SJan.Kryl@Sun.COM cmn_err(CE_PANIC, 2402*12184SJan.Kryl@Sun.COM "autofs: VFS_ROOT(vfs=%p) failed", 2403*12184SJan.Kryl@Sun.COM (void *)vfsp); 2404*12184SJan.Kryl@Sun.COM } 2405*12184SJan.Kryl@Sun.COM vn_vfsunlock(curvp); 2406*12184SJan.Kryl@Sun.COM VN_RELE(curvp); 2407*12184SJan.Kryl@Sun.COM currfnp = vntofn(newvp); 2408*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 2409*12184SJan.Kryl@Sun.COM continue; 2410*12184SJan.Kryl@Sun.COM } 2411*12184SJan.Kryl@Sun.COM vn_vfsunlock(curvp); 2412*12184SJan.Kryl@Sun.COM mutex_enter(&currfnp->fn_lock); 24130Sstevel@tonic-gate } 24140Sstevel@tonic-gate 24150Sstevel@tonic-gate /* 2416*12184SJan.Kryl@Sun.COM * Now we deal with the root node (currfnp's mutex is unlocked 2417*12184SJan.Kryl@Sun.COM * in try_unmount_node()). 24180Sstevel@tonic-gate */ 2419*12184SJan.Kryl@Sun.COM return (try_unmount_node(currfnp, force)); 2420*12184SJan.Kryl@Sun.COM } 2421*12184SJan.Kryl@Sun.COM 2422*12184SJan.Kryl@Sun.COM /* 2423*12184SJan.Kryl@Sun.COM * XXX unmount_tree() is not suspend-safe within the scope of 2424*12184SJan.Kryl@Sun.COM * the present model defined for cpr to suspend the system. Calls made 2425*12184SJan.Kryl@Sun.COM * by the unmount_tree() that have been identified to be unsafe are 2426*12184SJan.Kryl@Sun.COM * (1) RPC client handle setup and client calls to automountd which can 2427*12184SJan.Kryl@Sun.COM * block deep down in the RPC library, (2) kmem_alloc() calls with the 2428*12184SJan.Kryl@Sun.COM * KM_SLEEP flag which can block if memory is low, and (3) VFS_*() and 2429*12184SJan.Kryl@Sun.COM * VOP_*() calls which can result in over the wire calls to servers. 2430*12184SJan.Kryl@Sun.COM * The thread should be completely reevaluated to make it suspend-safe in 2431*12184SJan.Kryl@Sun.COM * case of future updates to the cpr model. 2432*12184SJan.Kryl@Sun.COM */ 2433*12184SJan.Kryl@Sun.COM void 2434*12184SJan.Kryl@Sun.COM unmount_tree(struct autofs_globals *fngp, boolean_t force) 2435*12184SJan.Kryl@Sun.COM { 2436*12184SJan.Kryl@Sun.COM callb_cpr_t cprinfo; 2437*12184SJan.Kryl@Sun.COM kmutex_t unmount_tree_cpr_lock; 2438*12184SJan.Kryl@Sun.COM fnnode_t *root, *fnp, *next; 2439*12184SJan.Kryl@Sun.COM 2440*12184SJan.Kryl@Sun.COM mutex_init(&unmount_tree_cpr_lock, NULL, MUTEX_DEFAULT, NULL); 2441*12184SJan.Kryl@Sun.COM CALLB_CPR_INIT(&cprinfo, &unmount_tree_cpr_lock, callb_generic_cpr, 2442*12184SJan.Kryl@Sun.COM "unmount_tree"); 2443*12184SJan.Kryl@Sun.COM 2444*12184SJan.Kryl@Sun.COM /* 2445*12184SJan.Kryl@Sun.COM * autofssys() will be calling in from the global zone and doing 2446*12184SJan.Kryl@Sun.COM * work on the behalf of the given zone, hence we can't always 2447*12184SJan.Kryl@Sun.COM * assert that we have the right credentials, nor that the 2448*12184SJan.Kryl@Sun.COM * caller is always in the correct zone. 2449*12184SJan.Kryl@Sun.COM * 2450*12184SJan.Kryl@Sun.COM * We do, however, know that if this is a "forced unmount" 2451*12184SJan.Kryl@Sun.COM * operation (which autofssys() does), then we won't go down to 2452*12184SJan.Kryl@Sun.COM * the krpc layers, so we don't need to fudge with the 2453*12184SJan.Kryl@Sun.COM * credentials. 2454*12184SJan.Kryl@Sun.COM */ 2455*12184SJan.Kryl@Sun.COM ASSERT(force || fngp->fng_zoneid == getzoneid()); 2456*12184SJan.Kryl@Sun.COM 2457*12184SJan.Kryl@Sun.COM /* 2458*12184SJan.Kryl@Sun.COM * If automountd is not running in this zone, 2459*12184SJan.Kryl@Sun.COM * don't attempt unmounting this round. 2460*12184SJan.Kryl@Sun.COM */ 2461*12184SJan.Kryl@Sun.COM if (force || auto_null_request(fngp->fng_zoneid, FALSE) == 0) { 24620Sstevel@tonic-gate /* 2463*12184SJan.Kryl@Sun.COM * Iterate over top level autofs filesystems and call 2464*12184SJan.Kryl@Sun.COM * unmount_subtree() for each of them. 24650Sstevel@tonic-gate */ 2466*12184SJan.Kryl@Sun.COM root = fngp->fng_rootfnnodep; 2467*12184SJan.Kryl@Sun.COM rw_enter(&root->fn_rwlock, RW_READER); 2468*12184SJan.Kryl@Sun.COM for (fnp = root->fn_dirents; fnp != NULL; fnp = next) { 2469*12184SJan.Kryl@Sun.COM VN_HOLD(fntovn(fnp)); 2470*12184SJan.Kryl@Sun.COM rw_exit(&root->fn_rwlock); 2471*12184SJan.Kryl@Sun.COM (void) unmount_subtree(fnp, force); 2472*12184SJan.Kryl@Sun.COM rw_enter(&root->fn_rwlock, RW_READER); 2473*12184SJan.Kryl@Sun.COM next = fnp->fn_next; 2474*12184SJan.Kryl@Sun.COM VN_RELE(fntovn(fnp)); 2475*12184SJan.Kryl@Sun.COM } 2476*12184SJan.Kryl@Sun.COM rw_exit(&root->fn_rwlock); 24770Sstevel@tonic-gate } 24780Sstevel@tonic-gate 24790Sstevel@tonic-gate mutex_enter(&unmount_tree_cpr_lock); 24800Sstevel@tonic-gate CALLB_CPR_EXIT(&cprinfo); 24810Sstevel@tonic-gate mutex_destroy(&unmount_tree_cpr_lock); 24820Sstevel@tonic-gate } 24830Sstevel@tonic-gate 24840Sstevel@tonic-gate static void 24850Sstevel@tonic-gate unmount_zone_tree(struct autofs_globals *fngp) 24860Sstevel@tonic-gate { 2487*12184SJan.Kryl@Sun.COM AUTOFS_DPRINT((5, "unmount_zone_tree started. Thread created.\n")); 2488*12184SJan.Kryl@Sun.COM 2489*12184SJan.Kryl@Sun.COM unmount_tree(fngp, B_FALSE); 24900Sstevel@tonic-gate mutex_enter(&fngp->fng_unmount_threads_lock); 24910Sstevel@tonic-gate fngp->fng_unmount_threads--; 24920Sstevel@tonic-gate mutex_exit(&fngp->fng_unmount_threads_lock); 24930Sstevel@tonic-gate 2494*12184SJan.Kryl@Sun.COM AUTOFS_DPRINT((5, "unmount_zone_tree done. Thread exiting.\n")); 24950Sstevel@tonic-gate 24960Sstevel@tonic-gate zthread_exit(); 24970Sstevel@tonic-gate /* NOTREACHED */ 24980Sstevel@tonic-gate } 24990Sstevel@tonic-gate 25000Sstevel@tonic-gate void 25010Sstevel@tonic-gate auto_do_unmount(struct autofs_globals *fngp) 25020Sstevel@tonic-gate { 25030Sstevel@tonic-gate callb_cpr_t cprinfo; 25040Sstevel@tonic-gate clock_t timeleft; 25050Sstevel@tonic-gate zone_t *zone = curproc->p_zone; 25060Sstevel@tonic-gate 25070Sstevel@tonic-gate CALLB_CPR_INIT(&cprinfo, &fngp->fng_unmount_threads_lock, 25086046Srmesta callb_generic_cpr, "auto_do_unmount"); 25090Sstevel@tonic-gate 25100Sstevel@tonic-gate for (;;) { /* forever */ 25110Sstevel@tonic-gate mutex_enter(&fngp->fng_unmount_threads_lock); 25120Sstevel@tonic-gate CALLB_CPR_SAFE_BEGIN(&cprinfo); 25130Sstevel@tonic-gate newthread: 25140Sstevel@tonic-gate mutex_exit(&fngp->fng_unmount_threads_lock); 251511066Srafael.vanoni@sun.com timeleft = zone_status_timedwait(zone, ddi_get_lbolt() + 25160Sstevel@tonic-gate autofs_unmount_thread_timer * hz, ZONE_IS_SHUTTING_DOWN); 25170Sstevel@tonic-gate mutex_enter(&fngp->fng_unmount_threads_lock); 25180Sstevel@tonic-gate 25190Sstevel@tonic-gate if (timeleft != -1) { /* didn't time out */ 25200Sstevel@tonic-gate ASSERT(zone_status_get(zone) >= ZONE_IS_SHUTTING_DOWN); 25210Sstevel@tonic-gate /* 25220Sstevel@tonic-gate * zone is exiting... don't create any new threads. 25230Sstevel@tonic-gate * fng_unmount_threads_lock is released implicitly by 25240Sstevel@tonic-gate * the below. 25250Sstevel@tonic-gate */ 25260Sstevel@tonic-gate CALLB_CPR_SAFE_END(&cprinfo, 25276046Srmesta &fngp->fng_unmount_threads_lock); 25280Sstevel@tonic-gate CALLB_CPR_EXIT(&cprinfo); 25290Sstevel@tonic-gate zthread_exit(); 25300Sstevel@tonic-gate /* NOTREACHED */ 25310Sstevel@tonic-gate } 25320Sstevel@tonic-gate if (fngp->fng_unmount_threads < autofs_unmount_threads) { 25330Sstevel@tonic-gate fngp->fng_unmount_threads++; 25340Sstevel@tonic-gate CALLB_CPR_SAFE_END(&cprinfo, 25356046Srmesta &fngp->fng_unmount_threads_lock); 25360Sstevel@tonic-gate mutex_exit(&fngp->fng_unmount_threads_lock); 25370Sstevel@tonic-gate 25380Sstevel@tonic-gate (void) zthread_create(NULL, 0, unmount_zone_tree, fngp, 25390Sstevel@tonic-gate 0, minclsyspri); 25400Sstevel@tonic-gate } else 25410Sstevel@tonic-gate goto newthread; 25420Sstevel@tonic-gate } 25430Sstevel@tonic-gate /* NOTREACHED */ 25440Sstevel@tonic-gate } 25450Sstevel@tonic-gate 25460Sstevel@tonic-gate /* 25470Sstevel@tonic-gate * Is nobrowse specified in option string? 25480Sstevel@tonic-gate * opts should be a null ('\0') terminated string. 25490Sstevel@tonic-gate * Returns non-zero if nobrowse has been specified. 25500Sstevel@tonic-gate */ 25510Sstevel@tonic-gate int 25520Sstevel@tonic-gate auto_nobrowse_option(char *opts) 25530Sstevel@tonic-gate { 25540Sstevel@tonic-gate char *buf; 25550Sstevel@tonic-gate char *p; 25560Sstevel@tonic-gate char *t; 25570Sstevel@tonic-gate int nobrowse = 0; 25580Sstevel@tonic-gate int last_opt = 0; 25590Sstevel@tonic-gate size_t len; 25600Sstevel@tonic-gate 25610Sstevel@tonic-gate len = strlen(opts) + 1; 25620Sstevel@tonic-gate p = buf = kmem_alloc(len, KM_SLEEP); 25630Sstevel@tonic-gate (void) strcpy(buf, opts); 25640Sstevel@tonic-gate do { 25650Sstevel@tonic-gate if (t = strchr(p, ',')) 25660Sstevel@tonic-gate *t++ = '\0'; 25670Sstevel@tonic-gate else 25680Sstevel@tonic-gate last_opt++; 25690Sstevel@tonic-gate if (strcmp(p, MNTOPT_NOBROWSE) == 0) 25700Sstevel@tonic-gate nobrowse = 1; 25710Sstevel@tonic-gate else if (strcmp(p, MNTOPT_BROWSE) == 0) 25720Sstevel@tonic-gate nobrowse = 0; 25730Sstevel@tonic-gate p = t; 25740Sstevel@tonic-gate } while (!last_opt); 25750Sstevel@tonic-gate kmem_free(buf, len); 25760Sstevel@tonic-gate 25770Sstevel@tonic-gate return (nobrowse); 25780Sstevel@tonic-gate } 25790Sstevel@tonic-gate 25800Sstevel@tonic-gate /* 25810Sstevel@tonic-gate * used to log warnings only if automountd is running 25820Sstevel@tonic-gate * with verbose mode set 25830Sstevel@tonic-gate */ 25842170Sevanl 25850Sstevel@tonic-gate void 25862170Sevanl auto_log(int verbose, zoneid_t zoneid, int level, const char *fmt, ...) 25870Sstevel@tonic-gate { 25882170Sevanl va_list args; 25890Sstevel@tonic-gate 25902170Sevanl if (verbose) { 25910Sstevel@tonic-gate va_start(args, fmt); 25922170Sevanl vzcmn_err(zoneid, level, fmt, args); 25930Sstevel@tonic-gate va_end(args); 25940Sstevel@tonic-gate } 25950Sstevel@tonic-gate } 25960Sstevel@tonic-gate 25970Sstevel@tonic-gate #ifdef DEBUG 25980Sstevel@tonic-gate static int autofs_debug = 0; 25990Sstevel@tonic-gate 26000Sstevel@tonic-gate /* 26010Sstevel@tonic-gate * Utilities used by both client and server 26020Sstevel@tonic-gate * Standard levels: 26030Sstevel@tonic-gate * 0) no debugging 26040Sstevel@tonic-gate * 1) hard failures 26050Sstevel@tonic-gate * 2) soft failures 26060Sstevel@tonic-gate * 3) current test software 26070Sstevel@tonic-gate * 4) main procedure entry points 26080Sstevel@tonic-gate * 5) main procedure exit points 26090Sstevel@tonic-gate * 6) utility procedure entry points 26100Sstevel@tonic-gate * 7) utility procedure exit points 26110Sstevel@tonic-gate * 8) obscure procedure entry points 26120Sstevel@tonic-gate * 9) obscure procedure exit points 26130Sstevel@tonic-gate * 10) random stuff 26140Sstevel@tonic-gate * 11) all <= 1 26150Sstevel@tonic-gate * 12) all <= 2 26160Sstevel@tonic-gate * 13) all <= 3 26170Sstevel@tonic-gate * ... 26180Sstevel@tonic-gate */ 26190Sstevel@tonic-gate /* PRINTFLIKE2 */ 26200Sstevel@tonic-gate void 26210Sstevel@tonic-gate auto_dprint(int level, const char *fmt, ...) 26220Sstevel@tonic-gate { 26230Sstevel@tonic-gate va_list args; 26240Sstevel@tonic-gate 26250Sstevel@tonic-gate if (autofs_debug == level || 26260Sstevel@tonic-gate (autofs_debug > 10 && (autofs_debug - 10) >= level)) { 26270Sstevel@tonic-gate va_start(args, fmt); 26280Sstevel@tonic-gate (void) vprintf(fmt, args); 26290Sstevel@tonic-gate va_end(args); 26300Sstevel@tonic-gate } 26310Sstevel@tonic-gate } 26320Sstevel@tonic-gate #endif /* DEBUG */ 2633