}
int main(int argc, char *argv[]) {
- bool need_umount, need_swapoff, need_loop_detach, need_dm_detach, need_md_detach, in_container, can_initrd;
_cleanup_free_ char *cgroup = NULL;
char *arguments[3];
- int cmd, r, umount_log_level = LOG_INFO;
+ int cmd, r;
static const char* const dirs[] = {SYSTEM_SHUTDOWN_PATH, NULL};
/* The log target defaults to console, but the original systemd process will pass its log target in through a
}
(void) cg_get_root_path(&cgroup);
- in_container = detect_container() > 0;
+ bool in_container = detect_container() > 0;
/* If the logging messages are going to KMSG, and if we are not running from a container, then try to
* update the sysctl kernel.printk current value in order to see "info" messages; This current log
log_info("Sending SIGKILL to remaining processes...");
broadcast_signal(SIGKILL, true, false, arg_timeout);
- need_umount = !in_container;
- need_swapoff = !in_container;
- need_loop_detach = !in_container;
- need_dm_detach = !in_container;
- need_md_detach = !in_container;
+ bool need_umount = !in_container, need_swapoff = !in_container, need_loop_detach = !in_container,
+ need_dm_detach = !in_container, need_md_detach = !in_container, can_initrd, last_try = false;
can_initrd = !in_container && !in_initrd() && access("/run/initramfs/shutdown", X_OK) == 0;
/* Unmount all mountpoints, swaps, and loopback devices */
if (need_umount) {
log_info("Unmounting file systems.");
- r = umount_all(&changed, umount_log_level);
+ r = umount_all(&changed, last_try);
if (r == 0) {
need_umount = false;
log_info("All filesystems unmounted.");
if (need_loop_detach) {
log_info("Detaching loop devices.");
- r = loopback_detach_all(&changed, umount_log_level);
+ r = loopback_detach_all(&changed, last_try);
if (r == 0) {
need_loop_detach = false;
log_info("All loop devices detached.");
if (need_md_detach) {
log_info("Stopping MD devices.");
- r = md_detach_all(&changed, umount_log_level);
+ r = md_detach_all(&changed, last_try);
if (r == 0) {
need_md_detach = false;
log_info("All MD devices stopped.");
if (need_dm_detach) {
log_info("Detaching DM devices.");
- r = dm_detach_all(&changed, umount_log_level);
+ r = dm_detach_all(&changed, last_try);
if (r == 0) {
need_dm_detach = false;
log_info("All DM devices detached.");
break;
}
- if (!changed && umount_log_level == LOG_INFO && !can_initrd) {
- /* There are things we cannot get rid of. Loop one more time
- * with LOG_ERR to inform the user. Note that we don't need
- * to do this if there is an initrd to switch to, because that
- * one is likely to get rid of the remaining mounts. If not,
- * it will log about them. */
- umount_log_level = LOG_ERR;
+ if (!changed && last_try && !can_initrd) {
+ /* There are things we cannot get rid of. Loop one more time in which we will log
+ * with higher priority to inform the user. Note that we don't need to do this if
+ * there is an initrd to switch to, because that one is likely to get rid of the
+ * remaining mounts. If not, it will log about them. */
+ last_try = true;
continue;
}
|| path_startswith(path, "/run/initramfs");
}
-static int remount_with_timeout(MountPoint *m, int umount_log_level) {
+static int remount_with_timeout(MountPoint *m, bool last_try) {
pid_t pid;
int r;
/* Start the mount operation here in the child */
r = mount(NULL, m->path, NULL, m->remount_flags, m->remount_options);
if (r < 0)
- log_full_errno(umount_log_level, errno, "Failed to remount '%s' read-only: %m", m->path);
+ log_full_errno(last_try ? LOG_ERR : LOG_INFO,
+ errno,
+ "Failed to remount '%s' read-only: %m",
+ m->path);
_exit(r < 0 ? EXIT_FAILURE : EXIT_SUCCESS);
}
return r;
}
-static int umount_with_timeout(MountPoint *m, int umount_log_level) {
+static int umount_with_timeout(MountPoint *m, bool last_try) {
pid_t pid;
int r;
* than return EBUSY). */
r = umount2(m->path, MNT_FORCE);
if (r < 0)
- log_full_errno(umount_log_level, errno, "Failed to unmount %s: %m", m->path);
+ log_full_errno(last_try ? LOG_ERR : LOG_INFO, errno, "Failed to unmount %s: %m", m->path);
_exit(r < 0 ? EXIT_FAILURE : EXIT_SUCCESS);
}
/* This includes remounting readonly, which changes the kernel mount options. Therefore the list passed to
* this function is invalidated, and should not be reused. */
-static int mount_points_list_umount(MountPoint **head, bool *changed, int umount_log_level) {
+static int mount_points_list_umount(MountPoint **head, bool *changed, bool last_try) {
int n_failed = 0;
assert(head);
*
* Since the remount can hang in the instance of remote filesystems, we remount
* asynchronously and skip the subsequent umount if it fails. */
- if (remount_with_timeout(m, umount_log_level) < 0) {
+ if (remount_with_timeout(m, last_try) < 0) {
/* Remount failed, but try unmounting anyway,
* unless this is a mount point we want to skip. */
if (nonunmountable_path(m->path)) {
continue;
/* Trying to umount */
- if (umount_with_timeout(m, umount_log_level) < 0)
+ if (umount_with_timeout(m, last_try) < 0)
n_failed++;
else
*changed = true;
return n_failed;
}
-static int loopback_points_list_detach(MountPoint **head, bool *changed, int umount_log_level) {
+static int loopback_points_list_detach(MountPoint **head, bool *changed, bool last_try) {
int n_failed = 0, r;
dev_t rootdev = 0;
log_info("Detaching loopback %s.", m->path);
r = delete_loopback(m->path);
if (r < 0) {
- log_full_errno(umount_log_level, r, "Could not detach loopback %s: %m", m->path);
+ log_full_errno(last_try ? LOG_ERR : LOG_INFO, r, "Could not detach loopback %s: %m", m->path);
n_failed++;
continue;
}
return n_failed;
}
-static int dm_points_list_detach(MountPoint **head, bool *changed, int umount_log_level) {
+static int dm_points_list_detach(MountPoint **head, bool *changed, bool last_try) {
int n_failed = 0, r;
dev_t rootdev = 0;
log_info("Detaching DM %s (%u:%u).", m->path, major(m->devnum), minor(m->devnum));
r = delete_dm(m);
if (r < 0) {
- log_full_errno(umount_log_level, r, "Could not detach DM %s: %m", m->path);
+ log_full_errno(last_try ? LOG_ERR : LOG_INFO, r, "Could not detach DM %s: %m", m->path);
n_failed++;
continue;
}
return n_failed;
}
-static int md_points_list_detach(MountPoint **head, bool *changed, int umount_log_level) {
+static int md_points_list_detach(MountPoint **head, bool *changed, bool last_try) {
int n_failed = 0, r;
dev_t rootdev = 0;
log_info("Stopping MD %s (%u:%u).", m->path, major(m->devnum), minor(m->devnum));
r = delete_md(m);
if (r < 0) {
- log_full_errno(umount_log_level, r, "Could not stop MD %s: %m", m->path);
+ log_full_errno(last_try ? LOG_ERR : LOG_INFO, r, "Could not stop MD %s: %m", m->path);
n_failed++;
continue;
}
return n_failed;
}
-static int umount_all_once(bool *changed, int umount_log_level) {
+static int umount_all_once(bool *changed, bool last_try) {
_cleanup_(mount_points_list_free) LIST_HEAD(MountPoint, mp_list_head);
int r;
if (r < 0)
return r;
- return mount_points_list_umount(&mp_list_head, changed, umount_log_level);
+ return mount_points_list_umount(&mp_list_head, changed, last_try);
}
-int umount_all(bool *changed, int umount_log_level) {
+int umount_all(bool *changed, bool last_try) {
bool umount_changed;
int r;
do {
umount_changed = false;
- r = umount_all_once(&umount_changed, umount_log_level);
+ r = umount_all_once(&umount_changed, last_try);
if (umount_changed)
*changed = true;
} while (umount_changed);
return swap_points_list_off(&swap_list_head, changed);
}
-int loopback_detach_all(bool *changed, int umount_log_level) {
+int loopback_detach_all(bool *changed, bool last_try) {
_cleanup_(mount_points_list_free) LIST_HEAD(MountPoint, loopback_list_head);
int r;
if (r < 0)
return r;
- return loopback_points_list_detach(&loopback_list_head, changed, umount_log_level);
+ return loopback_points_list_detach(&loopback_list_head, changed, last_try);
}
-int dm_detach_all(bool *changed, int umount_log_level) {
+int dm_detach_all(bool *changed, bool last_try) {
_cleanup_(mount_points_list_free) LIST_HEAD(MountPoint, dm_list_head);
int r;
if (r < 0)
return r;
- return dm_points_list_detach(&dm_list_head, changed, umount_log_level);
+ return dm_points_list_detach(&dm_list_head, changed, last_try);
}
-int md_detach_all(bool *changed, int umount_log_level) {
+int md_detach_all(bool *changed, bool last_try) {
_cleanup_(mount_points_list_free) LIST_HEAD(MountPoint, md_list_head);
int r;
if (r < 0)
return r;
- return md_points_list_detach(&md_list_head, changed, umount_log_level);
+ return md_points_list_detach(&md_list_head, changed, last_try);
}