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