summaryrefslogtreecommitdiffstats
path: root/sbin/mdconfig
diff options
context:
space:
mode:
authortrasz <trasz@FreeBSD.org>2012-01-27 11:48:44 +0000
committertrasz <trasz@FreeBSD.org>2012-01-27 11:48:44 +0000
commit1efa8b844757e2909b6ff42c6d940e38c8ce41c0 (patch)
tree6b40c5627f1f03a1768dfb48169f422c3d4de434 /sbin/mdconfig
parent137442a3ecbf41dae519d7c503b25fa7208a8bd4 (diff)
downloadFreeBSD-src-1efa8b844757e2909b6ff42c6d940e38c8ce41c0.zip
FreeBSD-src-1efa8b844757e2909b6ff42c6d940e38c8ce41c0.tar.gz
Rewrite option parsing in mdconfig(8). This makes it more user-friendly
by removing the ordering requirements and adding more descriptive error messages; it also makes it more readable and maintainable. Sponsored by: The FreeBSD Foundation
Diffstat (limited to 'sbin/mdconfig')
-rw-r--r--sbin/mdconfig/mdconfig.c249
1 files changed, 131 insertions, 118 deletions
diff --git a/sbin/mdconfig/mdconfig.c b/sbin/mdconfig/mdconfig.c
index 0e44b1a..55bda26 100644
--- a/sbin/mdconfig/mdconfig.c
+++ b/sbin/mdconfig/mdconfig.c
@@ -1,7 +1,11 @@
/*-
* Copyright (c) 2000-2004 Poul-Henning Kamp <phk@FreeBSD.org>
+ * Copyright (c) 2012 The FreeBSD Foundation
* All rights reserved.
*
+ * Portions of this software were developed by Edward Tomasz Napierala
+ * under sponsorship from the FreeBSD Foundation.
+ *
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
@@ -49,7 +53,6 @@
#include <string.h>
#include <unistd.h>
-
static struct md_ioctl mdio;
static enum {UNSET, ATTACH, DETACH, LIST} action = UNSET;
static int nflag;
@@ -72,6 +75,7 @@ static void md_prthumanval(char *length);
static void
usage(void)
{
+
fprintf(stderr,
"usage: mdconfig -a -t type [-n] [-o [no]option] ... [-f file]\n"
" [-s size] [-S sectorsize] [-u unit]\n"
@@ -92,8 +96,7 @@ main(int argc, char **argv)
{
int ch, fd, i, vflag;
char *p;
- int cmdline = 0;
- char *mdunit = NULL;
+ char *fflag = NULL, *tflag = NULL, *uflag = NULL;
bzero(&mdio, sizeof(mdio));
mdio.md_file = malloc(PATH_MAX);
@@ -101,79 +104,59 @@ main(int argc, char **argv)
err(1, "could not allocate memory");
vflag = 0;
bzero(mdio.md_file, PATH_MAX);
+
+ if (argc == 1)
+ usage();
+
while ((ch = getopt(argc, argv, "ab:df:lno:s:S:t:u:vx:y:")) != -1) {
switch (ch) {
case 'a':
- if (cmdline != 0)
- usage();
+ if (action != UNSET && action != ATTACH)
+ errx(1,
+ "-a is mutually exclusive with -d and -l");
action = ATTACH;
- cmdline = 1;
break;
case 'd':
- if (cmdline != 0)
- usage();
+ if (action != UNSET && action != DETACH)
+ errx(1,
+ "-d is mutually exclusive with -a and -l");
action = DETACH;
- mdio.md_options = MD_AUTOUNIT;
- cmdline = 3;
+ mdio.md_options |= MD_AUTOUNIT;
break;
case 'l':
- if (cmdline != 0)
- usage();
+ if (action != UNSET && action != LIST)
+ errx(1,
+ "-l is mutually exclusive with -a and -d");
action = LIST;
- mdio.md_options = MD_AUTOUNIT;
- cmdline = 3;
+ mdio.md_options |= MD_AUTOUNIT;
break;
case 'n':
nflag = 1;
break;
case 't':
- if (cmdline != 1)
- usage();
+ if (tflag != NULL)
+ errx(1, "-t can be passed only once");
+ tflag = optarg;
if (!strcmp(optarg, "malloc")) {
mdio.md_type = MD_MALLOC;
- mdio.md_options = MD_AUTOUNIT | MD_COMPRESS;
+ mdio.md_options |= MD_AUTOUNIT | MD_COMPRESS;
} else if (!strcmp(optarg, "preload")) {
mdio.md_type = MD_PRELOAD;
- mdio.md_options = 0;
} else if (!strcmp(optarg, "vnode")) {
mdio.md_type = MD_VNODE;
- mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
+ mdio.md_options |= MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
} else if (!strcmp(optarg, "swap")) {
mdio.md_type = MD_SWAP;
- mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
- } else {
- usage();
- }
- cmdline=2;
+ mdio.md_options |= MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
+ } else
+ errx(1, "unknown type: %s", optarg);
break;
case 'f':
- if (cmdline == 0) {
- action = ATTACH;
- cmdline = 1;
- }
- if (cmdline == 1) {
- /* Imply ``-t vnode'' */
- mdio.md_type = MD_VNODE;
- mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
- cmdline = 2;
- }
- if (cmdline != 2)
- usage();
- md_set_file(optarg);
+ if (fflag != NULL)
+ errx(1, "-f can be passed only once");
+ fflag = optarg;
break;
case 'o':
- if (action == DETACH) {
- if (!strcmp(optarg, "force"))
- mdio.md_options |= MD_FORCE;
- else if (!strcmp(optarg, "noforce"))
- mdio.md_options &= ~MD_FORCE;
- else
- errx(1, "Unknown option: %s.", optarg);
- break;
- }
-
- if (cmdline != 2)
- usage();
if (!strcmp(optarg, "async"))
mdio.md_options |= MD_ASYNC;
else if (!strcmp(optarg, "noasync"))
@@ -199,27 +182,12 @@ main(int argc, char **argv)
else if (!strcmp(optarg, "noreserve"))
mdio.md_options &= ~MD_RESERVE;
else
- errx(1, "Unknown option: %s.", optarg);
+ errx(1, "unknown option: %s", optarg);
break;
case 'S':
- if (cmdline != 2)
- usage();
mdio.md_sectorsize = strtoul(optarg, &p, 0);
break;
case 's':
- if (cmdline == 0) {
- /* Imply ``-a'' */
- action = ATTACH;
- cmdline = 1;
- }
- if (cmdline == 1) {
- /* Imply ``-t swap'' */
- mdio.md_type = MD_SWAP;
- mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
- cmdline = 2;
- }
- if (cmdline != 2)
- usage();
mdio.md_mediasize = (off_t)strtoumax(optarg, &p, 0);
if (p == NULL || *p == '\0')
mdio.md_mediasize *= DEV_BSIZE;
@@ -235,34 +203,22 @@ main(int argc, char **argv)
mdio.md_mediasize <<= 30;
mdio.md_mediasize <<= 10;
} else
- errx(1, "Unknown suffix on -s argument");
+ errx(1, "unknown suffix on -s argument");
break;
case 'u':
- if (cmdline != 2 && cmdline != 3)
- usage();
if (!strncmp(optarg, "/dev/", 5))
optarg += 5;
if (!strncmp(optarg, MD_NAME, sizeof(MD_NAME) - 1))
optarg += sizeof(MD_NAME) - 1;
- mdio.md_unit = strtoul(optarg, &p, 0);
- if (mdio.md_unit == (unsigned)ULONG_MAX || *p != '\0')
- errx(1, "bad unit: %s", optarg);
- mdunit = optarg;
- mdio.md_options &= ~MD_AUTOUNIT;
+ uflag = optarg;
break;
case 'v':
- if (cmdline != 3)
- usage();
vflag = OPT_VERBOSE;
break;
case 'x':
- if (cmdline != 2)
- usage();
mdio.md_fwsectors = strtoul(optarg, &p, 0);
break;
case 'y':
- if (cmdline != 2)
- usage();
mdio.md_fwheads = strtoul(optarg, &p, 0);
break;
default:
@@ -272,14 +228,88 @@ main(int argc, char **argv)
argc -= optind;
argv += optind;
- if (action == UNSET) {
- if (argc != 1)
- usage();
+
+ if (action == UNSET)
action = ATTACH;
- mdio.md_type = MD_VNODE;
- mdio.md_options = MD_CLUSTER | MD_AUTOUNIT | MD_COMPRESS;
- cmdline = 2;
- md_set_file(*argv);
+
+ if (action == ATTACH) {
+ if (tflag == NULL) {
+ /*
+ * Try to infer the type based on other arguments.
+ */
+ if (fflag != NULL || argc > 0) {
+ /* Imply ``-t vnode'' */
+ mdio.md_type = MD_VNODE;
+ mdio.md_options |= MD_CLUSTER | MD_AUTOUNIT |
+ MD_COMPRESS;
+ } else if (mdio.md_mediasize != 0) {
+ /* Imply ``-t swap'' */
+ mdio.md_type = MD_SWAP;
+ mdio.md_options |= MD_CLUSTER | MD_AUTOUNIT |
+ MD_COMPRESS;
+ } else
+ errx(1, "unable to determine type");
+ }
+
+ if ((fflag != NULL || argc > 0) && mdio.md_type != MD_VNODE)
+ errx(1, "only -t vnode can be used with file name");
+
+ if (mdio.md_type == MD_VNODE) {
+ if (fflag != NULL) {
+ if (argc != 0)
+ usage();
+ md_set_file(fflag);
+ } else {
+ if (argc != 1)
+ usage();
+ md_set_file(*argv);
+ }
+
+ if ((mdio.md_options & MD_READONLY) == 0 &&
+ access(mdio.md_file, W_OK) < 0 &&
+ (errno == EACCES || errno == EPERM ||
+ errno == EROFS)) {
+ warnx("WARNING: opening backing store: %s "
+ "readonly", mdio.md_file);
+ mdio.md_options |= MD_READONLY;
+ }
+ }
+
+ if ((mdio.md_type == MD_MALLOC || mdio.md_type == MD_SWAP) &&
+ mdio.md_mediasize == 0)
+ errx(1, "must specify -s for -t malloc or -t swap");
+ if (mdio.md_type == MD_VNODE && mdio.md_file[0] == '\0')
+ errx(1, "must specify -f for -t vnode");
+ } else {
+ if (mdio.md_sectorsize != 0)
+ errx(1, "-S can only be used with -a");
+ if (mdio.md_mediasize != 0)
+ errx(1, "-s can only be used with -a");
+ if (mdio.md_fwsectors != 0)
+ errx(1, "-x can only be used with -a");
+ if (mdio.md_fwheads != 0)
+ errx(1, "-y can only be used with -a");
+ if (fflag != NULL)
+ errx(1, "-f can only be used with -a");
+ if (tflag != NULL)
+ errx(1, "-t can only be used with -a");
+ if (argc > 0)
+ errx(1, "file can only be used with -a");
+ if (action != DETACH && (mdio.md_options & ~MD_AUTOUNIT) != 0)
+ errx(1, "-o can only be used with -a and -d");
+ if (action == DETACH &&
+ (mdio.md_options & ~(MD_FORCE | MD_AUTOUNIT)) != 0)
+ errx(1, "only -o [no]force can be used with -d");
+ }
+
+ if (action != LIST && vflag == OPT_VERBOSE)
+ errx(1, "-v can only be used with -l");
+
+ if (uflag != NULL) {
+ mdio.md_unit = strtoul(uflag, &p, 0);
+ if (mdio.md_unit == (unsigned)ULONG_MAX || *p != '\0')
+ errx(1, "bad unit: %s", uflag);
+ mdio.md_options &= ~MD_AUTOUNIT;
}
mdio.md_version = MDIOVERSION;
@@ -290,36 +320,8 @@ main(int argc, char **argv)
fd = open("/dev/" MDCTL_NAME, O_RDWR, 0);
if (fd < 0)
err(1, "open(/dev/%s)", MDCTL_NAME);
- if (cmdline == 2
- && (mdio.md_type == MD_MALLOC || mdio.md_type == MD_SWAP))
- if (mdio.md_mediasize == 0)
- errx(1, "must specify -s for -t malloc or -t swap");
- if (cmdline == 2 && mdio.md_type == MD_VNODE)
- if (mdio.md_file[0] == '\0')
- errx(1, "must specify -f for -t vnode");
- if (mdio.md_type == MD_VNODE &&
- (mdio.md_options & MD_READONLY) == 0) {
- if (access(mdio.md_file, W_OK) < 0 &&
- (errno == EACCES || errno == EPERM || errno == EROFS)) {
- fprintf(stderr,
- "WARNING: opening backing store: %s readonly\n",
- mdio.md_file);
- mdio.md_options |= MD_READONLY;
- }
- }
- if (action == LIST) {
- if (mdio.md_options & MD_AUTOUNIT) {
- /*
- * Listing all devices. This is why we pass NULL
- * together with OPT_LIST.
- */
- md_list(NULL, OPT_LIST | vflag);
- } else {
- return (md_query(mdunit));
- }
- } else if (action == ATTACH) {
- if (cmdline < 2)
- usage();
+
+ if (action == ATTACH) {
i = ioctl(fd, MDIOCATTACH, &mdio);
if (i < 0)
err(1, "ioctl(/dev/%s)", MDCTL_NAME);
@@ -327,13 +329,23 @@ main(int argc, char **argv)
printf("%s%d\n", nflag ? "" : MD_NAME, mdio.md_unit);
} else if (action == DETACH) {
if (mdio.md_options & MD_AUTOUNIT)
- usage();
+ errx(1, "-d requires -u");
i = ioctl(fd, MDIOCDETACH, &mdio);
if (i < 0)
err(1, "ioctl(/dev/%s)", MDCTL_NAME);
+ } else if (action == LIST) {
+ if (mdio.md_options & MD_AUTOUNIT) {
+ /*
+ * Listing all devices. This is why we pass NULL
+ * together with OPT_LIST.
+ */
+ md_list(NULL, OPT_LIST | vflag);
+ } else
+ return (md_query(uflag));
+
} else
usage();
- close (fd);
+ close(fd);
return (0);
}
@@ -507,5 +519,6 @@ md_prthumanval(char *length)
int
md_query(char *name)
{
+
return (md_list(name, OPT_UNIT));
}
OpenPOWER on IntegriCloud