⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ptrace.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
	case PTRACE_GETREGS64: {		struct pt_regs __user *pregs = (struct pt_regs __user *) addr;		struct pt_regs *cregs = child->thread_info->kregs;		unsigned long tpc = cregs->tpc;		int rval;		if ((child->thread_info->flags & _TIF_32BIT) != 0)			tpc &= 0xffffffff;		if (__put_user(cregs->tstate, (&pregs->tstate)) ||		    __put_user(tpc, (&pregs->tpc)) ||		    __put_user(cregs->tnpc, (&pregs->tnpc)) ||		    __put_user(cregs->y, (&pregs->y))) {			pt_error_return(regs, EFAULT);			goto out_tsk;		}		for (rval = 1; rval < 16; rval++)			if (__put_user(cregs->u_regs[rval], (&pregs->u_regs[rval - 1]))) {				pt_error_return(regs, EFAULT);				goto out_tsk;			}		pt_succ_return(regs, 0);#ifdef DEBUG_PTRACE		printk ("PC=%lx nPC=%lx o7=%lx\n", cregs->tpc, cregs->tnpc, cregs->u_regs [15]);#endif		goto out_tsk;	}	case PTRACE_SETREGS: {		struct pt_regs32 __user *pregs =			(struct pt_regs32 __user *) addr;		struct pt_regs *cregs = child->thread_info->kregs;		unsigned int psr, pc, npc, y;		int i;		/* Must be careful, tracing process can only set certain		 * bits in the psr.		 */		if (__get_user(psr, (&pregs->psr)) ||		    __get_user(pc, (&pregs->pc)) ||		    __get_user(npc, (&pregs->npc)) ||		    __get_user(y, (&pregs->y))) {			pt_error_return(regs, EFAULT);			goto out_tsk;		}		cregs->tstate &= ~(TSTATE_ICC);		cregs->tstate |= psr_to_tstate_icc(psr);               	if (!((pc | npc) & 3)) {			cregs->tpc = pc;			cregs->tnpc = npc;		}		cregs->y = y;		for (i = 1; i < 16; i++) {			if (__get_user(cregs->u_regs[i], (&pregs->u_regs[i-1]))) {				pt_error_return(regs, EFAULT);				goto out_tsk;			}		}		pt_succ_return(regs, 0);		goto out_tsk;	}	case PTRACE_SETREGS64: {		struct pt_regs __user *pregs = (struct pt_regs __user *) addr;		struct pt_regs *cregs = child->thread_info->kregs;		unsigned long tstate, tpc, tnpc, y;		int i;		/* Must be careful, tracing process can only set certain		 * bits in the psr.		 */		if (__get_user(tstate, (&pregs->tstate)) ||		    __get_user(tpc, (&pregs->tpc)) ||		    __get_user(tnpc, (&pregs->tnpc)) ||		    __get_user(y, (&pregs->y))) {			pt_error_return(regs, EFAULT);			goto out_tsk;		}		if ((child->thread_info->flags & _TIF_32BIT) != 0) {			tpc &= 0xffffffff;			tnpc &= 0xffffffff;		}		tstate &= (TSTATE_ICC | TSTATE_XCC);		cregs->tstate &= ~(TSTATE_ICC | TSTATE_XCC);		cregs->tstate |= tstate;		if (!((tpc | tnpc) & 3)) {			cregs->tpc = tpc;			cregs->tnpc = tnpc;		}		cregs->y = y;		for (i = 1; i < 16; i++) {			if (__get_user(cregs->u_regs[i], (&pregs->u_regs[i-1]))) {				pt_error_return(regs, EFAULT);				goto out_tsk;			}		}		pt_succ_return(regs, 0);		goto out_tsk;	}	case PTRACE_GETFPREGS: {		struct fps {			unsigned int regs[32];			unsigned int fsr;			unsigned int flags;			unsigned int extra;			unsigned int fpqd;			struct fq {				unsigned int insnaddr;				unsigned int insn;			} fpq[16];		};		struct fps __user *fps = (struct fps __user *) addr;		unsigned long *fpregs = child->thread_info->fpregs;		if (copy_to_user(&fps->regs[0], fpregs,				 (32 * sizeof(unsigned int))) ||		    __put_user(child->thread_info->xfsr[0], (&fps->fsr)) ||		    __put_user(0, (&fps->fpqd)) ||		    __put_user(0, (&fps->flags)) ||		    __put_user(0, (&fps->extra)) ||		    clear_user(&fps->fpq[0], 32 * sizeof(unsigned int))) {			pt_error_return(regs, EFAULT);			goto out_tsk;		}		pt_succ_return(regs, 0);		goto out_tsk;	}	case PTRACE_GETFPREGS64: {		struct fps {			unsigned int regs[64];			unsigned long fsr;		};		struct fps __user *fps = (struct fps __user *) addr;		unsigned long *fpregs = child->thread_info->fpregs;		if (copy_to_user(&fps->regs[0], fpregs,				 (64 * sizeof(unsigned int))) ||		    __put_user(child->thread_info->xfsr[0], (&fps->fsr))) {			pt_error_return(regs, EFAULT);			goto out_tsk;		}		pt_succ_return(regs, 0);		goto out_tsk;	}	case PTRACE_SETFPREGS: {		struct fps {			unsigned int regs[32];			unsigned int fsr;			unsigned int flags;			unsigned int extra;			unsigned int fpqd;			struct fq {				unsigned int insnaddr;				unsigned int insn;			} fpq[16];		};		struct fps __user *fps = (struct fps __user *) addr;		unsigned long *fpregs = child->thread_info->fpregs;		unsigned fsr;		if (copy_from_user(fpregs, &fps->regs[0],				   (32 * sizeof(unsigned int))) ||		    __get_user(fsr, (&fps->fsr))) {			pt_error_return(regs, EFAULT);			goto out_tsk;		}		child->thread_info->xfsr[0] &= 0xffffffff00000000UL;		child->thread_info->xfsr[0] |= fsr;		if (!(child->thread_info->fpsaved[0] & FPRS_FEF))			child->thread_info->gsr[0] = 0;		child->thread_info->fpsaved[0] |= (FPRS_FEF | FPRS_DL);		pt_succ_return(regs, 0);		goto out_tsk;	}	case PTRACE_SETFPREGS64: {		struct fps {			unsigned int regs[64];			unsigned long fsr;		};		struct fps __user *fps = (struct fps __user *) addr;		unsigned long *fpregs = child->thread_info->fpregs;		if (copy_from_user(fpregs, &fps->regs[0],				   (64 * sizeof(unsigned int))) ||		    __get_user(child->thread_info->xfsr[0], (&fps->fsr))) {			pt_error_return(regs, EFAULT);			goto out_tsk;		}		if (!(child->thread_info->fpsaved[0] & FPRS_FEF))			child->thread_info->gsr[0] = 0;		child->thread_info->fpsaved[0] |= (FPRS_FEF | FPRS_DL | FPRS_DU);		pt_succ_return(regs, 0);		goto out_tsk;	}	case PTRACE_READTEXT:	case PTRACE_READDATA: {		int res = ptrace_readdata(child, addr,					  (char __user *)addr2, data);		if (res == data) {			pt_succ_return(regs, 0);			goto out_tsk;		}		if (res >= 0)			res = -EIO;		pt_error_return(regs, -res);		goto out_tsk;	}	case PTRACE_WRITETEXT:	case PTRACE_WRITEDATA: {		int res = ptrace_writedata(child, (char __user *) addr2,					   addr, data);		if (res == data) {			pt_succ_return(regs, 0);			goto out_tsk;		}		if (res >= 0)			res = -EIO;		pt_error_return(regs, -res);		goto out_tsk;	}	case PTRACE_SYSCALL: /* continue and stop at (return from) syscall */		addr = 1;	case PTRACE_CONT: { /* restart after signal. */		if (!valid_signal(data)) {			pt_error_return(regs, EIO);			goto out_tsk;		}		if (request == PTRACE_SYSCALL) {			set_tsk_thread_flag(child, TIF_SYSCALL_TRACE);		} else {			clear_tsk_thread_flag(child, TIF_SYSCALL_TRACE);		}		child->exit_code = data;#ifdef DEBUG_PTRACE		printk("CONT: %s [%d]: set exit_code = %x %lx %lx\n", child->comm,			child->pid, child->exit_code,			child->thread_info->kregs->tpc,			child->thread_info->kregs->tnpc);		       #endif		wake_up_process(child);		pt_succ_return(regs, 0);		goto out_tsk;	}/* * make the child exit.  Best I can do is send it a sigkill.  * perhaps it should be put in the status that it wants to  * exit. */	case PTRACE_KILL: {		if (child->exit_state == EXIT_ZOMBIE) {	/* already dead */			pt_succ_return(regs, 0);			goto out_tsk;		}		child->exit_code = SIGKILL;		wake_up_process(child);		pt_succ_return(regs, 0);		goto out_tsk;	}	case PTRACE_SUNDETACH: { /* detach a process that was attached. */		int error = ptrace_detach(child, data);		if (error) {			pt_error_return(regs, EIO);			goto out_tsk;		}		pt_succ_return(regs, 0);		goto out_tsk;	}	/* PTRACE_DUMPCORE unsupported... */	default: {		int err = ptrace_request(child, request, addr, data);		if (err)			pt_error_return(regs, -err);		else			pt_succ_return(regs, 0);		goto out_tsk;	}	}out_tsk:	if (child)		put_task_struct(child);out:	unlock_kernel();}asmlinkage void syscall_trace(struct pt_regs *regs, int syscall_exit_p){	/* do the secure computing check first */	secure_computing(regs->u_regs[UREG_G1]);	if (unlikely(current->audit_context) && syscall_exit_p) {		unsigned long tstate = regs->tstate;		int result = AUDITSC_SUCCESS;		if (unlikely(tstate & (TSTATE_XCARRY | TSTATE_ICARRY)))			result = AUDITSC_FAILURE;		audit_syscall_exit(current, result, regs->u_regs[UREG_I0]);	}	if (!(current->ptrace & PT_PTRACED))		goto out;	if (!test_thread_flag(TIF_SYSCALL_TRACE))		goto out;	ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)				 ? 0x80 : 0));	/*	 * this isn't the same as continuing with a signal, but it will do	 * for normal use.  strace only continues with a signal if the	 * stopping signal is not SIGTRAP.  -brl	 */	if (current->exit_code) {		send_sig(current->exit_code, current, 1);		current->exit_code = 0;	}out:	if (unlikely(current->audit_context) && !syscall_exit_p)		audit_syscall_entry(current,				    (test_thread_flag(TIF_32BIT) ?				     AUDIT_ARCH_SPARC :				     AUDIT_ARCH_SPARC64),				    regs->u_regs[UREG_G1],				    regs->u_regs[UREG_I0],				    regs->u_regs[UREG_I1],				    regs->u_regs[UREG_I2],				    regs->u_regs[UREG_I3]);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -