| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * Copyright 2018, Breno Leitao, Gustavo Romero, IBM Corp. |
| * |
| * This test raises a SIGUSR1 signal, and toggle the MSR[TS] |
| * fields at the signal handler. With MSR[TS] being set, the kernel will |
| * force a recheckpoint, which may cause a segfault when returning to |
| * user space. Since the test needs to re-run, the segfault needs to be |
| * caught and handled. |
| * |
| * In order to continue the test even after a segfault, the context is |
| * saved prior to the signal being raised, and it is restored when there is |
| * a segmentation fault. This happens for COUNT_MAX times. |
| * |
| * This test never fails (as returning EXIT_FAILURE). It either succeeds, |
| * or crash the kernel (on a buggy kernel). |
| */ |
| |
| #define _GNU_SOURCE |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <signal.h> |
| #include <string.h> |
| #include <ucontext.h> |
| #include <unistd.h> |
| #include <sys/mman.h> |
| |
| #include "tm.h" |
| #include "utils.h" |
| #include "reg.h" |
| |
| #define COUNT_MAX 5000 /* Number of interactions */ |
| |
| /* |
| * This test only runs on 64 bits system. Unsetting MSR_TS_S to avoid |
| * compilation issue on 32 bits system. There is no side effect, since the |
| * whole test will be skipped if it is not running on 64 bits system. |
| */ |
| #ifndef __powerpc64__ |
| #undef MSR_TS_S |
| #define MSR_TS_S 0 |
| #endif |
| |
| /* Setting contexts because the test will crash and we want to recover */ |
| ucontext_t init_context; |
| |
| /* count is changed in the signal handler, so it must be volatile */ |
| static volatile int count; |
| |
| void usr_signal_handler(int signo, siginfo_t *si, void *uc) |
| { |
| ucontext_t *ucp = uc; |
| int ret; |
| |
| /* |
| * Allocating memory in a signal handler, and never freeing it on |
| * purpose, forcing the heap increase, so, the memory leak is what |
| * we want here. |
| */ |
| ucp->uc_link = mmap(NULL, sizeof(ucontext_t), |
| PROT_READ | PROT_WRITE, |
| MAP_PRIVATE | MAP_ANONYMOUS, 0, 0); |
| if (ucp->uc_link == (void *)-1) { |
| perror("Mmap failed"); |
| exit(-1); |
| } |
| |
| /* Forcing the page to be allocated in a page fault */ |
| ret = madvise(ucp->uc_link, sizeof(ucontext_t), MADV_DONTNEED); |
| if (ret) { |
| perror("madvise failed"); |
| exit(-1); |
| } |
| |
| memcpy(&ucp->uc_link->uc_mcontext, &ucp->uc_mcontext, |
| sizeof(ucp->uc_mcontext)); |
| |
| /* Forcing to enable MSR[TM] */ |
| UCONTEXT_MSR(ucp) |= MSR_TS_S; |
| |
| /* |
| * A fork inside a signal handler seems to be more efficient than a |
| * fork() prior to the signal being raised. |
| */ |
| if (fork() == 0) { |
| /* |
| * Both child and parent will return, but, child returns |
| * with count set so it will exit in the next segfault. |
| * Parent will continue to loop. |
| */ |
| count = COUNT_MAX; |
| } |
| |
| /* |
| * If the change above does not hit the bug, it will cause a |
| * segmentation fault, since the ck structures are NULL. |
| */ |
| } |
| |
| void seg_signal_handler(int signo, siginfo_t *si, void *uc) |
| { |
| count++; |
| |
| /* Reexecute the test */ |
| setcontext(&init_context); |
| } |
| |
| void tm_trap_test(void) |
| { |
| struct sigaction usr_sa, seg_sa; |
| stack_t ss; |
| |
| usr_sa.sa_flags = SA_SIGINFO | SA_ONSTACK; |
| usr_sa.sa_sigaction = usr_signal_handler; |
| |
| seg_sa.sa_flags = SA_SIGINFO; |
| seg_sa.sa_sigaction = seg_signal_handler; |
| |
| /* |
| * Set initial context. Will get back here from |
| * seg_signal_handler() |
| */ |
| getcontext(&init_context); |
| |
| while (count < COUNT_MAX) { |
| /* Allocated an alternative signal stack area */ |
| ss.ss_sp = mmap(NULL, SIGSTKSZ, PROT_READ | PROT_WRITE, |
| MAP_PRIVATE | MAP_ANONYMOUS, 0, 0); |
| ss.ss_size = SIGSTKSZ; |
| ss.ss_flags = 0; |
| |
| if (ss.ss_sp == (void *)-1) { |
| perror("mmap error\n"); |
| exit(-1); |
| } |
| |
| /* Force the allocation through a page fault */ |
| if (madvise(ss.ss_sp, SIGSTKSZ, MADV_DONTNEED)) { |
| perror("madvise\n"); |
| exit(-1); |
| } |
| |
| /* |
| * Setting an alternative stack to generate a page fault when |
| * the signal is raised. |
| */ |
| if (sigaltstack(&ss, NULL)) { |
| perror("sigaltstack\n"); |
| exit(-1); |
| } |
| |
| /* The signal handler will enable MSR_TS */ |
| sigaction(SIGUSR1, &usr_sa, NULL); |
| /* If it does not crash, it might segfault, avoid it to retest */ |
| sigaction(SIGSEGV, &seg_sa, NULL); |
| |
| raise(SIGUSR1); |
| count++; |
| } |
| } |
| |
| int tm_signal_context_force_tm(void) |
| { |
| SKIP_IF(!have_htm()); |
| /* |
| * Skipping if not running on 64 bits system, since I think it is |
| * not possible to set mcontext's [MSR] with TS, due to it being 32 |
| * bits. |
| */ |
| SKIP_IF(!is_ppc64le()); |
| |
| tm_trap_test(); |
| |
| return EXIT_SUCCESS; |
| } |
| |
| int main(int argc, char **argv) |
| { |
| test_harness(tm_signal_context_force_tm, "tm_signal_context_force_tm"); |
| } |