mount_unionfs.c revision 101829
11558Srgrimes/* 21558Srgrimes * Copyright (c) 1992, 1993, 1994 31558Srgrimes * The Regents of the University of California. All rights reserved. 41558Srgrimes * 51558Srgrimes * This code is derived from software donated to Berkeley by 61558Srgrimes * Jan-Simon Pendry. 71558Srgrimes * 81558Srgrimes * Redistribution and use in source and binary forms, with or without 91558Srgrimes * modification, are permitted provided that the following conditions 101558Srgrimes * are met: 111558Srgrimes * 1. Redistributions of source code must retain the above copyright 121558Srgrimes * notice, this list of conditions and the following disclaimer. 131558Srgrimes * 2. Redistributions in binary form must reproduce the above copyright 141558Srgrimes * notice, this list of conditions and the following disclaimer in the 151558Srgrimes * documentation and/or other materials provided with the distribution. 161558Srgrimes * 3. All advertising materials mentioning features or use of this software 171558Srgrimes * must display the following acknowledgement: 181558Srgrimes * This product includes software developed by the University of 191558Srgrimes * California, Berkeley and its contributors. 201558Srgrimes * 4. Neither the name of the University nor the names of its contributors 211558Srgrimes * may be used to endorse or promote products derived from this software 221558Srgrimes * without specific prior written permission. 231558Srgrimes * 241558Srgrimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 251558Srgrimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 261558Srgrimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 271558Srgrimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 281558Srgrimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 291558Srgrimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 301558Srgrimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 311558Srgrimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 321558Srgrimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 331558Srgrimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 341558Srgrimes * SUCH DAMAGE. 351558Srgrimes */ 361558Srgrimes 371558Srgrimes#ifndef lint 3828629Sstevestatic const char copyright[] = 391558Srgrimes"@(#) Copyright (c) 1992, 1993, 1994\n\ 401558Srgrimes The Regents of the University of California. All rights reserved.\n"; 411558Srgrimes#endif /* not lint */ 421558Srgrimes 431558Srgrimes#ifndef lint 4428629Ssteve#if 0 451558Srgrimesstatic char sccsid[] = "@(#)mount_union.c 8.5 (Berkeley) 3/27/94"; 4628629Ssteve#else 4728629Sstevestatic const char rcsid[] = 4850476Speter "$FreeBSD: head/sbin/mount_unionfs/mount_unionfs.c 101829 2002-08-13 16:07:39Z mux $"; 4928629Ssteve#endif 501558Srgrimes#endif /* not lint */ 511558Srgrimes 521558Srgrimes#include <sys/param.h> 531558Srgrimes#include <sys/mount.h> 54101829Smux#include <sys/uio.h> 551558Srgrimes 561558Srgrimes#include <err.h> 571558Srgrimes#include <stdio.h> 581558Srgrimes#include <stdlib.h> 591558Srgrimes#include <string.h> 6015770Swollman#include <sysexits.h> 611558Srgrimes#include <unistd.h> 621558Srgrimes 631558Srgrimes#include "mntopts.h" 641558Srgrimes 6515770Swollmanstatic struct mntopt mopts[] = { 661558Srgrimes MOPT_STDOPTS, 671558Srgrimes { NULL } 681558Srgrimes}; 691558Srgrimes 7092882Simpstatic int subdir(const char *, const char *); 7192882Simpstatic void usage (void) __dead2; 721558Srgrimes 731558Srgrimesint 741558Srgrimesmain(argc, argv) 751558Srgrimes int argc; 761558Srgrimes char *argv[]; 771558Srgrimes{ 7897196Smux struct iovec iov[8]; 7998266Smux int ch, mntflags; 8025275Sdfr char source[MAXPATHLEN]; 811558Srgrimes char target[MAXPATHLEN]; 82101270Smux int iovcnt; 831558Srgrimes 8498266Smux iovcnt = 6; 851558Srgrimes mntflags = 0; 8624359Simp while ((ch = getopt(argc, argv, "bo:r")) != -1) 871558Srgrimes switch (ch) { 881558Srgrimes case 'b': 8998266Smux iov[6].iov_base = "below"; 9098266Smux iov[6].iov_len = strlen(iov[6].iov_base) + 1; 9198266Smux iov[7].iov_base = NULL; 9298266Smux iov[7].iov_len = 0; 9398266Smux iovcnt = 8; 941558Srgrimes break; 951558Srgrimes case 'o': 964065Swollman getmntopts(optarg, mopts, &mntflags, 0); 971558Srgrimes break; 981558Srgrimes case 'r': 9998266Smux iov[6].iov_base = "replace"; 10098266Smux iov[6].iov_len = strlen(iov[6].iov_base) + 1; 10198266Smux iov[7].iov_base = NULL; 10298266Smux iov[7].iov_len = 0; 10398266Smux iovcnt = 8; 1041558Srgrimes break; 1051558Srgrimes case '?': 1061558Srgrimes default: 1071558Srgrimes usage(); 1081558Srgrimes /* NOTREACHED */ 1091558Srgrimes } 1101558Srgrimes argc -= optind; 1111558Srgrimes argv += optind; 1121558Srgrimes 1131558Srgrimes if (argc != 2) 1141558Srgrimes usage(); 1151558Srgrimes 11652055Sphk /* resolve both target and source with realpath(3) */ 11752055Sphk (void)checkpath(argv[0], target); 11852055Sphk (void)checkpath(argv[1], source); 1191558Srgrimes 12025275Sdfr if (subdir(target, source) || subdir(source, target)) 12125275Sdfr errx(EX_USAGE, "%s (%s) and %s (%s) are not distinct paths", 12225275Sdfr argv[0], target, argv[1], source); 12325275Sdfr 12497196Smux iov[0].iov_base = "fstype"; 12598266Smux iov[0].iov_len = strlen(iov[0].iov_base) + 1; 126101270Smux iov[1].iov_base = "unionfs"; 127101270Smux iov[1].iov_len = strlen(iov[1].iov_base) + 1; 12897196Smux iov[2].iov_base = "fspath"; 12998266Smux iov[2].iov_len = strlen(iov[2].iov_base) + 1; 13097196Smux iov[3].iov_base = source; 13197196Smux iov[3].iov_len = strlen(source) + 1; 13297196Smux iov[4].iov_base = "target"; 13398266Smux iov[4].iov_len = strlen(iov[4].iov_base) + 1; 13497196Smux iov[5].iov_base = target; 13597196Smux iov[5].iov_len = strlen(target) + 1; 13698266Smux if (nmount(iov, iovcnt, mntflags)) 13728629Ssteve err(EX_OSERR, "%s", target); 1381558Srgrimes exit(0); 1391558Srgrimes} 1401558Srgrimes 1411558Srgrimesint 1421558Srgrimessubdir(p, dir) 1431558Srgrimes const char *p; 1441558Srgrimes const char *dir; 1451558Srgrimes{ 1461558Srgrimes int l; 1471558Srgrimes 1481558Srgrimes l = strlen(dir); 1491558Srgrimes if (l <= 1) 1501558Srgrimes return (1); 1511558Srgrimes 1521558Srgrimes if ((strncmp(p, dir, l) == 0) && (p[l] == '/' || p[l] == '\0')) 1531558Srgrimes return (1); 1541558Srgrimes 1551558Srgrimes return (0); 1561558Srgrimes} 1571558Srgrimes 1581558Srgrimesvoid 1591558Srgrimesusage() 1601558Srgrimes{ 1611558Srgrimes (void)fprintf(stderr, 16277042Sru "usage: mount_unionfs [-br] [-o options] target_fs mount_point\n"); 16315770Swollman exit(EX_USAGE); 1641558Srgrimes} 165