1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2012 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
30 #include "unit-name.h"
31 #include "path-util.h"
32 #include "mount-setup.h"
37 static const char *arg_dest = "/tmp";
38 static bool arg_enabled = true;
40 static int device_name(const char *path, char **unit) {
45 if (!is_device_path(path))
48 p = unit_name_from_path(path, ".device");
56 static int mount_find_pri(struct mntent *me, int *ret) {
63 pri = hasmntopt(me, "pri");
70 r = strtoul(pri, &end, 10);
74 if (end == pri || (*end != ',' && *end != 0))
81 static int add_swap(const char *what, struct mntent *me) {
82 char *name = NULL, *unit = NULL, *lnk = NULL, *device = NULL;
90 r = mount_find_pri(me, &pri);
92 log_error("Failed to parse priority");
96 noauto = !!hasmntopt(me, "noauto");
97 nofail = !!hasmntopt(me, "nofail");
99 name = unit_name_from_path(what, ".swap");
105 unit = strjoin(arg_dest, "/", name, NULL);
111 f = fopen(unit, "wxe");
115 log_error("Failed to create swap unit file %s, as it already exists. Duplicate entry in /etc/fstab?", unit);
117 log_error("Failed to create unit file %s: %m", unit);
121 fputs("# Automatically generated by systemd-fstab-generator\n\n"
123 "SourcePath=/etc/fstab\n"
124 "DefaultDependencies=no\n"
125 "Conflicts=" SPECIAL_UMOUNT_TARGET "\n"
126 "Before=" SPECIAL_UMOUNT_TARGET "\n", f);
128 if (!noauto && !nofail)
129 fputs("Before=" SPECIAL_SWAP_TARGET "\n", f);
144 log_error("Failed to write unit file %s: %m", unit);
150 lnk = strjoin(arg_dest, "/" SPECIAL_SWAP_TARGET ".wants/", name, NULL);
156 mkdir_parents_label(lnk, 0755);
157 if (symlink(unit, lnk) < 0) {
158 log_error("Failed to create symlink %s: %m", lnk);
163 r = device_name(what, &device);
169 lnk = strjoin(arg_dest, "/", device, ".wants/", name, NULL);
175 mkdir_parents_label(lnk, 0755);
176 if (symlink(unit, lnk) < 0) {
177 log_error("Failed to create symlink %s: %m", lnk);
197 static bool mount_is_bind(struct mntent *me) {
201 hasmntopt(me, "bind") ||
202 streq(me->mnt_type, "bind");
205 static bool mount_is_network(struct mntent *me) {
209 hasmntopt(me, "_netdev") ||
210 fstype_is_network(me->mnt_type);
213 static int add_mount(const char *what, const char *where, struct mntent *me) {
214 char *name = NULL, *unit = NULL, *lnk = NULL, *device = NULL, *automount_name = NULL, *automount_unit = NULL;
216 bool noauto, nofail, automount, isbind, isnetwork;
218 const char *post, *pre;
224 if (streq(me->mnt_type, "autofs"))
227 if (!is_path(where)) {
228 log_warning("Mount point %s is not a valid path, ignoring.", where);
232 if (mount_point_is_api(where) ||
233 mount_point_ignore(where))
236 isnetwork = mount_is_network(me);
237 isbind = mount_is_bind(me);
239 noauto = !!hasmntopt(me, "noauto");
240 nofail = !!hasmntopt(me, "nofail");
242 hasmntopt(me, "comment=systemd.automount") ||
243 hasmntopt(me, "x-systemd.automount");
246 post = SPECIAL_REMOTE_FS_TARGET;
247 pre = SPECIAL_REMOTE_FS_PRE_TARGET;
249 post = SPECIAL_LOCAL_FS_TARGET;
250 pre = SPECIAL_LOCAL_FS_PRE_TARGET;
253 name = unit_name_from_path(where, ".mount");
259 unit = strjoin(arg_dest, "/", name, NULL);
265 f = fopen(unit, "wxe");
269 log_error("Failed to create mount unit file %s, as it already exists. Duplicate entry in /etc/fstab?", unit);
271 log_error("Failed to create unit file %s: %m", unit);
275 fputs("# Automatically generated by systemd-fstab-generator\n\n"
277 "SourcePath=/etc/fstab\n"
278 "DefaultDependencies=no\n", f);
280 if (!path_equal(where, "/"))
284 "Conflicts=" SPECIAL_UMOUNT_TARGET "\n"
285 "Before=" SPECIAL_UMOUNT_TARGET "\n",
290 if (!noauto && !nofail && !automount)
307 if (!isempty(me->mnt_opts) &&
308 !streq(me->mnt_opts, "defaults"))
315 log_error("Failed to write unit file %s: %m", unit);
321 lnk = strjoin(arg_dest, "/", post, nofail || automount ? ".wants/" : ".requires/", name, NULL);
327 mkdir_parents_label(lnk, 0755);
328 if (symlink(unit, lnk) < 0) {
329 log_error("Failed to create symlink %s: %m", lnk);
335 !path_equal(where, "/")) {
337 r = device_name(what, &device);
343 lnk = strjoin(arg_dest, "/", device, ".wants/", name, NULL);
349 mkdir_parents_label(lnk, 0755);
350 if (symlink(unit, lnk) < 0) {
351 log_error("Failed to create symlink %s: %m", lnk);
359 if (automount && !path_equal(where, "/")) {
360 automount_name = unit_name_from_path(where, ".automount");
366 automount_unit = strjoin(arg_dest, "/", automount_name, NULL);
367 if (!automount_unit) {
373 f = fopen(automount_unit, "wxe");
376 log_error("Failed to create unit file %s: %m", automount_unit);
381 "# Automatically generated by systemd-fstab-generator\n\n"
383 "SourcePath=/etc/fstab\n"
384 "DefaultDependencies=no\n"
385 "Conflicts=" SPECIAL_UMOUNT_TARGET "\n"
386 "Before=" SPECIAL_UMOUNT_TARGET " %s\n"
395 log_error("Failed to write unit file %s: %m", automount_unit);
401 lnk = strjoin(arg_dest, "/", post, nofail ? ".wants/" : ".requires/", automount_name, NULL);
407 mkdir_parents_label(lnk, 0755);
408 if (symlink(automount_unit, lnk) < 0) {
409 log_error("Failed to create symlink %s: %m", lnk);
424 free(automount_name);
425 free(automount_unit);
430 static int parse_fstab(void) {
436 f = setmntent("/etc/fstab", "r");
441 log_error("Failed to open /etc/fstab: %m");
445 while ((me = getmntent(f))) {
449 what = fstab_node_to_udev_node(me->mnt_fsname);
455 where = strdup(me->mnt_dir);
463 path_kill_slashes(where);
465 log_debug("Found entry what=%s where=%s type=%s", what, where, me->mnt_type);
467 if (streq(me->mnt_type, "swap"))
468 k = add_swap(what, me);
470 k = add_mount(what, where, me);
484 static int parse_proc_cmdline(void) {
485 char *line, *w, *state;
489 if (detect_container(NULL) > 0)
492 r = read_one_line_file("/proc/cmdline", &line);
494 log_warning("Failed to read /proc/cmdline, ignoring: %s", strerror(-r));
498 FOREACH_WORD_QUOTED(w, l, line, state) {
501 word = strndup(w, l);
507 if (startswith(word, "fstab=")) {
508 r = parse_boolean(word + 6);
510 log_warning("Failed to parse fstab switch %s. Ignoring.", word + 6);
514 } else if (startswith(word, "rd.fstab=")) {
517 r = parse_boolean(word + 6);
519 log_warning("Failed to parse fstab switch %s. Ignoring.", word + 6);
524 } else if (startswith(word, "fstab.") ||
525 (in_initrd() && startswith(word, "rd.fstab."))) {
527 log_warning("Unknown kernel switch %s. Ignoring.", word);
540 int main(int argc, char *argv[]) {
543 if (argc > 1 && argc != 4) {
544 log_error("This program takes three or no arguments.");
551 log_set_target(LOG_TARGET_SAFE);
552 log_parse_environment();
557 if (parse_proc_cmdline() < 0)
565 return r < 0 ? EXIT_FAILURE : EXIT_SUCCESS;