1 /* SPDX-License-Identifier: LGPL-2.1+ */
12 #include "exit-status.h"
14 #include "main-func.h"
15 #include "mount-setup.h"
16 #include "mount-util.h"
17 #include "path-util.h"
18 #include "process-util.h"
19 #include "signal-util.h"
23 /* Goes through /etc/fstab and remounts all API file systems, applying options that are in /etc/fstab that systemd
24 * might not have respected */
26 static int track_pid(Hashmap
**h
, const char *path
, pid_t pid
) {
27 _cleanup_free_
char *c
= NULL
;
32 assert(pid_is_valid(pid
));
34 r
= hashmap_ensure_allocated(h
, NULL
);
42 r
= hashmap_put(*h
, PID_TO_PTR(pid
), c
);
50 static int do_remount(const char *path
, bool force_rw
, Hashmap
**pids
) {
54 log_debug("Remounting %s...", path
);
56 r
= safe_fork(force_rw
? "(remount-rw)" : "(remount)",
57 FORK_RESET_SIGNALS
|FORK_DEATHSIG
|FORK_RLIMIT_NOFILE_SAFE
|FORK_LOG
, &pid
);
66 force_rw
? "remount,rw" : "remount"));
67 log_error_errno(errno
, "Failed to execute " MOUNT_PATH
": %m");
72 return track_pid(pids
, path
, pid
);
75 static int run(int argc
, char *argv
[]) {
76 _cleanup_hashmap_free_free_ Hashmap
*pids
= NULL
;
77 _cleanup_endmntent_
FILE *f
= NULL
;
78 bool has_root
= false;
85 return log_error_errno(SYNTHETIC_ERRNO(EINVAL
),
86 "This program takes no arguments.");
90 f
= setmntent("/etc/fstab", "re");
93 return log_error_errno(errno
, "Failed to open /etc/fstab: %m");
95 while ((me
= getmntent(f
))) {
96 /* Remount the root fs, /usr, and all API VFSs */
97 if (!mount_point_is_api(me
->mnt_dir
) &&
98 !PATH_IN_SET(me
->mnt_dir
, "/", "/usr"))
101 if (path_equal(me
->mnt_dir
, "/"))
104 r
= do_remount(me
->mnt_dir
, false, &pids
);
110 /* The $SYSTEMD_REMOUNT_ROOT_RW environment variable is set by systemd-gpt-auto-generator to tell us
111 * whether to remount things. We honour it only if there's no explicit line in /etc/fstab configured
112 * which takes precedence. */
114 r
= getenv_bool("SYSTEMD_REMOUNT_ROOT_RW");
115 if (r
< 0 && r
!= -ENXIO
)
116 log_warning_errno(r
, "Failed to parse $SYSTEMD_REMOUNT_ROOT_RW, ignoring: %m");
119 r
= do_remount("/", true, &pids
);
126 while (!hashmap_isempty(pids
)) {
127 _cleanup_free_
char *s
= NULL
;
130 if (waitid(P_ALL
, 0, &si
, WEXITED
) < 0) {
134 return log_error_errno(errno
, "waitid() failed: %m");
137 s
= hashmap_remove(pids
, PID_TO_PTR(si
.si_pid
));
139 !is_clean_exit(si
.si_code
, si
.si_status
, EXIT_CLEAN_COMMAND
, NULL
)) {
140 if (si
.si_code
== CLD_EXITED
)
141 log_error(MOUNT_PATH
" for %s exited with exit status %i.", s
, si
.si_status
);
143 log_error(MOUNT_PATH
" for %s terminated by signal %s.", s
, signal_to_string(si
.si_status
));
152 DEFINE_MAIN_FUNCTION(run
);