162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright 2015, Michael Neuling, IBM Corp. 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Test the kernel's signal return code to ensure that it doesn't 662306a36Sopenharmony_ci * crash when both the transactional and suspend MSR bits are set in 762306a36Sopenharmony_ci * the signal context. 862306a36Sopenharmony_ci * 962306a36Sopenharmony_ci * For this test, we send ourselves a SIGUSR1. In the SIGUSR1 handler 1062306a36Sopenharmony_ci * we modify the signal context to set both MSR TM S and T bits (which 1162306a36Sopenharmony_ci * is "reserved" by the PowerISA). When we return from the signal 1262306a36Sopenharmony_ci * handler (implicit sigreturn), the kernel should detect reserved MSR 1362306a36Sopenharmony_ci * value and send us with a SIGSEGV. 1462306a36Sopenharmony_ci */ 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci#include <stdlib.h> 1762306a36Sopenharmony_ci#include <stdio.h> 1862306a36Sopenharmony_ci#include <signal.h> 1962306a36Sopenharmony_ci#include <unistd.h> 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#include "utils.h" 2262306a36Sopenharmony_ci#include "tm.h" 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ciint segv_expected = 0; 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_civoid signal_segv(int signum) 2762306a36Sopenharmony_ci{ 2862306a36Sopenharmony_ci if (segv_expected && (signum == SIGSEGV)) 2962306a36Sopenharmony_ci _exit(0); 3062306a36Sopenharmony_ci _exit(1); 3162306a36Sopenharmony_ci} 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_civoid signal_usr1(int signum, siginfo_t *info, void *uc) 3462306a36Sopenharmony_ci{ 3562306a36Sopenharmony_ci ucontext_t *ucp = uc; 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci /* Link tm checkpointed context to normal context */ 3862306a36Sopenharmony_ci ucp->uc_link = ucp; 3962306a36Sopenharmony_ci /* Set all TM bits so that the context is now invalid */ 4062306a36Sopenharmony_ci#ifdef __powerpc64__ 4162306a36Sopenharmony_ci ucp->uc_mcontext.gp_regs[PT_MSR] |= (7ULL << 32); 4262306a36Sopenharmony_ci#else 4362306a36Sopenharmony_ci ucp->uc_mcontext.uc_regs->gregs[PT_MSR] |= (7ULL); 4462306a36Sopenharmony_ci#endif 4562306a36Sopenharmony_ci /* Should segv on return becuase of invalid context */ 4662306a36Sopenharmony_ci segv_expected = 1; 4762306a36Sopenharmony_ci} 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ciint tm_signal_msr_resv() 5062306a36Sopenharmony_ci{ 5162306a36Sopenharmony_ci struct sigaction act; 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci SKIP_IF(!have_htm()); 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci act.sa_sigaction = signal_usr1; 5662306a36Sopenharmony_ci sigemptyset(&act.sa_mask); 5762306a36Sopenharmony_ci act.sa_flags = SA_SIGINFO; 5862306a36Sopenharmony_ci if (sigaction(SIGUSR1, &act, NULL) < 0) { 5962306a36Sopenharmony_ci perror("sigaction sigusr1"); 6062306a36Sopenharmony_ci exit(1); 6162306a36Sopenharmony_ci } 6262306a36Sopenharmony_ci if (signal(SIGSEGV, signal_segv) == SIG_ERR) 6362306a36Sopenharmony_ci exit(1); 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci raise(SIGUSR1); 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci /* We shouldn't get here as we exit in the segv handler */ 6862306a36Sopenharmony_ci return 1; 6962306a36Sopenharmony_ci} 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ciint main(void) 7262306a36Sopenharmony_ci{ 7362306a36Sopenharmony_ci return test_harness(tm_signal_msr_resv, "tm_signal_msr_resv"); 7462306a36Sopenharmony_ci} 75