228 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			228 lines
		
	
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // 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");
 | |
| }
 |