| // SPDX-License-Identifier: GPL-2.0 | 
 | /* | 
 |  * A ptrace test for testing PTRACE_SYSEMU, PTRACE_SETREGS and | 
 |  * PTRACE_GETREG.  This test basically create a child process that executes | 
 |  * syscalls and the parent process check if it is being traced appropriated. | 
 |  * | 
 |  * This test is heavily based on tools/testing/selftests/x86/ptrace_syscall.c | 
 |  * test, and it was adapted to run on Powerpc by | 
 |  * Breno Leitao <leitao@debian.org> | 
 |  */ | 
 | #define _GNU_SOURCE | 
 |  | 
 | #include <sys/ptrace.h> | 
 | #include <sys/types.h> | 
 | #include <sys/wait.h> | 
 | #include <sys/syscall.h> | 
 | #include <sys/user.h> | 
 | #include <unistd.h> | 
 | #include <errno.h> | 
 | #include <stddef.h> | 
 | #include <stdio.h> | 
 | #include <err.h> | 
 | #include <string.h> | 
 | #include <sys/auxv.h> | 
 | #include "utils.h" | 
 |  | 
 | /* Bitness-agnostic defines for user_regs_struct fields. */ | 
 | #define user_syscall_nr	gpr[0] | 
 | #define user_arg0		gpr[3] | 
 | #define user_arg1		gpr[4] | 
 | #define user_arg2		gpr[5] | 
 | #define user_arg3		gpr[6] | 
 | #define user_arg4		gpr[7] | 
 | #define user_arg5		gpr[8] | 
 | #define user_ip		nip | 
 |  | 
 | #define PTRACE_SYSEMU		0x1d | 
 |  | 
 | static int nerrs; | 
 |  | 
 | static void wait_trap(pid_t chld) | 
 | { | 
 | 	siginfo_t si; | 
 |  | 
 | 	if (waitid(P_PID, chld, &si, WEXITED|WSTOPPED) != 0) | 
 | 		err(1, "waitid"); | 
 | 	if (si.si_pid != chld) | 
 | 		errx(1, "got unexpected pid in event\n"); | 
 | 	if (si.si_code != CLD_TRAPPED) | 
 | 		errx(1, "got unexpected event type %d\n", si.si_code); | 
 | } | 
 |  | 
 | static void test_ptrace_syscall_restart(void) | 
 | { | 
 | 	int status; | 
 | 	struct pt_regs regs; | 
 | 	pid_t chld; | 
 |  | 
 | 	printf("[RUN]\tptrace-induced syscall restart\n"); | 
 |  | 
 | 	chld = fork(); | 
 | 	if (chld < 0) | 
 | 		err(1, "fork"); | 
 |  | 
 | 	/* | 
 | 	 * Child process is running 4 syscalls after ptrace. | 
 | 	 * | 
 | 	 * 1) getpid() | 
 | 	 * 2) gettid() | 
 | 	 * 3) tgkill() -> Send SIGSTOP | 
 | 	 * 4) gettid() -> Where the tests will happen essentially | 
 | 	 */ | 
 | 	if (chld == 0) { | 
 | 		if (ptrace(PTRACE_TRACEME, 0, 0, 0) != 0) | 
 | 			err(1, "PTRACE_TRACEME"); | 
 |  | 
 | 		pid_t pid = getpid(), tid = syscall(SYS_gettid); | 
 |  | 
 | 		printf("\tChild will make one syscall\n"); | 
 | 		syscall(SYS_tgkill, pid, tid, SIGSTOP); | 
 |  | 
 | 		syscall(SYS_gettid, 10, 11, 12, 13, 14, 15); | 
 | 		_exit(0); | 
 | 	} | 
 | 	/* Parent process below */ | 
 |  | 
 | 	/* Wait for SIGSTOP sent by tgkill above. */ | 
 | 	if (waitpid(chld, &status, 0) != chld || !WIFSTOPPED(status)) | 
 | 		err(1, "waitpid"); | 
 |  | 
 | 	printf("[RUN]\tSYSEMU\n"); | 
 | 	if (ptrace(PTRACE_SYSEMU, chld, 0, 0) != 0) | 
 | 		err(1, "PTRACE_SYSEMU"); | 
 | 	wait_trap(chld); | 
 |  | 
 | 	if (ptrace(PTRACE_GETREGS, chld, 0, ®s) != 0) | 
 | 		err(1, "PTRACE_GETREGS"); | 
 |  | 
 | 	/* | 
 | 	 * Ptrace trapped prior to executing the syscall, thus r3 still has | 
 | 	 * the syscall number instead of the sys_gettid() result | 
 | 	 */ | 
 | 	if (regs.user_syscall_nr != SYS_gettid || | 
 | 	    regs.user_arg0 != 10 || regs.user_arg1 != 11 || | 
 | 	    regs.user_arg2 != 12 || regs.user_arg3 != 13 || | 
 | 	    regs.user_arg4 != 14 || regs.user_arg5 != 15) { | 
 | 		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); | 
 | 		 nerrs++; | 
 | 	} else { | 
 | 		printf("[OK]\tInitial nr and args are correct\n"); } | 
 |  | 
 | 	printf("[RUN]\tRestart the syscall (ip = 0x%lx)\n", | 
 | 	       (unsigned long)regs.user_ip); | 
 |  | 
 | 	/* | 
 | 	 * Rewind to retry the same syscall again. This will basically test | 
 | 	 * the rewind process together with PTRACE_SETREGS and PTRACE_GETREGS. | 
 | 	 */ | 
 | 	regs.user_ip -= 4; | 
 | 	if (ptrace(PTRACE_SETREGS, chld, 0, ®s) != 0) | 
 | 		err(1, "PTRACE_SETREGS"); | 
 |  | 
 | 	if (ptrace(PTRACE_SYSEMU, chld, 0, 0) != 0) | 
 | 		err(1, "PTRACE_SYSEMU"); | 
 | 	wait_trap(chld); | 
 |  | 
 | 	if (ptrace(PTRACE_GETREGS, chld, 0, ®s) != 0) | 
 | 		err(1, "PTRACE_GETREGS"); | 
 |  | 
 | 	if (regs.user_syscall_nr != SYS_gettid || | 
 | 	    regs.user_arg0 != 10 || regs.user_arg1 != 11 || | 
 | 	    regs.user_arg2 != 12 || regs.user_arg3 != 13 || | 
 | 	    regs.user_arg4 != 14 || regs.user_arg5 != 15) { | 
 | 		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); | 
 | 		nerrs++; | 
 | 	} else { | 
 | 		printf("[OK]\tRestarted nr and args are correct\n"); | 
 | 	} | 
 |  | 
 | 	printf("[RUN]\tChange nr and args and restart the syscall (ip = 0x%lx)\n", | 
 | 	       (unsigned long)regs.user_ip); | 
 |  | 
 | 	/* | 
 | 	 * Inject a new syscall (getpid) in the same place the previous | 
 | 	 * syscall (gettid), rewind and re-execute. | 
 | 	 */ | 
 | 	regs.user_syscall_nr = SYS_getpid; | 
 | 	regs.user_arg0 = 20; | 
 | 	regs.user_arg1 = 21; | 
 | 	regs.user_arg2 = 22; | 
 | 	regs.user_arg3 = 23; | 
 | 	regs.user_arg4 = 24; | 
 | 	regs.user_arg5 = 25; | 
 | 	regs.user_ip -= 4; | 
 |  | 
 | 	if (ptrace(PTRACE_SETREGS, chld, 0, ®s) != 0) | 
 | 		err(1, "PTRACE_SETREGS"); | 
 |  | 
 | 	if (ptrace(PTRACE_SYSEMU, chld, 0, 0) != 0) | 
 | 		err(1, "PTRACE_SYSEMU"); | 
 | 	wait_trap(chld); | 
 |  | 
 | 	if (ptrace(PTRACE_GETREGS, chld, 0, ®s) != 0) | 
 | 		err(1, "PTRACE_GETREGS"); | 
 |  | 
 | 	/* Check that ptrace stopped at the new syscall that was | 
 | 	 * injected, and guarantee that it haven't executed, i.e, user_args | 
 | 	 * contain the arguments and not the syscall return value, for | 
 | 	 * instance. | 
 | 	 */ | 
 | 	if (regs.user_syscall_nr != SYS_getpid | 
 | 		|| regs.user_arg0 != 20 || regs.user_arg1 != 21 | 
 | 		|| regs.user_arg2 != 22 || regs.user_arg3 != 23 | 
 | 		|| regs.user_arg4 != 24 || regs.user_arg5 != 25) { | 
 |  | 
 | 		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); | 
 | 		nerrs++; | 
 | 	} else { | 
 | 		printf("[OK]\tReplacement nr and args are correct\n"); | 
 | 	} | 
 |  | 
 | 	if (ptrace(PTRACE_CONT, chld, 0, 0) != 0) | 
 | 		err(1, "PTRACE_CONT"); | 
 |  | 
 | 	if (waitpid(chld, &status, 0) != chld) | 
 | 		err(1, "waitpid"); | 
 |  | 
 | 	/* Guarantee that the process executed properly, returning 0 */ | 
 | 	if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) { | 
 | 		printf("[FAIL]\tChild failed\n"); | 
 | 		nerrs++; | 
 | 	} else { | 
 | 		printf("[OK]\tChild exited cleanly\n"); | 
 | 	} | 
 | } | 
 |  | 
 | int ptrace_syscall(void) | 
 | { | 
 | 	test_ptrace_syscall_restart(); | 
 |  | 
 | 	return nerrs; | 
 | } | 
 |  | 
 | int main(void) | 
 | { | 
 | 	return test_harness(ptrace_syscall, "ptrace_syscall"); | 
 | } |