aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorCurt Wohlgemuth <curtw@google.com>2012-03-08 15:11:30 -0800
committerH. Peter Anvin <hpa@linux.intel.com>2012-05-15 15:29:39 -0700
commita0127d3dbd482a709711f8cd16643325883dcc3f (patch)
tree49ad3fb140c5d41193cf954e2aaf4cc1ba939e14
parent5cbb4b62b12bbdabb0b82804fc84c763553aeae2 (diff)
downloadklibc-a0127d3dbd482a709711f8cd16643325883dcc3f.tar.gz
[klibc] kinit: Add ability to mount filesystems via /etc/fstab or cmdline
This patch adds the ability to mount filesystems via an embedded fstab or via the kernel command line. When using the kernel command-line, the following format is required: 'kinit_mount=<fs_dev>;<fs_dir>;<fs_type>;<fs_opts>' Multiple mount options can be specified; they are evaluated in the order they appear in the command-line. Mount directories ('fs_dir') sent in will have '/root' prepended to them before the actual mount(2) call is made. Tested: Sent mount commands via kinit_mount= command line; also used an embedded /etc/fstab file. Signed-off-by: San Mehat <san@google.com> Signed-off-by: Curt Wohlgemuth <curtw@google.com> Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
-rw-r--r--usr/kinit/do_mounts.c129
1 files changed, 127 insertions, 2 deletions
diff --git a/usr/kinit/do_mounts.c b/usr/kinit/do_mounts.c
index 3ffac91967522..9c149fd8def8c 100644
--- a/usr/kinit/do_mounts.c
+++ b/usr/kinit/do_mounts.c
@@ -7,6 +7,7 @@
#include <string.h>
#include <unistd.h>
#include <inttypes.h>
+#include <mntent.h>
#include "do_mounts.h"
#include "kinit.h"
@@ -199,12 +200,120 @@ mount_root(int argc, char *argv[], dev_t root_dev, const char *root_dev_name)
return ret;
}
+/* Allocate a buffer and prepend '/root' onto 'src'. */
+static char *prepend_root_dir(const char *src)
+{
+ size_t len = strlen(src) + 6; /* "/root" */
+ char *p = malloc(len);
+
+ if (!p)
+ return NULL;
+
+ strcpy(p, "/root");
+ strcat(p, src);
+ return p;
+}
+
+int do_cmdline_mounts(int argc, char *argv[])
+{
+ int arg_i;
+
+ for (arg_i = 0; arg_i < argc; arg_i++) {
+ const char *fs_dev, *fs_dir, *fs_type;
+ char *fs_opts;
+ unsigned long flags = 0;
+ char new_fs_opts[128] = { 0 };
+ char *saveptr = NULL;
+ char *fs_opts_savedptr = NULL;
+ int opt_first = 1;
+ const char *opt;
+ char *new_dir;
+
+ if (strncmp(argv[arg_i], "kinit_mount=", 12))
+ continue;
+ /*
+ * Format:
+ * <fs_dev>;<dir>;<fs_type>;[opt1],[optn...]
+ */
+ fs_dev = strtok_r(&argv[arg_i][12], ";", &saveptr);
+ if (!fs_dev) {
+ fprintf(stderr, "Failed to parse fs_dev\n");
+ continue;
+ }
+ fs_dir = strtok_r(NULL, ";", &saveptr);
+ if (!fs_dir) {
+ fprintf(stderr, "Failed to parse fs_dir\n");
+ continue;
+ }
+ fs_type = strtok_r(NULL, ";", &saveptr);
+ if (!fs_type) {
+ fprintf(stderr, "Failed to parse fs_type\n");
+ continue;
+ }
+ fs_opts = strtok_r(NULL, ";", &saveptr);
+ if (!fs_opts) {
+ fprintf(stderr, "Failed to parse fs_opts\n");
+ continue;
+ }
+
+ /* If 'fs_opts' specifies 'ro' or 'bind', gobble it up. */
+ while ((opt = strtok_r((opt_first ? fs_opts : NULL),
+ ",", &fs_opts_savedptr))) {
+ if (!strcmp(opt, "ro"))
+ flags |= MS_RDONLY;
+ else if (!strcmp(opt, "bind"))
+ flags |= MS_BIND;
+ else {
+ if (!opt_first)
+ strcat(new_fs_opts, ",");
+ strcat(new_fs_opts, opt);
+ }
+ if (opt_first)
+ opt_first = 0;
+ }
+ new_dir = prepend_root_dir(fs_dir);
+ if (! new_dir)
+ return -ENOMEM;
+
+ if (!mount_block(fs_dev, new_dir, fs_type,
+ flags, new_fs_opts))
+ fprintf(stderr, "Skipping failed mount '%s'\n", fs_dev);
+
+ free(new_dir);
+ }
+ return 0;
+}
+
+int do_fstab_mounts(FILE *fp)
+{
+ struct mntent *ent = NULL;
+ char *new_dir;
+
+ while ((ent = getmntent(fp))) {
+ new_dir = prepend_root_dir(ent->mnt_dir);
+ if (! new_dir)
+ return -ENOMEM;
+ if (!mount_block(ent->mnt_fsname,
+ new_dir,
+ ent->mnt_type,
+ 0,
+ ent->mnt_opts)) {
+ fprintf(stderr, "Skipping failed mount '%s'\n",
+ ent->mnt_fsname);
+ }
+ free(new_dir);
+ }
+ return 0;
+}
+
int do_mounts(int argc, char *argv[])
{
const char *root_dev_name = get_arg(argc, argv, "root=");
const char *root_delay = get_arg(argc, argv, "rootdelay=");
const char *load_ramdisk = get_arg(argc, argv, "load_ramdisk=");
dev_t root_dev = 0;
+ int err;
+ FILE *fp;
dprintf("kinit: do_mounts\n");
@@ -241,6 +350,22 @@ int do_mounts(int argc, char *argv[])
}
if (root_dev == Root_MULTI)
- return mount_roots(argc, argv, root_dev_name);
- return mount_root(argc, argv, root_dev, root_dev_name);
+ err = mount_roots(argc, argv, root_dev_name);
+ else
+ err = mount_root(argc, argv, root_dev, root_dev_name);
+
+ if (err)
+ return err;
+
+ if ((fp = setmntent("/etc/fstab", "r"))) {
+ err = do_fstab_mounts(fp);
+ fclose(fp);
+ }
+
+ if (err)
+ return err;
+
+ if (get_arg(argc, argv, "kinit_mount="))
+ err = do_cmdline_mounts(argc, argv);
+ return err;
}