// SPDX-License-Identifier: GPL-2.0 #include <errno.h> #include <linux/unistd.h> #include <sys/ptrace.h> #include <sys/syscall.h> #include <unistd.h> #include <sysdep/tls.h> #ifndef PTRACE_GET_THREAD_AREA #define PTRACE_GET_THREAD_AREA 25 #endif #ifndef PTRACE_SET_THREAD_AREA #define PTRACE_SET_THREAD_AREA 26 #endif /* Checks whether host supports TLS, and sets *tls_min according to the value * valid on the host. * i386 host have it == 6; x86_64 host have it == 12, for i386 emulation. */ void check_host_supports_tls(int *supports_tls, int *tls_min) { /* Values for x86 and x86_64.*/ int val[] = {GDT_ENTRY_TLS_MIN_I386, GDT_ENTRY_TLS_MIN_X86_64}; int i; for (i = 0; i < ARRAY_SIZE(val); i++) { user_desc_t info; info.entry_number = val[i]; if (syscall(__NR_get_thread_area, &info) == 0) { *tls_min = val[i]; *supports_tls = 1; return; } else { if (errno == EINVAL) continue; else if (errno == ENOSYS) *supports_tls = 0; return; } } *supports_tls = 0; } int os_set_thread_area(user_desc_t *info, int pid) { int ret; ret = ptrace(PTRACE_SET_THREAD_AREA, pid, info->entry_number, (unsigned long) info); if (ret < 0) ret = -errno; return ret; } int os_get_thread_area(user_desc_t *info, int pid) { int ret; ret = ptrace(PTRACE_GET_THREAD_AREA, pid, info->entry_number, (unsigned long) info); if (ret < 0) ret = -errno; return ret; }