Skip to content

Commit 9c299b6

Browse files
committed
daemon: Use a private copy of /etc/lvm so we don't modify the real config.
In the libguestfs live case we need to be careful not to modify the real /etc/lvm/lvm.conf file (when setting the filter rule). When the daemon starts, make a complete copy of /etc/lvm in a temporary directory, and adjust LVM_SYSTEM_DIR to point to the copy. All changes are made in the temporary copy.
1 parent 6011b1f commit 9c299b6

File tree

3 files changed

+101
-16
lines changed

3 files changed

+101
-16
lines changed

daemon/daemon.h

+3
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,9 @@ extern int e2prog (char *name); /* Massive hack for RHEL 5. */
134134
/*-- in lvm.c --*/
135135
extern int lv_canonical (const char *device, char **ret);
136136

137+
/*-- in lvm-filter.c --*/
138+
extern void copy_lvm (void);
139+
137140
/*-- in proto.c --*/
138141
extern void main_loop (int sock) __attribute__((noreturn));
139142

daemon/guestfsd.c

+5
Original file line numberDiff line numberDiff line change
@@ -239,6 +239,11 @@ main (int argc, char *argv[])
239239
_umask (0);
240240
#endif
241241

242+
/* Make a private copy of /etc/lvm so we can change the config (see
243+
* daemon/lvm-filter.c).
244+
*/
245+
copy_lvm ();
246+
242247
/* Connect to virtio-serial channel. */
243248
int sock = open (VIRTIO_SERIAL_CHANNEL, O_RDWR | O_CLOEXEC);
244249
if (sock == -1) {

daemon/lvm-filter.c

+93-16
Original file line numberDiff line numberDiff line change
@@ -23,11 +23,79 @@
2323
#include <inttypes.h>
2424
#include <string.h>
2525
#include <unistd.h>
26+
#include <sys/stat.h>
2627

27-
#include "daemon.h"
2828
#include "c-ctype.h"
29+
#include "ignore-value.h"
30+
31+
#include "daemon.h"
2932
#include "actions.h"
3033

34+
/* This runs during daemon start up and creates a complete copy of
35+
* /etc/lvm so that we can modify it as we desire. We set
36+
* LVM_SYSTEM_DIR to point to the copy.
37+
*/
38+
static char lvm_system_dir[] = "/tmp/lvmXXXXXX";
39+
40+
static void rm_lvm_system_dir (void);
41+
42+
void
43+
copy_lvm (void)
44+
{
45+
struct stat statbuf;
46+
char cmd[64];
47+
int r;
48+
49+
/* If /etc/lvm directory doesn't exist (or isn't a directory) assume
50+
* that this system doesn't support LVM and do nothing.
51+
*/
52+
r = stat ("/etc/lvm", &statbuf);
53+
if (r == -1) {
54+
perror ("copy_lvm: stat: /etc/lvm");
55+
return;
56+
}
57+
if (! S_ISDIR (statbuf.st_mode)) {
58+
fprintf (stderr, "copy_lvm: warning: /etc/lvm is not a directory\n");
59+
return;
60+
}
61+
62+
if (mkdtemp (lvm_system_dir) == NULL) {
63+
perror (lvm_system_dir);
64+
exit (EXIT_FAILURE);
65+
}
66+
67+
/* Hopefully no dotfiles in there ... XXX */
68+
snprintf (cmd, sizeof cmd, "cp -a /etc/lvm/* %s", lvm_system_dir);
69+
r = system (cmd);
70+
if (r == -1) {
71+
perror (cmd);
72+
rmdir (lvm_system_dir);
73+
exit (EXIT_FAILURE);
74+
}
75+
76+
if (WEXITSTATUS (r) != 0) {
77+
fprintf (stderr, "cp command failed with return code %d\n",
78+
WEXITSTATUS (r));
79+
rmdir (lvm_system_dir);
80+
exit (EXIT_FAILURE);
81+
}
82+
83+
/* Set environment variable so we use the copy. */
84+
setenv ("LVM_SYSTEM_DIR", lvm_system_dir, 1);
85+
86+
/* Set a handler to remove the temporary directory at exit. */
87+
atexit (rm_lvm_system_dir);
88+
}
89+
90+
static void
91+
rm_lvm_system_dir (void)
92+
{
93+
char cmd[64];
94+
95+
snprintf (cmd, sizeof cmd, "rm -rf %s", lvm_system_dir);
96+
ignore_value (system (cmd));
97+
}
98+
3199
/* Does the current line match the regexp /^\s*filter\s*=/ */
32100
static int
33101
is_filter_line (const char *line)
@@ -52,18 +120,24 @@ is_filter_line (const char *line)
52120
return 1;
53121
}
54122

