2 * Copyright (c) 2009 The DragonFly Project. All rights reserved.
4 * This code is derived from software contributed to The DragonFly Project
5 * by Matthew Dillon <dillon@backplane.com>
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in
15 * the documentation and/or other materials provided with the
17 * 3. Neither the name of The DragonFly Project nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific, prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25 * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26 * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
29 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
30 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
31 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 static void config_get(const char *dirpath
, struct hammer_ioc_config
*config
);
38 static void config_set(const char *dirpath
, struct hammer_ioc_config
*config
);
39 static void config_remove_path(void);
41 static char *ConfigPath
;
44 * hammer config [<fs> [configfile]]
46 * Prints out the hammer cleanup configuration for the specified HAMMER
47 * filesystem(s) or the current filesystem.
50 hammer_cmd_config(char **av
, int ac
)
52 struct hammer_ioc_config config
;
57 bzero(&config
, sizeof(config
));
59 config_get(".", &config
);
60 if (config
.head
.error
== 0) {
61 printf("%s", config
.config
.text
);
63 errx(2, "hammer config: no configuration found");
70 config_get(dirpath
, &config
);
71 if (config
.head
.error
== 0) {
72 printf("%s", config
.config
.text
);
74 errx(2, "hammer config: no configuration found");
79 config_get(dirpath
, &config
); /* ignore errors */
80 config
.head
.error
= 0;
82 fd
= open(av
[1], O_RDONLY
);
84 err(2, "hammer config: %s", av
[1]);
87 n
= read(fd
, config
.config
.text
, sizeof(config
.config
.text
) - 1);
88 if (n
== sizeof(config
.config
.text
) - 1) {
89 err(2, "hammer config: config file too big, limit %zu bytes",
90 sizeof(config
.config
.text
) - 1);
93 bzero(config
.config
.text
+ n
, sizeof(config
.config
.text
) - n
);
94 config_set(dirpath
, &config
);
99 * hammer viconfig [<fs>]
102 hammer_cmd_viconfig(char **av
, int ac
)
104 struct hammer_ioc_config config
;
105 struct timeval times
[2];
108 char *runcmd
, *editor
, *tmp
;
114 errx(1, "hammer viconfig: 0 or 1 argument (<fs>) only");
121 config_get(dirpath
, &config
);
122 if (config
.head
.error
== ENOENT
) {
123 snprintf(config
.config
.text
, sizeof(config
.config
.text
),
125 "# No configuration present, here are some defaults\n"
126 "# you can uncomment. Also remove these instructions\n"
128 "#snapshots 1d 60d\n"
133 "#recopy 30d 10m\n");
134 config
.head
.error
= 0;
136 if (config
.head
.error
) {
137 errx(2, "hammer viconfig: read config failed error: %s",
138 strerror(config
.head
.error
));
143 * Edit a temporary file and write back if it was modified.
144 * Adjust the mtime back one second so a quick edit is not
145 * improperly detected as not having been modified.
147 snprintf(path
, sizeof(path
), "/tmp/configXXXXXXXXXX");
150 atexit(config_remove_path
);
152 fd
= open(path
, O_RDWR
|O_CREAT
|O_TRUNC
, 0600);
154 err(2, "hammer viconfig: creating temporary file %s", path
);
155 write(fd
, config
.config
.text
, strlen(config
.config
.text
));
156 if (fstat(fd
, &st
) < 0)
157 err(2, "hammer viconfig");
158 times
[0].tv_sec
= st
.st_mtime
- 1;
159 times
[0].tv_usec
= 0;
164 if ((tmp
= getenv("EDITOR")) != NULL
||
165 (tmp
= getenv("VISUAL")) != NULL
)
166 editor
= strdup(tmp
);
168 editor
= strdup("vi");
170 asprintf(&runcmd
, "%s %s", editor
, path
);
173 if (stat(path
, &st
) < 0)
174 err(2, "hammer viconfig: unable to stat file after vi");
175 if (times
[0].tv_sec
== st
.st_mtime
) {
176 printf("hammer viconfig: no changes were made\n");
180 fd
= open(path
, O_RDONLY
);
182 err(2, "hammer viconfig: unable to read %s", path
);
184 n
= read(fd
, config
.config
.text
, sizeof(config
.config
.text
) - 1);
186 err(2, "hammer viconfig: unable to read %s", path
);
187 if (n
== sizeof(config
.config
.text
) - 1) {
188 err(2, "hammer config: config file too big, limit %zu bytes",
189 sizeof(config
.config
.text
) - 1);
192 bzero(config
.config
.text
+ n
, sizeof(config
.config
.text
) - n
);
193 config_set(dirpath
, &config
);
199 config_get(const char *dirpath
, struct hammer_ioc_config
*config
)
201 struct hammer_ioc_version version
;
204 bzero(&version
, sizeof(version
));
205 if ((fd
= open(dirpath
, O_RDONLY
)) < 0)
206 err(2, "hammer config: unable to open directory %s", dirpath
);
207 if (ioctl(fd
, HAMMERIOC_GET_VERSION
, &version
) < 0)
208 errx(2, "hammer config: not a HAMMER filesystem!");
209 if (ioctl(fd
, HAMMERIOC_GET_CONFIG
, config
) < 0)
210 errx(2, "hammer config: config_get");
215 config_set(const char *dirpath
, struct hammer_ioc_config
*config
)
217 struct hammer_ioc_version version
;
220 bzero(&version
, sizeof(version
));
221 if ((fd
= open(dirpath
, O_RDONLY
)) < 0)
222 errx(2, "hammer config: unable to open directory %s", dirpath
);
223 if (ioctl(fd
, HAMMERIOC_GET_VERSION
, &version
) < 0)
224 errx(2, "hammer config: not a HAMMER filesystem!");
225 if (ioctl(fd
, HAMMERIOC_SET_CONFIG
, config
) < 0)
226 err(2, "hammer config");
231 config_remove_path(void)