mount_unionfs.c revision 97196
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 97196 2002-05-24 00:45:37Z mux $";
4928629Ssteve#endif
501558Srgrimes#endif /* not lint */
511558Srgrimes
521558Srgrimes#include <sys/param.h>
531558Srgrimes#include <sys/mount.h>
541558Srgrimes
5577031Sru#include <fs/unionfs/union.h>
561558Srgrimes
571558Srgrimes#include <err.h>
581558Srgrimes#include <stdio.h>
591558Srgrimes#include <stdlib.h>
601558Srgrimes#include <string.h>
6115770Swollman#include <sysexits.h>
621558Srgrimes#include <unistd.h>
631558Srgrimes
641558Srgrimes#include "mntopts.h"
651558Srgrimes
6615770Swollmanstatic struct mntopt mopts[] = {
671558Srgrimes	MOPT_STDOPTS,
681558Srgrimes	{ NULL }
691558Srgrimes};
701558Srgrimes
7192882Simpstatic int	subdir(const char *, const char *);
7292882Simpstatic void	usage (void) __dead2;
731558Srgrimes
741558Srgrimesint
751558Srgrimesmain(argc, argv)
761558Srgrimes	int argc;
771558Srgrimes	char *argv[];
781558Srgrimes{
7997196Smux	struct iovec iov[8];
8097196Smux	int ch, mntflags, unionflags;
8125275Sdfr	char source[MAXPATHLEN];
821558Srgrimes	char target[MAXPATHLEN];
8323680Speter	struct vfsconf vfc;
8423680Speter	int error;
851558Srgrimes
861558Srgrimes	mntflags = 0;
8797196Smux	unionflags = UNMNT_ABOVE;
8824359Simp	while ((ch = getopt(argc, argv, "bo:r")) != -1)
891558Srgrimes		switch (ch) {
901558Srgrimes		case 'b':
9197196Smux			unionflags &= ~UNMNT_OPMASK;
9297196Smux			unionflags |= UNMNT_BELOW;
931558Srgrimes			break;
941558Srgrimes		case 'o':
954065Swollman			getmntopts(optarg, mopts, &mntflags, 0);
961558Srgrimes			break;
971558Srgrimes		case 'r':
9897196Smux			unionflags &= ~UNMNT_OPMASK;
9997196Smux			unionflags |= UNMNT_REPLACE;
1001558Srgrimes			break;
1011558Srgrimes		case '?':
1021558Srgrimes		default:
1031558Srgrimes			usage();
1041558Srgrimes			/* NOTREACHED */
1051558Srgrimes		}
1061558Srgrimes	argc -= optind;
1071558Srgrimes	argv += optind;
1081558Srgrimes
1091558Srgrimes	if (argc != 2)
1101558Srgrimes		usage();
1111558Srgrimes
11252055Sphk	/* resolve both target and source with realpath(3) */
11352055Sphk	(void)checkpath(argv[0], target);
11452055Sphk	(void)checkpath(argv[1], source);
1151558Srgrimes
11625275Sdfr	if (subdir(target, source) || subdir(source, target))
11725275Sdfr		errx(EX_USAGE, "%s (%s) and %s (%s) are not distinct paths",
11825275Sdfr		    argv[0], target, argv[1], source);
11925275Sdfr
12077133Sru	error = getvfsbyname("unionfs", &vfc);
12177133Sru	if (error && vfsisloadable("unionfs")) {
12277133Sru		if (vfsload("unionfs"))
12377133Sru			err(EX_OSERR, "vfsload(unionfs)");
1242999Swollman		endvfsent();	/* flush cache */
12577133Sru		error = getvfsbyname("unionfs", &vfc);
1262999Swollman	}
12723680Speter	if (error)
12815770Swollman		errx(EX_OSERR, "union filesystem is not available");
1292999Swollman
13097196Smux	iov[0].iov_base = "fstype";
13197196Smux	iov[0].iov_len = sizeof("fstype");
13297196Smux	iov[1].iov_base = vfc.vfc_name;
13397196Smux	iov[1].iov_len = strlen(vfc.vfc_name) + 1;
13497196Smux	iov[2].iov_base = "fspath";
13597196Smux	iov[2].iov_len = sizeof("fspath");
13697196Smux	iov[3].iov_base = source;
13797196Smux	iov[3].iov_len = strlen(source) + 1;
13897196Smux	iov[4].iov_base = "target";
13997196Smux	iov[4].iov_len = sizeof("target");
14097196Smux	iov[5].iov_base = target;
14197196Smux	iov[5].iov_len = strlen(target) + 1;
14297196Smux	iov[6].iov_base = "unionflags";
14397196Smux	iov[6].iov_len = sizeof("unionflags");
14497196Smux	iov[7].iov_base = (char *)&unionflags;
14597196Smux	iov[7].iov_len = sizeof(unionflags);
14697196Smux	if (nmount(iov, 8, mntflags))
14728629Ssteve		err(EX_OSERR, "%s", target);
1481558Srgrimes	exit(0);
1491558Srgrimes}
1501558Srgrimes
1511558Srgrimesint
1521558Srgrimessubdir(p, dir)
1531558Srgrimes	const char *p;
1541558Srgrimes	const char *dir;
1551558Srgrimes{
1561558Srgrimes	int l;
1571558Srgrimes
1581558Srgrimes	l = strlen(dir);
1591558Srgrimes	if (l <= 1)
1601558Srgrimes		return (1);
1611558Srgrimes
1621558Srgrimes	if ((strncmp(p, dir, l) == 0) && (p[l] == '/' || p[l] == '\0'))
1631558Srgrimes		return (1);
1641558Srgrimes
1651558Srgrimes	return (0);
1661558Srgrimes}
1671558Srgrimes
1681558Srgrimesvoid
1691558Srgrimesusage()
1701558Srgrimes{
1711558Srgrimes	(void)fprintf(stderr,
17277042Sru		"usage: mount_unionfs [-br] [-o options] target_fs mount_point\n");
17315770Swollman	exit(EX_USAGE);
1741558Srgrimes}
175