ptrace.c
来自「linux 内核源代码」· C语言 代码 · 共 662 行 · 第 1/2 页
C
662 行
/* * PowerPC version * Copyright (C) 1995-1996 Gary Thomas (gdt@linuxppc.org) * * Derived from "arch/m68k/kernel/ptrace.c" * Copyright (C) 1994 by Hamish Macdonald * Taken from linux/kernel/ptrace.c and modified for M680x0. * linux/kernel/ptrace.c is by Ross Biro 1/23/92, edited by Linus Torvalds * * Modified by Cort Dougan (cort@hq.fsmlabs.com) * and Paul Mackerras (paulus@samba.org). * * This file is subject to the terms and conditions of the GNU General * Public License. See the file README.legal in the main directory of * this archive for more details. */#include <linux/kernel.h>#include <linux/sched.h>#include <linux/mm.h>#include <linux/smp.h>#include <linux/errno.h>#include <linux/ptrace.h>#include <linux/user.h>#include <linux/security.h>#include <linux/signal.h>#include <linux/seccomp.h>#include <linux/audit.h>#ifdef CONFIG_PPC32#include <linux/module.h>#endif#include <asm/uaccess.h>#include <asm/page.h>#include <asm/pgtable.h>#include <asm/system.h>/* * does not yet catch signals sent when the child dies. * in exit.c or in signal.c. *//* * Set of msr bits that gdb can change on behalf of a process. */#if defined(CONFIG_40x) || defined(CONFIG_BOOKE)#define MSR_DEBUGCHANGE 0#else#define MSR_DEBUGCHANGE (MSR_SE | MSR_BE)#endif/* * Max register writeable via put_reg */#ifdef CONFIG_PPC32#define PT_MAX_PUT_REG PT_MQ#else#define PT_MAX_PUT_REG PT_CCR#endif/* * Get contents of register REGNO in task TASK. */unsigned long ptrace_get_reg(struct task_struct *task, int regno){ unsigned long tmp = 0; if (task->thread.regs == NULL) return -EIO; if (regno == PT_MSR) { tmp = ((unsigned long *)task->thread.regs)[PT_MSR]; return tmp | task->thread.fpexc_mode; } if (regno < (sizeof(struct pt_regs) / sizeof(unsigned long))) return ((unsigned long *)task->thread.regs)[regno]; return -EIO;}/* * Write contents of register REGNO in task TASK. */int ptrace_put_reg(struct task_struct *task, int regno, unsigned long data){ if (task->thread.regs == NULL) return -EIO; if (regno <= PT_MAX_PUT_REG || regno == PT_TRAP) { if (regno == PT_MSR) data = (data & MSR_DEBUGCHANGE) | (task->thread.regs->msr & ~MSR_DEBUGCHANGE); /* We prevent mucking around with the reserved area of trap * which are used internally by the kernel */ if (regno == PT_TRAP) data &= 0xfff0; ((unsigned long *)task->thread.regs)[regno] = data; return 0; } return -EIO;}static int get_fpregs(void __user *data, struct task_struct *task, int has_fpscr){ unsigned int count = has_fpscr ? 33 : 32; if (copy_to_user(data, task->thread.fpr, count * sizeof(double))) return -EFAULT; return 0;}static int set_fpregs(void __user *data, struct task_struct *task, int has_fpscr){ unsigned int count = has_fpscr ? 33 : 32; if (copy_from_user(task->thread.fpr, data, count * sizeof(double))) return -EFAULT; return 0;}#ifdef CONFIG_ALTIVEC/* * Get/set all the altivec registers vr0..vr31, vscr, vrsave, in one go. * The transfer totals 34 quadword. Quadwords 0-31 contain the * corresponding vector registers. Quadword 32 contains the vscr as the * last word (offset 12) within that quadword. Quadword 33 contains the * vrsave as the first word (offset 0) within the quadword. * * This definition of the VMX state is compatible with the current PPC32 * ptrace interface. This allows signal handling and ptrace to use the * same structures. This also simplifies the implementation of a bi-arch * (combined (32- and 64-bit) gdb. *//* * Get contents of AltiVec register state in task TASK */static int get_vrregs(unsigned long __user *data, struct task_struct *task){ unsigned long regsize; /* copy AltiVec registers VR[0] .. VR[31] */ regsize = 32 * sizeof(vector128); if (copy_to_user(data, task->thread.vr, regsize)) return -EFAULT; data += (regsize / sizeof(unsigned long)); /* copy VSCR */ regsize = 1 * sizeof(vector128); if (copy_to_user(data, &task->thread.vscr, regsize)) return -EFAULT; data += (regsize / sizeof(unsigned long)); /* copy VRSAVE */ if (put_user(task->thread.vrsave, (u32 __user *)data)) return -EFAULT; return 0;}/* * Write contents of AltiVec register state into task TASK. */static int set_vrregs(struct task_struct *task, unsigned long __user *data){ unsigned long regsize; /* copy AltiVec registers VR[0] .. VR[31] */ regsize = 32 * sizeof(vector128); if (copy_from_user(task->thread.vr, data, regsize)) return -EFAULT; data += (regsize / sizeof(unsigned long)); /* copy VSCR */ regsize = 1 * sizeof(vector128); if (copy_from_user(&task->thread.vscr, data, regsize)) return -EFAULT; data += (regsize / sizeof(unsigned long)); /* copy VRSAVE */ if (get_user(task->thread.vrsave, (u32 __user *)data)) return -EFAULT; return 0;}#endif /* CONFIG_ALTIVEC */#ifdef CONFIG_SPE/* * For get_evrregs/set_evrregs functions 'data' has the following layout: * * struct { * u32 evr[32]; * u64 acc; * u32 spefscr; * } *//* * Get contents of SPE register state in task TASK. */static int get_evrregs(unsigned long *data, struct task_struct *task){ int i; if (!access_ok(VERIFY_WRITE, data, 35 * sizeof(unsigned long))) return -EFAULT; /* copy SPEFSCR */ if (__put_user(task->thread.spefscr, &data[34])) return -EFAULT; /* copy SPE registers EVR[0] .. EVR[31] */ for (i = 0; i < 32; i++, data++) if (__put_user(task->thread.evr[i], data)) return -EFAULT; /* copy ACC */ if (__put_user64(task->thread.acc, (unsigned long long *)data)) return -EFAULT; return 0;}/* * Write contents of SPE register state into task TASK. */static int set_evrregs(struct task_struct *task, unsigned long *data){ int i; if (!access_ok(VERIFY_READ, data, 35 * sizeof(unsigned long))) return -EFAULT; /* copy SPEFSCR */ if (__get_user(task->thread.spefscr, &data[34])) return -EFAULT; /* copy SPE registers EVR[0] .. EVR[31] */ for (i = 0; i < 32; i++, data++) if (__get_user(task->thread.evr[i], data)) return -EFAULT; /* copy ACC */ if (__get_user64(task->thread.acc, (unsigned long long*)data)) return -EFAULT; return 0;}#endif /* CONFIG_SPE */static void set_single_step(struct task_struct *task){ struct pt_regs *regs = task->thread.regs; if (regs != NULL) {#if defined(CONFIG_40x) || defined(CONFIG_BOOKE) task->thread.dbcr0 = DBCR0_IDM | DBCR0_IC; regs->msr |= MSR_DE;#else regs->msr |= MSR_SE;#endif } set_tsk_thread_flag(task, TIF_SINGLESTEP);}static void clear_single_step(struct task_struct *task){ struct pt_regs *regs = task->thread.regs; if (regs != NULL) {#if defined(CONFIG_40x) || defined(CONFIG_BOOKE) task->thread.dbcr0 = 0; regs->msr &= ~MSR_DE;#else regs->msr &= ~MSR_SE;#endif } clear_tsk_thread_flag(task, TIF_SINGLESTEP);}static int ptrace_set_debugreg(struct task_struct *task, unsigned long addr, unsigned long data){ /* We only support one DABR and no IABRS at the moment */ if (addr > 0) return -EINVAL; /* The bottom 3 bits are flags */ if ((data & ~0x7UL) >= TASK_SIZE) return -EIO; /* Ensure translation is on */ if (data && !(data & DABR_TRANSLATION)) return -EIO; task->thread.dabr = data; return 0;}/* * Called by kernel/ptrace.c when detaching.. * * Make sure single step bits etc are not set. */void ptrace_disable(struct task_struct *child){ /* make sure the single step bit is not set. */ clear_single_step(child);}/* * Here are the old "legacy" powerpc specific getregs/setregs ptrace calls, * we mark them as obsolete now, they will be removed in a future version */static long arch_ptrace_old(struct task_struct *child, long request, long addr, long data){ int ret = -EPERM; switch(request) { case PPC_PTRACE_GETREGS: { /* Get GPRs 0 - 31. */ int i; unsigned long *reg = &((unsigned long *)child->thread.regs)[0];
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?