1 /* libguestfs - the guestfsd daemon
2 * Copyright (C) 2009 Red Hat Inc.
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
25 #include "guestfs_protocol.h"
29 #include "ignore-value.h"
32 umount_ignore_fail (const char *path)
34 ignore_value (command (NULL, NULL, "umount", path, NULL));
38 do_command (char *const *argv)
42 char *sysroot_dev, *sysroot_dev_pts, *sysroot_proc,
43 *sysroot_selinux, *sysroot_sys;
44 int dev_ok, dev_pts_ok, proc_ok, selinux_ok, sys_ok;
46 /* We need a root filesystem mounted to do this. */
47 NEED_ROOT (, return NULL);
49 /* Conveniently, argv is already a NULL-terminated argv-style array
50 * of parameters, so we can pass it straight in to our internal
51 * commandv. We just have to check the list is non-empty.
53 if (argv[0] == NULL) {
54 reply_with_error ("passed an empty list");
58 /* While running the command, bind-mount /dev, /proc, /sys
59 * into the chroot. However we must be careful to unmount them
60 * afterwards because otherwise they would interfere with
61 * future mount and unmount operations.
63 * We deliberately allow these commands to fail silently, BUT
64 * if a mount fails, don't unmount the corresponding mount.
66 sysroot_dev = sysroot_path ("/dev");
67 sysroot_dev_pts = sysroot_path ("/dev/pts");
68 sysroot_proc = sysroot_path ("/proc");
69 sysroot_selinux = sysroot_path ("/selinux");
70 sysroot_sys = sysroot_path ("/sys");
72 if (sysroot_dev == NULL || sysroot_dev_pts == NULL ||
73 sysroot_proc == NULL || sysroot_selinux == NULL ||
74 sysroot_sys == NULL) {
75 reply_with_perror ("malloc");
77 free (sysroot_dev_pts);
79 free (sysroot_selinux);
84 r = command (NULL, NULL, "mount", "--bind", "/dev", sysroot_dev, NULL);
86 r = command (NULL, NULL, "mount", "--bind", "/dev/pts", sysroot_dev_pts, NULL);
88 r = command (NULL, NULL, "mount", "--bind", "/proc", sysroot_proc, NULL);
90 r = command (NULL, NULL, "mount", "--bind", "/selinux", sysroot_selinux, NULL);
92 r = command (NULL, NULL, "mount", "--bind", "/sys", sysroot_sys, NULL);
96 r = commandv (&out, &err, (const char * const *) argv);
99 if (sys_ok) umount_ignore_fail (sysroot_sys);
100 if (selinux_ok) umount_ignore_fail (sysroot_selinux);
101 if (proc_ok) umount_ignore_fail (sysroot_proc);
102 if (dev_pts_ok) umount_ignore_fail (sysroot_dev_pts);
103 if (dev_ok) umount_ignore_fail (sysroot_dev);
106 free (sysroot_dev_pts);
108 free (sysroot_selinux);
112 reply_with_error ("%s", err);
120 return out; /* Caller frees. */
124 do_command_lines (char *const *argv)
129 out = do_command (argv);
133 lines = split_lines (out);
139 return lines; /* Caller frees. */
143 do_sh (const char *cmd)
145 const char *argv[] = { "/bin/sh", "-c", cmd, NULL };
147 return do_command ((char **) argv);
151 do_sh_lines (const char *cmd)
153 const char *argv[] = { "/bin/sh", "-c", cmd, NULL };
155 return do_command_lines ((char **) argv);