1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22 /* 23 * Copyright 2008 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 */ 26 27 #pragma ident "%Z%%M% %I% %E% SMI" 28 29 #include <sys/param.h> 30 #include <sys/kmem.h> 31 #include <sys/errno.h> 32 #include <sys/proc.h> 33 #include <sys/disp.h> 34 #include <sys/vfs.h> 35 #include <sys/vnode.h> 36 #include <sys/pathname.h> 37 #include <sys/cred.h> 38 #include <sys/mount.h> 39 #include <sys/cmn_err.h> 40 #include <sys/debug.h> 41 #include <sys/systm.h> 42 #include <sys/dirent.h> 43 #include <fs/fs_subr.h> 44 #include <sys/fs/autofs.h> 45 #include <sys/callb.h> 46 #include <sys/sysmacros.h> 47 #include <sys/zone.h> 48 #include <sys/door.h> 49 #include <sys/fs/mntdata.h> 50 #include <nfs/mount.h> 51 #include <rpc/clnt.h> 52 #include <rpcsvc/autofs_prot.h> 53 #include <nfs/rnode.h> 54 #include <sys/utsname.h> 55 #include <sys/schedctl.h> 56 57 /* 58 * Autofs and Zones: 59 * 60 * Zones are delegated the responsibility of managing their own autofs mounts 61 * and maps. Each zone runs its own copy of automountd, with its own timeouts, 62 * and other logically "global" parameters. kRPC and virtualization in the 63 * loopback transport (tl) will prevent a zone from communicating with another 64 * zone's automountd. 65 * 66 * Each zone has its own "rootfnnode" and associated tree of auto nodes. 67 * 68 * Each zone also has its own set of "unmounter" kernel threads; these are 69 * created and run within the zone's context (ie, they are created via 70 * zthread_create()). 71 * 72 * Cross-zone mount triggers are disallowed. There is a check in 73 * auto_trigger_mount() to this effect; EPERM is returned to indicate that the 74 * mount is not owned by the caller. 75 * 76 * autofssys() enables a caller in the global zone to clean up in-kernel (as 77 * well as regular) autofs mounts via the unmount_tree() mechanism. This is 78 * routinely done when all mounts are removed as part of zone shutdown. 79 */ 80 #define TYPICALMAXPATHLEN 64 81 82 static kmutex_t autofs_nodeid_lock; 83 84 static int auto_perform_link(fnnode_t *, struct linka *, cred_t *); 85 static int auto_perform_actions(fninfo_t *, fnnode_t *, 86 action_list *, cred_t *); 87 static int auto_getmntpnt(vnode_t *, char *, vnode_t **, cred_t *); 88 static int auto_lookup_request(fninfo_t *, char *, struct linka *, 89 bool_t, bool_t *, cred_t *); 90 static int auto_mount_request(fninfo_t *, char *, action_list **, cred_t *, 91 bool_t); 92 93 /* 94 * Clears the MF_INPROG flag, and wakes up those threads sleeping on 95 * fn_cv_mount if MF_WAITING is set. 96 */ 97 void 98 auto_unblock_others( 99 fnnode_t *fnp, 100 uint_t operation) /* either MF_INPROG or MF_LOOKUP */ 101 { 102 ASSERT(operation & (MF_INPROG | MF_LOOKUP)); 103 fnp->fn_flags &= ~operation; 104 if (fnp->fn_flags & MF_WAITING) { 105 fnp->fn_flags &= ~MF_WAITING; 106 cv_broadcast(&fnp->fn_cv_mount); 107 } 108 } 109 110 int 111 auto_wait4mount(fnnode_t *fnp) 112 { 113 int error; 114 k_sigset_t smask; 115 116 AUTOFS_DPRINT((4, "auto_wait4mount: fnp=%p\n", (void *)fnp)); 117 118 mutex_enter(&fnp->fn_lock); 119 while (fnp->fn_flags & (MF_INPROG | MF_LOOKUP)) { 120 /* 121 * There is a mount or a lookup in progress. 122 */ 123 fnp->fn_flags |= MF_WAITING; 124 sigintr(&smask, 1); 125 if (!cv_wait_sig(&fnp->fn_cv_mount, &fnp->fn_lock)) { 126 /* 127 * Decided not to wait for operation to 128 * finish after all. 129 */ 130 sigunintr(&smask); 131 mutex_exit(&fnp->fn_lock); 132 return (EINTR); 133 } 134 sigunintr(&smask); 135 } 136 error = fnp->fn_error; 137 138 if (error == EINTR) { 139 /* 140 * The thread doing the mount got interrupted, we need to 141 * try again, by returning EAGAIN. 142 */ 143 error = EAGAIN; 144 } 145 mutex_exit(&fnp->fn_lock); 146 147 AUTOFS_DPRINT((5, "auto_wait4mount: fnp=%p error=%d\n", (void *)fnp, 148 error)); 149 return (error); 150 } 151 152 int 153 auto_lookup_aux(fnnode_t *fnp, char *name, cred_t *cred) 154 { 155 struct fninfo *fnip; 156 struct linka link; 157 bool_t mountreq = FALSE; 158 int error = 0; 159 160 fnip = vfstofni(fntovn(fnp)->v_vfsp); 161 bzero(&link, sizeof (link)); 162 error = auto_lookup_request(fnip, name, &link, TRUE, &mountreq, cred); 163 if (!error) { 164 if (link.link != NULL || link.link != '\0') { 165 /* 166 * This node should be a symlink 167 */ 168 error = auto_perform_link(fnp, &link, cred); 169 } else if (mountreq) { 170 /* 171 * The automount daemon is requesting a mount, 172 * implying this entry must be a wildcard match and 173 * therefore in need of verification that the entry 174 * exists on the server. 175 */ 176 mutex_enter(&fnp->fn_lock); 177 AUTOFS_BLOCK_OTHERS(fnp, MF_INPROG); 178 fnp->fn_error = 0; 179 180 /* 181 * Unblock other lookup requests on this node, 182 * this is needed to let the lookup generated by 183 * the mount call to complete. The caveat is 184 * other lookups on this node can also get by, 185 * i.e., another lookup on this node that occurs 186 * while this lookup is attempting the mount 187 * would return a positive result no matter what. 188 * Therefore two lookups on the this node could 189 * potentially get disparate results. 190 */ 191 AUTOFS_UNBLOCK_OTHERS(fnp, MF_LOOKUP); 192 mutex_exit(&fnp->fn_lock); 193 /* 194 * auto_new_mount_thread fires up a new thread which 195 * calls automountd finishing up the work 196 */ 197 auto_new_mount_thread(fnp, name, cred); 198 199 /* 200 * At this point, we are simply another thread 201 * waiting for the mount to complete 202 */ 203 error = auto_wait4mount(fnp); 204 if (error == AUTOFS_SHUTDOWN) 205 error = ENOENT; 206 } 207 } 208 209 if (link.link) 210 kmem_free(link.link, strlen(link.link) + 1); 211 if (link.dir) 212 kmem_free(link.dir, strlen(link.dir) + 1); 213 mutex_enter(&fnp->fn_lock); 214 fnp->fn_error = error; 215 216 /* 217 * Notify threads waiting for lookup/mount that 218 * it's done. 219 */ 220 if (mountreq) { 221 AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 222 } else { 223 AUTOFS_UNBLOCK_OTHERS(fnp, MF_LOOKUP); 224 } 225 mutex_exit(&fnp->fn_lock); 226 return (error); 227 } 228 229 /* 230 * Starting point for thread to handle mount requests with automountd. 231 * XXX auto_mount_thread() is not suspend-safe within the scope of 232 * the present model defined for cpr to suspend the system. Calls 233 * made by the auto_mount_thread() that have been identified to be unsafe 234 * are (1) RPC client handle setup and client calls to automountd which 235 * can block deep down in the RPC library, (2) kmem_alloc() calls with the 236 * KM_SLEEP flag which can block if memory is low, and (3) VFS_*(), and 237 * lookuppnvp() calls which can result in over the wire calls to servers. 238 * The thread should be completely reevaluated to make it suspend-safe in 239 * case of future updates to the cpr model. 240 */ 241 static void 242 auto_mount_thread(struct autofs_callargs *argsp) 243 { 244 struct fninfo *fnip; 245 fnnode_t *fnp; 246 vnode_t *vp; 247 char *name; 248 size_t namelen; 249 cred_t *cred; 250 action_list *alp = NULL; 251 int error; 252 callb_cpr_t cprinfo; 253 kmutex_t auto_mount_thread_cpr_lock; 254 255 mutex_init(&auto_mount_thread_cpr_lock, NULL, MUTEX_DEFAULT, NULL); 256 CALLB_CPR_INIT(&cprinfo, &auto_mount_thread_cpr_lock, 257 callb_generic_cpr, "auto_mount_thread"); 258 259 fnp = argsp->fnc_fnp; 260 vp = fntovn(fnp); 261 fnip = vfstofni(vp->v_vfsp); 262 name = argsp->fnc_name; 263 cred = argsp->fnc_cred; 264 ASSERT(crgetzoneid(argsp->fnc_cred) == fnip->fi_zoneid); 265 266 error = auto_mount_request(fnip, name, &alp, cred, TRUE); 267 if (!error) 268 error = auto_perform_actions(fnip, fnp, alp, cred); 269 mutex_enter(&fnp->fn_lock); 270 fnp->fn_error = error; 271 272 /* 273 * Notify threads waiting for mount that 274 * it's done. 275 */ 276 AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 277 mutex_exit(&fnp->fn_lock); 278 279 VN_RELE(vp); 280 crfree(argsp->fnc_cred); 281 namelen = strlen(argsp->fnc_name) + 1; 282 kmem_free(argsp->fnc_name, namelen); 283 kmem_free(argsp, sizeof (*argsp)); 284 285 mutex_enter(&auto_mount_thread_cpr_lock); 286 CALLB_CPR_EXIT(&cprinfo); 287 mutex_destroy(&auto_mount_thread_cpr_lock); 288 zthread_exit(); 289 /* NOTREACHED */ 290 } 291 292 static int autofs_thr_success = 0; 293 294 /* 295 * Creates new thread which calls auto_mount_thread which does 296 * the bulk of the work calling automountd, via 'auto_perform_actions'. 297 */ 298 void 299 auto_new_mount_thread(fnnode_t *fnp, char *name, cred_t *cred) 300 { 301 struct autofs_callargs *argsp; 302 303 argsp = kmem_alloc(sizeof (*argsp), KM_SLEEP); 304 VN_HOLD(fntovn(fnp)); 305 argsp->fnc_fnp = fnp; 306 argsp->fnc_name = kmem_alloc(strlen(name) + 1, KM_SLEEP); 307 (void) strcpy(argsp->fnc_name, name); 308 argsp->fnc_origin = curthread; 309 crhold(cred); 310 argsp->fnc_cred = cred; 311 312 (void) zthread_create(NULL, 0, auto_mount_thread, argsp, 0, 313 minclsyspri); 314 autofs_thr_success++; 315 } 316 317 int 318 auto_calldaemon( 319 zoneid_t zoneid, 320 int which, 321 xdrproc_t xarg_func, 322 void *argsp, 323 xdrproc_t xresp_func, 324 void *resp, 325 int reslen, 326 bool_t hard) /* retry forever? */ 327 { 328 329 int retry, error = 0; 330 k_sigset_t smask; 331 door_arg_t door_args; 332 door_handle_t dh; 333 XDR xdrarg, xdrres; 334 struct autofs_globals *fngp = NULL; 335 void *orig_resp = NULL; 336 int orig_reslen = reslen; 337 autofs_door_args_t *xdr_argsp; 338 int xdr_len = 0; 339 int printed_not_running_msg = 0; 340 klwp_t *lwp = ttolwp(curthread); 341 342 /* 343 * We know that the current thread is doing work on 344 * behalf of its own zone, so it's ok to use 345 * curproc->p_zone. 346 */ 347 ASSERT(zoneid == getzoneid()); 348 if (zone_status_get(curproc->p_zone) >= 349 ZONE_IS_SHUTTING_DOWN) { 350 /* 351 * There's no point in trying to talk to 352 * automountd. Plus, zone_shutdown() is 353 * waiting for us. 354 */ 355 return (ECONNREFUSED); 356 } 357 358 do { 359 retry = 0; 360 mutex_enter(&autofs_minor_lock); 361 fngp = zone_getspecific(autofs_key, curproc->p_zone); 362 mutex_exit(&autofs_minor_lock); 363 if (fngp == NULL) { 364 if (hard) { 365 AUTOFS_DPRINT((5, 366 "auto_calldaemon: "\ 367 "failed to get door handle\n")); 368 if (!printed_not_running_msg) { 369 printed_not_running_msg = 1; 370 zprintf(zoneid, "automountd not "\ 371 "running, retrying\n"); 372 } 373 delay(hz); 374 retry = 1; 375 } else { 376 /* 377 * There is no global data so no door. 378 * There's no point in attempting to talk 379 * to automountd if we can't get the door 380 * handle. 381 */ 382 return (ECONNREFUSED); 383 } 384 } 385 } while (retry); 386 387 if (printed_not_running_msg) { 388 fngp->fng_printed_not_running_msg = printed_not_running_msg; 389 } 390 391 ASSERT(fngp != NULL); 392 393 if (argsp != NULL && (xdr_len = xdr_sizeof(xarg_func, argsp)) == 0) 394 return (EINVAL); 395 xdr_argsp = kmem_zalloc(xdr_len + sizeof (*xdr_argsp), KM_SLEEP); 396 xdr_argsp->xdr_len = xdr_len; 397 xdr_argsp->cmd = which; 398 399 if (argsp) { 400 xdrmem_create(&xdrarg, (char *)&xdr_argsp->xdr_arg, 401 xdr_argsp->xdr_len, XDR_ENCODE); 402 403 if (!(*xarg_func)(&xdrarg, argsp)) { 404 kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 405 return (EINVAL); 406 } 407 } 408 409 /* 410 * We're saving off the original pointer and length due to the 411 * possibility that the results buffer returned by the door 412 * upcall can be different then what we passed in. This is because 413 * the door will allocate new memory if the results buffer passed 414 * in isn't large enough to hold what we need to send back. 415 * In this case we need to free the memory originally allocated 416 * for that buffer. 417 */ 418 if (orig_reslen) 419 orig_resp = kmem_zalloc(orig_reslen, KM_SLEEP); 420 421 do { 422 retry = 0; 423 mutex_enter(&fngp->fng_autofs_daemon_lock); 424 dh = fngp->fng_autofs_daemon_dh; 425 if (dh) 426 door_ki_hold(dh); 427 mutex_exit(&fngp->fng_autofs_daemon_lock); 428 429 if (dh == NULL) { 430 if (orig_resp) 431 kmem_free(orig_resp, orig_reslen); 432 kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 433 return (ENOENT); 434 } 435 door_args.data_ptr = (char *)xdr_argsp; 436 door_args.data_size = sizeof (*xdr_argsp) + xdr_argsp->xdr_len; 437 door_args.desc_ptr = NULL; 438 door_args.desc_num = 0; 439 door_args.rbuf = orig_resp ? (char *)orig_resp : NULL; 440 door_args.rsize = reslen; 441 442 sigintr(&smask, 1); 443 error = door_ki_upcall(dh, &door_args); 444 sigunintr(&smask); 445 446 door_ki_rele(dh); 447 448 if (!error) { 449 autofs_door_res_t *adr = 450 (autofs_door_res_t *)door_args.rbuf; 451 if (door_args.rbuf != NULL && 452 (error = adr->res_status)) { 453 kmem_free(xdr_argsp, 454 xdr_len + sizeof (*xdr_argsp)); 455 if (orig_resp) 456 kmem_free(orig_resp, orig_reslen); 457 return (error); 458 } 459 continue; 460 } 461 switch (error) { 462 case EINTR: 463 /* 464 * interrupts should be handled properly by the 465 * door upcall. If the door doesn't handle the 466 * interupt completely then we need to bail out. 467 */ 468 if (lwp && (ISSIG(curthread, 469 JUSTLOOKING) || MUSTRETURN(curproc, curthread))) { 470 if (ISSIG(curthread, FORREAL) || 471 lwp->lwp_sysabort || 472 MUSTRETURN(curproc, curthread)) { 473 lwp->lwp_sysabort = 0; 474 return (EINTR); 475 } 476 } 477 /* 478 * We may have gotten EINTR for other reasons 479 * like the door being revoked on us. Instead 480 * of trying to extract this out of the door 481 * handle, sleep and try again, if still 482 * revoked we will get EBADF next time 483 * through. 484 * 485 * If we have a pending cancellation and we don't 486 * have cancellation disabled, we will get EINTR 487 * forever, no matter how many times we retry, 488 * so just get out now if this is the case. 489 */ 490 if (schedctl_cancel_pending()) 491 break; 492 /* FALLTHROUGH */ 493 case EAGAIN: /* process may be forking */ 494 /* 495 * Back off for a bit 496 */ 497 delay(hz); 498 retry = 1; 499 break; 500 case EBADF: /* Invalid door */ 501 case EINVAL: /* Not a door, wrong target */ 502 /* 503 * A fatal door error, if our failing door 504 * handle is the current door handle, clean 505 * up our state. 506 */ 507 mutex_enter(&fngp->fng_autofs_daemon_lock); 508 if (dh == fngp->fng_autofs_daemon_dh) { 509 door_ki_rele(fngp->fng_autofs_daemon_dh); 510 fngp->fng_autofs_daemon_dh = NULL; 511 } 512 mutex_exit(&fngp->fng_autofs_daemon_lock); 513 AUTOFS_DPRINT((5, 514 "auto_calldaemon error=%d\n", error)); 515 if (hard) { 516 if (!fngp->fng_printed_not_running_msg) { 517 fngp->fng_printed_not_running_msg = 1; 518 zprintf(zoneid, "automountd not "\ 519 "running, retrying\n"); 520 } 521 delay(hz); 522 retry = 1; 523 break; 524 } else { 525 error = ECONNREFUSED; 526 kmem_free(xdr_argsp, 527 xdr_len + sizeof (*xdr_argsp)); 528 if (orig_resp) 529 kmem_free(orig_resp, orig_reslen); 530 return (error); 531 } 532 default: /* Unknown must be fatal */ 533 error = ENOENT; 534 kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 535 if (orig_resp) 536 kmem_free(orig_resp, orig_reslen); 537 return (error); 538 } 539 } while (retry); 540 541 if (fngp->fng_printed_not_running_msg == 1) { 542 fngp->fng_printed_not_running_msg = 0; 543 zprintf(zoneid, "automountd OK\n"); 544 } 545 546 if (orig_resp && orig_reslen) { 547 autofs_door_res_t *door_resp; 548 door_resp = 549 (autofs_door_res_t *)door_args.rbuf; 550 if ((void *)door_args.rbuf != orig_resp) 551 kmem_free(orig_resp, orig_reslen); 552 xdrmem_create(&xdrres, (char *)&door_resp->xdr_res, 553 door_resp->xdr_len, XDR_DECODE); 554 if (!((*xresp_func)(&xdrres, resp))) 555 error = EINVAL; 556 kmem_free(door_args.rbuf, door_args.rsize); 557 } 558 kmem_free(xdr_argsp, xdr_len + sizeof (*xdr_argsp)); 559 return (error); 560 } 561 562 static int 563 auto_null_request(fninfo_t *fnip, bool_t hard) 564 { 565 int error; 566 struct autofs_globals *fngp = vntofn(fnip->fi_rootvp)->fn_globals; 567 568 AUTOFS_DPRINT((4, "\tauto_null_request\n")); 569 570 error = auto_calldaemon(fngp->fng_zoneid, 571 NULLPROC, 572 xdr_void, 573 NULL, 574 xdr_void, 575 NULL, 576 0, 577 hard); 578 579 AUTOFS_DPRINT((5, "\tauto_null_request: error=%d\n", error)); 580 return (error); 581 } 582 583 static int 584 auto_lookup_request( 585 fninfo_t *fnip, 586 char *key, 587 struct linka *lnp, 588 bool_t hard, 589 bool_t *mountreq, 590 cred_t *cred) 591 { 592 int error; 593 struct autofs_globals *fngp; 594 struct autofs_lookupargs reqst; 595 autofs_lookupres *resp; 596 struct linka *p; 597 598 599 AUTOFS_DPRINT((4, "auto_lookup_equest: path=%s name=%s\n", 600 fnip->fi_path, key)); 601 602 fngp = vntofn(fnip->fi_rootvp)->fn_globals; 603 604 reqst.map = fnip->fi_map; 605 reqst.path = fnip->fi_path; 606 607 if (fnip->fi_flags & MF_DIRECT) 608 reqst.name = fnip->fi_key; 609 else 610 reqst.name = key; 611 AUTOFS_DPRINT((4, "auto_lookup_request: using key=%s\n", reqst.name)); 612 613 reqst.subdir = fnip->fi_subdir; 614 reqst.opts = fnip->fi_opts; 615 reqst.isdirect = fnip->fi_flags & MF_DIRECT ? TRUE : FALSE; 616 reqst.uid = crgetuid(cred); 617 618 resp = kmem_zalloc(sizeof (*resp), KM_SLEEP); 619 620 error = auto_calldaemon(fngp->fng_zoneid, 621 AUTOFS_LOOKUP, 622 xdr_autofs_lookupargs, 623 &reqst, 624 xdr_autofs_lookupres, 625 (void *)resp, 626 sizeof (autofs_lookupres), 627 hard); 628 629 630 if (error) { 631 xdr_free(xdr_autofs_lookupres, (char *)resp); 632 kmem_free(resp, sizeof (*resp)); 633 return (error); 634 } 635 636 if (!error) { 637 fngp->fng_verbose = resp->lu_verbose; 638 switch (resp->lu_res) { 639 case AUTOFS_OK: 640 switch (resp->lu_type.action) { 641 case AUTOFS_MOUNT_RQ: 642 lnp->link = NULL; 643 lnp->dir = NULL; 644 *mountreq = TRUE; 645 break; 646 case AUTOFS_LINK_RQ: 647 p = 648 &resp->lu_type.lookup_result_type_u.lt_linka; 649 lnp->dir = kmem_alloc(strlen(p->dir) + 1, 650 KM_SLEEP); 651 (void) strcpy(lnp->dir, p->dir); 652 lnp->link = kmem_alloc(strlen(p->link) + 1, 653 KM_SLEEP); 654 (void) strcpy(lnp->link, p->link); 655 break; 656 case AUTOFS_NONE: 657 lnp->link = NULL; 658 lnp->dir = NULL; 659 break; 660 default: 661 auto_log(fngp->fng_verbose, 662 fngp->fng_zoneid, CE_WARN, 663 "auto_lookup_request: bad action type %d", 664 resp->lu_res); 665 error = ENOENT; 666 } 667 break; 668 case AUTOFS_NOENT: 669 error = ENOENT; 670 break; 671 default: 672 error = ENOENT; 673 auto_log(fngp->fng_verbose, fngp->fng_zoneid, CE_WARN, 674 "auto_lookup_request: unknown result: %d", 675 resp->lu_res); 676 break; 677 } 678 } 679 done: 680 xdr_free(xdr_autofs_lookupres, (char *)resp); 681 kmem_free(resp, sizeof (*resp)); 682 AUTOFS_DPRINT((5, "auto_lookup_request: path=%s name=%s error=%d\n", 683 fnip->fi_path, key, error)); 684 return (error); 685 } 686 687 static int 688 auto_mount_request( 689 fninfo_t *fnip, 690 char *key, 691 action_list **alpp, 692 cred_t *cred, 693 bool_t hard) 694 { 695 int error; 696 struct autofs_globals *fngp; 697 autofs_lookupargs reqst; 698 autofs_mountres *xdrres = NULL; 699 700 AUTOFS_DPRINT((4, "auto_mount_request: path=%s name=%s\n", 701 fnip->fi_path, key)); 702 703 fngp = vntofn(fnip->fi_rootvp)->fn_globals; 704 reqst.map = fnip->fi_map; 705 reqst.path = fnip->fi_path; 706 707 if (fnip->fi_flags & MF_DIRECT) 708 reqst.name = fnip->fi_key; 709 else 710 reqst.name = key; 711 712 AUTOFS_DPRINT((4, "auto_mount_request: using key=%s\n", reqst.name)); 713 714 reqst.subdir = fnip->fi_subdir; 715 reqst.opts = fnip->fi_opts; 716 reqst.isdirect = fnip->fi_flags & MF_DIRECT ? TRUE : FALSE; 717 reqst.uid = crgetuid(cred); 718 719 xdrres = kmem_zalloc(sizeof (*xdrres), KM_SLEEP); 720 721 error = auto_calldaemon(fngp->fng_zoneid, 722 AUTOFS_MNTINFO, 723 xdr_autofs_lookupargs, 724 &reqst, 725 xdr_autofs_mountres, 726 (void *)xdrres, 727 sizeof (autofs_mountres), 728 hard); 729 730 if (!error) { 731 fngp->fng_verbose = xdrres->mr_verbose; 732 switch (xdrres->mr_type.status) { 733 case AUTOFS_ACTION: 734 error = 0; 735 /* 736 * Save the action list since it is used by 737 * the caller. We NULL the action list pointer 738 * in 'result' so that xdr_free() will not free 739 * the list. 740 */ 741 *alpp = xdrres->mr_type.mount_result_type_u.list; 742 xdrres->mr_type.mount_result_type_u.list = NULL; 743 break; 744 case AUTOFS_DONE: 745 error = xdrres->mr_type.mount_result_type_u.error; 746 break; 747 default: 748 error = ENOENT; 749 auto_log(fngp->fng_verbose, fngp->fng_zoneid, CE_WARN, 750 "auto_mount_request: unknown status %d", 751 xdrres->mr_type.status); 752 break; 753 } 754 } 755 756 xdr_free(xdr_autofs_mountres, (char *)xdrres); 757 kmem_free(xdrres, sizeof (*xdrres)); 758 759 760 AUTOFS_DPRINT((5, "auto_mount_request: path=%s name=%s error=%d\n", 761 fnip->fi_path, key, error)); 762 return (error); 763 } 764 765 766 static int 767 auto_send_unmount_request( 768 fninfo_t *fnip, 769 umntrequest *ul, 770 bool_t hard) 771 { 772 int error; 773 umntres xdrres; 774 775 struct autofs_globals *fngp = vntofn(fnip->fi_rootvp)->fn_globals; 776 777 AUTOFS_DPRINT((4, "\tauto_send_unmount_request: fstype=%s " 778 " mntpnt=%s\n", ul->fstype, ul->mntpnt)); 779 780 bzero(&xdrres, sizeof (umntres)); 781 error = auto_calldaemon(fngp->fng_zoneid, 782 AUTOFS_UNMOUNT, 783 xdr_umntrequest, 784 (void *)ul, 785 xdr_umntres, 786 (void *)&xdrres, 787 sizeof (umntres), 788 hard); 789 790 if (!error) 791 error = xdrres.status; 792 793 AUTOFS_DPRINT((5, "\tauto_send_unmount_request: error=%d\n", error)); 794 795 return (error); 796 } 797 798 static int 799 auto_perform_link(fnnode_t *fnp, struct linka *linkp, cred_t *cred) 800 { 801 vnode_t *vp; 802 size_t len; 803 char *tmp; 804 805 AUTOFS_DPRINT((3, "auto_perform_link: fnp=%p dir=%s link=%s\n", 806 (void *)fnp, linkp->dir, linkp->link)); 807 808 len = strlen(linkp->link) + 1; /* include '\0' */ 809 tmp = kmem_zalloc(len, KM_SLEEP); 810 (void) kcopy(linkp->link, tmp, len); 811 mutex_enter(&fnp->fn_lock); 812 fnp->fn_symlink = tmp; 813 fnp->fn_symlinklen = (uint_t)len; 814 fnp->fn_flags |= MF_THISUID_MATCH_RQD; 815 crhold(cred); 816 fnp->fn_cred = cred; 817 mutex_exit(&fnp->fn_lock); 818 819 vp = fntovn(fnp); 820 vp->v_type = VLNK; 821 822 return (0); 823 } 824 825 static void 826 auto_free_autofs_args(struct mounta *m) 827 { 828 autofs_args *aargs = (autofs_args *)m->dataptr; 829 830 if (aargs->addr.buf) 831 kmem_free(aargs->addr.buf, aargs->addr.len); 832 if (aargs->path) 833 kmem_free(aargs->path, strlen(aargs->path) + 1); 834 if (aargs->opts) 835 kmem_free(aargs->opts, strlen(aargs->opts) + 1); 836 if (aargs->map) 837 kmem_free(aargs->map, strlen(aargs->map) + 1); 838 if (aargs->subdir) 839 kmem_free(aargs->subdir, strlen(aargs->subdir) + 1); 840 if (aargs->key) 841 kmem_free(aargs->key, strlen(aargs->key) + 1); 842 kmem_free(aargs, sizeof (*aargs)); 843 } 844 845 static void 846 auto_free_action_list(action_list *alp) 847 { 848 struct mounta *m; 849 action_list *lastalp; 850 char *fstype; 851 852 m = &alp->action.action_list_entry_u.mounta; 853 while (alp != NULL) { 854 fstype = alp->action.action_list_entry_u.mounta.fstype; 855 m = &alp->action.action_list_entry_u.mounta; 856 if (m->dataptr) { 857 if (strcmp(fstype, "autofs") == 0) { 858 auto_free_autofs_args(m); 859 } 860 } 861 if (m->spec) 862 kmem_free(m->spec, strlen(m->spec) + 1); 863 if (m->dir) 864 kmem_free(m->dir, strlen(m->dir) + 1); 865 if (m->fstype) 866 kmem_free(m->fstype, strlen(m->fstype) + 1); 867 if (m->optptr) 868 kmem_free(m->optptr, m->optlen); 869 lastalp = alp; 870 alp = alp->next; 871 kmem_free(lastalp, sizeof (*lastalp)); 872 } 873 } 874 875 static boolean_t 876 auto_invalid_autofs(fninfo_t *dfnip, fnnode_t *dfnp, action_list *p) 877 { 878 struct mounta *m; 879 struct autofs_args *argsp; 880 vnode_t *dvp; 881 char buff[AUTOFS_MAXPATHLEN]; 882 size_t len; 883 struct autofs_globals *fngp; 884 885 fngp = dfnp->fn_globals; 886 dvp = fntovn(dfnp); 887 888 m = &p->action.action_list_entry_u.mounta; 889 /* 890 * Make sure we aren't geting passed NULL values or a "dir" that 891 * isn't "." and doesn't begin with "./". 892 * 893 * We also only want to perform autofs mounts, so make sure 894 * no-one is trying to trick us into doing anything else. 895 */ 896 if (m->spec == NULL || m->dir == NULL || m->dir[0] != '.' || 897 (m->dir[1] != '/' && m->dir[1] != '\0') || 898 m->fstype == NULL || strcmp(m->fstype, "autofs") != 0 || 899 m->dataptr == NULL || m->datalen != sizeof (struct autofs_args) || 900 m->optptr == NULL) 901 return (B_TRUE); 902 /* 903 * We also don't like ".."s in the pathname. Symlinks are 904 * handled by the fact that we'll use NOFOLLOW when we do 905 * lookup()s. 906 */ 907 if (strstr(m->dir, "/../") != NULL || 908 (len = strlen(m->dir)) > sizeof ("/..") - 1 && 909 m->dir[len] == '.' && m->dir[len - 1] == '.' && 910 m->dir[len - 2] == '/') 911 return (B_TRUE); 912 argsp = (struct autofs_args *)m->dataptr; 913 /* 914 * We don't want NULL values here either. 915 */ 916 if (argsp->addr.buf == NULL || argsp->path == NULL || 917 argsp->opts == NULL || argsp->map == NULL || argsp->subdir == NULL) 918 return (B_TRUE); 919 /* 920 * We know what the claimed pathname *should* look like: 921 * 922 * If the parent (dfnp) is a mount point (VROOT), then 923 * the path should be (dfnip->fi_path + m->dir). 924 * 925 * Else, we know we're only two levels deep, so we use 926 * (dfnip->fi_path + dfnp->fn_name + m->dir). 927 * 928 * Furthermore, "." only makes sense if dfnp is a 929 * trigger node. 930 * 931 * At this point it seems like the passed-in path is 932 * redundant. 933 */ 934 if (dvp->v_flag & VROOT) { 935 if (m->dir[1] == '\0' && !(dfnp->fn_flags & MF_TRIGGER)) 936 return (B_TRUE); 937 (void) snprintf(buff, sizeof (buff), "%s%s", 938 dfnip->fi_path, m->dir + 1); 939 } else { 940 (void) snprintf(buff, sizeof (buff), "%s/%s%s", 941 dfnip->fi_path, dfnp->fn_name, m->dir + 1); 942 } 943 if (strcmp(argsp->path, buff) != 0) { 944 auto_log(fngp->fng_verbose, fngp->fng_zoneid, 945 CE_WARN, "autofs: expected path of '%s', " 946 "got '%s' instead.", buff, argsp->path); 947 return (B_TRUE); 948 } 949 return (B_FALSE); /* looks OK */ 950 } 951 952 /* 953 * auto_invalid_action will validate the action_list received. If all is good 954 * this function returns FALSE, if there is a problem it returns TRUE. 955 */ 956 static boolean_t 957 auto_invalid_action(fninfo_t *dfnip, fnnode_t *dfnp, action_list *alistpp) 958 { 959 960 /* 961 * Before we go any further, this better be a mount request. 962 */ 963 if (alistpp->action.action != AUTOFS_MOUNT_RQ) 964 return (B_TRUE); 965 return (auto_invalid_autofs(dfnip, dfnp, alistpp)); 966 967 } 968 969 static int 970 auto_perform_actions( 971 fninfo_t *dfnip, 972 fnnode_t *dfnp, 973 action_list *alp, 974 cred_t *cred) /* Credentials of the caller */ 975 { 976 977 action_list *p; 978 struct mounta *m, margs; 979 struct autofs_args *argsp; 980 int error, success = 0; 981 vnode_t *mvp, *dvp, *newvp; 982 fnnode_t *newfnp, *mfnp; 983 int auto_mount = 0; 984 int save_triggers = 0; 985 int update_times = 0; 986 char *mntpnt; 987 char buff[AUTOFS_MAXPATHLEN]; 988 timestruc_t now; 989 struct autofs_globals *fngp; 990 cred_t *zcred; 991 992 AUTOFS_DPRINT((4, "auto_perform_actions: alp=%p\n", 993 (void *)alp)); 994 995 fngp = dfnp->fn_globals; 996 dvp = fntovn(dfnp); 997 998 /* 999 * As automountd running in a zone may be compromised, and this may be 1000 * an attack, we can't trust everything passed in by automountd, and we 1001 * need to do argument verification. We'll issue a warning and drop 1002 * the request if it doesn't seem right. 1003 */ 1004 1005 for (p = alp; p != NULL; p = p->next) { 1006 if (auto_invalid_action(dfnip, dfnp, p)) { 1007 /* 1008 * This warning should be sent to the global zone, 1009 * since presumably the zone administrator is the same 1010 * as the attacker. 1011 */ 1012 cmn_err(CE_WARN, "autofs: invalid action list received " 1013 "by automountd in zone %s.", 1014 curproc->p_zone->zone_name); 1015 /* 1016 * This conversation is over. 1017 */ 1018 xdr_free(xdr_action_list, (char *)alp); 1019 return (EINVAL); 1020 } 1021 } 1022 1023 zcred = zone_get_kcred(getzoneid()); 1024 ASSERT(zcred != NULL); 1025 1026 if (vn_mountedvfs(dvp) != NULL) { 1027 /* 1028 * The daemon successfully mounted a filesystem 1029 * on the AUTOFS root node. 1030 */ 1031 mutex_enter(&dfnp->fn_lock); 1032 dfnp->fn_flags |= MF_MOUNTPOINT; 1033 ASSERT(dfnp->fn_dirents == NULL); 1034 mutex_exit(&dfnp->fn_lock); 1035 success++; 1036 } else { 1037 /* 1038 * Clear MF_MOUNTPOINT. 1039 */ 1040 mutex_enter(&dfnp->fn_lock); 1041 if (dfnp->fn_flags & MF_MOUNTPOINT) { 1042 AUTOFS_DPRINT((10, "autofs: clearing mountpoint " 1043 "flag on %s.", dfnp->fn_name)); 1044 ASSERT(dfnp->fn_dirents == NULL); 1045 ASSERT(dfnp->fn_trigger == NULL); 1046 } 1047 dfnp->fn_flags &= ~MF_MOUNTPOINT; 1048 mutex_exit(&dfnp->fn_lock); 1049 } 1050 1051 for (p = alp; p != NULL; p = p->next) { 1052 1053 vfs_t *vfsp; /* dummy argument */ 1054 vfs_t *mvfsp; 1055 1056 auto_mount = 0; 1057 1058 m = &p->action.action_list_entry_u.mounta; 1059 argsp = (struct autofs_args *)m->dataptr; 1060 ASSERT(strcmp(m->fstype, "autofs") == 0); 1061 /* 1062 * use the parent directory's timeout since it's the 1063 * one specified/inherited by automount. 1064 */ 1065 argsp->mount_to = dfnip->fi_mount_to; 1066 /* 1067 * The mountpoint is relative, and it is guaranteed to 1068 * begin with "." 1069 * 1070 */ 1071 ASSERT(m->dir[0] == '.'); 1072 if (m->dir[0] == '.' && m->dir[1] == '\0') { 1073 /* 1074 * mounting on the trigger node 1075 */ 1076 mvp = dvp; 1077 VN_HOLD(mvp); 1078 goto mount; 1079 } 1080 /* 1081 * ignore "./" in front of mountpoint 1082 */ 1083 ASSERT(m->dir[1] == '/'); 1084 mntpnt = m->dir + 2; 1085 1086 AUTOFS_DPRINT((10, "\tdfnip->fi_path=%s\n", dfnip->fi_path)); 1087 AUTOFS_DPRINT((10, "\tdfnip->fi_flags=%x\n", dfnip->fi_flags)); 1088 AUTOFS_DPRINT((10, "\tmntpnt=%s\n", mntpnt)); 1089 1090 if (dfnip->fi_flags & MF_DIRECT) { 1091 AUTOFS_DPRINT((10, "\tDIRECT\n")); 1092 (void) sprintf(buff, "%s/%s", dfnip->fi_path, 1093 mntpnt); 1094 } else { 1095 AUTOFS_DPRINT((10, "\tINDIRECT\n")); 1096 (void) sprintf(buff, "%s/%s/%s", 1097 dfnip->fi_path, 1098 dfnp->fn_name, mntpnt); 1099 } 1100 1101 if (vn_mountedvfs(dvp) == NULL) { 1102 /* 1103 * Daemon didn't mount anything on the root 1104 * We have to create the mountpoint if it 1105 * doesn't exist already 1106 * 1107 * We use the caller's credentials in case a 1108 * UID-match is required 1109 * (MF_THISUID_MATCH_RQD). 1110 */ 1111 rw_enter(&dfnp->fn_rwlock, RW_WRITER); 1112 error = auto_search(dfnp, mntpnt, &mfnp, cred); 1113 if (error == 0) { 1114 /* 1115 * AUTOFS mountpoint exists 1116 */ 1117 if (vn_mountedvfs(fntovn(mfnp)) != NULL) { 1118 cmn_err(CE_PANIC, 1119 "auto_perform_actions:" 1120 " mfnp=%p covered", 1121 (void *)mfnp); 1122 } 1123 } else { 1124 /* 1125 * Create AUTOFS mountpoint 1126 */ 1127 ASSERT((dfnp->fn_flags & MF_MOUNTPOINT) == 0); 1128 error = auto_enter(dfnp, mntpnt, &mfnp, cred); 1129 ASSERT(mfnp->fn_linkcnt == 1); 1130 mfnp->fn_linkcnt++; 1131 } 1132 if (!error) 1133 update_times = 1; 1134 rw_exit(&dfnp->fn_rwlock); 1135 ASSERT(error != EEXIST); 1136 if (!error) { 1137 /* 1138 * mfnp is already held. 1139 */ 1140 mvp = fntovn(mfnp); 1141 } else { 1142 auto_log(fngp->fng_verbose, fngp->fng_zoneid, 1143 CE_WARN, "autofs: mount of %s " 1144 "failed - can't create" 1145 " mountpoint.", buff); 1146 continue; 1147 } 1148 } else { 1149 /* 1150 * Find mountpoint in VFS mounted here. If not 1151 * found, fail the submount, though the overall 1152 * mount has succeeded since the root is 1153 * mounted. 1154 */ 1155 if (error = auto_getmntpnt(dvp, mntpnt, &mvp, 1156 kcred)) { 1157 auto_log(fngp->fng_verbose, 1158 fngp->fng_zoneid, 1159 CE_WARN, "autofs: mount of %s " 1160 "failed - mountpoint doesn't" 1161 " exist.", buff); 1162 continue; 1163 } 1164 if (mvp->v_type == VLNK) { 1165 auto_log(fngp->fng_verbose, 1166 fngp->fng_zoneid, 1167 CE_WARN, "autofs: %s symbolic " 1168 "link: not a valid mountpoint " 1169 "- mount failed", buff); 1170 VN_RELE(mvp); 1171 error = ENOENT; 1172 continue; 1173 } 1174 } 1175 mount: 1176 m->flags |= MS_SYSSPACE | MS_OPTIONSTR; 1177 1178 /* 1179 * Copy mounta struct here so we can substitute a 1180 * buffer that is large enough to hold the returned 1181 * option string, if that string is longer than the 1182 * input option string. 1183 * This can happen if there are default options enabled 1184 * that were not in the input option string. 1185 */ 1186 bcopy(m, &margs, sizeof (*m)); 1187 margs.optptr = kmem_alloc(MAX_MNTOPT_STR, KM_SLEEP); 1188 margs.optlen = MAX_MNTOPT_STR; 1189 (void) strcpy(margs.optptr, m->optptr); 1190 margs.dir = argsp->path; 1191 1192 /* 1193 * We use the zone's kcred because we don't want the 1194 * zone to be able to thus do something it wouldn't 1195 * normally be able to. 1196 */ 1197 error = domount(NULL, &margs, mvp, zcred, &vfsp); 1198 kmem_free(margs.optptr, MAX_MNTOPT_STR); 1199 if (error != 0) { 1200 auto_log(fngp->fng_verbose, fngp->fng_zoneid, 1201 CE_WARN, "autofs: domount of %s failed " 1202 "error=%d", buff, error); 1203 VN_RELE(mvp); 1204 continue; 1205 } 1206 VFS_RELE(vfsp); 1207 1208 /* 1209 * If mountpoint is an AUTOFS node, then I'm going to 1210 * flag it that the Filesystem mounted on top was 1211 * mounted in the kernel so that the unmount can be 1212 * done inside the kernel as well. 1213 * I don't care to flag non-AUTOFS mountpoints when an 1214 * AUTOFS in-kernel mount was done on top, because the 1215 * unmount routine already knows that such case was 1216 * done in the kernel. 1217 */ 1218 if (vfs_matchops(dvp->v_vfsp, 1219 vfs_getops(mvp->v_vfsp))) { 1220 mfnp = vntofn(mvp); 1221 mutex_enter(&mfnp->fn_lock); 1222 mfnp->fn_flags |= MF_IK_MOUNT; 1223 mutex_exit(&mfnp->fn_lock); 1224 } 1225 1226 (void) vn_vfswlock_wait(mvp); 1227 mvfsp = vn_mountedvfs(mvp); 1228 if (mvfsp != NULL) { 1229 vfs_lock_wait(mvfsp); 1230 vn_vfsunlock(mvp); 1231 error = VFS_ROOT(mvfsp, &newvp); 1232 vfs_unlock(mvfsp); 1233 if (error) { 1234 /* 1235 * We've dropped the locks, so let's 1236 * get the mounted vfs again in case 1237 * it changed. 1238 */ 1239 (void) vn_vfswlock_wait(mvp); 1240 mvfsp = vn_mountedvfs(mvp); 1241 if (mvfsp != NULL) { 1242 error = dounmount(mvfsp, 0, CRED()); 1243 if (error) { 1244 cmn_err(CE_WARN, 1245 "autofs: could" 1246 " not unmount" 1247 " vfs=%p", 1248 (void *)mvfsp); 1249 } 1250 } else 1251 vn_vfsunlock(mvp); 1252 VN_RELE(mvp); 1253 continue; 1254 } 1255 } else { 1256 vn_vfsunlock(mvp); 1257 VN_RELE(mvp); 1258 continue; 1259 } 1260 1261 auto_mount = vfs_matchops(dvp->v_vfsp, 1262 vfs_getops(newvp->v_vfsp)); 1263 newfnp = vntofn(newvp); 1264 newfnp->fn_parent = dfnp; 1265 1266 /* 1267 * At this time we want to save the AUTOFS filesystem 1268 * as a trigger node. (We only do this if the mount 1269 * occurred on a node different from the root. 1270 * We look at the trigger nodes during 1271 * the automatic unmounting to make sure we remove them 1272 * as a unit and remount them as a unit if the 1273 * filesystem mounted at the root could not be 1274 * unmounted. 1275 */ 1276 if (auto_mount && (error == 0) && (mvp != dvp)) { 1277 save_triggers++; 1278 /* 1279 * Add AUTOFS mount to hierarchy 1280 */ 1281 newfnp->fn_flags |= MF_TRIGGER; 1282 rw_enter(&newfnp->fn_rwlock, RW_WRITER); 1283 newfnp->fn_next = dfnp->fn_trigger; 1284 rw_exit(&newfnp->fn_rwlock); 1285 rw_enter(&dfnp->fn_rwlock, RW_WRITER); 1286 dfnp->fn_trigger = newfnp; 1287 rw_exit(&dfnp->fn_rwlock); 1288 /* 1289 * Don't VN_RELE(newvp) here since dfnp now 1290 * holds reference to it as its trigger node. 1291 */ 1292 AUTOFS_DPRINT((10, "\tadding trigger %s to %s\n", 1293 newfnp->fn_name, dfnp->fn_name)); 1294 AUTOFS_DPRINT((10, "\tfirst trigger is %s\n", 1295 dfnp->fn_trigger->fn_name)); 1296 if (newfnp->fn_next != NULL) 1297 AUTOFS_DPRINT((10, 1298 "\tnext trigger is %s\n", 1299 newfnp->fn_next->fn_name)); 1300 else 1301 AUTOFS_DPRINT((10, 1302 "\tno next trigger\n")); 1303 } else 1304 VN_RELE(newvp); 1305 1306 if (!error) 1307 success++; 1308 1309 if (update_times) { 1310 gethrestime(&now); 1311 dfnp->fn_atime = dfnp->fn_mtime = now; 1312 } 1313 1314 VN_RELE(mvp); 1315 } 1316 1317 if (save_triggers) { 1318 /* 1319 * Make sure the parent can't be freed while it has triggers. 1320 */ 1321 VN_HOLD(dvp); 1322 } 1323 1324 crfree(zcred); 1325 1326 done: 1327 /* 1328 * Return failure if daemon didn't mount anything, and all 1329 * kernel mounts attempted failed. 1330 */ 1331 error = success ? 0 : ENOENT; 1332 1333 if (alp != NULL) { 1334 if ((error == 0) && save_triggers) { 1335 /* 1336 * Save action_list information, so that we can use it 1337 * when it comes time to remount the trigger nodes 1338 * The action list is freed when the directory node 1339 * containing the reference to it is unmounted in 1340 * unmount_tree(). 1341 */ 1342 mutex_enter(&dfnp->fn_lock); 1343 ASSERT(dfnp->fn_alp == NULL); 1344 dfnp->fn_alp = alp; 1345 mutex_exit(&dfnp->fn_lock); 1346 } else { 1347 /* 1348 * free the action list now, 1349 */ 1350 xdr_free(xdr_action_list, (char *)alp); 1351 } 1352 } 1353 AUTOFS_DPRINT((5, "auto_perform_actions: error=%d\n", error)); 1354 return (error); 1355 } 1356 1357 fnnode_t * 1358 auto_makefnnode( 1359 vtype_t type, 1360 vfs_t *vfsp, 1361 char *name, 1362 cred_t *cred, 1363 struct autofs_globals *fngp) 1364 { 1365 fnnode_t *fnp; 1366 vnode_t *vp; 1367 char *tmpname; 1368 timestruc_t now; 1369 /* 1370 * autofs uses odd inode numbers 1371 * automountd uses even inode numbers 1372 * 1373 * To preserve the age-old semantics that inum+devid is unique across 1374 * the system, this variable must be global across zones. 1375 */ 1376 static ino_t nodeid = 3; 1377 1378 fnp = kmem_zalloc(sizeof (*fnp), KM_SLEEP); 1379 fnp->fn_vnode = vn_alloc(KM_SLEEP); 1380 1381 vp = fntovn(fnp); 1382 tmpname = kmem_alloc(strlen(name) + 1, KM_SLEEP); 1383 (void) strcpy(tmpname, name); 1384 fnp->fn_name = &tmpname[0]; 1385 fnp->fn_namelen = (int)strlen(tmpname) + 1; /* include '\0' */ 1386 fnp->fn_uid = crgetuid(cred); 1387 fnp->fn_gid = crgetgid(cred); 1388 /* 1389 * ".." is added in auto_enter and auto_mount. 1390 * "." is added in auto_mkdir and auto_mount. 1391 */ 1392 /* 1393 * Note that fn_size and fn_linkcnt are already 0 since 1394 * we used kmem_zalloc to allocated fnp 1395 */ 1396 fnp->fn_mode = AUTOFS_MODE; 1397 gethrestime(&now); 1398 fnp->fn_atime = fnp->fn_mtime = fnp->fn_ctime = now; 1399 fnp->fn_ref_time = now.tv_sec; 1400 mutex_enter(&autofs_nodeid_lock); 1401 fnp->fn_nodeid = nodeid; 1402 nodeid += 2; 1403 fnp->fn_globals = fngp; 1404 fngp->fng_fnnode_count++; 1405 mutex_exit(&autofs_nodeid_lock); 1406 vn_setops(vp, auto_vnodeops); 1407 vp->v_type = type; 1408 vp->v_data = (void *)fnp; 1409 vp->v_vfsp = vfsp; 1410 mutex_init(&fnp->fn_lock, NULL, MUTEX_DEFAULT, NULL); 1411 rw_init(&fnp->fn_rwlock, NULL, RW_DEFAULT, NULL); 1412 cv_init(&fnp->fn_cv_mount, NULL, CV_DEFAULT, NULL); 1413 vn_exists(vp); 1414 return (fnp); 1415 } 1416 1417 1418 void 1419 auto_freefnnode(fnnode_t *fnp) 1420 { 1421 vnode_t *vp = fntovn(fnp); 1422 1423 AUTOFS_DPRINT((4, "auto_freefnnode: fnp=%p\n", (void *)fnp)); 1424 1425 ASSERT(fnp->fn_linkcnt == 0); 1426 ASSERT(vp->v_count == 0); 1427 ASSERT(fnp->fn_dirents == NULL); 1428 ASSERT(fnp->fn_parent == NULL); 1429 1430 vn_invalid(vp); 1431 kmem_free(fnp->fn_name, fnp->fn_namelen); 1432 if (fnp->fn_symlink) { 1433 ASSERT(fnp->fn_flags & MF_THISUID_MATCH_RQD); 1434 kmem_free(fnp->fn_symlink, fnp->fn_symlinklen); 1435 } 1436 if (fnp->fn_cred) 1437 crfree(fnp->fn_cred); 1438 mutex_destroy(&fnp->fn_lock); 1439 rw_destroy(&fnp->fn_rwlock); 1440 cv_destroy(&fnp->fn_cv_mount); 1441 vn_free(vp); 1442 1443 mutex_enter(&autofs_nodeid_lock); 1444 fnp->fn_globals->fng_fnnode_count--; 1445 mutex_exit(&autofs_nodeid_lock); 1446 kmem_free(fnp, sizeof (*fnp)); 1447 } 1448 1449 void 1450 auto_disconnect( 1451 fnnode_t *dfnp, 1452 fnnode_t *fnp) 1453 { 1454 fnnode_t *tmp, **fnpp; 1455 vnode_t *vp = fntovn(fnp); 1456 timestruc_t now; 1457 1458 AUTOFS_DPRINT((4, 1459 "auto_disconnect: dfnp=%p fnp=%p linkcnt=%d\n v_count=%d", 1460 (void *)dfnp, (void *)fnp, fnp->fn_linkcnt, vp->v_count)); 1461 1462 ASSERT(RW_WRITE_HELD(&dfnp->fn_rwlock)); 1463 ASSERT(fnp->fn_linkcnt == 1); 1464 1465 if (vn_mountedvfs(vp) != NULL) { 1466 cmn_err(CE_PANIC, "auto_disconnect: vp %p mounted on", 1467 (void *)vp); 1468 } 1469 1470 /* 1471 * Decrement by 1 because we're removing the entry in dfnp. 1472 */ 1473 fnp->fn_linkcnt--; 1474 fnp->fn_size--; 1475 1476 /* 1477 * only changed while holding parent's (dfnp) rw_lock 1478 */ 1479 fnp->fn_parent = NULL; 1480 1481 fnpp = &dfnp->fn_dirents; 1482 for (;;) { 1483 tmp = *fnpp; 1484 if (tmp == NULL) { 1485 cmn_err(CE_PANIC, 1486 "auto_disconnect: %p not in %p dirent list", 1487 (void *)fnp, (void *)dfnp); 1488 } 1489 if (tmp == fnp) { 1490 *fnpp = tmp->fn_next; /* remove it from the list */ 1491 ASSERT(vp->v_count == 0); 1492 /* child had a pointer to parent ".." */ 1493 dfnp->fn_linkcnt--; 1494 dfnp->fn_size--; 1495 break; 1496 } 1497 fnpp = &tmp->fn_next; 1498 } 1499 1500 mutex_enter(&fnp->fn_lock); 1501 gethrestime(&now); 1502 fnp->fn_atime = fnp->fn_mtime = now; 1503 mutex_exit(&fnp->fn_lock); 1504 1505 AUTOFS_DPRINT((5, "auto_disconnect: done\n")); 1506 } 1507 1508 int 1509 auto_enter(fnnode_t *dfnp, char *name, fnnode_t **fnpp, cred_t *cred) 1510 { 1511 struct fnnode *cfnp, **spp; 1512 vnode_t *dvp = fntovn(dfnp); 1513 ushort_t offset = 0; 1514 ushort_t diff; 1515 1516 AUTOFS_DPRINT((4, "auto_enter: dfnp=%p, name=%s ", (void *)dfnp, name)); 1517 1518 ASSERT(RW_WRITE_HELD(&dfnp->fn_rwlock)); 1519 1520 cfnp = dfnp->fn_dirents; 1521 if (cfnp == NULL) { 1522 /* 1523 * offset = 0 for '.' and offset = 1 for '..' 1524 */ 1525 spp = &dfnp->fn_dirents; 1526 offset = 2; 1527 } 1528 1529 for (; cfnp; cfnp = cfnp->fn_next) { 1530 if (strcmp(cfnp->fn_name, name) == 0) { 1531 mutex_enter(&cfnp->fn_lock); 1532 if (cfnp->fn_flags & MF_THISUID_MATCH_RQD) { 1533 /* 1534 * "thisuser" kind of node, need to 1535 * match CREDs as well 1536 */ 1537 mutex_exit(&cfnp->fn_lock); 1538 if (crcmp(cfnp->fn_cred, cred) == 0) 1539 return (EEXIST); 1540 } else { 1541 mutex_exit(&cfnp->fn_lock); 1542 return (EEXIST); 1543 } 1544 } 1545 1546 if (cfnp->fn_next != NULL) { 1547 diff = (ushort_t) 1548 (cfnp->fn_next->fn_offset - cfnp->fn_offset); 1549 ASSERT(diff != 0); 1550 if (diff > 1 && offset == 0) { 1551 offset = (ushort_t)cfnp->fn_offset + 1; 1552 spp = &cfnp->fn_next; 1553 } 1554 } else if (offset == 0) { 1555 offset = (ushort_t)cfnp->fn_offset + 1; 1556 spp = &cfnp->fn_next; 1557 } 1558 } 1559 1560 *fnpp = auto_makefnnode(VDIR, dvp->v_vfsp, name, cred, 1561 dfnp->fn_globals); 1562 if (*fnpp == NULL) 1563 return (ENOMEM); 1564 1565 /* 1566 * I don't hold the mutex on fnpp because I created it, and 1567 * I'm already holding the writers lock for it's parent 1568 * directory, therefore nobody can reference it without me first 1569 * releasing the writers lock. 1570 */ 1571 (*fnpp)->fn_offset = offset; 1572 (*fnpp)->fn_next = *spp; 1573 *spp = *fnpp; 1574 (*fnpp)->fn_parent = dfnp; 1575 (*fnpp)->fn_linkcnt++; /* parent now holds reference to entry */ 1576 (*fnpp)->fn_size++; 1577 1578 /* 1579 * dfnp->fn_linkcnt and dfnp->fn_size protected by dfnp->rw_lock 1580 */ 1581 dfnp->fn_linkcnt++; /* child now holds reference to parent '..' */ 1582 dfnp->fn_size++; 1583 1584 dfnp->fn_ref_time = gethrestime_sec(); 1585 1586 AUTOFS_DPRINT((5, "*fnpp=%p\n", (void *)*fnpp)); 1587 return (0); 1588 } 1589 1590 int 1591 auto_search(fnnode_t *dfnp, char *name, fnnode_t **fnpp, cred_t *cred) 1592 { 1593 vnode_t *dvp; 1594 fnnode_t *p; 1595 int error = ENOENT, match = 0; 1596 1597 AUTOFS_DPRINT((4, "auto_search: dfnp=%p, name=%s...\n", 1598 (void *)dfnp, name)); 1599 1600 dvp = fntovn(dfnp); 1601 if (dvp->v_type != VDIR) { 1602 cmn_err(CE_PANIC, "auto_search: dvp=%p not a directory", 1603 (void *)dvp); 1604 } 1605 1606 ASSERT(RW_LOCK_HELD(&dfnp->fn_rwlock)); 1607 for (p = dfnp->fn_dirents; p != NULL; p = p->fn_next) { 1608 if (strcmp(p->fn_name, name) == 0) { 1609 mutex_enter(&p->fn_lock); 1610 if (p->fn_flags & MF_THISUID_MATCH_RQD) { 1611 /* 1612 * "thisuser" kind of node 1613 * Need to match CREDs as well 1614 */ 1615 mutex_exit(&p->fn_lock); 1616 match = crcmp(p->fn_cred, cred) == 0; 1617 } else { 1618 /* 1619 * No need to check CRED 1620 */ 1621 mutex_exit(&p->fn_lock); 1622 match = 1; 1623 } 1624 } 1625 if (match) { 1626 error = 0; 1627 if (fnpp) { 1628 *fnpp = p; 1629 VN_HOLD(fntovn(*fnpp)); 1630 } 1631 break; 1632 } 1633 } 1634 1635 AUTOFS_DPRINT((5, "auto_search: error=%d\n", error)); 1636 return (error); 1637 } 1638 1639 /* 1640 * If dvp is mounted on, get path's vnode in the mounted on 1641 * filesystem. Path is relative to dvp, ie "./path". 1642 * If successful, *mvp points to a the held mountpoint vnode. 1643 */ 1644 /* ARGSUSED */ 1645 static int 1646 auto_getmntpnt( 1647 vnode_t *dvp, 1648 char *path, 1649 vnode_t **mvpp, /* vnode for mountpoint */ 1650 cred_t *cred) 1651 { 1652 int error = 0; 1653 vnode_t *newvp; 1654 char namebuf[TYPICALMAXPATHLEN]; 1655 struct pathname lookpn; 1656 vfs_t *vfsp; 1657 1658 AUTOFS_DPRINT((4, "auto_getmntpnt: path=%s\n", path)); 1659 1660 if (error = vn_vfsrlock_wait(dvp)) 1661 return (error); 1662 1663 /* 1664 * Now that we have the vfswlock, check to see if dvp 1665 * is still mounted on. If not, then just bail out as 1666 * there is no need to remount the triggers since the 1667 * higher level mount point has gotten unmounted. 1668 */ 1669 vfsp = vn_mountedvfs(dvp); 1670 if (vfsp == NULL) { 1671 vn_vfsunlock(dvp); 1672 error = EBUSY; 1673 goto done; 1674 } 1675 /* 1676 * Since mounted on, lookup "path" in the new filesystem, 1677 * it is important that we do the filesystem jump here to 1678 * avoid lookuppn() calling auto_lookup on dvp and deadlock. 1679 */ 1680 error = VFS_ROOT(vfsp, &newvp); 1681 vn_vfsunlock(dvp); 1682 if (error) 1683 goto done; 1684 1685 /* 1686 * We do a VN_HOLD on newvp just in case the first call to 1687 * lookuppnvp() fails with ENAMETOOLONG. We should still have a 1688 * reference to this vnode for the second call to lookuppnvp(). 1689 */ 1690 VN_HOLD(newvp); 1691 1692 /* 1693 * Now create the pathname struct so we can make use of lookuppnvp, 1694 * and pn_getcomponent. 1695 * This code is similar to lookupname() in fs/lookup.c. 1696 */ 1697 error = pn_get_buf(path, UIO_SYSSPACE, &lookpn, 1698 namebuf, sizeof (namebuf)); 1699 if (error == 0) { 1700 error = lookuppnvp(&lookpn, NULL, NO_FOLLOW, NULLVPP, 1701 mvpp, rootdir, newvp, cred); 1702 } else 1703 VN_RELE(newvp); 1704 if (error == ENAMETOOLONG) { 1705 /* 1706 * This thread used a pathname > TYPICALMAXPATHLEN bytes long. 1707 * newvp is VN_RELE'd by this call to lookuppnvp. 1708 * 1709 * Using 'rootdir' in a zone's context is OK here: we already 1710 * ascertained that there are no '..'s in the path, and we're 1711 * not following symlinks. 1712 */ 1713 if ((error = pn_get(path, UIO_SYSSPACE, &lookpn)) == 0) { 1714 error = lookuppnvp(&lookpn, NULL, NO_FOLLOW, NULLVPP, 1715 mvpp, rootdir, newvp, cred); 1716 pn_free(&lookpn); 1717 } else 1718 VN_RELE(newvp); 1719 } else { 1720 /* 1721 * Need to release newvp here since we held it. 1722 */ 1723 VN_RELE(newvp); 1724 } 1725 1726 done: 1727 AUTOFS_DPRINT((5, "auto_getmntpnt: path=%s *mvpp=%p error=%d\n", 1728 path, (void *)*mvpp, error)); 1729 return (error); 1730 } 1731 1732 #define DEEPER(x) (((x)->fn_dirents != NULL) || \ 1733 (vn_mountedvfs(fntovn((x)))) != NULL) 1734 1735 /* 1736 * The caller, should have already VN_RELE'd its reference to the 1737 * root vnode of this filesystem. 1738 */ 1739 static int 1740 auto_inkernel_unmount(vfs_t *vfsp) 1741 { 1742 vnode_t *cvp = vfsp->vfs_vnodecovered; 1743 int error; 1744 1745 AUTOFS_DPRINT((4, 1746 "auto_inkernel_unmount: devid=%lx mntpnt(%p) count %u\n", 1747 vfsp->vfs_dev, (void *)cvp, cvp->v_count)); 1748 1749 ASSERT(vn_vfswlock_held(cvp)); 1750 1751 /* 1752 * Perform the unmount 1753 * The mountpoint has already been locked by the caller. 1754 */ 1755 error = dounmount(vfsp, 0, kcred); 1756 1757 AUTOFS_DPRINT((5, "auto_inkernel_unmount: exit count %u\n", 1758 cvp->v_count)); 1759 return (error); 1760 } 1761 1762 /* 1763 * unmounts trigger nodes in the kernel. 1764 */ 1765 static void 1766 unmount_triggers(fnnode_t *fnp, action_list **alp) 1767 { 1768 fnnode_t *tp, *next; 1769 int error = 0; 1770 vfs_t *vfsp; 1771 vnode_t *tvp; 1772 1773 AUTOFS_DPRINT((4, "unmount_triggers: fnp=%p\n", (void *)fnp)); 1774 ASSERT(RW_WRITE_HELD(&fnp->fn_rwlock)); 1775 1776 *alp = fnp->fn_alp; 1777 next = fnp->fn_trigger; 1778 while ((tp = next) != NULL) { 1779 tvp = fntovn(tp); 1780 ASSERT(tvp->v_count >= 2); 1781 next = tp->fn_next; 1782 /* 1783 * drop writer's lock since the unmount will end up 1784 * disconnecting this node from fnp and needs to acquire 1785 * the writer's lock again. 1786 * next has at least a reference count >= 2 since it's 1787 * a trigger node, therefore can not be accidentally freed 1788 * by a VN_RELE 1789 */ 1790 rw_exit(&fnp->fn_rwlock); 1791 1792 vfsp = tvp->v_vfsp; 1793 1794 /* 1795 * Its parent was holding a reference to it, since this 1796 * is a trigger vnode. 1797 */ 1798 VN_RELE(tvp); 1799 if (error = auto_inkernel_unmount(vfsp)) { 1800 cmn_err(CE_PANIC, "unmount_triggers: " 1801 "unmount of vp=%p failed error=%d", 1802 (void *)tvp, error); 1803 } 1804 /* 1805 * reacquire writer's lock 1806 */ 1807 rw_enter(&fnp->fn_rwlock, RW_WRITER); 1808 } 1809 1810 /* 1811 * We were holding a reference to our parent. Drop that. 1812 */ 1813 VN_RELE(fntovn(fnp)); 1814 fnp->fn_trigger = NULL; 1815 fnp->fn_alp = NULL; 1816 1817 AUTOFS_DPRINT((5, "unmount_triggers: finished\n")); 1818 } 1819 1820 /* 1821 * This routine locks the mountpoint of every trigger node if they're 1822 * not busy, or returns EBUSY if any node is busy. If a trigger node should 1823 * be unmounted first, then it sets nfnp to point to it, otherwise nfnp 1824 * points to NULL. 1825 */ 1826 static int 1827 triggers_busy(fnnode_t *fnp, fnnode_t **nfnp) 1828 { 1829 int error = 0, done; 1830 int lck_error = 0; 1831 fnnode_t *tp, *t1p; 1832 vfs_t *vfsp; 1833 1834 ASSERT(RW_WRITE_HELD(&fnp->fn_rwlock)); 1835 1836 *nfnp = NULL; 1837 for (tp = fnp->fn_trigger; tp != NULL; tp = tp->fn_next) { 1838 AUTOFS_DPRINT((10, "\ttrigger: %s\n", tp->fn_name)); 1839 vfsp = fntovn(tp)->v_vfsp; 1840 error = 0; 1841 /* 1842 * The vn_vfsunlock will be done in auto_inkernel_unmount. 1843 */ 1844 lck_error = vn_vfswlock(vfsp->vfs_vnodecovered); 1845 if (lck_error == 0) { 1846 mutex_enter(&tp->fn_lock); 1847 ASSERT((tp->fn_flags & MF_LOOKUP) == 0); 1848 if (tp->fn_flags & MF_INPROG) { 1849 /* 1850 * a mount is in progress 1851 */ 1852 error = EBUSY; 1853 } 1854 mutex_exit(&tp->fn_lock); 1855 } 1856 if (lck_error || error || DEEPER(tp) || 1857 ((fntovn(tp))->v_count) > 2) { 1858 /* 1859 * couldn't lock it because it's busy, 1860 * It is mounted on or has dirents? 1861 * If reference count is greater than two, then 1862 * somebody else is holding a reference to this vnode. 1863 * One reference is for the mountpoint, and the second 1864 * is for the trigger node. 1865 */ 1866 AUTOFS_DPRINT((10, "\ttrigger busy\n")); 1867 if ((lck_error == 0) && (error == 0)) { 1868 *nfnp = tp; 1869 /* 1870 * The matching VN_RELE is done in 1871 * unmount_tree(). 1872 */ 1873 VN_HOLD(fntovn(*nfnp)); 1874 } 1875 /* 1876 * Unlock previously locked mountpoints 1877 */ 1878 for (done = 0, t1p = fnp->fn_trigger; !done; 1879 t1p = t1p->fn_next) { 1880 /* 1881 * Unlock all nodes previously 1882 * locked. All nodes up to 'tp' 1883 * were successfully locked. If 'lck_err' is 1884 * set, then 'tp' was not locked, and thus 1885 * should not be unlocked. If 1886 * 'lck_err' is not set, then 'tp' was 1887 * successfully locked, and it should 1888 * be unlocked. 1889 */ 1890 if (t1p != tp || !lck_error) { 1891 vfsp = fntovn(t1p)->v_vfsp; 1892 vn_vfsunlock(vfsp->vfs_vnodecovered); 1893 } 1894 done = (t1p == tp); 1895 } 1896 error = EBUSY; 1897 break; 1898 } 1899 } 1900 1901 AUTOFS_DPRINT((4, "triggers_busy: error=%d\n", error)); 1902 return (error); 1903 } 1904 1905 /* 1906 * Unlock previously locked trigger nodes. 1907 */ 1908 static int 1909 triggers_unlock(fnnode_t *fnp) 1910 { 1911 fnnode_t *tp; 1912 vfs_t *vfsp; 1913 1914 ASSERT(RW_WRITE_HELD(&fnp->fn_rwlock)); 1915 1916 for (tp = fnp->fn_trigger; tp != NULL; tp = tp->fn_next) { 1917 AUTOFS_DPRINT((10, "\tunlock trigger: %s\n", tp->fn_name)); 1918 vfsp = fntovn(tp)->v_vfsp; 1919 vn_vfsunlock(vfsp->vfs_vnodecovered); 1920 } 1921 1922 return (0); 1923 } 1924 1925 /* 1926 * It is the caller's responsibility to grab the VVFSLOCK. 1927 * Releases the VVFSLOCK upon return. 1928 */ 1929 static int 1930 unmount_node(vnode_t *cvp, int force) 1931 { 1932 int error = 0; 1933 fnnode_t *cfnp; 1934 vfs_t *vfsp; 1935 umntrequest ul; 1936 fninfo_t *fnip; 1937 1938 AUTOFS_DPRINT((4, "\tunmount_node cvp=%p\n", (void *)cvp)); 1939 1940 ASSERT(vn_vfswlock_held(cvp)); 1941 cfnp = vntofn(cvp); 1942 vfsp = vn_mountedvfs(cvp); 1943 1944 if (force || cfnp->fn_flags & MF_IK_MOUNT) { 1945 /* 1946 * Mount was performed in the kernel, so 1947 * do an in-kernel unmount. auto_inkernel_unmount() 1948 * will vn_vfsunlock(cvp). 1949 */ 1950 error = auto_inkernel_unmount(vfsp); 1951 } else { 1952 zone_t *zone = NULL; 1953 refstr_t *mntpt, *resource; 1954 size_t mntoptslen; 1955 1956 /* 1957 * Get the mnttab information of the node 1958 * and ask the daemon to unmount it. 1959 */ 1960 bzero(&ul, sizeof (ul)); 1961 mntfs_getmntopts(vfsp, &ul.mntopts, &mntoptslen); 1962 if (ul.mntopts == NULL) { 1963 auto_log(cfnp->fn_globals->fng_verbose, 1964 cfnp->fn_globals->fng_zoneid, 1965 CE_WARN, "unmount_node: " 1966 "no memory"); 1967 vn_vfsunlock(cvp); 1968 error = ENOMEM; 1969 goto done; 1970 } 1971 if (mntoptslen > AUTOFS_MAXOPTSLEN) 1972 ul.mntopts[AUTOFS_MAXOPTSLEN - 1] = '\0'; 1973 1974 mntpt = vfs_getmntpoint(vfsp); 1975 ul.mntpnt = (char *)refstr_value(mntpt); 1976 resource = vfs_getresource(vfsp); 1977 ul.mntresource = (char *)refstr_value(resource); 1978 1979 fnip = vfstofni(cvp->v_vfsp); 1980 ul.isdirect = fnip->fi_flags & MF_DIRECT ? TRUE : FALSE; 1981 1982 /* 1983 * Since a zone'd automountd's view of the autofs mount points 1984 * differs from those in the kernel, we need to make sure we 1985 * give it consistent mount points. 1986 */ 1987 ASSERT(fnip->fi_zoneid == getzoneid()); 1988 zone = curproc->p_zone; 1989 1990 if (fnip->fi_zoneid != GLOBAL_ZONEID) { 1991 if (ZONE_PATH_VISIBLE(ul.mntpnt, zone)) { 1992 ul.mntpnt = 1993 ZONE_PATH_TRANSLATE(ul.mntpnt, zone); 1994 } 1995 if (ZONE_PATH_VISIBLE(ul.mntresource, zone)) { 1996 ul.mntresource = 1997 ZONE_PATH_TRANSLATE(ul.mntresource, zone); 1998 } 1999 } 2000 2001 ul.fstype = vfssw[vfsp->vfs_fstype].vsw_name; 2002 vn_vfsunlock(cvp); 2003 2004 error = auto_send_unmount_request(fnip, &ul, FALSE); 2005 kmem_free(ul.mntopts, mntoptslen); 2006 refstr_rele(mntpt); 2007 refstr_rele(resource); 2008 } 2009 2010 done: 2011 AUTOFS_DPRINT((5, "\tunmount_node cvp=%p error=%d\n", (void *)cvp, 2012 error)); 2013 return (error); 2014 } 2015 2016 /* 2017 * vp is the "root" of the AUTOFS filesystem. 2018 * return EBUSY if any thread is holding a reference to this vnode 2019 * other than us. 2020 */ 2021 static int 2022 check_auto_node(vnode_t *vp) 2023 { 2024 fnnode_t *fnp; 2025 int error = 0; 2026 /* 2027 * number of references to expect for 2028 * a non-busy vnode. 2029 */ 2030 uint_t count; 2031 2032 AUTOFS_DPRINT((4, "\tcheck_auto_node vp=%p ", (void *)vp)); 2033 fnp = vntofn(vp); 2034 ASSERT(fnp->fn_flags & MF_INPROG); 2035 ASSERT((fnp->fn_flags & MF_LOOKUP) == 0); 2036 2037 count = 1; /* we are holding a reference to vp */ 2038 if (fnp->fn_flags & MF_TRIGGER) { 2039 /* 2040 * parent holds a pointer to us (trigger) 2041 */ 2042 count++; 2043 } 2044 if (fnp->fn_trigger != NULL) { 2045 /* 2046 * The trigger nodes have a hold on us. 2047 */ 2048 count++; 2049 } 2050 mutex_enter(&vp->v_lock); 2051 if (vp->v_flag & VROOT) 2052 count++; 2053 ASSERT(vp->v_count > 0); 2054 AUTOFS_DPRINT((10, "\tcount=%u ", vp->v_count)); 2055 if (vp->v_count > count) 2056 error = EBUSY; 2057 mutex_exit(&vp->v_lock); 2058 2059 AUTOFS_DPRINT((5, "\tcheck_auto_node error=%d ", error)); 2060 return (error); 2061 } 2062 2063 /* 2064 * rootvp is the root of the AUTOFS filesystem. 2065 * If rootvp is busy (v_count > 1) returns EBUSY. 2066 * else removes every vnode under this tree. 2067 * ASSUMPTION: Assumes that the only node which can be busy is 2068 * the root vnode. This filesystem better be two levels deep only, 2069 * the root and its immediate subdirs. 2070 * The daemon will "AUTOFS direct-mount" only one level below the root. 2071 */ 2072 static int 2073 unmount_autofs(vnode_t *rootvp) 2074 { 2075 fnnode_t *fnp, *rootfnp, *nfnp; 2076 int error; 2077 2078 AUTOFS_DPRINT((4, "\tunmount_autofs rootvp=%p ", (void *)rootvp)); 2079 2080 error = check_auto_node(rootvp); 2081 if (error == 0) { 2082 /* 2083 * Remove all its immediate subdirectories. 2084 */ 2085 rootfnp = vntofn(rootvp); 2086 rw_enter(&rootfnp->fn_rwlock, RW_WRITER); 2087 nfnp = NULL; /* lint clean */ 2088 for (fnp = rootfnp->fn_dirents; fnp != NULL; fnp = nfnp) { 2089 ASSERT(fntovn(fnp)->v_count == 0); 2090 ASSERT(fnp->fn_dirents == NULL); 2091 ASSERT(fnp->fn_linkcnt == 2); 2092 fnp->fn_linkcnt--; 2093 auto_disconnect(rootfnp, fnp); 2094 nfnp = fnp->fn_next; 2095 auto_freefnnode(fnp); 2096 } 2097 rw_exit(&rootfnp->fn_rwlock); 2098 } 2099 AUTOFS_DPRINT((5, "\tunmount_autofs error=%d ", error)); 2100 return (error); 2101 } 2102 2103 /* 2104 * max number of unmount threads running 2105 */ 2106 static int autofs_unmount_threads = 5; 2107 2108 /* 2109 * XXX unmount_tree() is not suspend-safe within the scope of 2110 * the present model defined for cpr to suspend the system. Calls made 2111 * by the unmount_tree() that have been identified to be unsafe are 2112 * (1) RPC client handle setup and client calls to automountd which can 2113 * block deep down in the RPC library, (2) kmem_alloc() calls with the 2114 * KM_SLEEP flag which can block if memory is low, and (3) VFS_*() and 2115 * VOP_*() calls which can result in over the wire calls to servers. 2116 * The thread should be completely reevaluated to make it suspend-safe in 2117 * case of future updates to the cpr model. 2118 */ 2119 void 2120 unmount_tree(struct autofs_globals *fngp, int force) 2121 { 2122 vnode_t *vp, *newvp; 2123 vfs_t *vfsp; 2124 fnnode_t *fnp, *nfnp, *pfnp; 2125 action_list *alp; 2126 int error, ilocked_it = 0; 2127 fninfo_t *fnip; 2128 time_t ref_time; 2129 int autofs_busy_root, unmount_as_unit, unmount_done = 0; 2130 timestruc_t now; 2131 2132 callb_cpr_t cprinfo; 2133 kmutex_t unmount_tree_cpr_lock; 2134 2135 mutex_init(&unmount_tree_cpr_lock, NULL, MUTEX_DEFAULT, NULL); 2136 CALLB_CPR_INIT(&cprinfo, &unmount_tree_cpr_lock, callb_generic_cpr, 2137 "unmount_tree"); 2138 2139 /* 2140 * Got to release lock before attempting unmount in case 2141 * it hangs. 2142 */ 2143 rw_enter(&fngp->fng_rootfnnodep->fn_rwlock, RW_READER); 2144 if ((fnp = fngp->fng_rootfnnodep->fn_dirents) == NULL) { 2145 ASSERT(fngp->fng_fnnode_count == 1); 2146 /* 2147 * no autofs mounted, done. 2148 */ 2149 rw_exit(&fngp->fng_rootfnnodep->fn_rwlock); 2150 goto done; 2151 } 2152 VN_HOLD(fntovn(fnp)); 2153 rw_exit(&fngp->fng_rootfnnodep->fn_rwlock); 2154 2155 vp = fntovn(fnp); 2156 fnip = vfstofni(vp->v_vfsp); 2157 /* 2158 * autofssys() will be calling in from the global zone and doing 2159 * work on the behalf of the given zone, hence we can't always assert 2160 * that we have the right credentials, nor that the caller is always in 2161 * the correct zone. 2162 * 2163 * We do, however, know that if this is a "forced unmount" operation 2164 * (which autofssys() does), then we won't go down to the krpc layers, 2165 * so we don't need to fudge with the credentials. 2166 */ 2167 ASSERT(force || fnip->fi_zoneid == getzoneid()); 2168 if (!force && auto_null_request(fnip, FALSE) != 0) { 2169 /* 2170 * automountd not running in this zone, 2171 * don't attempt unmounting this round. 2172 */ 2173 VN_RELE(vp); 2174 goto done; 2175 } 2176 /* reference time for this unmount round */ 2177 ref_time = gethrestime_sec(); 2178 /* 2179 * If this an autofssys() call, we need to make sure we don't skip 2180 * nodes because we think we saw them recently. 2181 */ 2182 mutex_enter(&fnp->fn_lock); 2183 if (force && fnp->fn_unmount_ref_time >= ref_time) 2184 ref_time = fnp->fn_unmount_ref_time + 1; 2185 mutex_exit(&fnp->fn_lock); 2186 2187 AUTOFS_DPRINT((4, "unmount_tree (ID=%ld)\n", ref_time)); 2188 top: 2189 AUTOFS_DPRINT((10, "unmount_tree: %s\n", fnp->fn_name)); 2190 ASSERT(fnp); 2191 vp = fntovn(fnp); 2192 if (vp->v_type == VLNK) { 2193 /* 2194 * can't unmount symbolic links 2195 */ 2196 goto next; 2197 } 2198 fnip = vfstofni(vp->v_vfsp); 2199 ASSERT(vp->v_count > 0); 2200 error = 0; 2201 autofs_busy_root = unmount_as_unit = 0; 2202 alp = NULL; 2203 2204 ilocked_it = 0; 2205 mutex_enter(&fnp->fn_lock); 2206 if (fnp->fn_flags & (MF_INPROG | MF_LOOKUP)) { 2207 /* 2208 * Either a mount, lookup or another unmount of this 2209 * subtree is in progress, don't attempt to unmount at 2210 * this time. 2211 */ 2212 mutex_exit(&fnp->fn_lock); 2213 error = EBUSY; 2214 goto next; 2215 } 2216 if (fnp->fn_unmount_ref_time >= ref_time) { 2217 /* 2218 * Already been here, try next node. 2219 */ 2220 mutex_exit(&fnp->fn_lock); 2221 error = EBUSY; 2222 goto next; 2223 } 2224 fnp->fn_unmount_ref_time = ref_time; 2225 2226 /* 2227 * If forced operation ignore timeout values 2228 */ 2229 if (!force && fnp->fn_ref_time + fnip->fi_mount_to > 2230 gethrestime_sec()) { 2231 /* 2232 * Node has been referenced recently, try the 2233 * unmount of its children if any. 2234 */ 2235 mutex_exit(&fnp->fn_lock); 2236 AUTOFS_DPRINT((10, "fn_ref_time within range\n")); 2237 rw_enter(&fnp->fn_rwlock, RW_READER); 2238 if (fnp->fn_dirents) { 2239 /* 2240 * Has subdirectory, attempt their 2241 * unmount first 2242 */ 2243 nfnp = fnp->fn_dirents; 2244 VN_HOLD(fntovn(nfnp)); 2245 rw_exit(&fnp->fn_rwlock); 2246 2247 VN_RELE(vp); 2248 fnp = nfnp; 2249 goto top; 2250 } 2251 rw_exit(&fnp->fn_rwlock); 2252 /* 2253 * No children, try next node. 2254 */ 2255 error = EBUSY; 2256 goto next; 2257 } 2258 2259 AUTOFS_BLOCK_OTHERS(fnp, MF_INPROG); 2260 fnp->fn_error = 0; 2261 mutex_exit(&fnp->fn_lock); 2262 ilocked_it = 1; 2263 2264 rw_enter(&fnp->fn_rwlock, RW_WRITER); 2265 if (fnp->fn_trigger != NULL) { 2266 unmount_as_unit = 1; 2267 if ((vn_mountedvfs(vp) == NULL) && (check_auto_node(vp))) { 2268 /* 2269 * AUTOFS mountpoint is busy, there's 2270 * no point trying to unmount. Fall through 2271 * to attempt to unmount subtrees rooted 2272 * at a possible trigger node, but remember 2273 * not to unmount this tree. 2274 */ 2275 autofs_busy_root = 1; 2276 } 2277 2278 if (triggers_busy(fnp, &nfnp)) { 2279 rw_exit(&fnp->fn_rwlock); 2280 if (nfnp == NULL) { 2281 error = EBUSY; 2282 goto next; 2283 } 2284 /* 2285 * nfnp is busy, try to unmount it first 2286 */ 2287 mutex_enter(&fnp->fn_lock); 2288 AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2289 mutex_exit(&fnp->fn_lock); 2290 VN_RELE(vp); 2291 ASSERT(fntovn(nfnp)->v_count > 1); 2292 fnp = nfnp; 2293 goto top; 2294 } 2295 2296 /* 2297 * At this point, we know all trigger nodes are locked, 2298 * and they're not busy or mounted on. 2299 */ 2300 2301 if (autofs_busy_root) { 2302 /* 2303 * Got to unlock the the trigger nodes since 2304 * I'm not really going to unmount the filesystem. 2305 */ 2306 (void) triggers_unlock(fnp); 2307 } else { 2308 /* 2309 * Attempt to unmount all the trigger nodes, 2310 * save the action_list in case we need to 2311 * remount them later. The action_list will be 2312 * freed later if there was no need to remount the 2313 * trigger nodes. 2314 */ 2315 unmount_triggers(fnp, &alp); 2316 } 2317 } 2318 rw_exit(&fnp->fn_rwlock); 2319 2320 if (autofs_busy_root) 2321 goto next; 2322 2323 (void) vn_vfswlock_wait(vp); 2324 2325 vfsp = vn_mountedvfs(vp); 2326 if (vfsp != NULL) { 2327 /* 2328 * Node is mounted on. 2329 */ 2330 AUTOFS_DPRINT((10, "\tNode is mounted on\n")); 2331 2332 /* 2333 * Deal with /xfn/host/jurassic alikes here... 2334 */ 2335 if (vfs_matchops(vfsp, vfs_getops(vp->v_vfsp))) { 2336 /* 2337 * If the filesystem mounted here is AUTOFS, and it 2338 * is busy, try to unmount the tree rooted on it 2339 * first. We know this call to VFS_ROOT is safe to 2340 * call while holding VVFSLOCK, since it resolves 2341 * to a call to auto_root(). 2342 */ 2343 AUTOFS_DPRINT((10, "\t\tAUTOFS mounted here\n")); 2344 if (VFS_ROOT(vfsp, &newvp)) { 2345 cmn_err(CE_PANIC, 2346 "unmount_tree: VFS_ROOT(vfs=%p) failed", 2347 (void *)vfsp); 2348 } 2349 nfnp = vntofn(newvp); 2350 if (DEEPER(nfnp)) { 2351 vn_vfsunlock(vp); 2352 mutex_enter(&fnp->fn_lock); 2353 AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2354 mutex_exit(&fnp->fn_lock); 2355 VN_RELE(vp); 2356 fnp = nfnp; 2357 goto top; 2358 } 2359 /* 2360 * Fall through to unmount this filesystem 2361 */ 2362 VN_RELE(newvp); 2363 } 2364 2365 /* 2366 * vn_vfsunlock(vp) is done inside unmount_node() 2367 */ 2368 error = unmount_node(vp, force); 2369 if (error == ECONNRESET) { 2370 AUTOFS_DPRINT((10, "\tConnection dropped\n")); 2371 if (vn_mountedvfs(vp) == NULL) { 2372 /* 2373 * The filesystem was unmounted before the 2374 * daemon died. Unfortunately we can not 2375 * determine whether all the cleanup work was 2376 * successfully finished (i.e. update mnttab, 2377 * or notify NFS server of the unmount). 2378 * We should not retry the operation since the 2379 * filesystem has already been unmounted, and 2380 * may have already been removed from mnttab, 2381 * in such case the devid/rdevid we send to 2382 * the daemon will not be matched. So we have 2383 * to be content with the partial unmount. 2384 * Since the mountpoint is no longer covered, we 2385 * clear the error condition. 2386 */ 2387 error = 0; 2388 auto_log(fngp->fng_verbose, fngp->fng_zoneid, 2389 CE_WARN, 2390 "unmount_tree: automountd connection " 2391 "dropped"); 2392 if (fnip->fi_flags & MF_DIRECT) { 2393 auto_log(fngp->fng_verbose, 2394 fngp->fng_zoneid, CE_WARN, 2395 "unmount_tree: " 2396 "%s successfully unmounted - " 2397 "do not remount triggers", 2398 fnip->fi_path); 2399 } else { 2400 auto_log(fngp->fng_verbose, 2401 fngp->fng_zoneid, CE_WARN, 2402 "unmount_tree: " 2403 "%s/%s successfully unmounted - " 2404 "do not remount triggers", 2405 fnip->fi_path, fnp->fn_name); 2406 } 2407 } 2408 } 2409 } else { 2410 vn_vfsunlock(vp); 2411 AUTOFS_DPRINT((10, "\tNode is AUTOFS\n")); 2412 if (unmount_as_unit) { 2413 AUTOFS_DPRINT((10, "\tunmount as unit\n")); 2414 error = unmount_autofs(vp); 2415 } else { 2416 AUTOFS_DPRINT((10, "\tunmount one at a time\n")); 2417 rw_enter(&fnp->fn_rwlock, RW_READER); 2418 if (fnp->fn_dirents != NULL) { 2419 /* 2420 * Has subdirectory, attempt their 2421 * unmount first 2422 */ 2423 nfnp = fnp->fn_dirents; 2424 VN_HOLD(fntovn(nfnp)); 2425 rw_exit(&fnp->fn_rwlock); 2426 2427 mutex_enter(&fnp->fn_lock); 2428 AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2429 mutex_exit(&fnp->fn_lock); 2430 VN_RELE(vp); 2431 fnp = nfnp; 2432 goto top; 2433 } 2434 rw_exit(&fnp->fn_rwlock); 2435 goto next; 2436 } 2437 } 2438 2439 if (error) { 2440 AUTOFS_DPRINT((10, "\tUnmount failed\n")); 2441 if (alp != NULL) { 2442 /* 2443 * Unmount failed, got to remount triggers. 2444 */ 2445 ASSERT((fnp->fn_flags & MF_THISUID_MATCH_RQD) == 0); 2446 error = auto_perform_actions(fnip, fnp, alp, 2447 CRED()); 2448 if (error) { 2449 auto_log(fngp->fng_verbose, 2450 fngp->fng_zoneid, CE_WARN, 2451 "autofs: can't remount " 2452 "triggers fnp=%p error=%d", (void *)fnp, 2453 error); 2454 error = 0; 2455 /* 2456 * The action list should have been 2457 * free'd by auto_perform_actions 2458 * since an error occured 2459 */ 2460 alp = NULL; 2461 2462 } 2463 } 2464 } else { 2465 /* 2466 * The unmount succeeded, which will cause this node to 2467 * be removed from its parent if its an indirect mount, 2468 * therefore update the parent's atime and mtime now. 2469 * I don't update them in auto_disconnect() because I 2470 * don't want atime and mtime changing every time a 2471 * lookup goes to the daemon and creates a new node. 2472 */ 2473 unmount_done = 1; 2474 if ((fnip->fi_flags & MF_DIRECT) == 0) { 2475 gethrestime(&now); 2476 if (fnp->fn_parent == fngp->fng_rootfnnodep) 2477 fnp->fn_atime = fnp->fn_mtime = now; 2478 else 2479 fnp->fn_parent->fn_atime = 2480 fnp->fn_parent->fn_mtime = now; 2481 } 2482 2483 /* 2484 * Free the action list here 2485 */ 2486 if (alp != NULL) { 2487 xdr_free(xdr_action_list, (char *)alp); 2488 alp = NULL; 2489 } 2490 } 2491 2492 fnp->fn_ref_time = gethrestime_sec(); 2493 2494 next: 2495 /* 2496 * Obtain parent's readers lock before grabbing 2497 * reference to next sibling. 2498 * XXX Note that nodes in the top level list (mounted 2499 * in user space not by the daemon in the kernel) parent is itself, 2500 * therefore grabbing the lock makes no sense, but doesn't 2501 * hurt either. 2502 */ 2503 pfnp = fnp->fn_parent; 2504 ASSERT(pfnp != NULL); 2505 rw_enter(&pfnp->fn_rwlock, RW_READER); 2506 if ((nfnp = fnp->fn_next) != NULL) 2507 VN_HOLD(fntovn(nfnp)); 2508 rw_exit(&pfnp->fn_rwlock); 2509 2510 if (ilocked_it) { 2511 mutex_enter(&fnp->fn_lock); 2512 if (unmount_done) { 2513 /* 2514 * Other threads may be waiting for this unmount to 2515 * finish. We must let it know that in order to 2516 * proceed, it must trigger the mount itself. 2517 */ 2518 fnp->fn_flags &= ~MF_IK_MOUNT; 2519 if (fnp->fn_flags & MF_WAITING) 2520 fnp->fn_error = EAGAIN; 2521 unmount_done = 0; 2522 } 2523 AUTOFS_UNBLOCK_OTHERS(fnp, MF_INPROG); 2524 mutex_exit(&fnp->fn_lock); 2525 ilocked_it = 0; 2526 } 2527 2528 if (nfnp != NULL) { 2529 VN_RELE(vp); 2530 fnp = nfnp; 2531 /* 2532 * Unmount next element 2533 */ 2534 goto top; 2535 } 2536 2537 /* 2538 * We don't want to unmount rootfnnodep, so the check is made here 2539 */ 2540 ASSERT(pfnp != fnp); 2541 if (pfnp != fngp->fng_rootfnnodep) { 2542 /* 2543 * Now attempt to unmount my parent 2544 */ 2545 VN_HOLD(fntovn(pfnp)); 2546 VN_RELE(vp); 2547 fnp = pfnp; 2548 2549 goto top; 2550 } 2551 2552 VN_RELE(vp); 2553 2554 /* 2555 * At this point we've walked the entire tree and attempted to unmount 2556 * as much as we can one level at a time. 2557 */ 2558 done: 2559 mutex_enter(&unmount_tree_cpr_lock); 2560 CALLB_CPR_EXIT(&cprinfo); 2561 mutex_destroy(&unmount_tree_cpr_lock); 2562 } 2563 2564 static void 2565 unmount_zone_tree(struct autofs_globals *fngp) 2566 { 2567 unmount_tree(fngp, 0); 2568 mutex_enter(&fngp->fng_unmount_threads_lock); 2569 fngp->fng_unmount_threads--; 2570 mutex_exit(&fngp->fng_unmount_threads_lock); 2571 2572 AUTOFS_DPRINT((5, "unmount_tree done. Thread exiting.\n")); 2573 2574 zthread_exit(); 2575 /* NOTREACHED */ 2576 } 2577 2578 static int autofs_unmount_thread_timer = 120; /* in seconds */ 2579 2580 void 2581 auto_do_unmount(struct autofs_globals *fngp) 2582 { 2583 callb_cpr_t cprinfo; 2584 clock_t timeleft; 2585 zone_t *zone = curproc->p_zone; 2586 2587 CALLB_CPR_INIT(&cprinfo, &fngp->fng_unmount_threads_lock, 2588 callb_generic_cpr, "auto_do_unmount"); 2589 2590 for (;;) { /* forever */ 2591 mutex_enter(&fngp->fng_unmount_threads_lock); 2592 CALLB_CPR_SAFE_BEGIN(&cprinfo); 2593 newthread: 2594 mutex_exit(&fngp->fng_unmount_threads_lock); 2595 timeleft = zone_status_timedwait(zone, lbolt + 2596 autofs_unmount_thread_timer * hz, ZONE_IS_SHUTTING_DOWN); 2597 mutex_enter(&fngp->fng_unmount_threads_lock); 2598 2599 if (timeleft != -1) { /* didn't time out */ 2600 ASSERT(zone_status_get(zone) >= ZONE_IS_SHUTTING_DOWN); 2601 /* 2602 * zone is exiting... don't create any new threads. 2603 * fng_unmount_threads_lock is released implicitly by 2604 * the below. 2605 */ 2606 CALLB_CPR_SAFE_END(&cprinfo, 2607 &fngp->fng_unmount_threads_lock); 2608 CALLB_CPR_EXIT(&cprinfo); 2609 zthread_exit(); 2610 /* NOTREACHED */ 2611 } 2612 if (fngp->fng_unmount_threads < autofs_unmount_threads) { 2613 fngp->fng_unmount_threads++; 2614 CALLB_CPR_SAFE_END(&cprinfo, 2615 &fngp->fng_unmount_threads_lock); 2616 mutex_exit(&fngp->fng_unmount_threads_lock); 2617 2618 (void) zthread_create(NULL, 0, unmount_zone_tree, fngp, 2619 0, minclsyspri); 2620 } else 2621 goto newthread; 2622 } 2623 /* NOTREACHED */ 2624 } 2625 2626 /* 2627 * Is nobrowse specified in option string? 2628 * opts should be a null ('\0') terminated string. 2629 * Returns non-zero if nobrowse has been specified. 2630 */ 2631 int 2632 auto_nobrowse_option(char *opts) 2633 { 2634 char *buf; 2635 char *p; 2636 char *t; 2637 int nobrowse = 0; 2638 int last_opt = 0; 2639 size_t len; 2640 2641 len = strlen(opts) + 1; 2642 p = buf = kmem_alloc(len, KM_SLEEP); 2643 (void) strcpy(buf, opts); 2644 do { 2645 if (t = strchr(p, ',')) 2646 *t++ = '\0'; 2647 else 2648 last_opt++; 2649 if (strcmp(p, MNTOPT_NOBROWSE) == 0) 2650 nobrowse = 1; 2651 else if (strcmp(p, MNTOPT_BROWSE) == 0) 2652 nobrowse = 0; 2653 p = t; 2654 } while (!last_opt); 2655 kmem_free(buf, len); 2656 2657 return (nobrowse); 2658 } 2659 2660 /* 2661 * used to log warnings only if automountd is running 2662 * with verbose mode set 2663 */ 2664 2665 void 2666 auto_log(int verbose, zoneid_t zoneid, int level, const char *fmt, ...) 2667 { 2668 va_list args; 2669 2670 if (verbose) { 2671 va_start(args, fmt); 2672 vzcmn_err(zoneid, level, fmt, args); 2673 va_end(args); 2674 } 2675 } 2676 2677 #ifdef DEBUG 2678 static int autofs_debug = 0; 2679 2680 /* 2681 * Utilities used by both client and server 2682 * Standard levels: 2683 * 0) no debugging 2684 * 1) hard failures 2685 * 2) soft failures 2686 * 3) current test software 2687 * 4) main procedure entry points 2688 * 5) main procedure exit points 2689 * 6) utility procedure entry points 2690 * 7) utility procedure exit points 2691 * 8) obscure procedure entry points 2692 * 9) obscure procedure exit points 2693 * 10) random stuff 2694 * 11) all <= 1 2695 * 12) all <= 2 2696 * 13) all <= 3 2697 * ... 2698 */ 2699 /* PRINTFLIKE2 */ 2700 void 2701 auto_dprint(int level, const char *fmt, ...) 2702 { 2703 va_list args; 2704 2705 if (autofs_debug == level || 2706 (autofs_debug > 10 && (autofs_debug - 10) >= level)) { 2707 va_start(args, fmt); 2708 (void) vprintf(fmt, args); 2709 va_end(args); 2710 } 2711 } 2712 #endif /* DEBUG */ 2713