1a65329aaSBreno Leitao // SPDX-License-Identifier: GPL-2.0
2a65329aaSBreno Leitao /*
3a65329aaSBreno Leitao * Copyright 2018, Breno Leitao, Gustavo Romero, IBM Corp.
4a65329aaSBreno Leitao *
5a65329aaSBreno Leitao * This test raises a SIGUSR1 signal, and toggle the MSR[TS]
6a65329aaSBreno Leitao * fields at the signal handler. With MSR[TS] being set, the kernel will
7a65329aaSBreno Leitao * force a recheckpoint, which may cause a segfault when returning to
8a65329aaSBreno Leitao * user space. Since the test needs to re-run, the segfault needs to be
9a65329aaSBreno Leitao * caught and handled.
10a65329aaSBreno Leitao *
11a65329aaSBreno Leitao * In order to continue the test even after a segfault, the context is
12a65329aaSBreno Leitao * saved prior to the signal being raised, and it is restored when there is
13a65329aaSBreno Leitao * a segmentation fault. This happens for COUNT_MAX times.
14a65329aaSBreno Leitao *
15a65329aaSBreno Leitao * This test never fails (as returning EXIT_FAILURE). It either succeeds,
16a65329aaSBreno Leitao * or crash the kernel (on a buggy kernel).
17a65329aaSBreno Leitao */
18a65329aaSBreno Leitao
19a65329aaSBreno Leitao #define _GNU_SOURCE
20a65329aaSBreno Leitao #include <stdio.h>
21a65329aaSBreno Leitao #include <stdlib.h>
22a65329aaSBreno Leitao #include <signal.h>
23a65329aaSBreno Leitao #include <string.h>
24a65329aaSBreno Leitao #include <ucontext.h>
25a65329aaSBreno Leitao #include <unistd.h>
26a65329aaSBreno Leitao #include <sys/mman.h>
27a65329aaSBreno Leitao
28a65329aaSBreno Leitao #include "tm.h"
29a65329aaSBreno Leitao #include "utils.h"
30a65329aaSBreno Leitao #include "reg.h"
31a65329aaSBreno Leitao
32a65329aaSBreno Leitao #define COUNT_MAX 5000 /* Number of interactions */
33a65329aaSBreno Leitao
34a65329aaSBreno Leitao /*
35a65329aaSBreno Leitao * This test only runs on 64 bits system. Unsetting MSR_TS_S to avoid
36a65329aaSBreno Leitao * compilation issue on 32 bits system. There is no side effect, since the
37a65329aaSBreno Leitao * whole test will be skipped if it is not running on 64 bits system.
38a65329aaSBreno Leitao */
39a65329aaSBreno Leitao #ifndef __powerpc64__
40a65329aaSBreno Leitao #undef MSR_TS_S
41a65329aaSBreno Leitao #define MSR_TS_S 0
42a65329aaSBreno Leitao #endif
43a65329aaSBreno Leitao
44a65329aaSBreno Leitao /* Setting contexts because the test will crash and we want to recover */
45*0f8f554eSGustavo Luiz Duarte ucontext_t init_context;
46a65329aaSBreno Leitao
47*0f8f554eSGustavo Luiz Duarte /* count is changed in the signal handler, so it must be volatile */
48*0f8f554eSGustavo Luiz Duarte static volatile int count;
49a65329aaSBreno Leitao
usr_signal_handler(int signo,siginfo_t * si,void * uc)50a65329aaSBreno Leitao void usr_signal_handler(int signo, siginfo_t *si, void *uc)
51a65329aaSBreno Leitao {
52a65329aaSBreno Leitao ucontext_t *ucp = uc;
53a65329aaSBreno Leitao int ret;
54a65329aaSBreno Leitao
55a65329aaSBreno Leitao /*
56a65329aaSBreno Leitao * Allocating memory in a signal handler, and never freeing it on
57a65329aaSBreno Leitao * purpose, forcing the heap increase, so, the memory leak is what
58a65329aaSBreno Leitao * we want here.
59a65329aaSBreno Leitao */
60a65329aaSBreno Leitao ucp->uc_link = mmap(NULL, sizeof(ucontext_t),
61a65329aaSBreno Leitao PROT_READ | PROT_WRITE,
62a65329aaSBreno Leitao MAP_PRIVATE | MAP_ANONYMOUS, 0, 0);
63a65329aaSBreno Leitao if (ucp->uc_link == (void *)-1) {
64a65329aaSBreno Leitao perror("Mmap failed");
65a65329aaSBreno Leitao exit(-1);
66a65329aaSBreno Leitao }
67a65329aaSBreno Leitao
68a65329aaSBreno Leitao /* Forcing the page to be allocated in a page fault */
69a65329aaSBreno Leitao ret = madvise(ucp->uc_link, sizeof(ucontext_t), MADV_DONTNEED);
70a65329aaSBreno Leitao if (ret) {
71a65329aaSBreno Leitao perror("madvise failed");
72a65329aaSBreno Leitao exit(-1);
73a65329aaSBreno Leitao }
74a65329aaSBreno Leitao
75a65329aaSBreno Leitao memcpy(&ucp->uc_link->uc_mcontext, &ucp->uc_mcontext,
76a65329aaSBreno Leitao sizeof(ucp->uc_mcontext));
77a65329aaSBreno Leitao
78a65329aaSBreno Leitao /* Forcing to enable MSR[TM] */
79a65329aaSBreno Leitao UCONTEXT_MSR(ucp) |= MSR_TS_S;
80a65329aaSBreno Leitao
81a65329aaSBreno Leitao /*
82a65329aaSBreno Leitao * A fork inside a signal handler seems to be more efficient than a
83a65329aaSBreno Leitao * fork() prior to the signal being raised.
84a65329aaSBreno Leitao */
85a65329aaSBreno Leitao if (fork() == 0) {
86a65329aaSBreno Leitao /*
87a65329aaSBreno Leitao * Both child and parent will return, but, child returns
88a65329aaSBreno Leitao * with count set so it will exit in the next segfault.
89a65329aaSBreno Leitao * Parent will continue to loop.
90a65329aaSBreno Leitao */
91a65329aaSBreno Leitao count = COUNT_MAX;
92a65329aaSBreno Leitao }
93a65329aaSBreno Leitao
94a65329aaSBreno Leitao /*
95a65329aaSBreno Leitao * If the change above does not hit the bug, it will cause a
96a65329aaSBreno Leitao * segmentation fault, since the ck structures are NULL.
97a65329aaSBreno Leitao */
98a65329aaSBreno Leitao }
99a65329aaSBreno Leitao
seg_signal_handler(int signo,siginfo_t * si,void * uc)100a65329aaSBreno Leitao void seg_signal_handler(int signo, siginfo_t *si, void *uc)
101a65329aaSBreno Leitao {
102a65329aaSBreno Leitao count++;
103a65329aaSBreno Leitao
104a65329aaSBreno Leitao /* Reexecute the test */
105a65329aaSBreno Leitao setcontext(&init_context);
106a65329aaSBreno Leitao }
107a65329aaSBreno Leitao
tm_trap_test(void)108a65329aaSBreno Leitao void tm_trap_test(void)
109a65329aaSBreno Leitao {
110a65329aaSBreno Leitao struct sigaction usr_sa, seg_sa;
111a65329aaSBreno Leitao stack_t ss;
112a65329aaSBreno Leitao
113a65329aaSBreno Leitao usr_sa.sa_flags = SA_SIGINFO | SA_ONSTACK;
114a65329aaSBreno Leitao usr_sa.sa_sigaction = usr_signal_handler;
115a65329aaSBreno Leitao
116a65329aaSBreno Leitao seg_sa.sa_flags = SA_SIGINFO;
117a65329aaSBreno Leitao seg_sa.sa_sigaction = seg_signal_handler;
118a65329aaSBreno Leitao
119a65329aaSBreno Leitao /*
120a65329aaSBreno Leitao * Set initial context. Will get back here from
121a65329aaSBreno Leitao * seg_signal_handler()
122a65329aaSBreno Leitao */
123a65329aaSBreno Leitao getcontext(&init_context);
124a65329aaSBreno Leitao
125*0f8f554eSGustavo Luiz Duarte while (count < COUNT_MAX) {
126a65329aaSBreno Leitao /* Allocated an alternative signal stack area */
127a65329aaSBreno Leitao ss.ss_sp = mmap(NULL, SIGSTKSZ, PROT_READ | PROT_WRITE,
128a65329aaSBreno Leitao MAP_PRIVATE | MAP_ANONYMOUS, 0, 0);
129a65329aaSBreno Leitao ss.ss_size = SIGSTKSZ;
130a65329aaSBreno Leitao ss.ss_flags = 0;
131a65329aaSBreno Leitao
132a65329aaSBreno Leitao if (ss.ss_sp == (void *)-1) {
133a65329aaSBreno Leitao perror("mmap error\n");
134a65329aaSBreno Leitao exit(-1);
135a65329aaSBreno Leitao }
136a65329aaSBreno Leitao
137a65329aaSBreno Leitao /* Force the allocation through a page fault */
138a65329aaSBreno Leitao if (madvise(ss.ss_sp, SIGSTKSZ, MADV_DONTNEED)) {
139a65329aaSBreno Leitao perror("madvise\n");
140a65329aaSBreno Leitao exit(-1);
141a65329aaSBreno Leitao }
142a65329aaSBreno Leitao
143*0f8f554eSGustavo Luiz Duarte /*
144*0f8f554eSGustavo Luiz Duarte * Setting an alternative stack to generate a page fault when
145a65329aaSBreno Leitao * the signal is raised.
146a65329aaSBreno Leitao */
147a65329aaSBreno Leitao if (sigaltstack(&ss, NULL)) {
148a65329aaSBreno Leitao perror("sigaltstack\n");
149a65329aaSBreno Leitao exit(-1);
150a65329aaSBreno Leitao }
151a65329aaSBreno Leitao
152a65329aaSBreno Leitao /* The signal handler will enable MSR_TS */
153a65329aaSBreno Leitao sigaction(SIGUSR1, &usr_sa, NULL);
154*0f8f554eSGustavo Luiz Duarte /* If it does not crash, it might segfault, avoid it to retest */
155a65329aaSBreno Leitao sigaction(SIGSEGV, &seg_sa, NULL);
156a65329aaSBreno Leitao
157a65329aaSBreno Leitao raise(SIGUSR1);
158*0f8f554eSGustavo Luiz Duarte count++;
159*0f8f554eSGustavo Luiz Duarte }
160a65329aaSBreno Leitao }
161a65329aaSBreno Leitao
tm_signal_context_force_tm(void)162a65329aaSBreno Leitao int tm_signal_context_force_tm(void)
163a65329aaSBreno Leitao {
164a65329aaSBreno Leitao SKIP_IF(!have_htm());
165a65329aaSBreno Leitao /*
166a65329aaSBreno Leitao * Skipping if not running on 64 bits system, since I think it is
167a65329aaSBreno Leitao * not possible to set mcontext's [MSR] with TS, due to it being 32
168a65329aaSBreno Leitao * bits.
169a65329aaSBreno Leitao */
170a65329aaSBreno Leitao SKIP_IF(!is_ppc64le());
171a65329aaSBreno Leitao
172a65329aaSBreno Leitao tm_trap_test();
173a65329aaSBreno Leitao
174a65329aaSBreno Leitao return EXIT_SUCCESS;
175a65329aaSBreno Leitao }
176a65329aaSBreno Leitao
main(int argc,char ** argv)177a65329aaSBreno Leitao int main(int argc, char **argv)
178a65329aaSBreno Leitao {
179a65329aaSBreno Leitao test_harness(tm_signal_context_force_tm, "tm_signal_context_force_tm");
180a65329aaSBreno Leitao }
181