/* libguestfs * Copyright (C) 2010 Red Hat Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include #include #include #include #include #include #include #include #include #include #include #ifdef HAVE_SYS_TYPES_H #include #endif #include "guestfs.h" #include "guestfs-internal.h" #include "guestfs-internal-actions.h" #include "guestfs_protocol.h" static const char *kernel_name = "vmlinuz." REPO "." host_cpu; static const char *initrd_name = "initramfs." REPO "." host_cpu ".img"; static int find_path (guestfs_h *g, int (*pred) (guestfs_h *g, const char *pelem, void *data), void *data, char **pelem); static int dir_contains_file (const char *dir, const char *file); static int dir_contains_files (const char *dir, ...); static int contains_supermin_appliance (guestfs_h *g, const char *path, void *data); static int contains_ordinary_appliance (guestfs_h *g, const char *path, void *data); static char *calculate_supermin_checksum (guestfs_h *g, const char *supermin_path); static int check_for_cached_appliance (guestfs_h *g, const char *supermin_path, const char *checksum, char **kernel, char **initrd, char **appliance); static int build_supermin_appliance (guestfs_h *g, const char *supermin_path, const char *checksum, char **kernel, char **initrd, char **appliance); static int run_supermin_helper (guestfs_h *g, const char *supermin_path, const char *cachedir, size_t cdlen); /* Locate or build the appliance. * * This function locates or builds the appliance as necessary, * handling the supermin appliance, caching of supermin-built * appliances, or using an ordinary appliance. * * The return value is 0 = good, -1 = error. Returned in '*kernel' * will be the name of the kernel to use, '*initrd' the name of the * initrd, '*appliance' the name of the ext2 root filesystem. * '*appliance' can be NULL, meaning that we are using an ordinary * (non-ext2) appliance. All three strings must be freed by the * caller. However the referenced files themselves must not be * deleted. * * The process is as follows: * * (1) Look for the first element of g->path which contains a * supermin appliance skeleton. If no element has this, skip * straight to step (5). * (2) Calculate the checksum of this supermin appliance. * (3) Check whether $TMPDIR/$checksum/ directory exists, contains * a cached appliance, and passes basic security checks. If so, * return this appliance. * (4) Try to build the supermin appliance into $TMPDIR/$checksum/. * If this is successful, return it. * (5) Check each element of g->path, looking for an ordinary appliance. * If one is found, return it. */ int guestfs___build_appliance (guestfs_h *g, char **kernel, char **initrd, char **appliance) { int r; /* Step (1). */ char *supermin_path; r = find_path (g, contains_supermin_appliance, NULL, &supermin_path); if (r == -1) return -1; if (r == 1) { /* Step (2): calculate checksum. */ char *checksum = calculate_supermin_checksum (g, supermin_path); if (checksum) { /* Step (3): cached appliance exists? */ r = check_for_cached_appliance (g, supermin_path, checksum, kernel, initrd, appliance); if (r != 0) { free (supermin_path); free (checksum); return r == 1 ? 0 : -1; } /* Step (4): build supermin appliance. */ r = build_supermin_appliance (g, supermin_path, checksum, kernel, initrd, appliance); free (supermin_path); free (checksum); return r; } free (supermin_path); } /* Step (5). */ char *path; r = find_path (g, contains_ordinary_appliance, NULL, &path); if (r == -1) return -1; if (r == 1) { size_t len = strlen (path); *kernel = safe_malloc (g, len + strlen (kernel_name) + 2); *initrd = safe_malloc (g, len + strlen (initrd_name) + 2); sprintf (*kernel, "%s/%s", path, kernel_name); sprintf (*initrd, "%s/%s", path, initrd_name); *appliance = NULL; free (path); return 0; } error (g, _("cannot find any suitable libguestfs supermin or ordinary appliance on LIBGUESTFS_PATH (search path: %s)"), g->path); return -1; } static int contains_supermin_appliance (guestfs_h *g, const char *path, void *data) { return dir_contains_files (path, "supermin.d", "kmod.whitelist", NULL); } static int contains_ordinary_appliance (guestfs_h *g, const char *path, void *data) { return dir_contains_files (path, kernel_name, initrd_name, NULL); } /* supermin_path is a path which is known to contain a supermin * appliance. Using febootstrap-supermin-helper -f checksum calculate * the checksum so we can see if it is cached. */ static char * calculate_supermin_checksum (guestfs_h *g, const char *supermin_path) { size_t len = 2 * strlen (supermin_path) + 256; char cmd[len]; snprintf (cmd, len, "febootstrap-supermin-helper%s " "-f checksum " "-k '%s/kmod.whitelist' " "'%s/supermin.d' " host_cpu, g->verbose ? " --verbose" : "", supermin_path, supermin_path); if (g->verbose) guestfs___print_timestamped_message (g, "%s", cmd); /* Errors here are non-fatal, so we don't need to call error(). */ FILE *pp = popen (cmd, "r"); if (pp == NULL) return NULL; char checksum[256]; if (fgets (checksum, sizeof checksum, pp) == NULL) { pclose (pp); return NULL; } if (pclose (pp) == -1) { perror ("pclose"); return NULL; } len = strlen (checksum); if (len < 16) { /* sanity check */ fprintf (stderr, "libguestfs: internal error: febootstrap-supermin-helper -f checksum returned a short string\n"); return NULL; } if (len > 0 && checksum[len-1] == '\n') checksum[--len] = '\0'; return safe_strndup (g, checksum, len); } /* Check for cached appliance in $TMPDIR/$checksum. Check it exists * and passes some basic security checks. * * Returns: * 1 = exists, and passes * 0 = does not exist * -1 = error which should abort the whole launch process */ static int security_check_cache_file (guestfs_h *g, const char *filename, const struct stat *statbuf) { uid_t uid = geteuid (); if (statbuf->st_uid != uid) { error (g, ("libguestfs cached appliance %s is not owned by UID %d\n"), filename, uid); return -1; } if ((statbuf->st_mode & 0022) != 0) { error (g, ("libguestfs cached appliance %s is writable by group or other (mode %o)\n"), filename, statbuf->st_mode); return -1; } return 0; } static int check_for_cached_appliance (guestfs_h *g, const char *supermin_path, const char *checksum, char **kernel, char **initrd, char **appliance) { const char *tmpdir = guestfs_tmpdir (); size_t len = strlen (tmpdir) + strlen (checksum) + 2; char cachedir[len]; snprintf (cachedir, len, "%s/%s", tmpdir, checksum); /* Touch the directory to prevent it being deleting in a rare race * between us doing the checks and a tmp cleaner running. Note this * doesn't create the directory, and we ignore any error. */ (void) utime (cachedir, NULL); /* See if the cache directory exists and passes some simple checks * to make sure it has not been tampered with. Note that geteuid() * forms a part of the checksum. */ struct stat statbuf; if (lstat (cachedir, &statbuf) == -1) return 0; if (security_check_cache_file (g, cachedir, &statbuf) == -1) return -1; int ret; *kernel = safe_malloc (g, len + 8 /* / + "kernel" + \0 */); *initrd = safe_malloc (g, len + 8 /* / + "initrd" + \0 */); *appliance = safe_malloc (g, len + 6 /* / + "root" + \0 */); sprintf (*kernel, "%s/kernel", cachedir); sprintf (*initrd, "%s/initrd", cachedir); sprintf (*appliance, "%s/root", cachedir); /* Touch the files to prevent them being deleted, and to bring the * cache up to date. Note this doesn't create the files. */ (void) utime (*kernel, NULL); /* NB. *kernel is a symlink, so we want to check the kernel, not the * link (stat, not lstat). We don't do a security check on the * kernel since it's always under /boot. */ if (stat (*kernel, &statbuf) == -1) { ret = 0; goto error; } (void) utime (*initrd, NULL); if (lstat (*initrd, &statbuf) == -1) { ret = 0; goto error; } if (security_check_cache_file (g, *initrd, &statbuf) == -1) { ret = -1; goto error; } (void) utime (*appliance, NULL); if (lstat (*appliance, &statbuf) == -1) { ret = 0; goto error; } if (security_check_cache_file (g, *appliance, &statbuf) == -1) { ret = -1; goto error; } /* Exists! */ return 1; error: free (*kernel); free (*initrd); free (*appliance); return ret; } /* Build supermin appliance from supermin_path to $TMPDIR/$checksum. * * Returns: * 0 = built * -1 = error (aborts launch) */ static int build_supermin_appliance (guestfs_h *g, const char *supermin_path, const char *checksum, char **kernel, char **initrd, char **appliance) { if (g->verbose) guestfs___print_timestamped_message (g, "begin building supermin appliance"); const char *tmpdir = guestfs_tmpdir (); size_t cdlen = strlen (tmpdir) + strlen (checksum) + 2; char cachedir[cdlen]; snprintf (cachedir, cdlen, "%s/%s", tmpdir, checksum); /* Don't worry about this failing, because the * febootstrap-supermin-helper command will fail if the directory * doesn't exist. Note the directory might already exist, eg. if a * tmp cleaner has removed the existing appliance but not the * directory itself. */ (void) mkdir (cachedir, 0755); if (g->verbose) guestfs___print_timestamped_message (g, "run febootstrap-supermin-helper"); int r = run_supermin_helper (g, supermin_path, cachedir, cdlen); if (r == -1) return -1; if (g->verbose) guestfs___print_timestamped_message (g, "finished building supermin appliance"); *kernel = safe_malloc (g, cdlen + 8 /* / + "kernel" + \0 */); *initrd = safe_malloc (g, cdlen + 8 /* / + "initrd" + \0 */); *appliance = safe_malloc (g, cdlen + 6 /* / + "root" + \0 */); sprintf (*kernel, "%s/kernel", cachedir); sprintf (*initrd, "%s/initrd", cachedir); sprintf (*appliance, "%s/root", cachedir); return 0; } /* Run febootstrap-supermin-helper and tell it to generate the * appliance. Note that we have to do an explicit fork/exec here. * 'system' goes via the shell, and on systems that have bash, bash * has a misfeature where it resets the euid to uid which breaks * virt-v2v. 'posix_spawn' was also considered but that doesn't allow * us to reset the umask. */ static int run_supermin_helper (guestfs_h *g, const char *supermin_path, const char *cachedir, size_t cdlen) { pid_t pid = fork (); if (pid == -1) { perrorf (g, "fork"); return -1; } if (pid > 0) { /* Parent. */ int status; if (waitpid (pid, &status, 0) == -1) { perrorf (g, "waitpid"); return -1; } if (!WIFEXITED (status) || WEXITSTATUS (status) != 0) { error (g, _("external command failed, see earlier error messages")); return -1; } return 0; } /* Child. */ /* Set a sensible umask in the subprocess, so kernel and initrd * output files are world-readable (RHBZ#610880). */ umask (0022); /* Set uid/gid in the child. This is a workaround for a misfeature * in bash which breaks virt-v2v - see the comment at the top of * this function. */ if (getuid () == 0) { int egid = getegid (); int euid = geteuid (); if (egid != 0 || euid != 0) { if (seteuid (0) == -1) { perror ("seteuid"); _exit (EXIT_FAILURE); } if (setgid (egid) == -1) { perror ("setgid"); _exit (EXIT_FAILURE); } if (setuid (euid) == -1) { perror ("setuid"); _exit (EXIT_FAILURE); } } } size_t pathlen = strlen (supermin_path); const char *argv[30]; size_t i = 0; argv[i++] = "febootstrap-supermin-helper"; if (g->verbose) argv[i++] = "--verbose"; argv[i++] = "-f"; argv[i++] = "ext2"; argv[i++] = "-k"; char whitelist[pathlen + 32]; snprintf (whitelist, pathlen + 32, "%s/kmod.whitelist", supermin_path); argv[i++] = whitelist; char supermin_d[pathlen + 32]; snprintf (supermin_d, pathlen + 32, "%s/supermin.d", supermin_path); argv[i++] = supermin_d; argv[i++] = host_cpu; char kernel[cdlen + 32]; snprintf (kernel, cdlen + 32, "%s/kernel", cachedir); argv[i++] = kernel; char initrd[cdlen + 32]; snprintf (initrd, cdlen + 32, "%s/initrd", cachedir); argv[i++] = initrd; char root[cdlen + 32]; snprintf (root, cdlen + 32, "%s/root", cachedir); argv[i++] = root; argv[i++] = NULL; execvp ("febootstrap-supermin-helper", (char * const *) argv); perror ("execvp"); _exit (EXIT_FAILURE); } /* Search elements of g->path, returning the first path element which * matches the predicate function 'pred'. * * Function 'pred' must return a true or false value. If it returns * -1 then the entire search is aborted. * * Return values: * 1 = a path element matched, it is returned in *pelem_ret and must be * freed by the caller, * 0 = no path element matched, *pelem_ret is set to NULL, or * -1 = error which aborts the launch process */ static int find_path (guestfs_h *g, int (*pred) (guestfs_h *g, const char *pelem, void *data), void *data, char **pelem_ret) { size_t len; int r; const char *pelem = g->path; /* Note that if g->path is an empty string, we want to check the * current directory (for backwards compatibility with * libguestfs < 1.5.4). */ do { len = strcspn (pelem, ":"); /* Empty element or "." means current directory. */ if (len == 0) *pelem_ret = safe_strdup (g, "."); else *pelem_ret = safe_strndup (g, pelem, len); r = pred (g, *pelem_ret, data); if (r == -1) { free (*pelem_ret); return -1; } if (r != 0) /* predicate matched */ return 1; free (*pelem_ret); if (pelem[len] == ':') pelem += len + 1; else pelem += len; } while (*pelem); /* Predicate didn't match on any path element. */ *pelem_ret = NULL; return 0; } /* Returns true iff file is contained in dir. */ static int dir_contains_file (const char *dir, const char *file) { size_t dirlen = strlen (dir); size_t filelen = strlen (file); size_t len = dirlen + filelen + 2; char path[len]; snprintf (path, len, "%s/%s", dir, file); return access (path, F_OK) == 0; } /* Returns true iff every listed file is contained in 'dir'. */ static int dir_contains_files (const char *dir, ...) { va_list args; const char *file; va_start (args, dir); while ((file = va_arg (args, const char *)) != NULL) { if (!dir_contains_file (dir, file)) { va_end (args); return 0; } } va_end (args); return 1; }