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 + -
显示快捷键?