]> git.ipfire.org Git - thirdparty/systemd.git/blob - src/core/killall.c
core: go on a killing spree when transitioning from initrd to main system
[thirdparty/systemd.git] / src / core / killall.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3 /***
4 This file is part of systemd.
5
6 Copyright 2010 ProFUSION embedded systems
7
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.
12
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.
17
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/>.
20 ***/
21
22 #include <sys/wait.h>
23 #include <signal.h>
24 #include <errno.h>
25
26 #include "util.h"
27 #include "def.h"
28 #include "killall.h"
29
30 #define TIMEOUT_USEC (5 * USEC_PER_SEC)
31
32 static bool ignore_proc(pid_t pid) {
33 char buf[PATH_MAX];
34 FILE *f;
35 char c;
36 size_t count;
37 uid_t uid;
38 int r;
39
40 /* We are PID 1, let's not commit suicide */
41 if (pid == 1)
42 return true;
43
44 r = get_process_uid(pid, &uid);
45 if (r < 0)
46 return true; /* not really, but better safe than sorry */
47
48 /* Non-root processes otherwise are always subject to be killed */
49 if (uid != 0)
50 return false;
51
52 snprintf(buf, sizeof(buf), "/proc/%lu/cmdline", (unsigned long) pid);
53 char_array_0(buf);
54
55 f = fopen(buf, "re");
56 if (!f)
57 return true; /* not really, but has the desired effect */
58
59 count = fread(&c, 1, 1, f);
60 fclose(f);
61
62 /* Kernel threads have an empty cmdline */
63 if (count <= 0)
64 return true;
65
66 /* Processes with argv[0][0] = '@' we ignore from the killing
67 * spree.
68 *
69 * http://www.freedesktop.org/wiki/Software/systemd/RootStorageDaemons */
70 if (count == 1 && c == '@')
71 return true;
72
73 return false;
74 }
75
76 static void wait_for_children(int n_processes, sigset_t *mask) {
77 usec_t until;
78
79 assert(mask);
80
81 until = now(CLOCK_MONOTONIC) + TIMEOUT_USEC;
82 for (;;) {
83 struct timespec ts;
84 int k;
85 usec_t n;
86
87 for (;;) {
88 pid_t pid = waitpid(-1, NULL, WNOHANG);
89
90 if (pid == 0)
91 break;
92
93 if (pid < 0 && errno == ECHILD)
94 return;
95
96 if (n_processes > 0)
97 if (--n_processes == 0)
98 return;
99 }
100
101 n = now(CLOCK_MONOTONIC);
102 if (n >= until)
103 return;
104
105 timespec_store(&ts, until - n);
106
107 if ((k = sigtimedwait(mask, NULL, &ts)) != SIGCHLD) {
108
109 if (k < 0 && errno != EAGAIN) {
110 log_error("sigtimedwait() failed: %m");
111 return;
112 }
113
114 if (k >= 0)
115 log_warning("sigtimedwait() returned unexpected signal.");
116 }
117 }
118 }
119
120 static int killall(int sig) {
121 DIR *dir;
122 struct dirent *d;
123 unsigned int n_processes = 0;
124
125 dir = opendir("/proc");
126 if (!dir)
127 return -errno;
128
129 while ((d = readdir(dir))) {
130 pid_t pid;
131
132 if (d->d_type != DT_DIR &&
133 d->d_type != DT_UNKNOWN)
134 continue;
135
136 if (parse_pid(d->d_name, &pid) < 0)
137 continue;
138
139 if (ignore_proc(pid))
140 continue;
141
142 if (kill(pid, sig) >= 0)
143 n_processes++;
144 else if (errno != ENOENT)
145 log_warning("Could not kill %d: %m", pid);
146 }
147
148 closedir(dir);
149
150 return n_processes;
151 }
152
153 void broadcast_signal(int sig) {
154 sigset_t mask, oldmask;
155 int n_processes;
156
157 assert_se(sigemptyset(&mask) == 0);
158 assert_se(sigaddset(&mask, SIGCHLD) == 0);
159 assert_se(sigprocmask(SIG_BLOCK, &mask, &oldmask) == 0);
160
161 if (kill(-1, SIGSTOP) < 0 && errno != ESRCH)
162 log_warning("kill(-1, SIGSTOP) failed: %m");
163
164 n_processes = killall(sig);
165
166 if (kill(-1, SIGCONT) < 0 && errno != ESRCH)
167 log_warning("kill(-1, SIGCONT) failed: %m");
168
169 if (n_processes <= 0)
170 goto finish;
171
172 wait_for_children(n_processes, &mask);
173
174 finish:
175 sigprocmask(SIG_SETMASK, &oldmask, NULL);
176 }