/* ptrace.c */
|
/* ptrace.c */
|
/* By Ross Biro 1/23/92 */
|
/* By Ross Biro 1/23/92 */
|
/* edited by Linus Torvalds */
|
/* edited by Linus Torvalds */
|
/* edited for ARM by Russell King */
|
/* edited for ARM by Russell King */
|
|
|
#include <linux/head.h>
|
#include <linux/head.h>
|
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
#include <linux/sched.h>
|
#include <linux/sched.h>
|
#include <linux/mm.h>
|
#include <linux/mm.h>
|
#include <linux/errno.h>
|
#include <linux/errno.h>
|
#include <linux/ptrace.h>
|
#include <linux/ptrace.h>
|
#include <linux/user.h>
|
#include <linux/user.h>
|
|
|
#include <asm/segment.h>
|
#include <asm/segment.h>
|
#include <asm/pgtable.h>
|
#include <asm/pgtable.h>
|
#include <asm/system.h>
|
#include <asm/system.h>
|
|
|
/*
|
/*
|
* does not yet catch signals sent when the child dies.
|
* does not yet catch signals sent when the child dies.
|
* in exit.c or in signal.c.
|
* in exit.c or in signal.c.
|
*/
|
*/
|
|
|
/*
|
/*
|
* Breakpoint SWI instruction: SWI &9F0001
|
* Breakpoint SWI instruction: SWI &9F0001
|
*/
|
*/
|
#define BREAKINST 0xef9f0001
|
#define BREAKINST 0xef9f0001
|
|
|
/* change a pid into a task struct. */
|
/* change a pid into a task struct. */
|
static inline struct task_struct * get_task(int pid)
|
static inline struct task_struct * get_task(int pid)
|
{
|
{
|
int i;
|
int i;
|
|
|
for (i = 1; i < NR_TASKS; i++) {
|
for (i = 1; i < NR_TASKS; i++) {
|
if (task[i] != NULL && (task[i]->pid == pid))
|
if (task[i] != NULL && (task[i]->pid == pid))
|
return task[i];
|
return task[i];
|
}
|
}
|
return NULL;
|
return NULL;
|
}
|
}
|
|
|
/*
|
/*
|
* this routine will get a word off of the processes privileged stack.
|
* this routine will get a word off of the processes privileged stack.
|
* the offset is how far from the base addr as stored in the TSS.
|
* the offset is how far from the base addr as stored in the TSS.
|
* this routine assumes that all the privileged stacks are in our
|
* this routine assumes that all the privileged stacks are in our
|
* data space.
|
* data space.
|
*/
|
*/
|
static inline long get_stack_long(struct task_struct *task, int offset)
|
static inline long get_stack_long(struct task_struct *task, int offset)
|
{
|
{
|
unsigned char *stack;
|
unsigned char *stack;
|
|
|
stack = (unsigned char *)(task->kernel_stack_page + 4096 - sizeof(struct pt_regs));
|
stack = (unsigned char *)(task->kernel_stack_page + 4096 - sizeof(struct pt_regs));
|
stack += offset << 2;
|
stack += offset << 2;
|
return *(unsigned long *)stack;
|
return *(unsigned long *)stack;
|
}
|
}
|
|
|
/*
|
/*
|
* this routine will put a word on the processes privileged stack.
|
* this routine will put a word on the processes privileged stack.
|
* the offset is how far from the base addr as stored in the TSS.
|
* the offset is how far from the base addr as stored in the TSS.
|
* this routine assumes that all the privileged stacks are in our
|
* this routine assumes that all the privileged stacks are in our
|
* data space.
|
* data space.
|
*/
|
*/
|
static inline long put_stack_long(struct task_struct *task, int offset,
|
static inline long put_stack_long(struct task_struct *task, int offset,
|
unsigned long data)
|
unsigned long data)
|
{
|
{
|
unsigned char *stack;
|
unsigned char *stack;
|
|
|
stack = (unsigned char *)(task->kernel_stack_page + 4096 - sizeof(struct pt_regs));
|
stack = (unsigned char *)(task->kernel_stack_page + 4096 - sizeof(struct pt_regs));
|
stack += offset << 2;
|
stack += offset << 2;
|
*(unsigned long *) stack = data;
|
*(unsigned long *) stack = data;
|
return 0;
|
return 0;
|
}
|
}
|
|
|
|
|
|
|
inline
|
inline
|
static unsigned long get_long(struct task_struct * tsk,
|
static unsigned long get_long(struct task_struct * tsk,
|
struct vm_area_struct * vma, unsigned long addr)
|
struct vm_area_struct * vma, unsigned long addr)
|
{
|
{
|
return *(unsigned long*)addr;
|
return *(unsigned long*)addr;
|
}
|
}
|
|
|
inline
|
inline
|
static void put_long(struct task_struct * tsk, struct vm_area_struct * vma, unsigned long addr,
|
static void put_long(struct task_struct * tsk, struct vm_area_struct * vma, unsigned long addr,
|
unsigned long data)
|
unsigned long data)
|
{
|
{
|
*(unsigned long*)addr = data;
|
*(unsigned long*)addr = data;
|
}
|
}
|
|
|
|
|
inline
|
inline
|
static int read_long(struct task_struct * tsk, unsigned long addr,
|
static int read_long(struct task_struct * tsk, unsigned long addr,
|
unsigned long * result)
|
unsigned long * result)
|
{
|
{
|
*result = *(unsigned long *)addr;
|
*result = *(unsigned long *)addr;
|
return 0;
|
return 0;
|
}
|
}
|
|
|
inline
|
inline
|
static int write_long(struct task_struct * tsk, unsigned long addr,
|
static int write_long(struct task_struct * tsk, unsigned long addr,
|
unsigned long data)
|
unsigned long data)
|
{
|
{
|
*(unsigned long *)addr = data;
|
*(unsigned long *)addr = data;
|
return 0;
|
return 0;
|
}
|
}
|
|
|
|
|
/*
|
/*
|
* Get value of register `rn' (in the instruction)
|
* Get value of register `rn' (in the instruction)
|
*/
|
*/
|
static unsigned long ptrace_getrn (struct task_struct *child, unsigned long insn)
|
static unsigned long ptrace_getrn (struct task_struct *child, unsigned long insn)
|
{
|
{
|
unsigned int reg = (insn >> 16) & 15;
|
unsigned int reg = (insn >> 16) & 15;
|
unsigned long val;
|
unsigned long val;
|
|
|
if (reg == 15)
|
if (reg == 15)
|
val = pc_pointer (get_stack_long (child, reg));
|
val = pc_pointer (get_stack_long (child, reg));
|
else
|
else
|
val = get_stack_long (child, reg);
|
val = get_stack_long (child, reg);
|
|
|
printk ("r%02d=%08lX ", reg, val);
|
printk ("r%02d=%08lX ", reg, val);
|
return val;
|
return val;
|
}
|
}
|
|
|
/*
|
/*
|
* Get value of operand 2 (in an ALU instruction)
|
* Get value of operand 2 (in an ALU instruction)
|
*/
|
*/
|
static unsigned long ptrace_getaluop2 (struct task_struct *child, unsigned long insn)
|
static unsigned long ptrace_getaluop2 (struct task_struct *child, unsigned long insn)
|
{
|
{
|
unsigned long val;
|
unsigned long val;
|
int shift;
|
int shift;
|
int type;
|
int type;
|
|
|
printk ("op2=");
|
printk ("op2=");
|
if (insn & 1 << 25) {
|
if (insn & 1 << 25) {
|
val = insn & 255;
|
val = insn & 255;
|
shift = (insn >> 8) & 15;
|
shift = (insn >> 8) & 15;
|
type = 3;
|
type = 3;
|
printk ("(imm)");
|
printk ("(imm)");
|
} else {
|
} else {
|
val = get_stack_long (child, insn & 15);
|
val = get_stack_long (child, insn & 15);
|
|
|
if (insn & (1 << 4))
|
if (insn & (1 << 4))
|
shift = (int)get_stack_long (child, (insn >> 8) & 15);
|
shift = (int)get_stack_long (child, (insn >> 8) & 15);
|
else
|
else
|
shift = (insn >> 7) & 31;
|
shift = (insn >> 7) & 31;
|
|
|
type = (insn >> 5) & 3;
|
type = (insn >> 5) & 3;
|
printk ("(r%02ld)", insn & 15);
|
printk ("(r%02ld)", insn & 15);
|
}
|
}
|
printk ("sh%dx%d", type, shift);
|
printk ("sh%dx%d", type, shift);
|
switch (type) {
|
switch (type) {
|
case 0: val <<= shift; break;
|
case 0: val <<= shift; break;
|
case 1: val >>= shift; break;
|
case 1: val >>= shift; break;
|
case 2:
|
case 2:
|
val = (((signed long)val) >> shift);
|
val = (((signed long)val) >> shift);
|
break;
|
break;
|
case 3:
|
case 3:
|
__asm__ __volatile__("mov %0, %0, ror %1" : "=r" (val) : "0" (val), "r" (shift));
|
__asm__ __volatile__("mov %0, %0, ror %1" : "=r" (val) : "0" (val), "r" (shift));
|
break;
|
break;
|
}
|
}
|
printk ("=%08lX ", val);
|
printk ("=%08lX ", val);
|
return val;
|
return val;
|
}
|
}
|
|
|
/*
|
/*
|
* Get value of operand 2 (in a LDR instruction)
|
* Get value of operand 2 (in a LDR instruction)
|
*/
|
*/
|
static unsigned long ptrace_getldrop2 (struct task_struct *child, unsigned long insn)
|
static unsigned long ptrace_getldrop2 (struct task_struct *child, unsigned long insn)
|
{
|
{
|
unsigned long val;
|
unsigned long val;
|
int shift;
|
int shift;
|
int type;
|
int type;
|
|
|
val = get_stack_long (child, insn & 15);
|
val = get_stack_long (child, insn & 15);
|
shift = (insn >> 7) & 31;
|
shift = (insn >> 7) & 31;
|
type = (insn >> 5) & 3;
|
type = (insn >> 5) & 3;
|
|
|
printk ("op2=r%02ldsh%dx%d", insn & 15, shift, type);
|
printk ("op2=r%02ldsh%dx%d", insn & 15, shift, type);
|
switch (type) {
|
switch (type) {
|
case 0: val <<= shift; break;
|
case 0: val <<= shift; break;
|
case 1: val >>= shift; break;
|
case 1: val >>= shift; break;
|
case 2:
|
case 2:
|
val = (((signed long)val) >> shift);
|
val = (((signed long)val) >> shift);
|
break;
|
break;
|
case 3:
|
case 3:
|
__asm__ __volatile__("mov %0, %0, ror %1" : "=r" (val) : "0" (val), "r" (shift));
|
__asm__ __volatile__("mov %0, %0, ror %1" : "=r" (val) : "0" (val), "r" (shift));
|
break;
|
break;
|
}
|
}
|
printk ("=%08lX ", val);
|
printk ("=%08lX ", val);
|
return val;
|
return val;
|
}
|
}
|
#undef pc_pointer
|
#undef pc_pointer
|
#define pc_pointer(x) ((x) & 0x03fffffc)
|
#define pc_pointer(x) ((x) & 0x03fffffc)
|
int ptrace_set_bpt (struct task_struct *child)
|
int ptrace_set_bpt (struct task_struct *child)
|
{
|
{
|
unsigned long insn, pc, alt;
|
unsigned long insn, pc, alt;
|
int i, nsaved = 0, res;
|
int i, nsaved = 0, res;
|
|
|
pc = pc_pointer (get_stack_long (child, 15/*REG_PC*/));
|
pc = pc_pointer (get_stack_long (child, 15/*REG_PC*/));
|
|
|
res = read_long (child, pc, &insn);
|
res = read_long (child, pc, &insn);
|
if (res < 0)
|
if (res < 0)
|
return res;
|
return res;
|
|
|
child->debugreg[nsaved++] = alt = pc + 4;
|
child->debugreg[nsaved++] = alt = pc + 4;
|
printk ("ptrace_set_bpt: insn=%08lX pc=%08lX ", insn, pc);
|
printk ("ptrace_set_bpt: insn=%08lX pc=%08lX ", insn, pc);
|
switch (insn & 0x0e100000) {
|
switch (insn & 0x0e100000) {
|
case 0x00000000:
|
case 0x00000000:
|
case 0x00100000:
|
case 0x00100000:
|
case 0x02000000:
|
case 0x02000000:
|
case 0x02100000: /* data processing */
|
case 0x02100000: /* data processing */
|
printk ("data ");
|
printk ("data ");
|
switch (insn & 0x01e0f000) {
|
switch (insn & 0x01e0f000) {
|
case 0x0000f000:
|
case 0x0000f000:
|
alt = ptrace_getrn(child, insn) & ptrace_getaluop2(child, insn);
|
alt = ptrace_getrn(child, insn) & ptrace_getaluop2(child, insn);
|
break;
|
break;
|
case 0x0020f000:
|
case 0x0020f000:
|
alt = ptrace_getrn(child, insn) ^ ptrace_getaluop2(child, insn);
|
alt = ptrace_getrn(child, insn) ^ ptrace_getaluop2(child, insn);
|
break;
|
break;
|
case 0x0040f000:
|
case 0x0040f000:
|
alt = ptrace_getrn(child, insn) - ptrace_getaluop2(child, insn);
|
alt = ptrace_getrn(child, insn) - ptrace_getaluop2(child, insn);
|
break;
|
break;
|
case 0x0060f000:
|
case 0x0060f000:
|
alt = ptrace_getaluop2(child, insn) - ptrace_getrn(child, insn);
|
alt = ptrace_getaluop2(child, insn) - ptrace_getrn(child, insn);
|
break;
|
break;
|
case 0x0080f000:
|
case 0x0080f000:
|
alt = ptrace_getrn(child, insn) + ptrace_getaluop2(child, insn);
|
alt = ptrace_getrn(child, insn) + ptrace_getaluop2(child, insn);
|
break;
|
break;
|
case 0x00a0f000:
|
case 0x00a0f000:
|
alt = ptrace_getrn(child, insn) + ptrace_getaluop2(child, insn) +
|
alt = ptrace_getrn(child, insn) + ptrace_getaluop2(child, insn) +
|
(get_stack_long (child, 16/*REG_PSR*/) & CC_C_BIT ? 1 : 0);
|
(get_stack_long (child, 16/*REG_PSR*/) & CC_C_BIT ? 1 : 0);
|
break;
|
break;
|
case 0x00c0f000:
|
case 0x00c0f000:
|
alt = ptrace_getrn(child, insn) - ptrace_getaluop2(child, insn) +
|
alt = ptrace_getrn(child, insn) - ptrace_getaluop2(child, insn) +
|
(get_stack_long (child, 16/*REG_PSR*/) & CC_C_BIT ? 1 : 0);
|
(get_stack_long (child, 16/*REG_PSR*/) & CC_C_BIT ? 1 : 0);
|
break;
|
break;
|
case 0x00e0f000:
|
case 0x00e0f000:
|
alt = ptrace_getaluop2(child, insn) - ptrace_getrn(child, insn) +
|
alt = ptrace_getaluop2(child, insn) - ptrace_getrn(child, insn) +
|
(get_stack_long (child, 16/*REG_PSR*/) & CC_C_BIT ? 1 : 0);
|
(get_stack_long (child, 16/*REG_PSR*/) & CC_C_BIT ? 1 : 0);
|
break;
|
break;
|
case 0x0180f000:
|
case 0x0180f000:
|
alt = ptrace_getrn(child, insn) | ptrace_getaluop2(child, insn);
|
alt = ptrace_getrn(child, insn) | ptrace_getaluop2(child, insn);
|
break;
|
break;
|
case 0x01a0f000:
|
case 0x01a0f000:
|
alt = ptrace_getaluop2(child, insn);
|
alt = ptrace_getaluop2(child, insn);
|
break;
|
break;
|
case 0x01c0f000:
|
case 0x01c0f000:
|
alt = ptrace_getrn(child, insn) & ~ptrace_getaluop2(child, insn);
|
alt = ptrace_getrn(child, insn) & ~ptrace_getaluop2(child, insn);
|
break;
|
break;
|
case 0x01e0f000:
|
case 0x01e0f000:
|
alt = ~ptrace_getaluop2(child, insn);
|
alt = ~ptrace_getaluop2(child, insn);
|
break;
|
break;
|
}
|
}
|
break;
|
break;
|
|
|
case 0x04100000: /* ldr imm */
|
case 0x04100000: /* ldr imm */
|
if ((insn & 0xf000) == 0xf000) {
|
if ((insn & 0xf000) == 0xf000) {
|
printk ("ldrimm ");
|
printk ("ldrimm ");
|
alt = ptrace_getrn(child, insn);
|
alt = ptrace_getrn(child, insn);
|
if (insn & 1 << 24) {
|
if (insn & 1 << 24) {
|
if (insn & 1 << 23)
|
if (insn & 1 << 23)
|
alt += insn & 0xfff;
|
alt += insn & 0xfff;
|
else
|
else
|
alt -= insn & 0xfff;
|
alt -= insn & 0xfff;
|
}
|
}
|
if (read_long (child, alt, &alt) < 0)
|
if (read_long (child, alt, &alt) < 0)
|
alt = pc + 4; /* not valid */
|
alt = pc + 4; /* not valid */
|
else
|
else
|
alt = pc_pointer (alt);
|
alt = pc_pointer (alt);
|
}
|
}
|
break;
|
break;
|
|
|
case 0x06100000: /* ldr */
|
case 0x06100000: /* ldr */
|
if ((insn & 0xf000) == 0xf000) {
|
if ((insn & 0xf000) == 0xf000) {
|
printk ("ldr ");
|
printk ("ldr ");
|
alt = ptrace_getrn(child, insn);
|
alt = ptrace_getrn(child, insn);
|
if (insn & 1 << 24) {
|
if (insn & 1 << 24) {
|
if (insn & 1 << 23)
|
if (insn & 1 << 23)
|
alt += ptrace_getldrop2 (child, insn);
|
alt += ptrace_getldrop2 (child, insn);
|
else
|
else
|
alt -= ptrace_getldrop2 (child, insn);
|
alt -= ptrace_getldrop2 (child, insn);
|
}
|
}
|
if (read_long (child, alt, &alt) < 0)
|
if (read_long (child, alt, &alt) < 0)
|
alt = pc + 4; /* not valid */
|
alt = pc + 4; /* not valid */
|
else
|
else
|
alt = pc_pointer (alt);
|
alt = pc_pointer (alt);
|
}
|
}
|
break;
|
break;
|
|
|
case 0x08100000: /* ldm */
|
case 0x08100000: /* ldm */
|
if (insn & (1 << 15)) {
|
if (insn & (1 << 15)) {
|
unsigned long base;
|
unsigned long base;
|
int nr_regs;
|
int nr_regs;
|
printk ("ldm ");
|
printk ("ldm ");
|
|
|
if (insn & (1 << 23)) {
|
if (insn & (1 << 23)) {
|
nr_regs = insn & 65535;
|
nr_regs = insn & 65535;
|
|
|
nr_regs = (nr_regs & 0x5555) + ((nr_regs & 0xaaaa) >> 1);
|
nr_regs = (nr_regs & 0x5555) + ((nr_regs & 0xaaaa) >> 1);
|
nr_regs = (nr_regs & 0x3333) + ((nr_regs & 0xcccc) >> 2);
|
nr_regs = (nr_regs & 0x3333) + ((nr_regs & 0xcccc) >> 2);
|
nr_regs = (nr_regs & 0x0707) + ((nr_regs & 0x7070) >> 4);
|
nr_regs = (nr_regs & 0x0707) + ((nr_regs & 0x7070) >> 4);
|
nr_regs = (nr_regs & 0x000f) + ((nr_regs & 0x0f00) >> 8);
|
nr_regs = (nr_regs & 0x000f) + ((nr_regs & 0x0f00) >> 8);
|
nr_regs <<= 2;
|
nr_regs <<= 2;
|
|
|
if (!(insn & (1 << 24)))
|
if (!(insn & (1 << 24)))
|
nr_regs -= 4;
|
nr_regs -= 4;
|
} else {
|
} else {
|
if (insn & (1 << 24))
|
if (insn & (1 << 24))
|
nr_regs = -4;
|
nr_regs = -4;
|
else
|
else
|
nr_regs = 0;
|
nr_regs = 0;
|
}
|
}
|
|
|
base = ptrace_getrn (child, insn);
|
base = ptrace_getrn (child, insn);
|
|
|
if (read_long (child, base + nr_regs, &alt) < 0)
|
if (read_long (child, base + nr_regs, &alt) < 0)
|
alt = pc + 4; /* not valid */
|
alt = pc + 4; /* not valid */
|
else
|
else
|
alt = pc_pointer (alt);
|
alt = pc_pointer (alt);
|
break;
|
break;
|
}
|
}
|
break;
|
break;
|
|
|
case 0x0a000000:
|
case 0x0a000000:
|
case 0x0a100000: { /* bl or b */
|
case 0x0a100000: { /* bl or b */
|
signed long displ;
|
signed long displ;
|
printk ("b/bl ");
|
printk ("b/bl ");
|
/* It's a branch/branch link: instead of trying to
|
/* It's a branch/branch link: instead of trying to
|
* figure out whether the branch will be taken or not,
|
* figure out whether the branch will be taken or not,
|
* we'll put a breakpoint at either location. This is
|
* we'll put a breakpoint at either location. This is
|
* simpler, more reliable, and probably not a whole lot
|
* simpler, more reliable, and probably not a whole lot
|
* slower than the alternative approach of emulating the
|
* slower than the alternative approach of emulating the
|
* branch.
|
* branch.
|
*/
|
*/
|
displ = (insn & 0x00ffffff) << 8;
|
displ = (insn & 0x00ffffff) << 8;
|
displ = (displ >> 6) + 8;
|
displ = (displ >> 6) + 8;
|
if (displ != 0 && displ != 4)
|
if (displ != 0 && displ != 4)
|
alt = pc + displ;
|
alt = pc + displ;
|
}
|
}
|
break;
|
break;
|
}
|
}
|
printk ("=%08lX\n", alt);
|
printk ("=%08lX\n", alt);
|
if (alt != pc + 4)
|
if (alt != pc + 4)
|
child->debugreg[nsaved++] = alt;
|
child->debugreg[nsaved++] = alt;
|
|
|
for (i = 0; i < nsaved; i++) {
|
for (i = 0; i < nsaved; i++) {
|
res = read_long (child, child->debugreg[i], &insn);
|
res = read_long (child, child->debugreg[i], &insn);
|
if (res >= 0) {
|
if (res >= 0) {
|
child->debugreg[i + 2] = insn;
|
child->debugreg[i + 2] = insn;
|
res = write_long (child, child->debugreg[i], BREAKINST);
|
res = write_long (child, child->debugreg[i], BREAKINST);
|
__flush_entry_to_ram(child->debugreg[i]);
|
__flush_entry_to_ram(child->debugreg[i]);
|
}
|
}
|
if (res < 0) {
|
if (res < 0) {
|
child->debugreg[4] = 0;
|
child->debugreg[4] = 0;
|
return res;
|
return res;
|
}
|
}
|
}
|
}
|
child->debugreg[4] = nsaved;
|
child->debugreg[4] = nsaved;
|
return 0;
|
return 0;
|
}
|
}
|
|
|
/* Ensure no single-step breakpoint is pending. Returns non-zero
|
/* Ensure no single-step breakpoint is pending. Returns non-zero
|
* value if child was being single-stepped.
|
* value if child was being single-stepped.
|
*/
|
*/
|
int ptrace_cancel_bpt (struct task_struct *child)
|
int ptrace_cancel_bpt (struct task_struct *child)
|
{
|
{
|
int i, nsaved = child->debugreg[4];
|
int i, nsaved = child->debugreg[4];
|
|
|
child->debugreg[4] = 0;
|
child->debugreg[4] = 0;
|
|
|
if (nsaved > 2) {
|
if (nsaved > 2) {
|
printk ("ptrace_cancel_bpt: bogus nsaved: %d!\n", nsaved);
|
printk ("ptrace_cancel_bpt: bogus nsaved: %d!\n", nsaved);
|
nsaved = 2;
|
nsaved = 2;
|
}
|
}
|
for (i = 0; i < nsaved; i++) {
|
for (i = 0; i < nsaved; i++) {
|
write_long (child, child->debugreg[i], child->debugreg[i + 2]);
|
write_long (child, child->debugreg[i], child->debugreg[i + 2]);
|
__flush_entry_to_ram(child->debugreg[i]);
|
__flush_entry_to_ram(child->debugreg[i]);
|
}
|
}
|
return nsaved != 0;
|
return nsaved != 0;
|
}
|
}
|
|
|
asmlinkage int sys_ptrace(long request, long pid, long addr, long data)
|
asmlinkage int sys_ptrace(long request, long pid, long addr, long data)
|
{
|
{
|
struct task_struct *child;
|
struct task_struct *child;
|
struct user * dummy;
|
struct user * dummy;
|
|
|
dummy = NULL;
|
dummy = NULL;
|
|
|
if (request == PTRACE_TRACEME) {
|
if (request == PTRACE_TRACEME) {
|
/* are we already being traced? */
|
/* are we already being traced? */
|
if (current->flags & PF_PTRACED)
|
if (current->flags & PF_PTRACED)
|
return -EPERM;
|
return -EPERM;
|
/* set the ptrace bit in the process flags. */
|
/* set the ptrace bit in the process flags. */
|
current->flags |= PF_PTRACED;
|
current->flags |= PF_PTRACED;
|
return 0;
|
return 0;
|
}
|
}
|
if (pid == 1) /* you may not mess with init */
|
if (pid == 1) /* you may not mess with init */
|
return -EPERM;
|
return -EPERM;
|
if (!(child = get_task(pid)))
|
if (!(child = get_task(pid)))
|
return -ESRCH;
|
return -ESRCH;
|
if (request == PTRACE_ATTACH) {
|
if (request == PTRACE_ATTACH) {
|
if (child == current)
|
if (child == current)
|
return -EPERM;
|
return -EPERM;
|
if ((!child->dumpable ||
|
if ((!child->dumpable ||
|
(current->uid != child->euid) ||
|
(current->uid != child->euid) ||
|
(current->uid != child->suid) ||
|
(current->uid != child->suid) ||
|
(current->uid != child->uid) ||
|
(current->uid != child->uid) ||
|
(current->gid != child->egid) ||
|
(current->gid != child->egid) ||
|
(current->gid != child->sgid) ||
|
(current->gid != child->sgid) ||
|
(current->gid != child->gid)) && !suser())
|
(current->gid != child->gid)) && !suser())
|
return -EPERM;
|
return -EPERM;
|
/* the same process cannot be attached many times */
|
/* the same process cannot be attached many times */
|
if (child->flags & PF_PTRACED)
|
if (child->flags & PF_PTRACED)
|
return -EPERM;
|
return -EPERM;
|
child->flags |= PF_PTRACED;
|
child->flags |= PF_PTRACED;
|
if (child->p_pptr != current) {
|
if (child->p_pptr != current) {
|
REMOVE_LINKS(child);
|
REMOVE_LINKS(child);
|
child->p_pptr = current;
|
child->p_pptr = current;
|
SET_LINKS(child);
|
SET_LINKS(child);
|
}
|
}
|
send_sig(SIGSTOP, child, 1);
|
send_sig(SIGSTOP, child, 1);
|
return 0;
|
return 0;
|
}
|
}
|
if (!(child->flags & PF_PTRACED))
|
if (!(child->flags & PF_PTRACED))
|
return -ESRCH;
|
return -ESRCH;
|
if (child->state != TASK_STOPPED) {
|
if (child->state != TASK_STOPPED) {
|
if (request != PTRACE_KILL)
|
if (request != PTRACE_KILL)
|
return -ESRCH;
|
return -ESRCH;
|
}
|
}
|
if (child->p_pptr != current)
|
if (child->p_pptr != current)
|
return -ESRCH;
|
return -ESRCH;
|
|
|
switch (request) {
|
switch (request) {
|
case PTRACE_PEEKTEXT: /* read word at location addr. */
|
case PTRACE_PEEKTEXT: /* read word at location addr. */
|
case PTRACE_PEEKDATA: {
|
case PTRACE_PEEKDATA: {
|
unsigned long tmp;
|
unsigned long tmp;
|
int res;
|
int res;
|
|
|
res = read_long(child, addr, &tmp);
|
res = read_long(child, addr, &tmp);
|
if (res < 0)
|
if (res < 0)
|
return res;
|
return res;
|
res = verify_area(VERIFY_WRITE, (void *) data, sizeof(long));
|
res = verify_area(VERIFY_WRITE, (void *) data, sizeof(long));
|
if (!res)
|
if (!res)
|
put_fs_long(tmp,(unsigned long *) data);
|
put_fs_long(tmp,(unsigned long *) data);
|
return res;
|
return res;
|
}
|
}
|
|
|
case PTRACE_PEEKUSR: { /* read the word at location addr in the USER area. */
|
case PTRACE_PEEKUSR: { /* read the word at location addr in the USER area. */
|
unsigned long tmp;
|
unsigned long tmp;
|
int res;
|
int res;
|
|
|
if ((addr & 3) || addr < 0 || addr >= sizeof(struct user))
|
if ((addr & 3) || addr < 0 || addr >= sizeof(struct user))
|
return -EIO;
|
return -EIO;
|
|
|
res = verify_area(VERIFY_WRITE, (void *) data, sizeof(long));
|
res = verify_area(VERIFY_WRITE, (void *) data, sizeof(long));
|
if (res)
|
if (res)
|
return res;
|
return res;
|
|
|
tmp = 0; /* Default return condition */
|
tmp = 0; /* Default return condition */
|
|
|
if (addr < sizeof (struct pt_regs))
|
if (addr < sizeof (struct pt_regs))
|
tmp = get_stack_long(child, (int)addr >> 2);
|
tmp = get_stack_long(child, (int)addr >> 2);
|
put_fs_long(tmp,(unsigned long *) data);
|
put_fs_long(tmp,(unsigned long *) data);
|
return 0;
|
return 0;
|
}
|
}
|
|
|
case PTRACE_POKETEXT: /* write the word at location addr. */
|
case PTRACE_POKETEXT: /* write the word at location addr. */
|
case PTRACE_POKEDATA:
|
case PTRACE_POKEDATA:
|
return write_long(child,addr,data);
|
return write_long(child,addr,data);
|
|
|
case PTRACE_POKEUSR: /* write the word at location addr in the USER area */
|
case PTRACE_POKEUSR: /* write the word at location addr in the USER area */
|
if ((addr & 3) || addr < 0 || addr >= sizeof(struct user))
|
if ((addr & 3) || addr < 0 || addr >= sizeof(struct user))
|
return -EIO;
|
return -EIO;
|
|
|
if (addr < sizeof (struct pt_regs))
|
if (addr < sizeof (struct pt_regs))
|
put_stack_long(child, (int)addr >> 2, data);
|
put_stack_long(child, (int)addr >> 2, data);
|
else
|
else
|
return -EIO;
|
return -EIO;
|
return 0;
|
return 0;
|
|
|
case PTRACE_SYSCALL: /* continue and stop at next (return from) syscall */
|
case PTRACE_SYSCALL: /* continue and stop at next (return from) syscall */
|
case PTRACE_CONT: /* restart after signal. */
|
case PTRACE_CONT: /* restart after signal. */
|
if ((unsigned long) data > NSIG)
|
if ((unsigned long) data > NSIG)
|
return -EIO;
|
return -EIO;
|
if (request == PTRACE_SYSCALL)
|
if (request == PTRACE_SYSCALL)
|
child->flags |= PF_TRACESYS;
|
child->flags |= PF_TRACESYS;
|
else
|
else
|
child->flags &= ~PF_TRACESYS;
|
child->flags &= ~PF_TRACESYS;
|
child->exit_code = data;
|
child->exit_code = data;
|
wake_up_process (child);
|
wake_up_process (child);
|
/* make sure single-step breakpoint is gone. */
|
/* make sure single-step breakpoint is gone. */
|
ptrace_cancel_bpt (child);
|
ptrace_cancel_bpt (child);
|
return 0;
|
return 0;
|
|
|
/* make the child exit. Best I can do is send it a sigkill.
|
/* 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
|
* perhaps it should be put in the status that it wants to
|
* exit.
|
* exit.
|
*/
|
*/
|
case PTRACE_KILL:
|
case PTRACE_KILL:
|
if (child->state == TASK_ZOMBIE) /* already dead */
|
if (child->state == TASK_ZOMBIE) /* already dead */
|
return 0;
|
return 0;
|
wake_up_process (child);
|
wake_up_process (child);
|
child->exit_code = SIGKILL;
|
child->exit_code = SIGKILL;
|
ptrace_cancel_bpt (child);
|
ptrace_cancel_bpt (child);
|
/* make sure single-step breakpoint is gone. */
|
/* make sure single-step breakpoint is gone. */
|
ptrace_cancel_bpt (child);
|
ptrace_cancel_bpt (child);
|
return 0;
|
return 0;
|
|
|
case PTRACE_SINGLESTEP: /* execute single instruction. */
|
case PTRACE_SINGLESTEP: /* execute single instruction. */
|
if ((unsigned long) data > NSIG)
|
if ((unsigned long) data > NSIG)
|
return -EIO;
|
return -EIO;
|
child->debugreg[4] = -1;
|
child->debugreg[4] = -1;
|
child->flags &= ~PF_TRACESYS;
|
child->flags &= ~PF_TRACESYS;
|
wake_up_process(child);
|
wake_up_process(child);
|
child->exit_code = data;
|
child->exit_code = data;
|
/* give it a chance to run. */
|
/* give it a chance to run. */
|
return 0;
|
return 0;
|
|
|
case PTRACE_DETACH: /* detach a process that was attached. */
|
case PTRACE_DETACH: /* detach a process that was attached. */
|
if ((unsigned long) data > NSIG)
|
if ((unsigned long) data > NSIG)
|
return -EIO;
|
return -EIO;
|
child->flags &= ~(PF_PTRACED|PF_TRACESYS);
|
child->flags &= ~(PF_PTRACED|PF_TRACESYS);
|
wake_up_process (child);
|
wake_up_process (child);
|
child->exit_code = data;
|
child->exit_code = data;
|
REMOVE_LINKS(child);
|
REMOVE_LINKS(child);
|
child->p_pptr = child->p_opptr;
|
child->p_pptr = child->p_opptr;
|
SET_LINKS(child);
|
SET_LINKS(child);
|
/* make sure single-step breakpoint is gone. */
|
/* make sure single-step breakpoint is gone. */
|
ptrace_cancel_bpt (child);
|
ptrace_cancel_bpt (child);
|
return 0;
|
return 0;
|
|
|
default:
|
default:
|
return -EIO;
|
return -EIO;
|
}
|
}
|
}
|
}
|
|
|
asmlinkage void syscall_trace(void)
|
asmlinkage void syscall_trace(void)
|
{
|
{
|
if ((current->flags & (PF_PTRACED|PF_TRACESYS))
|
if ((current->flags & (PF_PTRACED|PF_TRACESYS))
|
!= (PF_PTRACED|PF_TRACESYS))
|
!= (PF_PTRACED|PF_TRACESYS))
|
return;
|
return;
|
current->exit_code = SIGTRAP;
|
current->exit_code = SIGTRAP;
|
current->state = TASK_STOPPED;
|
current->state = TASK_STOPPED;
|
notify_parent(current, SIGCHLD);
|
notify_parent(current, SIGCHLD);
|
schedule();
|
schedule();
|
/*
|
/*
|
* this isn't the same as continuing with a signal, but it will do
|
* 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
|
* for normal use. strace only continues with a signal if the
|
* stopping signal is not SIGTRAP. -brl
|
* stopping signal is not SIGTRAP. -brl
|
*/
|
*/
|
if (current->exit_code)
|
if (current->exit_code)
|
current->signal |= (1 << (current->exit_code - 1));
|
current->signal |= (1 << (current->exit_code - 1));
|
current->exit_code = 0;
|
current->exit_code = 0;
|
}
|
}
|
|
|