summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/systemctl.c32
1 files changed, 26 insertions, 6 deletions
diff --git a/src/systemctl.c b/src/systemctl.c
index 8a6277e75c..59ea7490e7 100644
--- a/src/systemctl.c
+++ b/src/systemctl.c
@@ -110,6 +110,8 @@ static enum dot {
static bool private_bus = false;
+static pid_t pager_pid = 0;
+
static int daemon_reload(DBusConnection *bus, char **args, unsigned n);
static bool on_tty(void) {
@@ -5283,10 +5285,12 @@ static int runlevel_main(void) {
}
static void pager_open(void) {
- pid_t pid;
int fd[2];
const char *pager;
+ if (pager_pid > 0)
+ return;
+
if (!on_tty() || arg_no_pager)
return;
@@ -5299,15 +5303,15 @@ static void pager_open(void) {
return;
}
- pid = fork();
- if (pid < 0) {
+ pager_pid = fork();
+ if (pager_pid < 0) {
log_error("Failed to fork pager: %m");
close_pipe(fd);
return;
}
- /* The original process turns into the PAGER */
- if (pid != 0) {
+ /* In the child start the pager */
+ if (pager_pid == 0) {
dup2(fd[0], STDIN_FILENO);
close_pipe(fd);
@@ -5315,6 +5319,8 @@ static void pager_open(void) {
if (!getenv("LESS"))
setenv("LESS", "FRSX", 0);
+ prctl(PR_SET_PDEATHSIG, SIGTERM);
+
if (pager) {
execlp(pager, pager, NULL);
execl("/bin/sh", "sh", "-c", pager, NULL);
@@ -5328,13 +5334,25 @@ static void pager_open(void) {
_exit(EXIT_FAILURE);
}
- /* Return in the child */
+ /* Return in the parent */
if (dup2(fd[1], STDOUT_FILENO) < 0)
log_error("Failed to duplicate pager pipe: %m");
close_pipe(fd);
}
+static void pager_close(void) {
+ siginfo_t dummy;
+
+ if (pager_pid <= 0)
+ return;
+
+ /* Inform pager that we are done */
+ fclose(stdout);
+ wait_for_terminate(pager_pid, &dummy);
+ pager_pid = 0;
+}
+
int main(int argc, char*argv[]) {
int r, retval = EXIT_FAILURE;
DBusConnection *bus = NULL;
@@ -5417,5 +5435,7 @@ finish:
strv_free(arg_property);
+ pager_close();
+
return retval;
}