55-
/* Rewrite the 'filter = [ ... ]' line in /etc/lvm/lvm.conf. */
123+
/* Rewrite the 'filter = [ ... ]' line in lvm.conf. */
56124
static int
57125
set_filter (const char *filter)
58126
{
59-
FILE *ifp = fopen ("/etc/lvm/lvm.conf", "r");
127+
char lvm_conf[64];
128+
snprintf (lvm_conf, sizeof lvm_conf, "%s/lvm.conf", lvm_system_dir);
129+
130+
char lvm_conf_new[64];
131+
snprintf (lvm_conf_new, sizeof lvm_conf, "%s/lvm.conf.new", lvm_system_dir);
132+
133+
FILE *ifp = fopen (lvm_conf, "r");
60134
if (ifp == NULL) {
61-
reply_with_perror ("open: /etc/lvm/lvm.conf");
135+
reply_with_perror ("open: %s", lvm_conf);
62136
return -1;
63137
}
64-
FILE *ofp = fopen ("/etc/lvm/lvm.conf.new", "w");
138+
FILE *ofp = fopen (lvm_conf_new, "w");
65139
if (ofp == NULL) {
66-
reply_with_perror ("open: /etc/lvm/lvm.conf.new");
140+
reply_with_perror ("open: %s", lvm_conf_new);
67141
fclose (ifp);
68142
return -1;
69143
}
@@ -79,32 +153,32 @@ set_filter (const char *filter)
79153
}
80154
if (r < 0) {
81155
/* NB. fprintf doesn't set errno on error. */
82-
reply_with_error ("/etc/lvm/lvm.conf.new: write failed");
156+
reply_with_error ("%s: write failed", lvm_conf_new);
83157
fclose (ifp);
84158
fclose (ofp);
85159
free (line);
86-
unlink ("/etc/lvm/lvm.conf.new");
160+
unlink (lvm_conf_new);
87161
return -1;
88162
}
89163
}
90164

91165
free (line);
92166

93167
if (fclose (ifp) == EOF) {
94-
reply_with_perror ("/etc/lvm/lvm.conf.new");
95-
unlink ("/etc/lvm/lvm.conf.new");
168+
reply_with_perror ("close: %s", lvm_conf);
169+
unlink (lvm_conf_new);
96170
fclose (ofp);
97171
return -1;
98172
}
99173
if (fclose (ofp) == EOF) {
100-
reply_with_perror ("/etc/lvm/lvm.conf.new");
101-
unlink ("/etc/lvm/lvm.conf.new");
174+
reply_with_perror ("close: %s", lvm_conf_new);
175+
unlink (lvm_conf_new);
102176
return -1;
103177
}
104178

105-
if (rename ("/etc/lvm/lvm.conf.new", "/etc/lvm/lvm.conf") == -1) {
106-
reply_with_perror ("rename: /etc/lvm/lvm.conf");
107-
unlink ("/etc/lvm/lvm.conf.new");
179+
if (rename (lvm_conf_new, lvm_conf) == -1) {
180+
reply_with_perror ("rename: %s", lvm_conf);
181+
unlink (lvm_conf_new);
108182
return -1;
109183
}
110184

@@ -144,7 +218,10 @@ reactivate (void)
144218
static int
145219
rescan (void)
146220
{
147-
unlink ("/etc/lvm/cache/.cache");
221+
char lvm_cache[64];
222+
snprintf (lvm_cache, sizeof lvm_cache, "%s/cache/.cache", lvm_system_dir);
223+
224+
unlink (lvm_cache);
148225

149226
char *err;
150227
int r = command (NULL, &err, "lvm", "vgscan", NULL);

0 commit comments

Comments
 (0)