Skip to content

Commit 6b64a65

Browse files
siddheshshuahkh
authored andcommitted
kselftest: Minimise dependency of get_size on C library interfaces
It was observed[1] on arm64 that __builtin_strlen led to an infinite loop in the get_size selftest. This is because __builtin_strlen (and other builtins) may sometimes result in a call to the C library function. The C library implementation of strlen uses an IFUNC resolver to load the most efficient strlen implementation for the underlying machine and hence has a PLT indirection even for static binaries. Because this binary avoids the C library startup routines, the PLT initialization never happens and hence the program gets stuck in an infinite loop. On x86_64 the __builtin_strlen just happens to expand inline and avoid the call but that is not always guaranteed. Further, while testing on x86_64 (Fedora 31), it was observed that the test also failed with a segfault inside write() because the generated code for the write function in glibc seems to access TLS before the syscall (probably due to the cancellation point check) and fails because TLS is not initialised. To mitigate these problems, this patch reduces the interface with the C library to just the syscall function. The syscall function still sets errno on failure, which is undesirable but for now it only affects cases where syscalls fail. [1] https://bugs.linaro.org/show_bug.cgi?id=5479 Signed-off-by: Siddhesh Poyarekar <[email protected]> Reported-by: Masami Hiramatsu <[email protected]> Tested-by: Masami Hiramatsu <[email protected]> Reviewed-by: Tim Bird <[email protected]> Signed-off-by: Shuah Khan <[email protected]>
1 parent e1dae51 commit 6b64a65

File tree

1 file changed

+18
-6
lines changed

1 file changed

+18
-6
lines changed

tools/testing/selftests/size/get_size.c

Lines changed: 18 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -11,23 +11,35 @@
1111
* own execution. It also attempts to have as few dependencies
1212
* on kernel features as possible.
1313
*
14-
* It should be statically linked, with startup libs avoided.
15-
* It uses no library calls, and only the following 3 syscalls:
14+
* It should be statically linked, with startup libs avoided. It uses
15+
* no library calls except the syscall() function for the following 3
16+
* syscalls:
1617
* sysinfo(), write(), and _exit()
1718
*
1819
* For output, it avoids printf (which in some C libraries
1920
* has large external dependencies) by implementing it's own
2021
* number output and print routines, and using __builtin_strlen()
22+
*
23+
* The test may crash if any of the above syscalls fails because in some
24+
* libc implementations (e.g. the GNU C Library) errno is saved in
25+
* thread-local storage, which does not get initialized due to avoiding
26+
* startup libs.
2127
*/
2228

2329
#include <sys/sysinfo.h>
2430
#include <unistd.h>
31+
#include <sys/syscall.h>
2532

2633
#define STDOUT_FILENO 1
2734

2835
static int print(const char *s)
2936
{
30-
return write(STDOUT_FILENO, s, __builtin_strlen(s));
37+
size_t len = 0;
38+
39+
while (s[len] != '\0')
40+
len++;
41+
42+
return syscall(SYS_write, STDOUT_FILENO, s, len);
3143
}
3244

3345
static inline char *num_to_str(unsigned long num, char *buf, int len)
@@ -79,12 +91,12 @@ void _start(void)
7991
print("TAP version 13\n");
8092
print("# Testing system size.\n");
8193

82-
ccode = sysinfo(&info);
94+
ccode = syscall(SYS_sysinfo, &info);
8395
if (ccode < 0) {
8496
print("not ok 1");
8597
print(test_name);
8698
print(" ---\n reason: \"could not get sysinfo\"\n ...\n");
87-
_exit(ccode);
99+
syscall(SYS_exit, ccode);
88100
}
89101
print("ok 1");
90102
print(test_name);
@@ -100,5 +112,5 @@ void _start(void)
100112
print(" ...\n");
101113
print("1..1\n");
102114

103-
_exit(0);
115+
syscall(SYS_exit, 0);
104116
}

0 commit comments

Comments
 (0)