/* * Copyright (C) 2002 Jeff Dike (jdike@karaya.com) * Licensed under the GPL */ #include <errno.h> #include <asm/sigcontext.h> #include <sys/ptrace.h> #include <linux/ptrace.h> #include "sysdep/ptrace.h" #include "sysdep/ptrace_user.h" #include "kern_util.h" #include "user.h" #include "sigcontext.h" extern int userspace_pid; int copy_sc_from_user_skas(struct uml_pt_regs *regs, void *from_ptr) { struct sigcontext sc, *from = from_ptr; unsigned long fpregs[FP_FRAME_SIZE]; int err; err = copy_from_user_proc(&sc, from, sizeof(sc)); err |= copy_from_user_proc(fpregs, sc.fpstate, sizeof(fpregs)); if(err) return(err); regs->mode.skas.regs[GS] = sc.gs; regs->mode.skas.regs[FS] = sc.fs; regs->mode.skas.regs[ES] = sc.es; regs->mode.skas.regs[DS] = sc.ds; regs->mode.skas.regs[EDI] = sc.edi; regs->mode.skas.regs[ESI] = sc.esi; regs->mode.skas.regs[EBP] = sc.ebp; regs->mode.skas.regs[UESP] = sc.esp; regs->mode.skas.regs[EBX] = sc.ebx; regs->mode.skas.regs[EDX] = sc.edx; regs->mode.skas.regs[ECX] = sc.ecx; regs->mode.skas.regs[EAX] = sc.eax; regs->mode.skas.regs[EIP] = sc.eip; regs->mode.skas.regs[CS] = sc.cs; regs->mode.skas.regs[EFL] = sc.eflags; regs->mode.skas.regs[UESP] = sc.esp_at_signal; regs->mode.skas.regs[SS] = sc.ss; regs->mode.skas.fault_addr = sc.cr2; regs->mode.skas.fault_type = FAULT_WRITE(sc.err); regs->mode.skas.trap_type = sc.trapno; err = ptrace(PTRACE_SETFPREGS, userspace_pid, 0, fpregs); if(err < 0){ printk("copy_sc_to_user - PTRACE_SETFPREGS failed, " "errno = %d\n", errno); return(1); } return(0); } int copy_sc_to_user_skas(void *to_ptr, struct uml_pt_regs *regs, unsigned long fault_addr, int fault_type) { struct sigcontext sc, *to = to_ptr; struct _fpstate *to_fp; unsigned long fpregs[FP_FRAME_SIZE]; int err; sc.gs = regs->mode.skas.regs[GS]; sc.fs = regs->mode.skas.regs[FS]; sc.es = regs->mode.skas.regs[ES]; sc.ds = regs->mode.skas.regs[DS]; sc.edi = regs->mode.skas.regs[EDI]; sc.esi = regs->mode.skas.regs[ESI]; sc.ebp = regs->mode.skas.regs[EBP]; sc.esp = regs->mode.skas.regs[UESP]; sc.ebx = regs->mode.skas.regs[EBX]; sc.edx = regs->mode.skas.regs[EDX]; sc.ecx = regs->mode.skas.regs[ECX]; sc.eax = regs->mode.skas.regs[EAX]; sc.eip = regs->mode.skas.regs[EIP]; sc.cs = regs->mode.skas.regs[CS]; sc.eflags = regs->mode.skas.regs[EFL]; sc.esp_at_signal = regs->mode.skas.regs[UESP]; sc.ss = regs->mode.skas.regs[SS]; sc.cr2 = fault_addr; sc.err = TO_SC_ERR(fault_type); sc.trapno = regs->mode.skas.trap_type; err = ptrace(PTRACE_GETFPREGS, userspace_pid, 0, fpregs); if(err < 0){ printk("copy_sc_to_user - PTRACE_GETFPREGS failed, " "errno = %d\n", errno); return(1); } to_fp = (struct _fpstate *)((unsigned long) to + sizeof(*to)); sc.fpstate = to_fp; if(err) return(err); return(copy_to_user_proc(to, &sc, sizeof(sc)) || copy_to_user_proc(to_fp, fpregs, sizeof(fpregs))); } /* * Overrides for Emacs so that we follow Linus's tabbing style. * Emacs will notice this stuff at the end of the file and automatically * adjust the settings for this buffer only. This must remain at the end * of the file. * --------------------------------------------------------------------------- * Local variables: * c-file-style: "linux" * End: */