23
23
#include <inttypes.h>
24
24
#include <string.h>
25
25
#include <unistd.h>
26
+ #include <sys/stat.h>
26
27
27
- #include "daemon.h"
28
28
#include "c-ctype.h"
29
+ #include "ignore-value.h"
30
+
31
+ #include "daemon.h"
29
32
#include "actions.h"
30
33
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
+
31
99
/* Does the current line match the regexp /^\s*filter\s*=/ */
32
100
static int
33
101
is_filter_line (const char * line )
@@ -52,18 +120,24 @@ is_filter_line (const char *line)
52
120
return 1 ;
53
121
}
54
122
55
- /* Rewrite the 'filter = [ ... ]' line in /etc/lvm/ lvm.conf. */
123
+ /* Rewrite the 'filter = [ ... ]' line in lvm.conf. */
56
124
static int
57
125
set_filter (const char * filter )
58
126
{
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" );
60
134
if (ifp == NULL ) {
61
- reply_with_perror ("open: /etc/lvm/lvm.conf" );
135
+ reply_with_perror ("open: %s" , lvm_conf );
62
136
return -1 ;
63
137
}
64
- FILE * ofp = fopen ("/etc/lvm/lvm.conf.new" , "w" );
138
+ FILE * ofp = fopen (lvm_conf_new , "w" );
65
139
if (ofp == NULL ) {
66
- reply_with_perror ("open: /etc/lvm/lvm.conf.new" );
140
+ reply_with_perror ("open: %s" , lvm_conf_new );
67
141
fclose (ifp );
68
142
return -1 ;
69
143
}
@@ -79,32 +153,32 @@ set_filter (const char *filter)
79
153
}
80
154
if (r < 0 ) {
81
155
/* 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 );
83
157
fclose (ifp );
84
158
fclose (ofp );
85
159
free (line );
86
- unlink ("/etc/lvm/lvm.conf.new" );
160
+ unlink (lvm_conf_new );
87
161
return -1 ;
88
162
}
89
163
}
90
164
91
165
free (line );
92
166
93
167
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 );
96
170
fclose (ofp );
97
171
return -1 ;
98
172
}
99
173
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 );
102
176
return -1 ;
103
177
}
104
178
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 );
108
182
return -1 ;
109
183
}
110
184
@@ -144,7 +218,10 @@ reactivate (void)
144
218
static int
145
219
rescan (void )
146
220
{
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 );
148
225
149
226
char * err ;
150
227
int r = command (NULL , & err , "lvm" , "vgscan" , NULL );
0 commit comments