// SPDX-License-Identifier: MIT // SPDX-FileCopyrightText: 2024 Michael Jeanson #ifndef _GNU_SOURCE #define _GNU_SOURCE #endif #include #include #include #include #include #include "rseq.h" static int sys_rseq(void *rseq_abi, uint32_t rseq_len, int flags, uint32_t sig) { return syscall(__NR_rseq, rseq_abi, rseq_len, flags, sig); } /* * Check the value of errno on some expected failures of the rseq syscall. */ int main(void) { struct rseq_abi *global_rseq = rseq_get_abi(); int ret; int errno_copy; if (!rseq_available()) { fprintf(stderr, "rseq syscall unavailable"); goto error; } /* The current thread is NOT registered. */ /* EINVAL */ errno = 0; ret = sys_rseq(global_rseq, 32, -1, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Registration with invalid flag fails with errno set to EINVAL (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret == 0 || errno_copy != EINVAL) goto error; errno = 0; ret = sys_rseq((char *) global_rseq + 1, 32, 0, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Registration with unaligned rseq_abi fails with errno set to EINVAL (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret == 0 || errno_copy != EINVAL) goto error; errno = 0; ret = sys_rseq(global_rseq, 31, 0, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Registration with invalid size fails with errno set to EINVAL (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret == 0 || errno_copy != EINVAL) goto error; #if defined(__LP64__) && (!defined(__s390__) && !defined(__s390x__)) /* * We haven't found a reliable way to find an invalid address when * running a 32bit userspace on a 64bit kernel, so only run this test * on 64bit builds for the moment. * * Also exclude architectures that select * CONFIG_ALTERNATE_USER_ADDRESS_SPACE where the kernel and userspace * have their own address space and this failure can't happen. */ /* EFAULT */ errno = 0; ret = sys_rseq((void *) -4096UL, 32, 0, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Registration with invalid address fails with errno set to EFAULT (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret == 0 || errno_copy != EFAULT) goto error; #endif errno = 0; ret = sys_rseq(global_rseq, 32, 0, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Registration succeeds for the current thread (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret != 0 && errno != 0) goto error; /* The current thread is registered. */ /* EBUSY */ errno = 0; ret = sys_rseq(global_rseq, 32, 0, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Double registration fails with errno set to EBUSY (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret == 0 || errno_copy != EBUSY) goto error; /* EPERM */ errno = 0; ret = sys_rseq(global_rseq, 32, RSEQ_ABI_FLAG_UNREGISTER, RSEQ_SIG + 1); errno_copy = errno; fprintf(stderr, "Unregistration with wrong RSEQ_SIG fails with errno to EPERM (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret == 0 || errno_copy != EPERM) goto error; errno = 0; ret = sys_rseq(global_rseq, 32, RSEQ_ABI_FLAG_UNREGISTER, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Unregistration succeeds for the current thread (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret != 0) goto error; errno = 0; ret = sys_rseq(global_rseq, 32, RSEQ_ABI_FLAG_UNREGISTER, RSEQ_SIG); errno_copy = errno; fprintf(stderr, "Double unregistration fails with errno set to EINVAL (ret = %d, errno = %s)\n", ret, strerrorname_np(errno_copy)); if (ret == 0 || errno_copy != EINVAL) goto error; return 0; error: return -1; }