]>
git.ipfire.org Git - thirdparty/systemd.git/blob - src/test/test-process-util.c
2 This file is part of systemd.
4 Copyright 2010 Lennart Poettering
5 Copyright 2013 Thomas H.P. Andersen
7 systemd is free software; you can redistribute it and/or modify it
8 under the terms of the GNU Lesser General Public License as published by
9 the Free Software Foundation; either version 2.1 of the License, or
10 (at your option) any later version.
12 systemd is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 Lesser General Public License for more details.
17 You should have received a copy of the GNU Lesser General Public License
18 along with systemd; If not, see <http://www.gnu.org/licenses/>.
22 #include <sys/mount.h>
23 #include <sys/personality.h>
24 #include <sys/prctl.h>
26 #include <sys/types.h>
29 #ifdef HAVE_VALGRIND_VALGRIND_H
30 #include <valgrind/valgrind.h>
33 #include "alloc-util.h"
34 #include "architecture.h"
38 #include "parse-util.h"
39 #include "process-util.h"
40 #include "stdio-util.h"
41 #include "string-util.h"
42 #include "terminal-util.h"
43 #include "test-helper.h"
47 static void test_get_process_comm(pid_t pid
) {
49 _cleanup_free_
char *a
= NULL
, *c
= NULL
, *d
= NULL
, *f
= NULL
, *i
= NULL
;
50 _cleanup_free_
char *env
= NULL
;
51 char path
[strlen("/proc//comm") + DECIMAL_STR_MAX(pid_t
)];
58 xsprintf(path
, "/proc/"PID_FMT
"/comm", pid
);
60 if (stat(path
, &st
) == 0) {
61 assert_se(get_process_comm(pid
, &a
) >= 0);
62 log_info("PID"PID_FMT
" comm: '%s'", pid
, a
);
64 log_warning("%s not exist.", path
);
66 assert_se(get_process_cmdline(pid
, 0, true, &c
) >= 0);
67 log_info("PID"PID_FMT
" cmdline: '%s'", pid
, c
);
69 assert_se(get_process_cmdline(pid
, 8, false, &d
) >= 0);
70 log_info("PID"PID_FMT
" cmdline truncated to 8: '%s'", pid
, d
);
73 assert_se(get_process_cmdline(pid
, 1, false, &d
) >= 0);
74 log_info("PID"PID_FMT
" cmdline truncated to 1: '%s'", pid
, d
);
76 assert_se(get_process_ppid(pid
, &e
) >= 0);
77 log_info("PID"PID_FMT
" PPID: "PID_FMT
, pid
, e
);
78 assert_se(pid
== 1 ? e
== 0 : e
> 0);
80 assert_se(is_kernel_thread(pid
) == 0 || pid
!= 1);
82 r
= get_process_exe(pid
, &f
);
83 assert_se(r
>= 0 || r
== -EACCES
);
84 log_info("PID"PID_FMT
" exe: '%s'", pid
, strna(f
));
86 assert_se(get_process_uid(pid
, &u
) == 0);
87 log_info("PID"PID_FMT
" UID: "UID_FMT
, pid
, u
);
88 assert_se(u
== 0 || pid
!= 1);
90 assert_se(get_process_gid(pid
, &g
) == 0);
91 log_info("PID"PID_FMT
" GID: "GID_FMT
, pid
, g
);
92 assert_se(g
== 0 || pid
!= 1);
94 r
= get_process_environ(pid
, &env
);
95 assert_se(r
>= 0 || r
== -EACCES
);
96 log_info("PID"PID_FMT
" strlen(environ): %zi", pid
, env
? (ssize_t
)strlen(env
) : (ssize_t
)-errno
);
98 if (!detect_container())
99 assert_se(get_ctty_devnr(pid
, &h
) == -ENXIO
|| pid
!= 1);
101 getenv_for_pid(pid
, "PATH", &i
);
102 log_info("PID"PID_FMT
" $PATH: '%s'", pid
, strna(i
));
105 static void test_pid_is_unwaited(void) {
115 waitpid(pid
, &status
, 0);
116 assert_se(!pid_is_unwaited(pid
));
118 assert_se(pid_is_unwaited(getpid()));
119 assert_se(!pid_is_unwaited(-1));
122 static void test_pid_is_alive(void) {
132 waitpid(pid
, &status
, 0);
133 assert_se(!pid_is_alive(pid
));
135 assert_se(pid_is_alive(getpid()));
136 assert_se(!pid_is_alive(-1));
139 static void test_personality(void) {
141 assert_se(personality_to_string(PER_LINUX
));
142 assert_se(!personality_to_string(PERSONALITY_INVALID
));
144 assert_se(streq(personality_to_string(PER_LINUX
), architecture_to_string(native_architecture())));
146 assert_se(personality_from_string(personality_to_string(PER_LINUX
)) == PER_LINUX
);
147 assert_se(personality_from_string(architecture_to_string(native_architecture())) == PER_LINUX
);
150 assert_se(streq_ptr(personality_to_string(PER_LINUX
), "x86-64"));
151 assert_se(streq_ptr(personality_to_string(PER_LINUX32
), "x86"));
153 assert_se(personality_from_string("x86-64") == PER_LINUX
);
154 assert_se(personality_from_string("x86") == PER_LINUX32
);
155 assert_se(personality_from_string("ia64") == PERSONALITY_INVALID
);
156 assert_se(personality_from_string(NULL
) == PERSONALITY_INVALID
);
158 assert_se(personality_from_string(personality_to_string(PER_LINUX32
)) == PER_LINUX32
);
162 static void test_get_process_cmdline_harder(void) {
163 char path
[] = "/tmp/test-cmdlineXXXXXX";
164 _cleanup_close_
int fd
= -1;
165 _cleanup_free_
char *line
= NULL
;
171 #ifdef HAVE_VALGRIND_VALGRIND_H
172 /* valgrind patches open(/proc//cmdline)
173 * so, test_get_process_cmdline_harder fails always
174 * See https://github.com/systemd/systemd/pull/3555#issuecomment-226564908 */
175 if (RUNNING_ON_VALGRIND
)
183 (void) wait_for_terminate(pid
, &si
);
185 assert_se(si
.si_code
== CLD_EXITED
);
186 assert_se(si
.si_status
== 0);
192 assert_se(unshare(CLONE_NEWNS
) >= 0);
194 fd
= mkostemp(path
, O_CLOEXEC
);
196 assert_se(mount(path
, "/proc/self/cmdline", "bind", MS_BIND
, NULL
) >= 0);
197 assert_se(unlink(path
) >= 0);
199 assert_se(prctl(PR_SET_NAME
, "testa") >= 0);
201 assert_se(get_process_cmdline(getpid(), 0, false, &line
) == -ENOENT
);
203 assert_se(get_process_cmdline(getpid(), 0, true, &line
) >= 0);
204 assert_se(streq(line
, "[testa]"));
207 assert_se(get_process_cmdline(getpid(), 1, true, &line
) >= 0);
208 assert_se(streq(line
, ""));
211 assert_se(get_process_cmdline(getpid(), 2, true, &line
) >= 0);
212 assert_se(streq(line
, "["));
215 assert_se(get_process_cmdline(getpid(), 3, true, &line
) >= 0);
216 assert_se(streq(line
, "[."));
219 assert_se(get_process_cmdline(getpid(), 4, true, &line
) >= 0);
220 assert_se(streq(line
, "[.."));
223 assert_se(get_process_cmdline(getpid(), 5, true, &line
) >= 0);
224 assert_se(streq(line
, "[..."));
227 assert_se(get_process_cmdline(getpid(), 6, true, &line
) >= 0);
228 assert_se(streq(line
, "[...]"));
231 assert_se(get_process_cmdline(getpid(), 7, true, &line
) >= 0);
232 assert_se(streq(line
, "[t...]"));
235 assert_se(get_process_cmdline(getpid(), 8, true, &line
) >= 0);
236 assert_se(streq(line
, "[testa]"));
239 assert_se(write(fd
, "\0\0\0\0\0\0\0\0\0", 10) == 10);
241 assert_se(get_process_cmdline(getpid(), 0, false, &line
) == -ENOENT
);
243 assert_se(get_process_cmdline(getpid(), 0, true, &line
) >= 0);
244 assert_se(streq(line
, "[testa]"));
247 assert_se(write(fd
, "foo\0bar\0\0\0\0\0", 10) == 10);
249 assert_se(get_process_cmdline(getpid(), 0, false, &line
) >= 0);
250 assert_se(streq(line
, "foo bar"));
253 assert_se(get_process_cmdline(getpid(), 0, true, &line
) >= 0);
254 assert_se(streq(line
, "foo bar"));
257 assert_se(write(fd
, "quux", 4) == 4);
258 assert_se(get_process_cmdline(getpid(), 0, false, &line
) >= 0);
259 assert_se(streq(line
, "foo bar quux"));
262 assert_se(get_process_cmdline(getpid(), 0, true, &line
) >= 0);
263 assert_se(streq(line
, "foo bar quux"));
266 assert_se(get_process_cmdline(getpid(), 1, true, &line
) >= 0);
267 assert_se(streq(line
, ""));
270 assert_se(get_process_cmdline(getpid(), 2, true, &line
) >= 0);
271 assert_se(streq(line
, "."));
274 assert_se(get_process_cmdline(getpid(), 3, true, &line
) >= 0);
275 assert_se(streq(line
, ".."));
278 assert_se(get_process_cmdline(getpid(), 4, true, &line
) >= 0);
279 assert_se(streq(line
, "..."));
282 assert_se(get_process_cmdline(getpid(), 5, true, &line
) >= 0);
283 assert_se(streq(line
, "f..."));
286 assert_se(get_process_cmdline(getpid(), 6, true, &line
) >= 0);
287 assert_se(streq(line
, "fo..."));
290 assert_se(get_process_cmdline(getpid(), 7, true, &line
) >= 0);
291 assert_se(streq(line
, "foo..."));
294 assert_se(get_process_cmdline(getpid(), 8, true, &line
) >= 0);
295 assert_se(streq(line
, "foo..."));
298 assert_se(get_process_cmdline(getpid(), 9, true, &line
) >= 0);
299 assert_se(streq(line
, "foo b..."));
302 assert_se(get_process_cmdline(getpid(), 10, true, &line
) >= 0);
303 assert_se(streq(line
, "foo ba..."));
306 assert_se(get_process_cmdline(getpid(), 11, true, &line
) >= 0);
307 assert_se(streq(line
, "foo bar..."));
310 assert_se(get_process_cmdline(getpid(), 12, true, &line
) >= 0);
311 assert_se(streq(line
, "foo bar..."));
314 assert_se(get_process_cmdline(getpid(), 13, true, &line
) >= 0);
315 assert_se(streq(line
, "foo bar quux"));
318 assert_se(get_process_cmdline(getpid(), 14, true, &line
) >= 0);
319 assert_se(streq(line
, "foo bar quux"));
322 assert_se(get_process_cmdline(getpid(), 1000, true, &line
) >= 0);
323 assert_se(streq(line
, "foo bar quux"));
326 assert_se(ftruncate(fd
, 0) >= 0);
327 assert_se(prctl(PR_SET_NAME
, "aaaa bbbb cccc") >= 0);
329 assert_se(get_process_cmdline(getpid(), 0, false, &line
) == -ENOENT
);
331 assert_se(get_process_cmdline(getpid(), 0, true, &line
) >= 0);
332 assert_se(streq(line
, "[aaaa bbbb cccc]"));
335 assert_se(get_process_cmdline(getpid(), 10, true, &line
) >= 0);
336 assert_se(streq(line
, "[aaaa...]"));
339 assert_se(get_process_cmdline(getpid(), 11, true, &line
) >= 0);
340 assert_se(streq(line
, "[aaaa...]"));
343 assert_se(get_process_cmdline(getpid(), 12, true, &line
) >= 0);
344 assert_se(streq(line
, "[aaaa b...]"));
351 int main(int argc
, char *argv
[]) {
352 log_parse_environment();
358 (void) parse_pid(argv
[1], &pid
);
359 test_get_process_comm(pid
);
361 TEST_REQ_RUNNING_SYSTEMD(test_get_process_comm(1));
362 test_get_process_comm(getpid());
365 test_pid_is_unwaited();
368 test_get_process_cmdline_harder();