diff options
author | Alexey Dobriyan <adobriyan@gmail.com> | 2018-06-07 20:10:20 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2018-06-07 20:34:38 -0400 |
commit | b2f5de0334f03e90ae7dee72a7fc597ef555a9a8 (patch) | |
tree | 6eddd34b28ecd67582164b588f862951bf392680 /tools/testing/selftests/proc/proc.h | |
parent | 5d008fb414f7c73e0761835d941943e17d32463d (diff) |
tools/testing/selftests/proc: test /proc/*/fd a bit (+ PF_KTHREAD is ABI!)
* Test lookup in /proc/self/fd.
"map_files" lookup story showed that lookup is not that simple.
* Test that all those symlinks open the same file.
Check with (st_dev, st_info).
* Test that kernel threads do not have anything in their /proc/*/fd/
directory.
Now this is where things get interesting.
First, kernel threads aren't pinned by /proc/self or equivalent,
thus some "atomicity" is required.
Second, ->comm can contain whitespace and ')'.
No, they are not escaped.
Third, the only reliable way to check if process is kernel thread
appears to be field #9 in /proc/*/stat.
This field is struct task_struct::flags in decimal!
Check is done by testing PF_KTHREAD flags like we do in kernel.
PF_KTREAD value is a part of userspace ABI !!!
Other methods for determining kernel threadness are not reliable:
* RSS can be 0 if everything is swapped, even while reading
from /proc/self.
* ->total_vm CAN BE ZERO if process is finishing
munmap(NULL, whole address space);
* /proc/*/maps and similar files can be empty because unmapping
everything works. Read returning 0 can't distinguish between
kernel thread and such suicide process.
Link: http://lkml.kernel.org/r/20180505000414.GA15090@avx2
Signed-off-by: Alexey Dobriyan <adobriyan@gmail.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'tools/testing/selftests/proc/proc.h')
-rw-r--r-- | tools/testing/selftests/proc/proc.h | 39 |
1 files changed, 39 insertions, 0 deletions
diff --git a/tools/testing/selftests/proc/proc.h b/tools/testing/selftests/proc/proc.h new file mode 100644 index 000000000000..4e178166fd84 --- /dev/null +++ b/tools/testing/selftests/proc/proc.h | |||
@@ -0,0 +1,39 @@ | |||
1 | #pragma once | ||
2 | #undef NDEBUG | ||
3 | #include <assert.h> | ||
4 | #include <dirent.h> | ||
5 | #include <errno.h> | ||
6 | #include <stdbool.h> | ||
7 | #include <stdlib.h> | ||
8 | #include <string.h> | ||
9 | |||
10 | static inline bool streq(const char *s1, const char *s2) | ||
11 | { | ||
12 | return strcmp(s1, s2) == 0; | ||
13 | } | ||
14 | |||
15 | static unsigned long long xstrtoull(const char *p, char **end) | ||
16 | { | ||
17 | if (*p == '0') { | ||
18 | *end = (char *)p + 1; | ||
19 | return 0; | ||
20 | } else if ('1' <= *p && *p <= '9') { | ||
21 | unsigned long long val; | ||
22 | |||
23 | errno = 0; | ||
24 | val = strtoull(p, end, 10); | ||
25 | assert(errno == 0); | ||
26 | return val; | ||
27 | } else | ||
28 | assert(0); | ||
29 | } | ||
30 | |||
31 | static struct dirent *xreaddir(DIR *d) | ||
32 | { | ||
33 | struct dirent *de; | ||
34 | |||
35 | errno = 0; | ||
36 | de = readdir(d); | ||
37 | assert(de || errno == 0); | ||
38 | return de; | ||
39 | } | ||