2  *  linux/arch/h8300/kernel/ptrace.c
 
   4  *  Yoshinori Sato <ysato@users.sourceforge.jp>
 
   7  *  linux/arch/m68k/kernel/ptrace.c
 
   9  *  Copyright (C) 1994 by Hamish Macdonald
 
  10  *  Taken from linux/kernel/ptrace.c and modified for M680x0.
 
  11  *  linux/kernel/ptrace.c is by Ross Biro 1/23/92, edited by Linus Torvalds
 
  13  * This file is subject to the terms and conditions of the GNU General
 
  14  * Public License.  See the file COPYING in the main directory of
 
  15  * this archive for more details.
 
  18 #include <linux/kernel.h>
 
  19 #include <linux/sched.h>
 
  21 #include <linux/smp.h>
 
  22 #include <linux/smp_lock.h>
 
  23 #include <linux/errno.h>
 
  24 #include <linux/ptrace.h>
 
  25 #include <linux/user.h>
 
  26 #include <linux/config.h>
 
  27 #include <linux/signal.h>
 
  29 #include <asm/uaccess.h>
 
  31 #include <asm/pgtable.h>
 
  32 #include <asm/system.h>
 
  33 #include <asm/processor.h>
 
  34 #include <asm/signal.h>
 
  36 /* cpu depend functions */
 
  37 extern long h8300_get_reg(struct task_struct *task, int regno);
 
  38 extern int  h8300_put_reg(struct task_struct *task, int regno, unsigned long data);
 
  39 extern void h8300_disable_trace(struct task_struct *child);
 
  40 extern void h8300_enable_trace(struct task_struct *child);
 
  43  * does not yet catch signals sent when the child dies.
 
  44  * in exit.c or in signal.c.
 
  48 static int read_long(struct task_struct * tsk, unsigned long addr,
 
  49         unsigned long * result)
 
  51         *result = *(unsigned long *)addr;
 
  55 void ptrace_disable(struct task_struct *child)
 
  57         h8300_disable_trace(child);
 
  60 asmlinkage int sys_ptrace(long request, long pid, long addr, long data)
 
  62         struct task_struct *child;
 
  67         if (request == PTRACE_TRACEME) {
 
  68                 /* are we already being traced? */
 
  69                 if (current->ptrace & PT_PTRACED)
 
  71                 /* set the ptrace bit in the process flags. */
 
  72                 current->ptrace |= PT_PTRACED;
 
  77         read_lock(&tasklist_lock);
 
  78         child = find_task_by_pid(pid);
 
  80                 get_task_struct(child);
 
  81         read_unlock(&tasklist_lock);
 
  86         if (pid == 1)           /* you may not mess with init */
 
  89         if (request == PTRACE_ATTACH) {
 
  90                 ret = ptrace_attach(child);
 
  93         ret = ptrace_check_attach(child, request == PTRACE_KILL);
 
  98                 case PTRACE_PEEKTEXT: /* read word at location addr. */ 
 
  99                 case PTRACE_PEEKDATA: {
 
 102                         ret = read_long(child, addr, &tmp);
 
 105                         ret = put_user(tmp, (unsigned long *) data);
 
 109         /* read the word at location addr in the USER area. */
 
 110                 case PTRACE_PEEKUSR: {
 
 111                         unsigned long tmp = 0;
 
 113                         if ((addr & 3) || addr < 0 || addr >= sizeof(struct user)) {
 
 118                         ret = 0;  /* Default return condition */
 
 119                         addr = addr >> 2; /* temporary hack. */
 
 121                         if (addr < H8300_REGS_NO)
 
 122                                 tmp = h8300_get_reg(child, addr);
 
 126                                         tmp = child->mm->start_code;
 
 129                                         tmp = child->mm->start_data;
 
 132                                         tmp = child->mm->end_code;
 
 135                                         tmp = child->mm->end_data;
 
 142                                 ret = put_user(tmp,(unsigned long *) data);
 
 146       /* when I and D space are separate, this will have to be fixed. */
 
 147                 case PTRACE_POKETEXT: /* write the word at location addr. */
 
 148                 case PTRACE_POKEDATA:
 
 150                         if (access_process_vm(child, addr, &data, sizeof(data), 1) == sizeof(data))
 
 155                 case PTRACE_POKEUSR: /* write the word at location addr in the USER area */
 
 156                         if ((addr & 3) || addr < 0 || addr >= sizeof(struct user)) {
 
 160                         addr = addr >> 2; /* temporary hack. */
 
 162                         if (addr == PT_ORIG_ER0) {
 
 166                         if (addr < H8300_REGS_NO) {
 
 167                                 ret = h8300_put_reg(child, addr, data);
 
 172                 case PTRACE_SYSCALL: /* continue and stop at next (return from) syscall */
 
 173                 case PTRACE_CONT: { /* restart after signal. */
 
 175                         if (!valid_signal(data))
 
 177                         if (request == PTRACE_SYSCALL)
 
 178                                 set_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
 
 180                                 clear_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
 
 181                         child->exit_code = data;
 
 182                         wake_up_process(child);
 
 183                         /* make sure the single step bit is not set. */
 
 184                         h8300_disable_trace(child);
 
 189  * make the child exit.  Best I can do is send it a sigkill. 
 
 190  * perhaps it should be put in the status that it wants to 
 
 196                         if (child->exit_state == EXIT_ZOMBIE) /* already dead */
 
 198                         child->exit_code = SIGKILL;
 
 199                         h8300_disable_trace(child);
 
 200                         wake_up_process(child);
 
 204                 case PTRACE_SINGLESTEP: {  /* set the trap flag. */
 
 206                         if (!valid_signal(data))
 
 208                         clear_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
 
 209                         child->exit_code = data;
 
 210                         h8300_enable_trace(child);
 
 211                         wake_up_process(child);
 
 216                 case PTRACE_DETACH:     /* detach a process that was attached. */
 
 217                         ret = ptrace_detach(child, data);
 
 220                 case PTRACE_GETREGS: { /* Get all gp regs from the child. */
 
 223                         for (i = 0; i < H8300_REGS_NO; i++) {
 
 224                             tmp = h8300_get_reg(child, i);
 
 225                             if (put_user(tmp, (unsigned long *) data)) {
 
 229                             data += sizeof(long);
 
 235                 case PTRACE_SETREGS: { /* Set all gp regs in the child. */
 
 238                         for (i = 0; i < H8300_REGS_NO; i++) {
 
 239                             if (get_user(tmp, (unsigned long *) data)) {
 
 243                             h8300_put_reg(child, i, tmp);
 
 244                             data += sizeof(long);
 
 255         put_task_struct(child);
 
 261 asmlinkage void syscall_trace(void)
 
 263         if (!test_thread_flag(TIF_SYSCALL_TRACE))
 
 265         if (!(current->ptrace & PT_PTRACED))
 
 267         ptrace_notify(SIGTRAP | ((current->ptrace & PT_TRACESYSGOOD)
 
 270          * this isn't the same as continuing with a signal, but it will do
 
 271          * for normal use.  strace only continues with a signal if the
 
 272          * stopping signal is not SIGTRAP.  -brl
 
 274         if (current->exit_code) {
 
 275                 send_sig(current->exit_code, current, 1);
 
 276                 current->exit_code = 0;