X-Git-Url: http://git.annexia.org/?p=libguestfs.git;a=blobdiff_plain;f=daemon%2Fmount.c;h=ccd07c6bacd9143362dae5a6019df26287e64ca5;hp=440ec0de40c1a7fad8a83d081e4ca90ba61c8090;hb=4d900cdac8258daa2e99c6ceb2a4985154e94150;hpb=99bd425a0a8fb02c27e0c22b32dafa804773a7b1 diff --git a/daemon/mount.c b/daemon/mount.c index 440ec0d..ccd07c6 100644 --- a/daemon/mount.c +++ b/daemon/mount.c @@ -1,5 +1,5 @@ /* libguestfs - the guestfsd daemon - * Copyright (C) 2009 Red Hat Inc. + * Copyright (C) 2009 Red Hat Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -22,6 +22,8 @@ #include #include #include +#include +#include #include "daemon.h" #include "actions.h" @@ -30,7 +32,8 @@ int root_mounted = 0; /* The "simple mount" call offers no complex options, you can just - * mount a device on a mountpoint. + * mount a device on a mountpoint. The variations like mount_ro, + * mount_options and mount_vfs let you set progressively more things. * * It's tempting to try a direct mount(2) syscall, but that doesn't * do any autodetection, so we are better off calling out to @@ -38,33 +41,37 @@ int root_mounted = 0; */ int -do_mount (const char *device, const char *mountpoint) +do_mount_vfs (const char *options, const char *vfstype, + const char *device, const char *mountpoint) { - int len, r, is_root; + int r, is_root; char *mp; char *error; - is_root = strcmp (mountpoint, "/") == 0; + ABS_PATH (mountpoint, 0, return -1); + + is_root = STREQ (mountpoint, "/"); if (!root_mounted && !is_root) { - reply_with_error ("mount: you must mount something on / first"); + reply_with_error ("you must mount something on / first"); return -1; } - len = strlen (mountpoint) + 9; - - mp = malloc (len); + mp = sysroot_path (mountpoint); if (!mp) { reply_with_perror ("malloc"); return -1; } - snprintf (mp, len, "/sysroot%s", mountpoint); - - r = command (NULL, &error, - "mount", "-o", "sync,noatime", device, mp, NULL); + if (vfstype) + r = command (NULL, &error, + "mount", "-o", options, "-t", vfstype, device, mp, NULL); + else + r = command (NULL, &error, + "mount", "-o", options, device, mp, NULL); + free (mp); if (r == -1) { - reply_with_error ("mount: %s on %s: %s", device, mountpoint, error); + reply_with_error ("%s on %s: %s", device, mountpoint, error); free (error); return -1; } @@ -74,3 +81,337 @@ do_mount (const char *device, const char *mountpoint) return 0; } + +int +do_mount (const char *device, const char *mountpoint) +{ + return do_mount_vfs ("sync,noatime", NULL, device, mountpoint); +} + +int +do_mount_ro (const char *device, const char *mountpoint) +{ + return do_mount_vfs ("ro", NULL, device, mountpoint); +} + +int +do_mount_options (const char *options, const char *device, + const char *mountpoint) +{ + return do_mount_vfs (options, NULL, device, mountpoint); +} + +/* Again, use the external /bin/umount program, so that /etc/mtab + * is kept updated. + */ +int +do_umount (const char *pathordevice) +{ + int r; + char *err; + char *buf; + int is_dev; + + is_dev = STREQLEN (pathordevice, "/dev/", 5); + buf = is_dev ? strdup (pathordevice) + : sysroot_path (pathordevice); + if (buf == NULL) { + reply_with_perror ("malloc"); + return -1; + } + + if (is_dev) + RESOLVE_DEVICE (buf, 0, { free (buf); return -1; }); + + r = command (NULL, &err, "umount", buf, NULL); + free (buf); + + if (r == -1) { + reply_with_error ("%s: %s", pathordevice, err); + free (err); + return -1; + } + + free (err); + + /* update root_mounted? */ + + return 0; +} + +static char ** +mounts_or_mountpoints (int mp) +{ + char *out, *err; + int r; + char **ret = NULL; + int size = 0, alloc = 0; + char *p, *pend, *p2; + int len; + char matching[5 + sysroot_len]; + size_t i; + + r = command (&out, &err, "mount", NULL); + if (r == -1) { + reply_with_error ("mount: %s", err); + free (out); + free (err); + return NULL; + } + + free (err); + + /* Lines have the format: + * /dev/foo on /mountpoint type ... + */ + snprintf (matching, 5 + sysroot_len, " on %s", sysroot); + + p = out; + while (p) { + pend = strchr (p, '\n'); + if (pend) { + *pend = '\0'; + pend++; + } + + p2 = strstr (p, matching); + if (p2 != NULL) { + *p2 = '\0'; + if (add_string (&ret, &size, &alloc, p) == -1) { + free (out); + return NULL; + } + if (mp) { + p2 += 4 + sysroot_len; /* skip " on /sysroot" */ + len = strcspn (p2, " "); + + if (len == 0) /* .. just /sysroot, so we turn it into "/" */ + p2 = (char *) "/"; + else + p2[len] = '\0'; + + if (add_string (&ret, &size, &alloc, p2) == -1) { + free (out); + return NULL; + } + } + } + + p = pend; + } + + free (out); + + if (add_string (&ret, &size, &alloc, NULL) == -1) + return NULL; + + /* Convert /dev/mapper LV paths into canonical paths (RHBZ#646432). */ + for (i = 0; ret[i] != NULL; i += mp ? 2 : 1) { + if (STRPREFIX (ret[i], "/dev/mapper/") || STRPREFIX (ret[i], "/dev/dm-")) { + char *canonical; + r = lv_canonical (ret[i], &canonical); + if (r == -1) { + free_strings (ret); + return NULL; + } + if (r == 1) { + free (ret[i]); + ret[i] = canonical; + } + /* Ignore the case where r == 0. This might happen where + * eg. a LUKS /dev/mapper device is mounted, but that won't + * correspond to any LV. + */ + } + } + + return ret; +} + +char ** +do_mounts (void) +{ + return mounts_or_mountpoints (0); +} + +char ** +do_mountpoints (void) +{ + return mounts_or_mountpoints (1); +} + +/* Unmount everything mounted under /sysroot. + * + * We have to unmount in the correct order, so we sort the paths by + * longest first to ensure that child paths are unmounted by parent + * paths. + * + * This call is more important than it appears at first, because it + * is widely used by both test and production code in order to + * get back to a known state (nothing mounted, everything synchronized). + */ +static int +compare_longest_first (const void *vp1, const void *vp2) +{ + char * const *p1 = (char * const *) vp1; + char * const *p2 = (char * const *) vp2; + int n1 = strlen (*p1); + int n2 = strlen (*p2); + return n2 - n1; +} + +int +do_umount_all (void) +{ + char *out, *err; + int i, r; + char **mounts = NULL; + int size = 0, alloc = 0; + char *p, *p2, *p3, *pend; + char matching[5 + sysroot_len]; + + r = command (&out, &err, "mount", NULL); + if (r == -1) { + reply_with_error ("mount: %s", err); + free (out); + free (err); + return -1; + } + + free (err); + + /* Lines have the format: + * /dev/foo on /mountpoint type ... + */ + snprintf (matching, 5 + sysroot_len, " on %s", sysroot); + + p = out; + while (p) { + pend = strchr (p, '\n'); + if (pend) { + *pend = '\0'; + pend++; + } + + p2 = strstr (p, matching); + if (p2 != NULL) { + p2 += 4; + p3 = p2 + strcspn (p2, " "); + *p3 = '\0'; + if (add_string (&mounts, &size, &alloc, p2) == -1) { + free (out); + return -1; + } + } + + p = pend; + } + free (out); + + qsort (mounts, size, sizeof (char *), compare_longest_first); + + /* Unmount them. */ + for (i = 0; i < size; ++i) { + r = command (NULL, &err, "umount", mounts[i], NULL); + if (r == -1) { + reply_with_error ("umount: %s: %s", mounts[i], err); + free (err); + free_stringslen (mounts, size); + return -1; + } + free (err); + } + + free_stringslen (mounts, size); + + /* We've unmounted root now, so ... */ + root_mounted = 0; + + return 0; +} + +/* Mount using the loopback device. You can't use the generic + * do_mount call for this because the first parameter isn't a + * device. + */ +int +do_mount_loop (const char *file, const char *mountpoint) +{ + int r; + char *buf, *mp; + char *error; + + /* We have to prefix /sysroot on both the filename and the mountpoint. */ + mp = sysroot_path (mountpoint); + if (!mp) { + reply_with_perror ("malloc"); + return -1; + } + + buf = sysroot_path (file); + if (!file) { + reply_with_perror ("malloc"); + free (mp); + return -1; + } + + r = command (NULL, &error, "mount", "-o", "loop", buf, mp, NULL); + free (mp); + free (buf); + if (r == -1) { + reply_with_error ("%s on %s: %s", file, mountpoint, error); + free (error); + return -1; + } + + return 0; +} + +/* Specialized calls mkmountpoint and rmmountpoint are really + * variations on mkdir and rmdir which do no checking and (in the + * mkmountpoint case) set the root_mounted flag. + */ +int +do_mkmountpoint (const char *path) +{ + int r; + + /* NEED_ROOT (return -1); - we don't want this test for this call. */ + ABS_PATH (path, 0, return -1); + + CHROOT_IN; + r = mkdir (path, 0777); + CHROOT_OUT; + + if (r == -1) { + reply_with_perror ("%s", path); + return -1; + } + + /* Set the flag so that filesystems can be mounted here, + * not just on /sysroot. + */ + root_mounted = 1; + + return 0; +} + +int +do_rmmountpoint (const char *path) +{ + int r; + + /* NEED_ROOT (return -1); - we don't want this test for this call. */ + ABS_PATH (path, 0, return -1); + + CHROOT_IN; + r = rmdir (path); + CHROOT_OUT; + + if (r == -1) { + reply_with_perror ("%s", path); + return -1; + } + + return 0; +}