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