[PATCH] uml: S390 preparation, peekusr/pokeusr defined by subarch
s390 needs to change some parts of arch/um/kernel/ptrace.c. Thus, the code regarding PEEKUSER and POKEUSER are shifted to arch/um/sys-<subarch>/ptrace.c. Also s390 debug registers need to be updated, when singlestepping is switched on / off. Thus, setting/resetting of singlestepping is centralized in the new function set_singlestep(), which also inserts the macro SUBARCH_SET_SINGLESTEP(mode), if defined. Finally, s390 has the "ieee_instruction_pointer" in its registers, which also is allowed to be read via ptrace( PTRACE_PEEKUSER, getpid(), PT_IEEE_IP, 0); To implement this feature, sys_ptrace inserts the macro SUBARCH_PTRACE_SPECIAL, if defined. Signed-off-by: Bodo Stroesser <bstroesser@fujitsu-siemens.com> Signed-off-by: Jeff Dike <jdike@addtoit.com> Cc: Paolo 'Blaisorblade' Giarrusso <blaisorblade@yahoo.it> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>wifi-calibration
parent
16c1116301
commit
82c1c11bdd
|
@ -19,15 +19,30 @@
|
||||||
#include "skas_ptrace.h"
|
#include "skas_ptrace.h"
|
||||||
#include "sysdep/ptrace.h"
|
#include "sysdep/ptrace.h"
|
||||||
|
|
||||||
|
static inline void set_singlestepping(struct task_struct *child, int on)
|
||||||
|
{
|
||||||
|
if (on)
|
||||||
|
child->ptrace |= PT_DTRACE;
|
||||||
|
else
|
||||||
|
child->ptrace &= ~PT_DTRACE;
|
||||||
|
child->thread.singlestep_syscall = 0;
|
||||||
|
|
||||||
|
#ifdef SUBARCH_SET_SINGLESTEPPING
|
||||||
|
SUBARCH_SET_SINGLESTEPPING(child, on)
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Called by kernel/ptrace.c when detaching..
|
* Called by kernel/ptrace.c when detaching..
|
||||||
*/
|
*/
|
||||||
void ptrace_disable(struct task_struct *child)
|
void ptrace_disable(struct task_struct *child)
|
||||||
{
|
{
|
||||||
child->ptrace &= ~PT_DTRACE;
|
set_singlestepping(child,0);
|
||||||
child->thread.singlestep_syscall = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern int peek_user(struct task_struct * child, long addr, long data);
|
||||||
|
extern int poke_user(struct task_struct * child, long addr, long data);
|
||||||
|
|
||||||
long sys_ptrace(long request, long pid, long addr, long data)
|
long sys_ptrace(long request, long pid, long addr, long data)
|
||||||
{
|
{
|
||||||
struct task_struct *child;
|
struct task_struct *child;
|
||||||
|
@ -67,6 +82,10 @@ long sys_ptrace(long request, long pid, long addr, long data)
|
||||||
goto out_tsk;
|
goto out_tsk;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef SUBACH_PTRACE_SPECIAL
|
||||||
|
SUBARCH_PTRACE_SPECIAL(child,request,addr,data)
|
||||||
|
#endif
|
||||||
|
|
||||||
ret = ptrace_check_attach(child, request == PTRACE_KILL);
|
ret = ptrace_check_attach(child, request == PTRACE_KILL);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out_tsk;
|
goto out_tsk;
|
||||||
|
@ -87,29 +106,10 @@ long sys_ptrace(long request, long pid, long addr, long data)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* read the word at location addr in the USER area. */
|
/* read the word at location addr in the USER area. */
|
||||||
case PTRACE_PEEKUSR: {
|
case PTRACE_PEEKUSR:
|
||||||
unsigned long tmp;
|
ret = peek_user(child, addr, data);
|
||||||
|
|
||||||
ret = -EIO;
|
|
||||||
if ((addr & 3) || addr < 0)
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
tmp = 0; /* Default return condition */
|
|
||||||
if(addr < MAX_REG_OFFSET){
|
|
||||||
tmp = getreg(child, addr);
|
|
||||||
}
|
|
||||||
#if defined(CONFIG_UML_X86) && !defined(CONFIG_64BIT)
|
|
||||||
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
|
||||||
(addr <= offsetof(struct user, u_debugreg[7]))){
|
|
||||||
addr -= offsetof(struct user, u_debugreg[0]);
|
|
||||||
addr = addr >> 2;
|
|
||||||
tmp = child->thread.arch.debugregs[addr];
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
ret = put_user(tmp, (unsigned long __user *) data);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* when I and D space are separate, this will have to be fixed. */
|
/* when I and D space are separate, this will have to be fixed. */
|
||||||
case PTRACE_POKETEXT: /* write the word at location addr. */
|
case PTRACE_POKETEXT: /* write the word at location addr. */
|
||||||
case PTRACE_POKEDATA:
|
case PTRACE_POKEDATA:
|
||||||
|
@ -121,25 +121,7 @@ long sys_ptrace(long request, long pid, long addr, long data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
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 */
|
||||||
ret = -EIO;
|
ret = poke_user(child, addr, data);
|
||||||
if ((addr & 3) || addr < 0)
|
|
||||||
break;
|
|
||||||
|
|
||||||
if (addr < MAX_REG_OFFSET) {
|
|
||||||
ret = putreg(child, addr, data);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#if defined(CONFIG_UML_X86) && !defined(CONFIG_64BIT)
|
|
||||||
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
|
||||||
(addr <= offsetof(struct user, u_debugreg[7]))){
|
|
||||||
addr -= offsetof(struct user, u_debugreg[0]);
|
|
||||||
addr = addr >> 2;
|
|
||||||
if((addr == 4) || (addr == 5)) break;
|
|
||||||
child->thread.arch.debugregs[addr] = data;
|
|
||||||
ret = 0;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PTRACE_SYSCALL: /* continue and stop at next (return from) syscall */
|
case PTRACE_SYSCALL: /* continue and stop at next (return from) syscall */
|
||||||
|
@ -148,8 +130,7 @@ long sys_ptrace(long request, long pid, long addr, long data)
|
||||||
if (!valid_signal(data))
|
if (!valid_signal(data))
|
||||||
break;
|
break;
|
||||||
|
|
||||||
child->ptrace &= ~PT_DTRACE;
|
set_singlestepping(child, 0);
|
||||||
child->thread.singlestep_syscall = 0;
|
|
||||||
if (request == PTRACE_SYSCALL) {
|
if (request == PTRACE_SYSCALL) {
|
||||||
set_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
|
set_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
|
||||||
}
|
}
|
||||||
|
@ -172,8 +153,7 @@ long sys_ptrace(long request, long pid, long addr, long data)
|
||||||
if (child->exit_state == EXIT_ZOMBIE) /* already dead */
|
if (child->exit_state == EXIT_ZOMBIE) /* already dead */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
child->ptrace &= ~PT_DTRACE;
|
set_singlestepping(child, 0);
|
||||||
child->thread.singlestep_syscall = 0;
|
|
||||||
child->exit_code = SIGKILL;
|
child->exit_code = SIGKILL;
|
||||||
wake_up_process(child);
|
wake_up_process(child);
|
||||||
break;
|
break;
|
||||||
|
@ -184,8 +164,7 @@ long sys_ptrace(long request, long pid, long addr, long data)
|
||||||
if (!valid_signal(data))
|
if (!valid_signal(data))
|
||||||
break;
|
break;
|
||||||
clear_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
|
clear_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
|
||||||
child->ptrace |= PT_DTRACE;
|
set_singlestepping(child, 1);
|
||||||
child->thread.singlestep_syscall = 0;
|
|
||||||
child->exit_code = data;
|
child->exit_code = data;
|
||||||
/* give it a chance to run. */
|
/* give it a chance to run. */
|
||||||
wake_up_process(child);
|
wake_up_process(child);
|
||||||
|
|
|
@ -73,6 +73,25 @@ int putreg(struct task_struct *child, int regno, unsigned long value)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int poke_user(struct task_struct *child, long addr, long data)
|
||||||
|
{
|
||||||
|
if ((addr & 3) || addr < 0)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
if (addr < MAX_REG_OFFSET)
|
||||||
|
return putreg(child, addr, data);
|
||||||
|
|
||||||
|
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
||||||
|
(addr <= offsetof(struct user, u_debugreg[7]))){
|
||||||
|
addr -= offsetof(struct user, u_debugreg[0]);
|
||||||
|
addr = addr >> 2;
|
||||||
|
if((addr == 4) || (addr == 5)) return -EIO;
|
||||||
|
child->thread.arch.debugregs[addr] = data;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
unsigned long getreg(struct task_struct *child, int regno)
|
unsigned long getreg(struct task_struct *child, int regno)
|
||||||
{
|
{
|
||||||
unsigned long retval = ~0UL;
|
unsigned long retval = ~0UL;
|
||||||
|
@ -93,6 +112,27 @@ unsigned long getreg(struct task_struct *child, int regno)
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int peek_user(struct task_struct *child, long addr, long data)
|
||||||
|
{
|
||||||
|
/* read the word at location addr in the USER area. */
|
||||||
|
unsigned long tmp;
|
||||||
|
|
||||||
|
if ((addr & 3) || addr < 0)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
tmp = 0; /* Default return condition */
|
||||||
|
if(addr < MAX_REG_OFFSET){
|
||||||
|
tmp = getreg(child, addr);
|
||||||
|
}
|
||||||
|
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
||||||
|
(addr <= offsetof(struct user, u_debugreg[7]))){
|
||||||
|
addr -= offsetof(struct user, u_debugreg[0]);
|
||||||
|
addr = addr >> 2;
|
||||||
|
tmp = child->thread.arch.debugregs[addr];
|
||||||
|
}
|
||||||
|
return put_user(tmp, (unsigned long *) data);
|
||||||
|
}
|
||||||
|
|
||||||
struct i387_fxsave_struct {
|
struct i387_fxsave_struct {
|
||||||
unsigned short cwd;
|
unsigned short cwd;
|
||||||
unsigned short swd;
|
unsigned short swd;
|
||||||
|
|
|
@ -8,6 +8,25 @@ int putreg(struct task_struct *child, unsigned long regno,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int poke_user(struct task_struct *child, long addr, long data)
|
||||||
|
{
|
||||||
|
if ((addr & 3) || addr < 0)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
if (addr < MAX_REG_OFFSET)
|
||||||
|
return putreg(child, addr, data);
|
||||||
|
|
||||||
|
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
||||||
|
(addr <= offsetof(struct user, u_debugreg[7]))){
|
||||||
|
addr -= offsetof(struct user, u_debugreg[0]);
|
||||||
|
addr = addr >> 2;
|
||||||
|
if((addr == 4) || (addr == 5)) return -EIO;
|
||||||
|
child->thread.arch.debugregs[addr] = data;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
unsigned long getreg(struct task_struct *child, unsigned long regno)
|
unsigned long getreg(struct task_struct *child, unsigned long regno)
|
||||||
{
|
{
|
||||||
unsigned long retval = ~0UL;
|
unsigned long retval = ~0UL;
|
||||||
|
@ -16,6 +35,27 @@ unsigned long getreg(struct task_struct *child, unsigned long regno)
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int peek_user(struct task_struct *child, long addr, long data)
|
||||||
|
{
|
||||||
|
/* read the word at location addr in the USER area. */
|
||||||
|
unsigned long tmp;
|
||||||
|
|
||||||
|
if ((addr & 3) || addr < 0)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
tmp = 0; /* Default return condition */
|
||||||
|
if(addr < MAX_REG_OFFSET){
|
||||||
|
tmp = getreg(child, addr);
|
||||||
|
}
|
||||||
|
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
||||||
|
(addr <= offsetof(struct user, u_debugreg[7]))){
|
||||||
|
addr -= offsetof(struct user, u_debugreg[0]);
|
||||||
|
addr = addr >> 2;
|
||||||
|
tmp = child->thread.arch.debugregs[addr];
|
||||||
|
}
|
||||||
|
return put_user(tmp, (unsigned long *) data);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Overrides for Emacs so that we follow Linus's tabbing style.
|
* Overrides for Emacs so that we follow Linus's tabbing style.
|
||||||
* Emacs will notice this stuff at the end of the file and automatically
|
* Emacs will notice this stuff at the end of the file and automatically
|
||||||
|
|
|
@ -62,6 +62,27 @@ int putreg(struct task_struct *child, int regno, unsigned long value)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int poke_user(struct task_struct *child, long addr, long data)
|
||||||
|
{
|
||||||
|
if ((addr & 3) || addr < 0)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
if (addr < MAX_REG_OFFSET)
|
||||||
|
return putreg(child, addr, data);
|
||||||
|
|
||||||
|
#if 0 /* Need x86_64 debugregs handling */
|
||||||
|
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
||||||
|
(addr <= offsetof(struct user, u_debugreg[7]))){
|
||||||
|
addr -= offsetof(struct user, u_debugreg[0]);
|
||||||
|
addr = addr >> 2;
|
||||||
|
if((addr == 4) || (addr == 5)) return -EIO;
|
||||||
|
child->thread.arch.debugregs[addr] = data;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
unsigned long getreg(struct task_struct *child, int regno)
|
unsigned long getreg(struct task_struct *child, int regno)
|
||||||
{
|
{
|
||||||
unsigned long retval = ~0UL;
|
unsigned long retval = ~0UL;
|
||||||
|
@ -84,6 +105,29 @@ unsigned long getreg(struct task_struct *child, int regno)
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int peek_user(struct task_struct *child, long addr, long data)
|
||||||
|
{
|
||||||
|
/* read the word at location addr in the USER area. */
|
||||||
|
unsigned long tmp;
|
||||||
|
|
||||||
|
if ((addr & 3) || addr < 0)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
tmp = 0; /* Default return condition */
|
||||||
|
if(addr < MAX_REG_OFFSET){
|
||||||
|
tmp = getreg(child, addr);
|
||||||
|
}
|
||||||
|
#if 0 /* Need x86_64 debugregs handling */
|
||||||
|
else if((addr >= offsetof(struct user, u_debugreg[0])) &&
|
||||||
|
(addr <= offsetof(struct user, u_debugreg[7]))){
|
||||||
|
addr -= offsetof(struct user, u_debugreg[0]);
|
||||||
|
addr = addr >> 2;
|
||||||
|
tmp = child->thread.arch.debugregs[addr];
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return put_user(tmp, (unsigned long *) data);
|
||||||
|
}
|
||||||
|
|
||||||
void arch_switch(void)
|
void arch_switch(void)
|
||||||
{
|
{
|
||||||
/* XXX
|
/* XXX
|
||||||
|
|
Loading…
Reference in New Issue