Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

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 <siddhesh@gotplt.org>
Reported-by: Masami Hiramatsu <masami.hiramatsu@linaro.org>
Tested-by: Masami Hiramatsu <masami.hiramatsu@linaro.org>
Reviewed-by: Tim Bird <tim.bird@sony.com>
Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>

authored by

Siddhesh Poyarekar and committed by
Shuah Khan
6b64a650 e1dae517

+18 -6
+18 -6
tools/testing/selftests/size/get_size.c
··· 11 11 * own execution. It also attempts to have as few dependencies 12 12 * on kernel features as possible. 13 13 * 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: 16 17 * sysinfo(), write(), and _exit() 17 18 * 18 19 * For output, it avoids printf (which in some C libraries 19 20 * has large external dependencies) by implementing it's own 20 21 * 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. 21 27 */ 22 28 23 29 #include <sys/sysinfo.h> 24 30 #include <unistd.h> 31 + #include <sys/syscall.h> 25 32 26 33 #define STDOUT_FILENO 1 27 34 28 35 static int print(const char *s) 29 36 { 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); 31 43 } 32 44 33 45 static inline char *num_to_str(unsigned long num, char *buf, int len) ··· 91 79 print("TAP version 13\n"); 92 80 print("# Testing system size.\n"); 93 81 94 - ccode = sysinfo(&info); 82 + ccode = syscall(SYS_sysinfo, &info); 95 83 if (ccode < 0) { 96 84 print("not ok 1"); 97 85 print(test_name); 98 86 print(" ---\n reason: \"could not get sysinfo\"\n ...\n"); 99 - _exit(ccode); 87 + syscall(SYS_exit, ccode); 100 88 } 101 89 print("ok 1"); 102 90 print(test_name); ··· 112 100 print(" ...\n"); 113 101 print("1..1\n"); 114 102 115 - _exit(0); 103 + syscall(SYS_exit, 0); 116 104 }