From: Jeff Dike From: Bodo Stroesser Patch 1/3 to implement usage of PTRACE_O_TRACESYSGOOD This is necessary, to fix UMLs bad behavior when a process does a systemcall with syscall-number less than 0. Insert a check for availability and function of ptrace(PTRACE_SETOPTIONS,,,PTRACE_O_TRACESYSGOOD) into the normal ptrace checks at startup. Patch 2/3 to implement usage of PTRACE_O_TRACESYSGOOD This is necessary, to fix UMLs bad behavior when a process does a systemcall with syscall-number less than 0. This patch makes SKAS-mode use PTRACE_O_TRACESYSGOOD and fixes the problems in SKAS. Patch 3/3 to implement usage of PTRACE_O_TRACESYSGOOD This is necessary, to fix UMLs bad behavior when a process does a systemcall with syscall-number less than 0. This patch makes TT-mode use PTRACE_O_TRACESYSGOOD and fixes the problems in TT. I'm not quite sure, that this patch doesn't cause problems with debugger usage. It should be testet by someone, who has more know how about TT-mode debugger. Signed-off-by: Bodo Stroesser Signed-off-by: Jeff Dike Signed-off-by: Andrew Morton --- 25-akpm/arch/um/kernel/process.c | 8 ++++++-- 25-akpm/arch/um/kernel/skas/process.c | 22 +++++++++++++--------- 25-akpm/arch/um/kernel/tt/exec_user.c | 4 ++++ 25-akpm/arch/um/kernel/tt/include/tt.h | 1 + 25-akpm/arch/um/kernel/tt/syscall_user.c | 14 ++++++-------- 25-akpm/arch/um/kernel/tt/tracer.c | 21 ++++++++++++++++++--- 6 files changed, 48 insertions(+), 22 deletions(-) diff -puN arch/um/kernel/process.c~uml-add-tracesysgood-support arch/um/kernel/process.c --- 25/arch/um/kernel/process.c~uml-add-tracesysgood-support Fri Dec 3 13:50:30 2004 +++ 25-akpm/arch/um/kernel/process.c Fri Dec 3 13:50:30 2004 @@ -13,6 +13,7 @@ #include #include #include +#include #include #include #include @@ -285,6 +286,9 @@ void __init check_ptrace(void) printk("Checking that ptrace can change system call numbers..."); pid = start_ptraced_child(&stack); + if(ptrace(PTRACE_SETOPTIONS, pid, 0, (void *)PTRACE_O_TRACESYSGOOD) < 0) + panic("check_ptrace: PTRACE_SETOPTIONS failed, errno = %d", errno); + while(1){ if(ptrace(PTRACE_SYSCALL, pid, 0, 0) < 0) panic("check_ptrace : ptrace failed, errno = %d", @@ -292,8 +296,8 @@ void __init check_ptrace(void) CATCH_EINTR(n = waitpid(pid, &status, WUNTRACED)); if(n < 0) panic("check_ptrace : wait failed, errno = %d", errno); - if(!WIFSTOPPED(status) || (WSTOPSIG(status) != SIGTRAP)) - panic("check_ptrace : expected SIGTRAP, " + if(!WIFSTOPPED(status) || (WSTOPSIG(status) != (SIGTRAP + 0x80))) + panic("check_ptrace : expected SIGTRAP + 0x80, " "got status = %d", status); syscall = ptrace(PTRACE_PEEKUSER, pid, PT_SYSCALL_NR_OFFSET, diff -puN arch/um/kernel/skas/process.c~uml-add-tracesysgood-support arch/um/kernel/skas/process.c --- 25/arch/um/kernel/skas/process.c~uml-add-tracesysgood-support Fri Dec 3 13:50:30 2004 +++ 25-akpm/arch/um/kernel/skas/process.c Fri Dec 3 13:50:30 2004 @@ -11,6 +11,7 @@ #include #include #include +#include #include #include #include @@ -60,14 +61,9 @@ static void handle_segv(int pid) /*To use the same value of using_sysemu as the caller, ask it that value (in local_using_sysemu)*/ static void handle_trap(int pid, union uml_pt_regs *regs, int local_using_sysemu) { - int err, syscall_nr, status; + int err, status; - syscall_nr = PT_SYSCALL_NR(regs->skas.regs); - UPT_SYSCALL_NR(regs) = syscall_nr; - if(syscall_nr < 0){ - relay_signal(SIGTRAP, regs); - return; - } + UPT_SYSCALL_NR(regs) = PT_SYSCALL_NR(regs->skas.regs); if (!local_using_sysemu) { @@ -82,7 +78,7 @@ static void handle_trap(int pid, union u "errno = %d\n", errno); CATCH_EINTR(err = waitpid(pid, &status, WUNTRACED)); - if((err < 0) || !WIFSTOPPED(status) || (WSTOPSIG(status) != SIGTRAP)) + if((err < 0) || !WIFSTOPPED(status) || (WSTOPSIG(status) != (SIGTRAP + 0x80))) panic("handle_trap - failed to wait at end of syscall, " "errno = %d, status = %d\n", errno, status); } @@ -131,6 +127,10 @@ void start_userspace(int cpu) panic("start_userspace : expected SIGSTOP, got status = %d", status); + if (ptrace(PTRACE_SETOPTIONS, pid, NULL, (void *)PTRACE_O_TRACESYSGOOD) < 0) + panic("start_userspace : PTRACE_SETOPTIONS failed, errno=%d\n", + errno); + if(munmap(stack, PAGE_SIZE) < 0) panic("start_userspace : munmap failed, errno = %d\n", errno); @@ -166,9 +166,13 @@ void userspace(union uml_pt_regs *regs) case SIGSEGV: handle_segv(pid); break; - case SIGTRAP: + case SIGTRAP + 0x80: handle_trap(pid, regs, local_using_sysemu); break; + case SIGTRAP: + UPT_SYSCALL_NR(regs) = -1; + relay_signal(SIGTRAP, regs); + break; case SIGIO: case SIGVTALRM: case SIGILL: diff -puN arch/um/kernel/tt/exec_user.c~uml-add-tracesysgood-support arch/um/kernel/tt/exec_user.c --- 25/arch/um/kernel/tt/exec_user.c~uml-add-tracesysgood-support Fri Dec 3 13:50:30 2004 +++ 25-akpm/arch/um/kernel/tt/exec_user.c Fri Dec 3 13:50:30 2004 @@ -10,6 +10,7 @@ #include #include #include +#include #include #include "user_util.h" #include "kern_util.h" @@ -37,6 +38,9 @@ void do_exec(int old_pid, int new_pid) kill(old_pid, SIGKILL); + if (ptrace(PTRACE_SETOPTIONS, new_pid, 0, (void *)PTRACE_O_TRACESYSGOOD) < 0) + tracer_panic("do_exec: PTRACE_SETOPTIONS failed, errno = %d", errno); + if(ptrace_setregs(new_pid, regs) < 0) tracer_panic("do_exec failed to start new proc - errno = %d", errno); diff -puN arch/um/kernel/tt/include/tt.h~uml-add-tracesysgood-support arch/um/kernel/tt/include/tt.h --- 25/arch/um/kernel/tt/include/tt.h~uml-add-tracesysgood-support Fri Dec 3 13:50:30 2004 +++ 25-akpm/arch/um/kernel/tt/include/tt.h Fri Dec 3 13:50:30 2004 @@ -27,6 +27,7 @@ extern int is_tracing(void *task); extern void syscall_handler(int sig, union uml_pt_regs *regs); extern void exit_kernel(int pid, void *task); extern int do_syscall(void *task, int pid, int local_using_sysemu); +extern void do_sigtrap(void *task); extern int is_valid_pid(int pid); extern void remap_data(void *segment_start, void *segment_end, int w); diff -puN arch/um/kernel/tt/syscall_user.c~uml-add-tracesysgood-support arch/um/kernel/tt/syscall_user.c --- 25/arch/um/kernel/tt/syscall_user.c~uml-add-tracesysgood-support Fri Dec 3 13:50:30 2004 +++ 25-akpm/arch/um/kernel/tt/syscall_user.c Fri Dec 3 13:50:30 2004 @@ -43,21 +43,19 @@ void syscall_handler_tt(int sig, union u record_syscall_end(index, result); } +void do_sigtrap(void *task) +{ + UPT_SYSCALL_NR(TASK_REGS(task)) = -1; +} + int do_syscall(void *task, int pid, int local_using_sysemu) { unsigned long proc_regs[FRAME_SIZE]; - union uml_pt_regs *regs; - int syscall; if(ptrace_getregs(pid, proc_regs) < 0) tracer_panic("Couldn't read registers"); - syscall = PT_SYSCALL_NR(proc_regs); - - regs = TASK_REGS(task); - UPT_SYSCALL_NR(regs) = syscall; - if(syscall < 0) - return(0); + UPT_SYSCALL_NR(TASK_REGS(task)) = PT_SYSCALL_NR(proc_regs); if((syscall != __NR_sigreturn) && ((unsigned long *) PT_IP(proc_regs) >= &_stext) && diff -puN arch/um/kernel/tt/tracer.c~uml-add-tracesysgood-support arch/um/kernel/tt/tracer.c --- 25/arch/um/kernel/tt/tracer.c~uml-add-tracesysgood-support Fri Dec 3 13:50:30 2004 +++ 25-akpm/arch/um/kernel/tt/tracer.c Fri Dec 3 13:50:30 2004 @@ -13,6 +13,7 @@ #include #include #include +#include #include #include #include "user.h" @@ -71,6 +72,8 @@ void attach_process(int pid) (ptrace(PTRACE_CONT, pid, 0, 0) < 0)) tracer_panic("OP_FORK failed to attach pid"); wait_for_stop(pid, SIGSTOP, PTRACE_CONT, NULL); + if (ptrace(PTRACE_SETOPTIONS, pid, 0, (void *)PTRACE_O_TRACESYSGOOD) < 0) + tracer_panic("OP_FORK: PTRACE_SETOPTIONS failed, errno = %d", errno); if(ptrace(PTRACE_CONT, pid, 0, 0) < 0) tracer_panic("OP_FORK failed to continue process"); } @@ -140,7 +143,7 @@ static void sleeping_process_signal(int * any more, the trace of those will land here. So, we need to just * PTRACE_SYSCALL it. */ - case SIGTRAP: + case (SIGTRAP + 0x80): if(ptrace(PTRACE_SYSCALL, pid, 0, 0) < 0) tracer_panic("sleeping_process_signal : Failed to " "PTRACE_SYSCALL pid %d, errno = %d\n", @@ -196,6 +199,10 @@ int tracer(int (*init_proc)(void *), voi printf("waitpid on idle thread failed, errno = %d\n", errno); exit(1); } + if (ptrace(PTRACE_SETOPTIONS, pid, 0, (void *)PTRACE_O_TRACESYSGOOD) < 0) { + printf("Failed to PTRACE_SETOPTIONS for idle thread, errno = %d\n", errno); + exit(1); + } if((ptrace(PTRACE_CONT, pid, 0, 0) < 0)){ printf("Failed to continue idle thread, errno = %d\n", errno); exit(1); @@ -326,14 +333,22 @@ int tracer(int (*init_proc)(void *), voi */ pid = cpu_tasks[proc_id].pid; break; + case (SIGTRAP + 0x80): + if(!tracing && (debugger_pid != -1)){ + child_signal(pid, status & 0x7fff); + continue; + } + tracing = 0; + do_syscall(task, pid, local_using_sysemu); + sig = SIGUSR2; + break; case SIGTRAP: if(!tracing && (debugger_pid != -1)){ child_signal(pid, status); continue; } tracing = 0; - if(do_syscall(task, pid, local_using_sysemu)) - sig = SIGUSR2; + do_sigtrap(task); break; case SIGPROF: if(tracing) sig = 0; _