xref: /linux/tools/testing/selftests/x86/ptrace_syscall.c (revision ead5d1f4d877e92c051e1a1ade623d0d30e71619)
1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
204235c00SAndy Lutomirski #define _GNU_SOURCE
304235c00SAndy Lutomirski 
404235c00SAndy Lutomirski #include <sys/ptrace.h>
504235c00SAndy Lutomirski #include <sys/types.h>
604235c00SAndy Lutomirski #include <sys/wait.h>
704235c00SAndy Lutomirski #include <sys/syscall.h>
804235c00SAndy Lutomirski #include <sys/user.h>
904235c00SAndy Lutomirski #include <unistd.h>
1004235c00SAndy Lutomirski #include <errno.h>
1104235c00SAndy Lutomirski #include <stddef.h>
1204235c00SAndy Lutomirski #include <stdio.h>
1304235c00SAndy Lutomirski #include <err.h>
1404235c00SAndy Lutomirski #include <string.h>
1504235c00SAndy Lutomirski #include <asm/ptrace-abi.h>
1604235c00SAndy Lutomirski #include <sys/auxv.h>
1704235c00SAndy Lutomirski 
1804235c00SAndy Lutomirski /* Bitness-agnostic defines for user_regs_struct fields. */
1904235c00SAndy Lutomirski #ifdef __x86_64__
2004235c00SAndy Lutomirski # define user_syscall_nr	orig_rax
2104235c00SAndy Lutomirski # define user_arg0		rdi
2204235c00SAndy Lutomirski # define user_arg1		rsi
2304235c00SAndy Lutomirski # define user_arg2		rdx
2404235c00SAndy Lutomirski # define user_arg3		r10
2504235c00SAndy Lutomirski # define user_arg4		r8
2604235c00SAndy Lutomirski # define user_arg5		r9
2704235c00SAndy Lutomirski # define user_ip		rip
2804235c00SAndy Lutomirski # define user_ax		rax
2904235c00SAndy Lutomirski #else
3004235c00SAndy Lutomirski # define user_syscall_nr	orig_eax
3104235c00SAndy Lutomirski # define user_arg0		ebx
3204235c00SAndy Lutomirski # define user_arg1		ecx
3304235c00SAndy Lutomirski # define user_arg2		edx
3404235c00SAndy Lutomirski # define user_arg3		esi
3504235c00SAndy Lutomirski # define user_arg4		edi
3604235c00SAndy Lutomirski # define user_arg5		ebp
3704235c00SAndy Lutomirski # define user_ip		eip
3804235c00SAndy Lutomirski # define user_ax		eax
3904235c00SAndy Lutomirski #endif
4004235c00SAndy Lutomirski 
4104235c00SAndy Lutomirski static int nerrs = 0;
4204235c00SAndy Lutomirski 
4304235c00SAndy Lutomirski struct syscall_args32 {
4404235c00SAndy Lutomirski 	uint32_t nr, arg0, arg1, arg2, arg3, arg4, arg5;
4504235c00SAndy Lutomirski };
4604235c00SAndy Lutomirski 
4704235c00SAndy Lutomirski #ifdef __i386__
4804235c00SAndy Lutomirski extern void sys32_helper(struct syscall_args32 *, void *);
4904235c00SAndy Lutomirski extern void int80_and_ret(void);
5004235c00SAndy Lutomirski #endif
5104235c00SAndy Lutomirski 
5204235c00SAndy Lutomirski /*
5304235c00SAndy Lutomirski  * Helper to invoke int80 with controlled regs and capture the final regs.
5404235c00SAndy Lutomirski  */
do_full_int80(struct syscall_args32 * args)5504235c00SAndy Lutomirski static void do_full_int80(struct syscall_args32 *args)
5604235c00SAndy Lutomirski {
5704235c00SAndy Lutomirski #ifdef __x86_64__
5804235c00SAndy Lutomirski 	register unsigned long bp asm("bp") = args->arg5;
5904235c00SAndy Lutomirski 	asm volatile ("int $0x80"
6004235c00SAndy Lutomirski 		      : "+a" (args->nr),
6104235c00SAndy Lutomirski 			"+b" (args->arg0), "+c" (args->arg1), "+d" (args->arg2),
622a4d0c62SDmitry Safonov 			"+S" (args->arg3), "+D" (args->arg4), "+r" (bp)
632a4d0c62SDmitry Safonov 			: : "r8", "r9", "r10", "r11");
6404235c00SAndy Lutomirski 	args->arg5 = bp;
6504235c00SAndy Lutomirski #else
6604235c00SAndy Lutomirski 	sys32_helper(args, int80_and_ret);
6704235c00SAndy Lutomirski #endif
6804235c00SAndy Lutomirski }
6904235c00SAndy Lutomirski 
7004235c00SAndy Lutomirski #ifdef __i386__
7104235c00SAndy Lutomirski static void (*vsyscall32)(void);
7204235c00SAndy Lutomirski 
7304235c00SAndy Lutomirski /*
7404235c00SAndy Lutomirski  * Nasty helper to invoke AT_SYSINFO (i.e. __kernel_vsyscall) with
7504235c00SAndy Lutomirski  * controlled regs and capture the final regs.  This is so nasty that it
7604235c00SAndy Lutomirski  * crashes my copy of gdb :)
7704235c00SAndy Lutomirski  */
do_full_vsyscall32(struct syscall_args32 * args)7804235c00SAndy Lutomirski static void do_full_vsyscall32(struct syscall_args32 *args)
7904235c00SAndy Lutomirski {
8004235c00SAndy Lutomirski 	sys32_helper(args, vsyscall32);
8104235c00SAndy Lutomirski }
8204235c00SAndy Lutomirski #endif
8304235c00SAndy Lutomirski 
wait_trap(pid_t chld)8404235c00SAndy Lutomirski static siginfo_t wait_trap(pid_t chld)
8504235c00SAndy Lutomirski {
8604235c00SAndy Lutomirski 	siginfo_t si;
8704235c00SAndy Lutomirski 	if (waitid(P_PID, chld, &si, WEXITED|WSTOPPED) != 0)
8804235c00SAndy Lutomirski 		err(1, "waitid");
8904235c00SAndy Lutomirski 	if (si.si_pid != chld)
9004235c00SAndy Lutomirski 		errx(1, "got unexpected pid in event\n");
9104235c00SAndy Lutomirski 	if (si.si_code != CLD_TRAPPED)
9204235c00SAndy Lutomirski 		errx(1, "got unexpected event type %d\n", si.si_code);
9304235c00SAndy Lutomirski 	return si;
9404235c00SAndy Lutomirski }
9504235c00SAndy Lutomirski 
sethandler(int sig,void (* handler)(int,siginfo_t *,void *),int flags)9604235c00SAndy Lutomirski static void sethandler(int sig, void (*handler)(int, siginfo_t *, void *),
9704235c00SAndy Lutomirski 		       int flags)
9804235c00SAndy Lutomirski {
9904235c00SAndy Lutomirski 	struct sigaction sa;
10004235c00SAndy Lutomirski 	memset(&sa, 0, sizeof(sa));
10104235c00SAndy Lutomirski 	sa.sa_sigaction = handler;
10204235c00SAndy Lutomirski 	sa.sa_flags = SA_SIGINFO | flags;
10304235c00SAndy Lutomirski 	sigemptyset(&sa.sa_mask);
10404235c00SAndy Lutomirski 	if (sigaction(sig, &sa, 0))
10504235c00SAndy Lutomirski 		err(1, "sigaction");
10604235c00SAndy Lutomirski }
10704235c00SAndy Lutomirski 
setsigign(int sig,int flags)10840361343SAndy Lutomirski static void setsigign(int sig, int flags)
10940361343SAndy Lutomirski {
11040361343SAndy Lutomirski 	struct sigaction sa;
11140361343SAndy Lutomirski 	memset(&sa, 0, sizeof(sa));
11240361343SAndy Lutomirski 	sa.sa_sigaction = (void *)SIG_IGN;
11340361343SAndy Lutomirski 	sa.sa_flags = flags;
11440361343SAndy Lutomirski 	sigemptyset(&sa.sa_mask);
11540361343SAndy Lutomirski 	if (sigaction(sig, &sa, 0))
11640361343SAndy Lutomirski 		err(1, "sigaction");
11740361343SAndy Lutomirski }
11840361343SAndy Lutomirski 
clearhandler(int sig)11904235c00SAndy Lutomirski static void clearhandler(int sig)
12004235c00SAndy Lutomirski {
12104235c00SAndy Lutomirski 	struct sigaction sa;
12204235c00SAndy Lutomirski 	memset(&sa, 0, sizeof(sa));
12304235c00SAndy Lutomirski 	sa.sa_handler = SIG_DFL;
12404235c00SAndy Lutomirski 	sigemptyset(&sa.sa_mask);
12504235c00SAndy Lutomirski 	if (sigaction(sig, &sa, 0))
12604235c00SAndy Lutomirski 		err(1, "sigaction");
12704235c00SAndy Lutomirski }
12804235c00SAndy Lutomirski 
12904235c00SAndy Lutomirski #ifdef __x86_64__
13004235c00SAndy Lutomirski # define REG_BP REG_RBP
13104235c00SAndy Lutomirski #else
13204235c00SAndy Lutomirski # define REG_BP REG_EBP
13304235c00SAndy Lutomirski #endif
13404235c00SAndy Lutomirski 
empty_handler(int sig,siginfo_t * si,void * ctx_void)13504235c00SAndy Lutomirski static void empty_handler(int sig, siginfo_t *si, void *ctx_void)
13604235c00SAndy Lutomirski {
13704235c00SAndy Lutomirski }
13804235c00SAndy Lutomirski 
test_sys32_regs(void (* do_syscall)(struct syscall_args32 *))13904235c00SAndy Lutomirski static void test_sys32_regs(void (*do_syscall)(struct syscall_args32 *))
14004235c00SAndy Lutomirski {
14104235c00SAndy Lutomirski 	struct syscall_args32 args = {
14204235c00SAndy Lutomirski 		.nr = 224,	/* gettid */
14304235c00SAndy Lutomirski 		.arg0 = 10, .arg1 = 11, .arg2 = 12,
14404235c00SAndy Lutomirski 		.arg3 = 13, .arg4 = 14, .arg5 = 15,
14504235c00SAndy Lutomirski 	};
14604235c00SAndy Lutomirski 
14704235c00SAndy Lutomirski 	do_syscall(&args);
14804235c00SAndy Lutomirski 
14904235c00SAndy Lutomirski 	if (args.nr != getpid() ||
15004235c00SAndy Lutomirski 	    args.arg0 != 10 || args.arg1 != 11 || args.arg2 != 12 ||
15104235c00SAndy Lutomirski 	    args.arg3 != 13 || args.arg4 != 14 || args.arg5 != 15) {
1521d723de7SColin Ian King 		printf("[FAIL]\tgetpid() failed to preserve regs\n");
15304235c00SAndy Lutomirski 		nerrs++;
15404235c00SAndy Lutomirski 	} else {
15504235c00SAndy Lutomirski 		printf("[OK]\tgetpid() preserves regs\n");
15604235c00SAndy Lutomirski 	}
15704235c00SAndy Lutomirski 
15804235c00SAndy Lutomirski 	sethandler(SIGUSR1, empty_handler, 0);
15904235c00SAndy Lutomirski 
16004235c00SAndy Lutomirski 	args.nr = 37;	/* kill */
16104235c00SAndy Lutomirski 	args.arg0 = getpid();
16204235c00SAndy Lutomirski 	args.arg1 = SIGUSR1;
16304235c00SAndy Lutomirski 	do_syscall(&args);
16404235c00SAndy Lutomirski 	if (args.nr != 0 ||
16504235c00SAndy Lutomirski 	    args.arg0 != getpid() || args.arg1 != SIGUSR1 || args.arg2 != 12 ||
16604235c00SAndy Lutomirski 	    args.arg3 != 13 || args.arg4 != 14 || args.arg5 != 15) {
1671d723de7SColin Ian King 		printf("[FAIL]\tkill(getpid(), SIGUSR1) failed to preserve regs\n");
16804235c00SAndy Lutomirski 		nerrs++;
16904235c00SAndy Lutomirski 	} else {
17004235c00SAndy Lutomirski 		printf("[OK]\tkill(getpid(), SIGUSR1) preserves regs\n");
17104235c00SAndy Lutomirski 	}
17204235c00SAndy Lutomirski 	clearhandler(SIGUSR1);
17304235c00SAndy Lutomirski }
17404235c00SAndy Lutomirski 
test_ptrace_syscall_restart(void)17504235c00SAndy Lutomirski static void test_ptrace_syscall_restart(void)
17604235c00SAndy Lutomirski {
17704235c00SAndy Lutomirski 	printf("[RUN]\tptrace-induced syscall restart\n");
17804235c00SAndy Lutomirski 	pid_t chld = fork();
17904235c00SAndy Lutomirski 	if (chld < 0)
18004235c00SAndy Lutomirski 		err(1, "fork");
18104235c00SAndy Lutomirski 
18204235c00SAndy Lutomirski 	if (chld == 0) {
18304235c00SAndy Lutomirski 		if (ptrace(PTRACE_TRACEME, 0, 0, 0) != 0)
18404235c00SAndy Lutomirski 			err(1, "PTRACE_TRACEME");
18504235c00SAndy Lutomirski 
1864b0b37d4SAndy Lutomirski 		pid_t pid = getpid(), tid = syscall(SYS_gettid);
1874b0b37d4SAndy Lutomirski 
18804235c00SAndy Lutomirski 		printf("\tChild will make one syscall\n");
1894b0b37d4SAndy Lutomirski 		syscall(SYS_tgkill, pid, tid, SIGSTOP);
19004235c00SAndy Lutomirski 
19104235c00SAndy Lutomirski 		syscall(SYS_gettid, 10, 11, 12, 13, 14, 15);
19204235c00SAndy Lutomirski 		_exit(0);
19304235c00SAndy Lutomirski 	}
19404235c00SAndy Lutomirski 
19504235c00SAndy Lutomirski 	int status;
19604235c00SAndy Lutomirski 
19704235c00SAndy Lutomirski 	/* Wait for SIGSTOP. */
19804235c00SAndy Lutomirski 	if (waitpid(chld, &status, 0) != chld || !WIFSTOPPED(status))
19904235c00SAndy Lutomirski 		err(1, "waitpid");
20004235c00SAndy Lutomirski 
20104235c00SAndy Lutomirski 	struct user_regs_struct regs;
20204235c00SAndy Lutomirski 
20304235c00SAndy Lutomirski 	printf("[RUN]\tSYSEMU\n");
20404235c00SAndy Lutomirski 	if (ptrace(PTRACE_SYSEMU, chld, 0, 0) != 0)
205adcfd23eSAndy Lutomirski 		err(1, "PTRACE_SYSEMU");
20604235c00SAndy Lutomirski 	wait_trap(chld);
20704235c00SAndy Lutomirski 
20804235c00SAndy Lutomirski 	if (ptrace(PTRACE_GETREGS, chld, 0, &regs) != 0)
20904235c00SAndy Lutomirski 		err(1, "PTRACE_GETREGS");
21004235c00SAndy Lutomirski 
21104235c00SAndy Lutomirski 	if (regs.user_syscall_nr != SYS_gettid ||
21204235c00SAndy Lutomirski 	    regs.user_arg0 != 10 || regs.user_arg1 != 11 ||
21304235c00SAndy Lutomirski 	    regs.user_arg2 != 12 || regs.user_arg3 != 13 ||
21404235c00SAndy Lutomirski 	    regs.user_arg4 != 14 || regs.user_arg5 != 15) {
21504235c00SAndy Lutomirski 		printf("[FAIL]\tInitial args are wrong (nr=%lu, args=%lu %lu %lu %lu %lu %lu)\n", (unsigned long)regs.user_syscall_nr, (unsigned long)regs.user_arg0, (unsigned long)regs.user_arg1, (unsigned long)regs.user_arg2, (unsigned long)regs.user_arg3, (unsigned long)regs.user_arg4, (unsigned long)regs.user_arg5);
21604235c00SAndy Lutomirski 		nerrs++;
21704235c00SAndy Lutomirski 	} else {
21804235c00SAndy Lutomirski 		printf("[OK]\tInitial nr and args are correct\n");
21904235c00SAndy Lutomirski 	}
22004235c00SAndy Lutomirski 
22104235c00SAndy Lutomirski 	printf("[RUN]\tRestart the syscall (ip = 0x%lx)\n",
22204235c00SAndy Lutomirski 	       (unsigned long)regs.user_ip);
22304235c00SAndy Lutomirski 
22404235c00SAndy Lutomirski 	/*
22504235c00SAndy Lutomirski 	 * This does exactly what it appears to do if syscall is int80 or
22604235c00SAndy Lutomirski 	 * SYSCALL64.  For SYSCALL32 or SYSENTER, though, this is highly
22704235c00SAndy Lutomirski 	 * magical.  It needs to work so that ptrace and syscall restart
22804235c00SAndy Lutomirski 	 * work as expected.
22904235c00SAndy Lutomirski 	 */
23004235c00SAndy Lutomirski 	regs.user_ax = regs.user_syscall_nr;
23104235c00SAndy Lutomirski 	regs.user_ip -= 2;
23204235c00SAndy Lutomirski 	if (ptrace(PTRACE_SETREGS, chld, 0, &regs) != 0)
23304235c00SAndy Lutomirski 		err(1, "PTRACE_SETREGS");
23404235c00SAndy Lutomirski 
23504235c00SAndy Lutomirski 	if (ptrace(PTRACE_SYSEMU, chld, 0, 0) != 0)
236adcfd23eSAndy Lutomirski 		err(1, "PTRACE_SYSEMU");
23704235c00SAndy Lutomirski 	wait_trap(chld);
23804235c00SAndy Lutomirski 
23904235c00SAndy Lutomirski 	if (ptrace(PTRACE_GETREGS, chld, 0, &regs) != 0)
24004235c00SAndy Lutomirski 		err(1, "PTRACE_GETREGS");
24104235c00SAndy Lutomirski 
24204235c00SAndy Lutomirski 	if (regs.user_syscall_nr != SYS_gettid ||
24304235c00SAndy Lutomirski 	    regs.user_arg0 != 10 || regs.user_arg1 != 11 ||
24404235c00SAndy Lutomirski 	    regs.user_arg2 != 12 || regs.user_arg3 != 13 ||
24504235c00SAndy Lutomirski 	    regs.user_arg4 != 14 || regs.user_arg5 != 15) {
24604235c00SAndy Lutomirski 		printf("[FAIL]\tRestart nr or args are wrong (nr=%lu, args=%lu %lu %lu %lu %lu %lu)\n", (unsigned long)regs.user_syscall_nr, (unsigned long)regs.user_arg0, (unsigned long)regs.user_arg1, (unsigned long)regs.user_arg2, (unsigned long)regs.user_arg3, (unsigned long)regs.user_arg4, (unsigned long)regs.user_arg5);
24704235c00SAndy Lutomirski 		nerrs++;
24804235c00SAndy Lutomirski 	} else {
24904235c00SAndy Lutomirski 		printf("[OK]\tRestarted nr and args are correct\n");
25004235c00SAndy Lutomirski 	}
25104235c00SAndy Lutomirski 
25204235c00SAndy Lutomirski 	printf("[RUN]\tChange nr and args and restart the syscall (ip = 0x%lx)\n",
25304235c00SAndy Lutomirski 	       (unsigned long)regs.user_ip);
25404235c00SAndy Lutomirski 
25504235c00SAndy Lutomirski 	regs.user_ax = SYS_getpid;
25604235c00SAndy Lutomirski 	regs.user_arg0 = 20;
25704235c00SAndy Lutomirski 	regs.user_arg1 = 21;
25804235c00SAndy Lutomirski 	regs.user_arg2 = 22;
25904235c00SAndy Lutomirski 	regs.user_arg3 = 23;
26004235c00SAndy Lutomirski 	regs.user_arg4 = 24;
26104235c00SAndy Lutomirski 	regs.user_arg5 = 25;
26204235c00SAndy Lutomirski 	regs.user_ip -= 2;
26304235c00SAndy Lutomirski 
26404235c00SAndy Lutomirski 	if (ptrace(PTRACE_SETREGS, chld, 0, &regs) != 0)
26504235c00SAndy Lutomirski 		err(1, "PTRACE_SETREGS");
26604235c00SAndy Lutomirski 
26704235c00SAndy Lutomirski 	if (ptrace(PTRACE_SYSEMU, chld, 0, 0) != 0)
268adcfd23eSAndy Lutomirski 		err(1, "PTRACE_SYSEMU");
26904235c00SAndy Lutomirski 	wait_trap(chld);
27004235c00SAndy Lutomirski 
27104235c00SAndy Lutomirski 	if (ptrace(PTRACE_GETREGS, chld, 0, &regs) != 0)
27204235c00SAndy Lutomirski 		err(1, "PTRACE_GETREGS");
27304235c00SAndy Lutomirski 
27404235c00SAndy Lutomirski 	if (regs.user_syscall_nr != SYS_getpid ||
27504235c00SAndy Lutomirski 	    regs.user_arg0 != 20 || regs.user_arg1 != 21 || regs.user_arg2 != 22 ||
27604235c00SAndy Lutomirski 	    regs.user_arg3 != 23 || regs.user_arg4 != 24 || regs.user_arg5 != 25) {
27704235c00SAndy Lutomirski 		printf("[FAIL]\tRestart nr or args are wrong (nr=%lu, args=%lu %lu %lu %lu %lu %lu)\n", (unsigned long)regs.user_syscall_nr, (unsigned long)regs.user_arg0, (unsigned long)regs.user_arg1, (unsigned long)regs.user_arg2, (unsigned long)regs.user_arg3, (unsigned long)regs.user_arg4, (unsigned long)regs.user_arg5);
27804235c00SAndy Lutomirski 		nerrs++;
27904235c00SAndy Lutomirski 	} else {
28004235c00SAndy Lutomirski 		printf("[OK]\tReplacement nr and args are correct\n");
28104235c00SAndy Lutomirski 	}
28204235c00SAndy Lutomirski 
28304235c00SAndy Lutomirski 	if (ptrace(PTRACE_CONT, chld, 0, 0) != 0)
28404235c00SAndy Lutomirski 		err(1, "PTRACE_CONT");
28504235c00SAndy Lutomirski 	if (waitpid(chld, &status, 0) != chld)
28604235c00SAndy Lutomirski 		err(1, "waitpid");
28704235c00SAndy Lutomirski 	if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) {
28804235c00SAndy Lutomirski 		printf("[FAIL]\tChild failed\n");
28904235c00SAndy Lutomirski 		nerrs++;
29004235c00SAndy Lutomirski 	} else {
29104235c00SAndy Lutomirski 		printf("[OK]\tChild exited cleanly\n");
29204235c00SAndy Lutomirski 	}
29304235c00SAndy Lutomirski }
29404235c00SAndy Lutomirski 
test_restart_under_ptrace(void)29540361343SAndy Lutomirski static void test_restart_under_ptrace(void)
29640361343SAndy Lutomirski {
29740361343SAndy Lutomirski 	printf("[RUN]\tkernel syscall restart under ptrace\n");
29840361343SAndy Lutomirski 	pid_t chld = fork();
29940361343SAndy Lutomirski 	if (chld < 0)
30040361343SAndy Lutomirski 		err(1, "fork");
30140361343SAndy Lutomirski 
30240361343SAndy Lutomirski 	if (chld == 0) {
30340361343SAndy Lutomirski 		if (ptrace(PTRACE_TRACEME, 0, 0, 0) != 0)
30440361343SAndy Lutomirski 			err(1, "PTRACE_TRACEME");
30540361343SAndy Lutomirski 
3064b0b37d4SAndy Lutomirski 		pid_t pid = getpid(), tid = syscall(SYS_gettid);
3074b0b37d4SAndy Lutomirski 
30840361343SAndy Lutomirski 		printf("\tChild will take a nap until signaled\n");
30940361343SAndy Lutomirski 		setsigign(SIGUSR1, SA_RESTART);
3104b0b37d4SAndy Lutomirski 		syscall(SYS_tgkill, pid, tid, SIGSTOP);
31140361343SAndy Lutomirski 
31240361343SAndy Lutomirski 		syscall(SYS_pause, 0, 0, 0, 0, 0, 0);
31340361343SAndy Lutomirski 		_exit(0);
31440361343SAndy Lutomirski 	}
31540361343SAndy Lutomirski 
31640361343SAndy Lutomirski 	int status;
31740361343SAndy Lutomirski 
31840361343SAndy Lutomirski 	/* Wait for SIGSTOP. */
31940361343SAndy Lutomirski 	if (waitpid(chld, &status, 0) != chld || !WIFSTOPPED(status))
32040361343SAndy Lutomirski 		err(1, "waitpid");
32140361343SAndy Lutomirski 
32240361343SAndy Lutomirski 	struct user_regs_struct regs;
32340361343SAndy Lutomirski 
32440361343SAndy Lutomirski 	printf("[RUN]\tSYSCALL\n");
32540361343SAndy Lutomirski 	if (ptrace(PTRACE_SYSCALL, chld, 0, 0) != 0)
32640361343SAndy Lutomirski 		err(1, "PTRACE_SYSCALL");
32740361343SAndy Lutomirski 	wait_trap(chld);
32840361343SAndy Lutomirski 
32940361343SAndy Lutomirski 	/* We should be stopped at pause(2) entry. */
33040361343SAndy Lutomirski 
33140361343SAndy Lutomirski 	if (ptrace(PTRACE_GETREGS, chld, 0, &regs) != 0)
33240361343SAndy Lutomirski 		err(1, "PTRACE_GETREGS");
33340361343SAndy Lutomirski 
33440361343SAndy Lutomirski 	if (regs.user_syscall_nr != SYS_pause ||
33540361343SAndy Lutomirski 	    regs.user_arg0 != 0 || regs.user_arg1 != 0 ||
33640361343SAndy Lutomirski 	    regs.user_arg2 != 0 || regs.user_arg3 != 0 ||
33740361343SAndy Lutomirski 	    regs.user_arg4 != 0 || regs.user_arg5 != 0) {
33840361343SAndy Lutomirski 		printf("[FAIL]\tInitial args are wrong (nr=%lu, args=%lu %lu %lu %lu %lu %lu)\n", (unsigned long)regs.user_syscall_nr, (unsigned long)regs.user_arg0, (unsigned long)regs.user_arg1, (unsigned long)regs.user_arg2, (unsigned long)regs.user_arg3, (unsigned long)regs.user_arg4, (unsigned long)regs.user_arg5);
33940361343SAndy Lutomirski 		nerrs++;
34040361343SAndy Lutomirski 	} else {
34140361343SAndy Lutomirski 		printf("[OK]\tInitial nr and args are correct\n");
34240361343SAndy Lutomirski 	}
34340361343SAndy Lutomirski 
34440361343SAndy Lutomirski 	/* Interrupt it. */
34540361343SAndy Lutomirski 	kill(chld, SIGUSR1);
34640361343SAndy Lutomirski 
34740361343SAndy Lutomirski 	/* Advance.  We should be stopped at exit. */
34840361343SAndy Lutomirski 	printf("[RUN]\tSYSCALL\n");
34940361343SAndy Lutomirski 	if (ptrace(PTRACE_SYSCALL, chld, 0, 0) != 0)
35040361343SAndy Lutomirski 		err(1, "PTRACE_SYSCALL");
35140361343SAndy Lutomirski 	wait_trap(chld);
35240361343SAndy Lutomirski 
35340361343SAndy Lutomirski 	if (ptrace(PTRACE_GETREGS, chld, 0, &regs) != 0)
35440361343SAndy Lutomirski 		err(1, "PTRACE_GETREGS");
35540361343SAndy Lutomirski 
35640361343SAndy Lutomirski 	if (regs.user_syscall_nr != SYS_pause ||
35740361343SAndy Lutomirski 	    regs.user_arg0 != 0 || regs.user_arg1 != 0 ||
35840361343SAndy Lutomirski 	    regs.user_arg2 != 0 || regs.user_arg3 != 0 ||
35940361343SAndy Lutomirski 	    regs.user_arg4 != 0 || regs.user_arg5 != 0) {
36040361343SAndy Lutomirski 		printf("[FAIL]\tArgs after SIGUSR1 are wrong (nr=%lu, args=%lu %lu %lu %lu %lu %lu)\n", (unsigned long)regs.user_syscall_nr, (unsigned long)regs.user_arg0, (unsigned long)regs.user_arg1, (unsigned long)regs.user_arg2, (unsigned long)regs.user_arg3, (unsigned long)regs.user_arg4, (unsigned long)regs.user_arg5);
36140361343SAndy Lutomirski 		nerrs++;
36240361343SAndy Lutomirski 	} else {
36340361343SAndy Lutomirski 		printf("[OK]\tArgs after SIGUSR1 are correct (ax = %ld)\n",
36440361343SAndy Lutomirski 		       (long)regs.user_ax);
36540361343SAndy Lutomirski 	}
36640361343SAndy Lutomirski 
36740361343SAndy Lutomirski 	/* Poke the regs back in.  This must not break anything. */
36840361343SAndy Lutomirski 	if (ptrace(PTRACE_SETREGS, chld, 0, &regs) != 0)
36940361343SAndy Lutomirski 		err(1, "PTRACE_SETREGS");
37040361343SAndy Lutomirski 
37140361343SAndy Lutomirski 	/* Catch the (ignored) SIGUSR1. */
37240361343SAndy Lutomirski 	if (ptrace(PTRACE_CONT, chld, 0, 0) != 0)
37340361343SAndy Lutomirski 		err(1, "PTRACE_CONT");
37440361343SAndy Lutomirski 	if (waitpid(chld, &status, 0) != chld)
37540361343SAndy Lutomirski 		err(1, "waitpid");
37640361343SAndy Lutomirski 	if (!WIFSTOPPED(status)) {
37740361343SAndy Lutomirski 		printf("[FAIL]\tChild was stopped for SIGUSR1 (status = 0x%x)\n", status);
37840361343SAndy Lutomirski 		nerrs++;
37940361343SAndy Lutomirski 	} else {
38040361343SAndy Lutomirski 		printf("[OK]\tChild got SIGUSR1\n");
38140361343SAndy Lutomirski 	}
38240361343SAndy Lutomirski 
38340361343SAndy Lutomirski 	/* The next event should be pause(2) again. */
38440361343SAndy Lutomirski 	printf("[RUN]\tStep again\n");
38540361343SAndy Lutomirski 	if (ptrace(PTRACE_SYSCALL, chld, 0, 0) != 0)
38640361343SAndy Lutomirski 		err(1, "PTRACE_SYSCALL");
38740361343SAndy Lutomirski 	wait_trap(chld);
38840361343SAndy Lutomirski 
38940361343SAndy Lutomirski 	/* We should be stopped at pause(2) entry. */
39040361343SAndy Lutomirski 
39140361343SAndy Lutomirski 	if (ptrace(PTRACE_GETREGS, chld, 0, &regs) != 0)
39240361343SAndy Lutomirski 		err(1, "PTRACE_GETREGS");
39340361343SAndy Lutomirski 
39440361343SAndy Lutomirski 	if (regs.user_syscall_nr != SYS_pause ||
39540361343SAndy Lutomirski 	    regs.user_arg0 != 0 || regs.user_arg1 != 0 ||
39640361343SAndy Lutomirski 	    regs.user_arg2 != 0 || regs.user_arg3 != 0 ||
39740361343SAndy Lutomirski 	    regs.user_arg4 != 0 || regs.user_arg5 != 0) {
39840361343SAndy Lutomirski 		printf("[FAIL]\tpause did not restart (nr=%lu, args=%lu %lu %lu %lu %lu %lu)\n", (unsigned long)regs.user_syscall_nr, (unsigned long)regs.user_arg0, (unsigned long)regs.user_arg1, (unsigned long)regs.user_arg2, (unsigned long)regs.user_arg3, (unsigned long)regs.user_arg4, (unsigned long)regs.user_arg5);
39940361343SAndy Lutomirski 		nerrs++;
40040361343SAndy Lutomirski 	} else {
40140361343SAndy Lutomirski 		printf("[OK]\tpause(2) restarted correctly\n");
40240361343SAndy Lutomirski 	}
40340361343SAndy Lutomirski 
40440361343SAndy Lutomirski 	/* Kill it. */
40540361343SAndy Lutomirski 	kill(chld, SIGKILL);
40640361343SAndy Lutomirski 	if (waitpid(chld, &status, 0) != chld)
40740361343SAndy Lutomirski 		err(1, "waitpid");
40840361343SAndy Lutomirski }
40940361343SAndy Lutomirski 
main()41004235c00SAndy Lutomirski int main()
41104235c00SAndy Lutomirski {
41204235c00SAndy Lutomirski 	printf("[RUN]\tCheck int80 return regs\n");
41304235c00SAndy Lutomirski 	test_sys32_regs(do_full_int80);
41404235c00SAndy Lutomirski 
41504235c00SAndy Lutomirski #if defined(__i386__) && (!defined(__GLIBC__) || __GLIBC__ > 2 || __GLIBC_MINOR__ >= 16)
41604235c00SAndy Lutomirski 	vsyscall32 = (void *)getauxval(AT_SYSINFO);
417*630b99abSAndy Lutomirski 	if (vsyscall32) {
41804235c00SAndy Lutomirski 		printf("[RUN]\tCheck AT_SYSINFO return regs\n");
41904235c00SAndy Lutomirski 		test_sys32_regs(do_full_vsyscall32);
420*630b99abSAndy Lutomirski 	} else {
421*630b99abSAndy Lutomirski 		printf("[SKIP]\tAT_SYSINFO is not available\n");
422*630b99abSAndy Lutomirski 	}
42304235c00SAndy Lutomirski #endif
42404235c00SAndy Lutomirski 
42504235c00SAndy Lutomirski 	test_ptrace_syscall_restart();
42604235c00SAndy Lutomirski 
42740361343SAndy Lutomirski 	test_restart_under_ptrace();
42840361343SAndy Lutomirski 
42904235c00SAndy Lutomirski 	return 0;
43004235c00SAndy Lutomirski }
431