1 /*- 2 * Copyright (c) 1980, 1989, 1993 3 * The Regents of the University of California. All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 3. All advertising materials mentioning features or use of this software 14 * must display the following acknowledgement: 15 * This product includes software developed by the University of 16 * California, Berkeley and its contributors. 17 * 4. Neither the name of the University nor the names of its contributors 18 * may be used to endorse or promote products derived from this software 19 * without specific prior written permission. 20 * 21 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 24 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31 * SUCH DAMAGE. 32 */ 33 34 #ifndef lint 35 static const char copyright[] = 36 "@(#) Copyright (c) 1980, 1989, 1993\n\ 37 The Regents of the University of California. All rights reserved.\n"; 38 #endif /* not lint */ 39 40 #ifndef lint 41 #if 0 42 static char sccsid[] = "@(#)umount.c 8.8 (Berkeley) 5/8/95"; 43 #endif 44 static const char rcsid[] = 45 "$FreeBSD$"; 46 #endif /* not lint */ 47 48 #include <sys/param.h> 49 #include <sys/mount.h> 50 #include <sys/socket.h> 51 52 #include <netdb.h> 53 #include <rpc/rpc.h> 54 #include <nfs/rpcv2.h> 55 56 #include <err.h> 57 #include <fstab.h> 58 #include <stdio.h> 59 #include <stdlib.h> 60 #include <string.h> 61 #include <unistd.h> 62 63 #include "mounttab.h" 64 65 #define ISDOT(x) ((x)[0] == '.' && (x)[1] == '\0') 66 #define ISDOTDOT(x) ((x)[0] == '.' && (x)[1] == '.' && (x)[2] == '\0') 67 68 typedef enum { MNTON, MNTFROM, NOTHING } mntwhat; 69 typedef enum { MARK, UNMARK, NAME, COUNT, FREE } dowhat; 70 71 struct addrinfo *nfshost_ai = NULL; 72 int fflag, vflag; 73 char *nfshost; 74 75 void checkmntlist (char *, char **, char **, char **); 76 int checkvfsname (const char *, char **); 77 char *getmntname (const char *, const char *, 78 mntwhat, char **, dowhat); 79 char *getrealname(char *, char *resolved_path); 80 char **makevfslist (const char *); 81 size_t mntinfo (struct statfs **); 82 int namematch (struct addrinfo *); 83 int sacmp (struct sockaddr *, struct sockaddr *); 84 int umountall (char **); 85 int checkname (char *, char **); 86 int umountfs (char *, char *, char *); 87 void usage (void); 88 int xdr_dir (XDR *, char *); 89 90 int 91 main(int argc, char *argv[]) 92 { 93 int all, errs, ch, mntsize, error; 94 char **typelist = NULL, *mntonname, *mntfromname; 95 char *type, *mntfromnamerev, *mntonnamerev; 96 struct statfs *mntbuf; 97 struct addrinfo hints; 98 99 /* Start disks transferring immediately. */ 100 sync(); 101 102 all = errs = 0; 103 while ((ch = getopt(argc, argv, "AaF:fh:t:v")) != -1) 104 switch (ch) { 105 case 'A': 106 all = 2; 107 break; 108 case 'a': 109 all = 1; 110 break; 111 case 'F': 112 setfstab(optarg); 113 break; 114 case 'f': 115 fflag = MNT_FORCE; 116 break; 117 case 'h': /* -h implies -A. */ 118 all = 2; 119 nfshost = optarg; 120 break; 121 case 't': 122 if (typelist != NULL) 123 err(1, "only one -t option may be specified"); 124 typelist = makevfslist(optarg); 125 break; 126 case 'v': 127 vflag = 1; 128 break; 129 default: 130 usage(); 131 /* NOTREACHED */ 132 } 133 argc -= optind; 134 argv += optind; 135 136 if ((argc == 0 && !all) || (argc != 0 && all)) 137 usage(); 138 139 /* -h implies "-t nfs" if no -t flag. */ 140 if ((nfshost != NULL) && (typelist == NULL)) 141 typelist = makevfslist("nfs"); 142 143 if (nfshost != NULL) { 144 memset(&hints, 0, sizeof hints); 145 error = getaddrinfo(nfshost, NULL, &hints, &nfshost_ai); 146 if (error) 147 errx(1, "%s: %s", nfshost, gai_strerror(error)); 148 } 149 150 switch (all) { 151 case 2: 152 if ((mntsize = mntinfo(&mntbuf)) <= 0) 153 break; 154 /* 155 * We unmount the nfs-mounts in the reverse order 156 * that they were mounted. 157 */ 158 for (errs = 0, mntsize--; mntsize > 0; mntsize--) { 159 if (checkvfsname(mntbuf[mntsize].f_fstypename, 160 typelist)) 161 continue; 162 /* 163 * Check if a mountpoint is laid over by another mount. 164 * A warning will be printed to stderr if this is 165 * the case. The laid over mount remains unmounted. 166 */ 167 mntonname = mntbuf[mntsize].f_mntonname; 168 mntfromname = mntbuf[mntsize].f_mntfromname; 169 mntonnamerev = getmntname(getmntname(mntonname, 170 NULL, MNTFROM, &type, NAME), NULL, 171 MNTON, &type, NAME); 172 173 mntfromnamerev = getmntname(mntonnamerev, 174 NULL, MNTFROM, &type, NAME); 175 176 if (strcmp(mntonnamerev, mntonname) == 0 && 177 strcmp(mntfromnamerev, mntfromname ) != 0) 178 warnx("cannot umount %s, %s\n " 179 "is mounted there, umount it first", 180 mntonname, mntfromnamerev); 181 182 if (checkname(mntbuf[mntsize].f_mntonname, 183 typelist) != 0) 184 errs = 1; 185 } 186 free(mntbuf); 187 break; 188 case 1: 189 if (setfsent() == 0) 190 err(1, "%s", getfstab()); 191 errs = umountall(typelist); 192 break; 193 case 0: 194 for (errs = 0; *argv != NULL; ++argv) 195 if (checkname(*argv, typelist) != 0) 196 errs = 1; 197 break; 198 } 199 (void)getmntname(NULL, NULL, NOTHING, NULL, FREE); 200 exit(errs); 201 } 202 203 int 204 umountall(char **typelist) 205 { 206 struct xvfsconf vfc; 207 struct fstab *fs; 208 int rval; 209 char *cp; 210 static int firstcall = 1; 211 212 if ((fs = getfsent()) != NULL) 213 firstcall = 0; 214 else if (firstcall) 215 errx(1, "fstab reading failure"); 216 else 217 return (0); 218 do { 219 /* Ignore the root. */ 220 if (strcmp(fs->fs_file, "/") == 0) 221 continue; 222 /* 223 * !!! 224 * Historic practice: ignore unknown FSTAB_* fields. 225 */ 226 if (strcmp(fs->fs_type, FSTAB_RW) && 227 strcmp(fs->fs_type, FSTAB_RO) && 228 strcmp(fs->fs_type, FSTAB_RQ)) 229 continue; 230 /* Ignore unknown file system types. */ 231 if (getvfsbyname(fs->fs_vfstype, &vfc) == -1) 232 continue; 233 if (checkvfsname(fs->fs_vfstype, typelist)) 234 continue; 235 236 /* 237 * We want to unmount the file systems in the reverse order 238 * that they were mounted. So, we save off the file name 239 * in some allocated memory, and then call recursively. 240 */ 241 if ((cp = malloc((size_t)strlen(fs->fs_file) + 1)) == NULL) 242 err(1, "malloc failed"); 243 (void)strcpy(cp, fs->fs_file); 244 rval = umountall(typelist); 245 rval = checkname(cp, typelist) || rval; 246 free(cp); 247 return (rval); 248 } while ((fs = getfsent()) != NULL); 249 return (0); 250 } 251 252 /* 253 * Do magic checks on mountpoint and device or hand over 254 * it to unmount(2) if everything fails. 255 */ 256 int 257 checkname(char *name, char **typelist) 258 { 259 size_t len; 260 int speclen; 261 char *mntonname, *mntfromname; 262 char *mntfromnamerev; 263 char *resolved, realname[MAXPATHLEN]; 264 char *type, *hostp, *delimp, *origname; 265 266 len = 0; 267 mntfromname = mntonname = delimp = hostp = NULL; 268 269 /* 270 * 1. Check if the name exists in the mounttable. 271 */ 272 (void)checkmntlist(name, &mntfromname, &mntonname, &type); 273 /* 274 * 2. Remove trailing slashes if there are any. After that 275 * we look up the name in the mounttable again. 276 */ 277 if (mntfromname == NULL && mntonname == NULL) { 278 speclen = strlen(name); 279 for (speclen = strlen(name); 280 speclen > 1 && name[speclen - 1] == '/'; 281 speclen--) 282 name[speclen - 1] = '\0'; 283 (void)checkmntlist(name, &mntfromname, &mntonname, &type); 284 resolved = name; 285 /* Save off original name in origname */ 286 if ((origname = strdup(name)) == NULL) 287 err(1, "strdup"); 288 /* 289 * 3. Check if the deprecated nfs-syntax with an '@' 290 * has been used and translate it to the ':' syntax. 291 * Look up the name in the mounttable again. 292 */ 293 if (mntfromname == NULL && mntonname == NULL) { 294 if ((delimp = strrchr(name, '@')) != NULL) { 295 hostp = delimp + 1; 296 if (*hostp != '\0') { 297 /* 298 * Make both '@' and ':' 299 * notations equal 300 */ 301 char *host = strdup(hostp); 302 len = strlen(hostp); 303 if (host == NULL) 304 err(1, "strdup"); 305 memmove(name + len + 1, name, 306 (size_t)(delimp - name)); 307 name[len] = ':'; 308 memmove(name, host, len); 309 free(host); 310 } 311 for (speclen = strlen(name); 312 speclen > 1 && name[speclen - 1] == '/'; 313 speclen--) 314 name[speclen - 1] = '\0'; 315 name[len + speclen + 1] = '\0'; 316 (void)checkmntlist(name, &mntfromname, 317 &mntonname, &type); 318 resolved = name; 319 } 320 /* 321 * 4. Check if a relative mountpoint has been 322 * specified. This should happen as last check, 323 * the order is important. To prevent possible 324 * nfs-hangs, we just call realpath(3) on the 325 * basedir of mountpoint and add the dirname again. 326 * Check the name in mounttable one last time. 327 */ 328 if (mntfromname == NULL && mntonname == NULL) { 329 (void)strcpy(name, origname); 330 if ((getrealname(name, realname)) != NULL) { 331 (void)checkmntlist(realname, 332 &mntfromname, &mntonname, &type); 333 resolved = realname; 334 } 335 /* 336 * 5. All tests failed, just hand over the 337 * mountpoint to the kernel, maybe the statfs 338 * structure has been truncated or is not 339 * useful anymore because of a chroot(2). 340 * Please note that nfs will not be able to 341 * notify the nfs-server about unmounting. 342 * These things can change in future when the 343 * fstat structure get's more reliable, 344 * but at the moment we cannot thrust it. 345 */ 346 if (mntfromname == NULL && mntonname == NULL) { 347 (void)strcpy(name, origname); 348 if (umountfs(NULL, origname, 349 "none") == 0) {; 350 warnx("%s not found in " 351 "mount table, " 352 "unmounted it anyway", 353 origname); 354 free(origname); 355 return (0); 356 } else 357 free(origname); 358 return (1); 359 } 360 } 361 } 362 free(origname); 363 } else 364 resolved = name; 365 366 if (checkvfsname(type, typelist)) 367 return (1); 368 369 /* 370 * Check if the reverse entrys of the mounttable are really the 371 * same as the normal ones. 372 */ 373 if ((mntfromnamerev = strdup(getmntname(getmntname(mntfromname, 374 NULL, MNTON, &type, NAME), NULL, MNTFROM, &type, NAME))) == NULL) 375 err(1, "strdup"); 376 /* 377 * Mark the uppermost mount as unmounted. 378 */ 379 (void)getmntname(mntfromname, mntonname, NOTHING, &type, MARK); 380 /* 381 * If several equal mounts are in the mounttable, check the order 382 * and warn the user if necessary. 383 */ 384 if (strcmp(mntfromnamerev, mntfromname ) != 0 && 385 strcmp(resolved, mntonname) != 0) { 386 warnx("cannot umount %s, %s\n " 387 "is mounted there, umount it first", 388 mntonname, mntfromnamerev); 389 390 /* call getmntname again to set mntcheck[i] to 0 */ 391 (void)getmntname(mntfromname, mntonname, 392 NOTHING, &type, UNMARK); 393 return (1); 394 } 395 free(mntfromnamerev); 396 return (umountfs(mntfromname, mntonname, type)); 397 } 398 399 /* 400 * NFS stuff and unmount(2) call 401 */ 402 int 403 umountfs(char *mntfromname, char *mntonname, char *type) 404 { 405 enum clnt_stat clnt_stat; 406 struct timeval try; 407 struct addrinfo *ai, hints; 408 int do_rpc; 409 CLIENT *clp; 410 char *nfsdirname, *orignfsdirname; 411 char *hostp, *delimp; 412 413 ai = NULL; 414 do_rpc = 0; 415 hostp = NULL; 416 nfsdirname = delimp = orignfsdirname = NULL; 417 memset(&hints, 0, sizeof hints); 418 419 if (strcmp(type, "nfs") == 0) { 420 if ((nfsdirname = strdup(mntfromname)) == NULL) 421 err(1, "strdup"); 422 orignfsdirname = nfsdirname; 423 if ((delimp = strrchr(nfsdirname, ':')) != NULL) { 424 *delimp = '\0'; 425 hostp = nfsdirname; 426 getaddrinfo(hostp, NULL, &hints, &ai); 427 if (ai == NULL) { 428 warnx("can't get net id for host"); 429 } 430 nfsdirname = delimp + 1; 431 } 432 433 /* 434 * Check if we have to start the rpc-call later. 435 * If there are still identical nfs-names mounted, 436 * we skip the rpc-call. Obviously this has to 437 * happen before unmount(2), but it should happen 438 * after the previous namecheck. 439 * A non-NULL return means that this is the last 440 * mount from mntfromname that is still mounted. 441 */ 442 if (getmntname(mntfromname, NULL, NOTHING, &type, COUNT) 443 != NULL) 444 do_rpc = 1; 445 } 446 447 if (!namematch(ai)) 448 return (1); 449 if (unmount(mntonname, fflag) != 0 ) { 450 warn("unmount of %s failed", mntonname); 451 return (1); 452 } 453 if (vflag) 454 (void)printf("%s: unmount from %s\n", mntfromname, mntonname); 455 /* 456 * Report to mountd-server which nfsname 457 * has been unmounted. 458 */ 459 if (ai != NULL && !(fflag & MNT_FORCE) && do_rpc) { 460 clp = clnt_create(hostp, RPCPROG_MNT, RPCMNT_VER1, "udp"); 461 if (clp == NULL) { 462 warnx("%s: %s", hostp, 463 clnt_spcreateerror("RPCPROG_MNT")); 464 return (1); 465 } 466 clp->cl_auth = authsys_create_default(); 467 try.tv_sec = 20; 468 try.tv_usec = 0; 469 clnt_stat = clnt_call(clp, RPCMNT_UMOUNT, (xdrproc_t)xdr_dir, 470 nfsdirname, (xdrproc_t)xdr_void, (caddr_t)0, try); 471 if (clnt_stat != RPC_SUCCESS) { 472 warnx("%s: %s", hostp, 473 clnt_sperror(clp, "RPCMNT_UMOUNT")); 474 return (1); 475 } 476 /* 477 * Remove the unmounted entry from /var/db/mounttab. 478 */ 479 if (read_mtab()) { 480 clean_mtab(hostp, nfsdirname, vflag); 481 if(!write_mtab(vflag)) 482 warnx("cannot remove mounttab entry %s:%s", 483 hostp, nfsdirname); 484 free_mtab(); 485 } 486 free(orignfsdirname); 487 auth_destroy(clp->cl_auth); 488 clnt_destroy(clp); 489 } 490 return (0); 491 } 492 493 char * 494 getmntname(const char *fromname, const char *onname, 495 mntwhat what, char **type, dowhat mark) 496 { 497 static struct statfs *mntbuf; 498 static size_t mntsize = 0; 499 static char *mntcheck = NULL; 500 static char *mntcount = NULL; 501 int i, count; 502 503 if (mntsize <= 0) { 504 if ((mntsize = mntinfo(&mntbuf)) <= 0) 505 return (NULL); 506 } 507 if (mntcheck == NULL) { 508 if ((mntcheck = calloc(mntsize + 1, sizeof(int))) == NULL || 509 (mntcount = calloc(mntsize + 1, sizeof(int))) == NULL) 510 err(1, "calloc"); 511 } 512 /* 513 * We want to get the file systems in the reverse order 514 * that they were mounted. Mounted and unmounted file systems 515 * are marked or unmarked in a table called 'mntcheck'. 516 * Unmount(const char *dir, int flags) does only take the 517 * mountpoint as argument, not the destination. If we don't pay 518 * attention to the order, it can happen that an overlaying 519 * file system gets unmounted instead of the one the user 520 * has choosen. 521 */ 522 switch (mark) { 523 case NAME: 524 /* Return only the specific name */ 525 for (i = mntsize - 1; i >= 0; i--) { 526 if (fromname != NULL && what == MNTON && 527 !strcmp(mntbuf[i].f_mntfromname, fromname) && 528 mntcheck[i] != 1) { 529 if (type) 530 *type = mntbuf[i].f_fstypename; 531 return (mntbuf[i].f_mntonname); 532 } 533 if (fromname != NULL && what == MNTFROM && 534 !strcmp(mntbuf[i].f_mntonname, fromname) && 535 mntcheck[i] != 1) { 536 if (type) 537 *type = mntbuf[i].f_fstypename; 538 return (mntbuf[i].f_mntfromname); 539 } 540 } 541 return (NULL); 542 case MARK: 543 /* Mark current mount with '1' and return name */ 544 for (i = mntsize - 1; i >= 0; i--) { 545 if (mntcheck[i] == 0 && 546 (strcmp(mntbuf[i].f_mntonname, onname) == 0) && 547 (strcmp(mntbuf[i].f_mntfromname, fromname) == 0)) { 548 mntcheck[i] = 1; 549 return (mntbuf[i].f_mntonname); 550 } 551 } 552 return (NULL); 553 case UNMARK: 554 /* Unmark current mount with '0' and return name */ 555 for (i = 0; i < mntsize; i++) { 556 if (mntcheck[i] == 1 && 557 (strcmp(mntbuf[i].f_mntonname, onname) == 0) && 558 (strcmp(mntbuf[i].f_mntfromname, fromname) == 0)) { 559 mntcheck[i] = 0; 560 return (mntbuf[i].f_mntonname); 561 } 562 } 563 return (NULL); 564 case COUNT: 565 /* Count the equal mntfromnames */ 566 count = 0; 567 for (i = mntsize - 1; i >= 0; i--) { 568 if (strcmp(mntbuf[i].f_mntfromname, fromname) == 0) 569 count++; 570 } 571 /* Mark the already unmounted mounts and return 572 * mntfromname if count <= 1. Else return NULL. 573 */ 574 for (i = mntsize - 1; i >= 0; i--) { 575 if (strcmp(mntbuf[i].f_mntfromname, fromname) == 0) { 576 if (mntcount[i] == 1) 577 count--; 578 else { 579 mntcount[i] = 1; 580 break; 581 } 582 } 583 } 584 if (count <= 1) 585 return (mntbuf[i].f_mntonname); 586 else 587 return (NULL); 588 case FREE: 589 free(mntbuf); 590 free(mntcheck); 591 free(mntcount); 592 return (NULL); 593 default: 594 return (NULL); 595 } 596 } 597 598 int 599 sacmp(struct sockaddr *sa1, struct sockaddr *sa2) 600 { 601 void *p1, *p2; 602 int len; 603 604 if (sa1->sa_family != sa2->sa_family) 605 return (1); 606 607 switch (sa1->sa_family) { 608 case AF_INET: 609 p1 = &((struct sockaddr_in *)sa1)->sin_addr; 610 p2 = &((struct sockaddr_in *)sa2)->sin_addr; 611 len = 4; 612 break; 613 case AF_INET6: 614 p1 = &((struct sockaddr_in6 *)sa1)->sin6_addr; 615 p2 = &((struct sockaddr_in6 *)sa2)->sin6_addr; 616 len = 16; 617 if (((struct sockaddr_in6 *)sa1)->sin6_scope_id != 618 ((struct sockaddr_in6 *)sa2)->sin6_scope_id) 619 return (1); 620 break; 621 default: 622 return (1); 623 } 624 625 return memcmp(p1, p2, len); 626 } 627 628 int 629 namematch(struct addrinfo *ai) 630 { 631 struct addrinfo *aip; 632 633 if (nfshost == NULL || nfshost_ai == NULL) 634 return (1); 635 636 while (ai != NULL) { 637 aip = nfshost_ai; 638 while (aip != NULL) { 639 if (sacmp(ai->ai_addr, aip->ai_addr) == 0) 640 return (1); 641 aip = aip->ai_next; 642 } 643 ai = ai->ai_next; 644 } 645 646 return (0); 647 } 648 649 void 650 checkmntlist(char *name, char **fromname, char **onname, char **type) 651 { 652 653 *fromname = getmntname(name, NULL, MNTFROM, type, NAME); 654 if (*fromname == NULL) { 655 *onname = getmntname(name, NULL, MNTON, type, NAME); 656 if (*onname != NULL) 657 *fromname = name; 658 } else 659 *onname = name; 660 } 661 662 size_t 663 mntinfo(struct statfs **mntbuf) 664 { 665 static struct statfs *origbuf; 666 size_t bufsize; 667 int mntsize; 668 669 mntsize = getfsstat(NULL, 0, MNT_NOWAIT); 670 if (mntsize <= 0) 671 return (0); 672 bufsize = (mntsize + 1) * sizeof(struct statfs); 673 if ((origbuf = malloc(bufsize)) == NULL) 674 err(1, "malloc"); 675 mntsize = getfsstat(origbuf, (long)bufsize, MNT_NOWAIT); 676 *mntbuf = origbuf; 677 return (mntsize); 678 } 679 680 char * 681 getrealname(char *name, char *realname) 682 { 683 char *dirname; 684 int havedir; 685 size_t baselen; 686 size_t dirlen; 687 688 dirname = '\0'; 689 havedir = 0; 690 if (*name == '/') { 691 if (ISDOT(name + 1) || ISDOTDOT(name + 1)) 692 strcpy(realname, "/"); 693 else { 694 if ((dirname = strrchr(name + 1, '/')) == NULL) 695 snprintf(realname, MAXPATHLEN, "%s", name); 696 else 697 havedir = 1; 698 } 699 } else { 700 if (ISDOT(name) || ISDOTDOT(name)) 701 (void)realpath(name, realname); 702 else { 703 if ((dirname = strrchr(name, '/')) == NULL) { 704 if ((realpath(name, realname)) == NULL) 705 return (NULL); 706 } else 707 havedir = 1; 708 } 709 } 710 if (havedir) { 711 *dirname++ = '\0'; 712 if (ISDOT(dirname)) { 713 *dirname = '\0'; 714 if ((realpath(name, realname)) == NULL) 715 return (NULL); 716 } else if (ISDOTDOT(dirname)) { 717 *--dirname = '/'; 718 if ((realpath(name, realname)) == NULL) 719 return (NULL); 720 } else { 721 if ((realpath(name, realname)) == NULL) 722 return (NULL); 723 baselen = strlen(realname); 724 dirlen = strlen(dirname); 725 if (baselen + dirlen + 1 > MAXPATHLEN) 726 return (NULL); 727 if (realname[1] == '\0') { 728 memmove(realname + 1, dirname, dirlen); 729 realname[dirlen + 1] = '\0'; 730 } else { 731 realname[baselen] = '/'; 732 memmove(realname + baselen + 1, 733 dirname, dirlen); 734 realname[baselen + dirlen + 1] = '\0'; 735 } 736 } 737 } 738 return (realname); 739 } 740 741 /* 742 * xdr routines for mount rpc's 743 */ 744 int 745 xdr_dir(XDR *xdrsp, char *dirp) 746 { 747 748 return (xdr_string(xdrsp, &dirp, RPCMNT_PATHLEN)); 749 } 750 751 void 752 usage() 753 { 754 755 (void)fprintf(stderr, "%s\n%s\n", 756 "usage: umount [-fv] special | node", 757 " umount -a | -A [ -F fstab] [-fv] [-h host] [-t type]"); 758 exit(1); 759 } 760