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