184 lines
4.7 KiB
C
184 lines
4.7 KiB
C
/* $NetBSD: mount_puffs.c,v 1.5 2016/11/23 14:33:29 pho Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2010 Antti Kantee. All Rights Reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
|
|
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
/*
|
|
* This is to support -o getargs without having to replicate it in
|
|
* every file server. It also allows puffs filesystems to be mounted
|
|
* via "mount -a".
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
#ifndef lint
|
|
__RCSID("$NetBSD: mount_puffs.c,v 1.5 2016/11/23 14:33:29 pho Exp $");
|
|
#endif /* !lint */
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/mount.h>
|
|
|
|
#include <fs/puffs/puffs_msgif.h>
|
|
|
|
#include <err.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/cdefs.h>
|
|
#include <unistd.h>
|
|
#include <util.h>
|
|
|
|
static int
|
|
usage(void)
|
|
{
|
|
|
|
fprintf(stderr, "usage: %s [-o options] program[#source] mountpoint\n", getprogname());
|
|
return 1;
|
|
}
|
|
|
|
static int show_puffs_mount_args(const char *mountpoint)
|
|
{
|
|
const char *vtypes[] = { VNODE_TYPES };
|
|
struct puffs_kargs kargs;
|
|
|
|
if (mount(MOUNT_PUFFS, mountpoint, MNT_GETARGS, &kargs, sizeof(kargs)) == -1)
|
|
err(1, "mount");
|
|
|
|
printf("version=%d, ", kargs.pa_vers);
|
|
printf("flags=0x%x, ", kargs.pa_flags);
|
|
|
|
printf("root cookie=%p, ", kargs.pa_root_cookie);
|
|
printf("root type=%s", vtypes[kargs.pa_root_vtype]);
|
|
|
|
if (kargs.pa_root_vtype != VDIR)
|
|
printf(", root size=%llu",
|
|
(unsigned long long)kargs.pa_root_vsize);
|
|
if (kargs.pa_root_vtype == VCHR || kargs.pa_root_vtype == VBLK)
|
|
printf(", root rdev=0x%" PRIx64, (uint64_t)kargs.pa_root_rdev);
|
|
printf("\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
mount_puffs_filesystem(const char *program, const char *opts,
|
|
const char *source, const char *mountpoint)
|
|
{
|
|
int argc = 0;
|
|
const char **argv;
|
|
int rv = 0;
|
|
|
|
/* Construct an argument vector:
|
|
* program [-o opts] [source] mountpoint */
|
|
argv = ecalloc(1 + 2 + 1 + 1, sizeof(*argv));
|
|
argv[argc++] = program;
|
|
if (opts != NULL) {
|
|
argv[argc++] = "-o";
|
|
argv[argc++] = opts;
|
|
}
|
|
if (source != NULL) {
|
|
argv[argc++] = source;
|
|
}
|
|
argv[argc++] = mountpoint;
|
|
argv[argc] = NULL;
|
|
|
|
/* We intentionally use execvp(3) here because the program can
|
|
* actually be a basename. */
|
|
if (execvp(program, __UNCONST(argv)) == -1) {
|
|
warn("Cannot execute %s", program);
|
|
rv = 1;
|
|
}
|
|
|
|
free(argv);
|
|
return rv;
|
|
}
|
|
|
|
static void add_opt(char **opts, const char *opt)
|
|
{
|
|
const size_t orig_len = *opts == NULL ? 0 : strlen(*opts);
|
|
|
|
*opts = erealloc(*opts, orig_len + 1 + strlen(opt) + 1);
|
|
|
|
if (orig_len == 0) {
|
|
strcpy(*opts, opt);
|
|
}
|
|
else {
|
|
strcat(*opts, ",");
|
|
strcat(*opts, opt);
|
|
}
|
|
}
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
int mntflags = 0;
|
|
int ch;
|
|
char *opts = NULL;
|
|
int rv = 0;
|
|
|
|
while ((ch = getopt(argc, argv, "o:")) != -1) {
|
|
switch (ch) {
|
|
case 'o':
|
|
for (char *opt = optarg; (opt = strtok(opt, ",")) != NULL; opt = NULL) {
|
|
if (strcmp(opt, "getargs") == 0) {
|
|
mntflags |= MNT_GETARGS;
|
|
break; /* No need to parse it any further. */
|
|
}
|
|
else {
|
|
add_opt(&opts, opt);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
rv = usage();
|
|
goto free_opts;
|
|
}
|
|
}
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
if (argc != 2) {
|
|
rv = usage();
|
|
goto free_opts;
|
|
}
|
|
|
|
if (mntflags & MNT_GETARGS) {
|
|
/* Special case for -o getargs: retrieve kernel arguments for
|
|
* an already mounted filesystem. */
|
|
rv = show_puffs_mount_args(argv[1]);
|
|
}
|
|
else {
|
|
/* Split the program name and source. This is to allow
|
|
* filesystems to be mounted via "mount -a" i.e. /etc/fstab */
|
|
char *source = argv[0];
|
|
char *program = strsep(&source, "#");
|
|
|
|
rv = mount_puffs_filesystem(program, opts, source, argv[1]);
|
|
}
|
|
|
|
free_opts:
|
|
free(opts);
|
|
return rv;
|
|
}
|