mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-11-01 09:13:37 +00:00 
			
		
		
		
	The sys_ptrace boilerplate code (everything outside the big switch statement for the arch-specific requests) is shared by most architectures. This patch moves it to kernel/ptrace.c and leaves the arch-specific code as arch_ptrace. Some architectures have a too different ptrace so we have to exclude them. They continue to keep their implementations. For sh64 I had to add a sh64_ptrace wrapper because it does some initialization on the first call. For um I removed an ifdefed SUBARCH_PTRACE_SPECIAL block, but SUBARCH_PTRACE_SPECIAL isn't defined anywhere in the tree. Signed-off-by: Christoph Hellwig <hch@lst.de> Acked-by: Paul Mackerras <paulus@samba.org> Acked-by: Ralf Baechle <ralf@linux-mips.org> Acked-By: David Howells <dhowells@redhat.com> Acked-by: Russell King <rmk+kernel@arm.linux.org.uk> Acked-by: Paul Mundt <lethal@linux-sh.org> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
		
			
				
	
	
		
			724 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			724 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* ptrace.c: FRV specific parts of process tracing
 | 
						|
 *
 | 
						|
 * Copyright (C) 2003-5 Red Hat, Inc. All Rights Reserved.
 | 
						|
 * Written by David Howells (dhowells@redhat.com)
 | 
						|
 * - Derived from arch/m68k/kernel/ptrace.c
 | 
						|
 *
 | 
						|
 * This program is free software; you can redistribute it and/or
 | 
						|
 * modify it under the terms of the GNU General Public License
 | 
						|
 * as published by the Free Software Foundation; either version
 | 
						|
 * 2 of the License, or (at your option) any later version.
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/kernel.h>
 | 
						|
#include <linux/sched.h>
 | 
						|
#include <linux/mm.h>
 | 
						|
#include <linux/smp.h>
 | 
						|
#include <linux/smp_lock.h>
 | 
						|
#include <linux/errno.h>
 | 
						|
#include <linux/ptrace.h>
 | 
						|
#include <linux/user.h>
 | 
						|
#include <linux/config.h>
 | 
						|
#include <linux/security.h>
 | 
						|
#include <linux/signal.h>
 | 
						|
 | 
						|
#include <asm/uaccess.h>
 | 
						|
#include <asm/page.h>
 | 
						|
#include <asm/pgtable.h>
 | 
						|
#include <asm/system.h>
 | 
						|
#include <asm/processor.h>
 | 
						|
#include <asm/unistd.h>
 | 
						|
 | 
						|
/*
 | 
						|
 * does not yet catch signals sent when the child dies.
 | 
						|
 * in exit.c or in signal.c.
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 * Get contents of register REGNO in task TASK.
 | 
						|
 */
 | 
						|
static inline long get_reg(struct task_struct *task, int regno)
 | 
						|
{
 | 
						|
	struct user_context *user = task->thread.user;
 | 
						|
 | 
						|
	if (regno < 0 || regno >= PT__END)
 | 
						|
		return 0;
 | 
						|
 | 
						|
	return ((unsigned long *) user)[regno];
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Write contents of register REGNO in task TASK.
 | 
						|
 */
 | 
						|
static inline int put_reg(struct task_struct *task, int regno,
 | 
						|
			  unsigned long data)
 | 
						|
{
 | 
						|
	struct user_context *user = task->thread.user;
 | 
						|
 | 
						|
	if (regno < 0 || regno >= PT__END)
 | 
						|
		return -EIO;
 | 
						|
 | 
						|
	switch (regno) {
 | 
						|
	case PT_GR(0):
 | 
						|
		return 0;
 | 
						|
	case PT_PSR:
 | 
						|
	case PT__STATUS:
 | 
						|
		return -EIO;
 | 
						|
	default:
 | 
						|
		((unsigned long *) user)[regno] = data;
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * check that an address falls within the bounds of the target process's memory mappings
 | 
						|
 */
 | 
						|
static inline int is_user_addr_valid(struct task_struct *child,
 | 
						|
				     unsigned long start, unsigned long len)
 | 
						|
{
 | 
						|
#ifdef CONFIG_MMU
 | 
						|
	if (start >= PAGE_OFFSET || len > PAGE_OFFSET - start)
 | 
						|
		return -EIO;
 | 
						|
	return 0;
 | 
						|
#else
 | 
						|
	struct vm_list_struct *vml;
 | 
						|
 | 
						|
	for (vml = child->mm->context.vmlist; vml; vml = vml->next)
 | 
						|
		if (start >= vml->vma->vm_start && start + len <= vml->vma->vm_end)
 | 
						|
			return 0;
 | 
						|
 | 
						|
	return -EIO;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Called by kernel/ptrace.c when detaching..
 | 
						|
 *
 | 
						|
 * Control h/w single stepping
 | 
						|
 */
 | 
						|
void ptrace_disable(struct task_struct *child)
 | 
						|
{
 | 
						|
	child->thread.frame0->__status &= ~REG__STATUS_STEP;
 | 
						|
}
 | 
						|
 | 
						|
void ptrace_enable(struct task_struct *child)
 | 
						|
{
 | 
						|
	child->thread.frame0->__status |= REG__STATUS_STEP;
 | 
						|
}
 | 
						|
 | 
						|
long arch_ptrace(struct task_struct *child, long request, long addr, long data)
 | 
						|
{
 | 
						|
	unsigned long tmp;
 | 
						|
	int ret;
 | 
						|
 | 
						|
	switch (request) {
 | 
						|
		/* when I and D space are separate, these will need to be fixed. */
 | 
						|
	case PTRACE_PEEKTEXT: /* read word at location addr. */
 | 
						|
	case PTRACE_PEEKDATA: {
 | 
						|
		int copied;
 | 
						|
 | 
						|
		ret = -EIO;
 | 
						|
		if (is_user_addr_valid(child, addr, sizeof(tmp)) < 0)
 | 
						|
			break;
 | 
						|
 | 
						|
		copied = access_process_vm(child, addr, &tmp, sizeof(tmp), 0);
 | 
						|
		if (copied != sizeof(tmp))
 | 
						|
			break;
 | 
						|
 | 
						|
		ret = put_user(tmp,(unsigned long *) data);
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
		/* read the word at location addr in the USER area. */
 | 
						|
	case PTRACE_PEEKUSR: {
 | 
						|
		tmp = 0;
 | 
						|
		ret = -EIO;
 | 
						|
		if ((addr & 3) || addr < 0)
 | 
						|
			break;
 | 
						|
 | 
						|
		ret = 0;
 | 
						|
		switch (addr >> 2) {
 | 
						|
		case 0 ... PT__END - 1:
 | 
						|
			tmp = get_reg(child, addr >> 2);
 | 
						|
			break;
 | 
						|
 | 
						|
		case PT__END + 0:
 | 
						|
			tmp = child->mm->end_code - child->mm->start_code;
 | 
						|
			break;
 | 
						|
 | 
						|
		case PT__END + 1:
 | 
						|
			tmp = child->mm->end_data - child->mm->start_data;
 | 
						|
			break;
 | 
						|
 | 
						|
		case PT__END + 2:
 | 
						|
			tmp = child->mm->start_stack - child->mm->start_brk;
 | 
						|
			break;
 | 
						|
 | 
						|
		case PT__END + 3:
 | 
						|
			tmp = child->mm->start_code;
 | 
						|
			break;
 | 
						|
 | 
						|
		case PT__END + 4:
 | 
						|
			tmp = child->mm->start_stack;
 | 
						|
			break;
 | 
						|
 | 
						|
		default:
 | 
						|
			ret = -EIO;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		if (ret == 0)
 | 
						|
			ret = put_user(tmp, (unsigned long *) data);
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
		/* when I and D space are separate, this will have to be fixed. */
 | 
						|
	case PTRACE_POKETEXT: /* write the word at location addr. */
 | 
						|
	case PTRACE_POKEDATA:
 | 
						|
		ret = -EIO;
 | 
						|
		if (is_user_addr_valid(child, addr, sizeof(tmp)) < 0)
 | 
						|
			break;
 | 
						|
		if (access_process_vm(child, addr, &data, sizeof(data), 1) != sizeof(data))
 | 
						|
			break;
 | 
						|
		ret = 0;
 | 
						|
		break;
 | 
						|
 | 
						|
	case PTRACE_POKEUSR: /* write the word at location addr in the USER area */
 | 
						|
		ret = -EIO;
 | 
						|
		if ((addr & 3) || addr < 0)
 | 
						|
			break;
 | 
						|
 | 
						|
		ret = 0;
 | 
						|
		switch (addr >> 2) {
 | 
						|
		case 0 ... PT__END-1:
 | 
						|
			ret = put_reg(child, addr >> 2, data);
 | 
						|
			break;
 | 
						|
 | 
						|
		default:
 | 
						|
			ret = -EIO;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		break;
 | 
						|
 | 
						|
	case PTRACE_SYSCALL: /* continue and stop at next (return from) syscall */
 | 
						|
	case PTRACE_CONT: /* restart after signal. */
 | 
						|
		ret = -EIO;
 | 
						|
		if (!valid_signal(data))
 | 
						|
			break;
 | 
						|
		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;
 | 
						|
		ptrace_disable(child);
 | 
						|
		wake_up_process(child);
 | 
						|
		ret = 0;
 | 
						|
		break;
 | 
						|
 | 
						|
		/* 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:
 | 
						|
		ret = 0;
 | 
						|
		if (child->exit_state == EXIT_ZOMBIE)	/* already dead */
 | 
						|
			break;
 | 
						|
		child->exit_code = SIGKILL;
 | 
						|
		clear_tsk_thread_flag(child, TIF_SINGLESTEP);
 | 
						|
		ptrace_disable(child);
 | 
						|
		wake_up_process(child);
 | 
						|
		break;
 | 
						|
 | 
						|
	case PTRACE_SINGLESTEP:  /* set the trap flag. */
 | 
						|
		ret = -EIO;
 | 
						|
		if (!valid_signal(data))
 | 
						|
			break;
 | 
						|
		clear_tsk_thread_flag(child, TIF_SYSCALL_TRACE);
 | 
						|
		ptrace_enable(child);
 | 
						|
		child->exit_code = data;
 | 
						|
		wake_up_process(child);
 | 
						|
		ret = 0;
 | 
						|
		break;
 | 
						|
 | 
						|
	case PTRACE_DETACH:	/* detach a process that was attached. */
 | 
						|
		ret = ptrace_detach(child, data);
 | 
						|
		break;
 | 
						|
 | 
						|
	case PTRACE_GETREGS: { /* Get all integer regs from the child. */
 | 
						|
		int i;
 | 
						|
		for (i = 0; i < PT__GPEND; i++) {
 | 
						|
			tmp = get_reg(child, i);
 | 
						|
			if (put_user(tmp, (unsigned long *) data)) {
 | 
						|
				ret = -EFAULT;
 | 
						|
				break;
 | 
						|
			}
 | 
						|
			data += sizeof(long);
 | 
						|
		}
 | 
						|
		ret = 0;
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	case PTRACE_SETREGS: { /* Set all integer regs in the child. */
 | 
						|
		int i;
 | 
						|
		for (i = 0; i < PT__GPEND; i++) {
 | 
						|
			if (get_user(tmp, (unsigned long *) data)) {
 | 
						|
				ret = -EFAULT;
 | 
						|
				break;
 | 
						|
			}
 | 
						|
			put_reg(child, i, tmp);
 | 
						|
			data += sizeof(long);
 | 
						|
		}
 | 
						|
		ret = 0;
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	case PTRACE_GETFPREGS: { /* Get the child FP/Media state. */
 | 
						|
		ret = 0;
 | 
						|
		if (copy_to_user((void *) data,
 | 
						|
				 &child->thread.user->f,
 | 
						|
				 sizeof(child->thread.user->f)))
 | 
						|
			ret = -EFAULT;
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	case PTRACE_SETFPREGS: { /* Set the child FP/Media state. */
 | 
						|
		ret = 0;
 | 
						|
		if (copy_from_user(&child->thread.user->f,
 | 
						|
				   (void *) data,
 | 
						|
				   sizeof(child->thread.user->f)))
 | 
						|
			ret = -EFAULT;
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
	case PTRACE_GETFDPIC:
 | 
						|
		tmp = 0;
 | 
						|
		switch (addr) {
 | 
						|
		case PTRACE_GETFDPIC_EXEC:
 | 
						|
			tmp = child->mm->context.exec_fdpic_loadmap;
 | 
						|
			break;
 | 
						|
		case PTRACE_GETFDPIC_INTERP:
 | 
						|
			tmp = child->mm->context.interp_fdpic_loadmap;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		ret = 0;
 | 
						|
		if (put_user(tmp, (unsigned long *) data)) {
 | 
						|
			ret = -EFAULT;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		break;
 | 
						|
 | 
						|
	default:
 | 
						|
		ret = -EIO;
 | 
						|
		break;
 | 
						|
	}
 | 
						|
	return ret;
 | 
						|
}
 | 
						|
 | 
						|
int __nongprelbss kstrace;
 | 
						|
 | 
						|
static const struct {
 | 
						|
	const char	*name;
 | 
						|
	unsigned	argmask;
 | 
						|
} __syscall_name_table[NR_syscalls] = {
 | 
						|
	[0]	= { "restart_syscall"			},
 | 
						|
	[1]	= { "exit",		0x000001	},
 | 
						|
	[2]	= { "fork",		0xffffff	},
 | 
						|
	[3]	= { "read",		0x000141	},
 | 
						|
	[4]	= { "write",		0x000141	},
 | 
						|
	[5]	= { "open",		0x000235	},
 | 
						|
	[6]	= { "close",		0x000001	},
 | 
						|
	[7]	= { "waitpid",		0x000141	},
 | 
						|
	[8]	= { "creat",		0x000025	},
 | 
						|
	[9]	= { "link",		0x000055	},
 | 
						|
	[10]	= { "unlink",		0x000005	},
 | 
						|
	[11]	= { "execve",		0x000445	},
 | 
						|
	[12]	= { "chdir",		0x000005	},
 | 
						|
	[13]	= { "time",		0x000004	},
 | 
						|
	[14]	= { "mknod",		0x000325	},
 | 
						|
	[15]	= { "chmod",		0x000025	},
 | 
						|
	[16]	= { "lchown",		0x000025	},
 | 
						|
	[17]	= { "break" },
 | 
						|
	[18]	= { "oldstat",		0x000045	},
 | 
						|
	[19]	= { "lseek",		0x000131	},
 | 
						|
	[20]	= { "getpid",		0xffffff	},
 | 
						|
	[21]	= { "mount",		0x043555	},
 | 
						|
	[22]	= { "umount",		0x000005	},
 | 
						|
	[23]	= { "setuid",		0x000001	},
 | 
						|
	[24]	= { "getuid",		0xffffff	},
 | 
						|
	[25]	= { "stime",		0x000004	},
 | 
						|
	[26]	= { "ptrace",		0x004413	},
 | 
						|
	[27]	= { "alarm",		0x000001	},
 | 
						|
	[28]	= { "oldfstat",		0x000041	},
 | 
						|
	[29]	= { "pause",		0xffffff	},
 | 
						|
	[30]	= { "utime",		0x000045	},
 | 
						|
	[31]	= { "stty" },
 | 
						|
	[32]	= { "gtty" },
 | 
						|
	[33]	= { "access",		0x000025	},
 | 
						|
	[34]	= { "nice",		0x000001	},
 | 
						|
	[35]	= { "ftime" },
 | 
						|
	[36]	= { "sync",		0xffffff	},
 | 
						|
	[37]	= { "kill",		0x000011	},
 | 
						|
	[38]	= { "rename",		0x000055	},
 | 
						|
	[39]	= { "mkdir",		0x000025	},
 | 
						|
	[40]	= { "rmdir",		0x000005	},
 | 
						|
	[41]	= { "dup",		0x000001	},
 | 
						|
	[42]	= { "pipe",		0x000004	},
 | 
						|
	[43]	= { "times",		0x000004	},
 | 
						|
	[44]	= { "prof" },
 | 
						|
	[45]	= { "brk",		0x000004	},
 | 
						|
	[46]	= { "setgid",		0x000001	},
 | 
						|
	[47]	= { "getgid",		0xffffff	},
 | 
						|
	[48]	= { "signal",		0x000041	},
 | 
						|
	[49]	= { "geteuid",		0xffffff	},
 | 
						|
	[50]	= { "getegid",		0xffffff	},
 | 
						|
	[51]	= { "acct",		0x000005	},
 | 
						|
	[52]	= { "umount2",		0x000035	},
 | 
						|
	[53]	= { "lock" },
 | 
						|
	[54]	= { "ioctl",		0x000331	},
 | 
						|
	[55]	= { "fcntl",		0x000331	},
 | 
						|
	[56]	= { "mpx" },
 | 
						|
	[57]	= { "setpgid",		0x000011	},
 | 
						|
	[58]	= { "ulimit" },
 | 
						|
	[60]	= { "umask",		0x000002	},
 | 
						|
	[61]	= { "chroot",		0x000005	},
 | 
						|
	[62]	= { "ustat",		0x000043	},
 | 
						|
	[63]	= { "dup2",		0x000011	},
 | 
						|
	[64]	= { "getppid",		0xffffff	},
 | 
						|
	[65]	= { "getpgrp",		0xffffff	},
 | 
						|
	[66]	= { "setsid",		0xffffff	},
 | 
						|
	[67]	= { "sigaction" },
 | 
						|
	[68]	= { "sgetmask" },
 | 
						|
	[69]	= { "ssetmask" },
 | 
						|
	[70]	= { "setreuid" },
 | 
						|
	[71]	= { "setregid" },
 | 
						|
	[72]	= { "sigsuspend" },
 | 
						|
	[73]	= { "sigpending" },
 | 
						|
	[74]	= { "sethostname" },
 | 
						|
	[75]	= { "setrlimit" },
 | 
						|
	[76]	= { "getrlimit" },
 | 
						|
	[77]	= { "getrusage" },
 | 
						|
	[78]	= { "gettimeofday" },
 | 
						|
	[79]	= { "settimeofday" },
 | 
						|
	[80]	= { "getgroups" },
 | 
						|
	[81]	= { "setgroups" },
 | 
						|
	[82]	= { "select" },
 | 
						|
	[83]	= { "symlink" },
 | 
						|
	[84]	= { "oldlstat" },
 | 
						|
	[85]	= { "readlink" },
 | 
						|
	[86]	= { "uselib" },
 | 
						|
	[87]	= { "swapon" },
 | 
						|
	[88]	= { "reboot" },
 | 
						|
	[89]	= { "readdir" },
 | 
						|
	[91]	= { "munmap",		0x000034	},
 | 
						|
	[92]	= { "truncate" },
 | 
						|
	[93]	= { "ftruncate" },
 | 
						|
	[94]	= { "fchmod" },
 | 
						|
	[95]	= { "fchown" },
 | 
						|
	[96]	= { "getpriority" },
 | 
						|
	[97]	= { "setpriority" },
 | 
						|
	[99]	= { "statfs" },
 | 
						|
	[100]	= { "fstatfs" },
 | 
						|
	[102]	= { "socketcall" },
 | 
						|
	[103]	= { "syslog" },
 | 
						|
	[104]	= { "setitimer" },
 | 
						|
	[105]	= { "getitimer" },
 | 
						|
	[106]	= { "stat" },
 | 
						|
	[107]	= { "lstat" },
 | 
						|
	[108]	= { "fstat" },
 | 
						|
	[111]	= { "vhangup" },
 | 
						|
	[114]	= { "wait4" },
 | 
						|
	[115]	= { "swapoff" },
 | 
						|
	[116]	= { "sysinfo" },
 | 
						|
	[117]	= { "ipc" },
 | 
						|
	[118]	= { "fsync" },
 | 
						|
	[119]	= { "sigreturn" },
 | 
						|
	[120]	= { "clone" },
 | 
						|
	[121]	= { "setdomainname" },
 | 
						|
	[122]	= { "uname" },
 | 
						|
	[123]	= { "modify_ldt" },
 | 
						|
	[123]	= { "cacheflush" },
 | 
						|
	[124]	= { "adjtimex" },
 | 
						|
	[125]	= { "mprotect" },
 | 
						|
	[126]	= { "sigprocmask" },
 | 
						|
	[127]	= { "create_module" },
 | 
						|
	[128]	= { "init_module" },
 | 
						|
	[129]	= { "delete_module" },
 | 
						|
	[130]	= { "get_kernel_syms" },
 | 
						|
	[131]	= { "quotactl" },
 | 
						|
	[132]	= { "getpgid" },
 | 
						|
	[133]	= { "fchdir" },
 | 
						|
	[134]	= { "bdflush" },
 | 
						|
	[135]	= { "sysfs" },
 | 
						|
	[136]	= { "personality" },
 | 
						|
	[137]	= { "afs_syscall" },
 | 
						|
	[138]	= { "setfsuid" },
 | 
						|
	[139]	= { "setfsgid" },
 | 
						|
	[140]	= { "_llseek",			0x014331	},
 | 
						|
	[141]	= { "getdents" },
 | 
						|
	[142]	= { "_newselect",		0x000141	},
 | 
						|
	[143]	= { "flock" },
 | 
						|
	[144]	= { "msync" },
 | 
						|
	[145]	= { "readv" },
 | 
						|
	[146]	= { "writev" },
 | 
						|
	[147]	= { "getsid",			0x000001	},
 | 
						|
	[148]	= { "fdatasync",		0x000001	},
 | 
						|
	[149]	= { "_sysctl",			0x000004	},
 | 
						|
	[150]	= { "mlock" },
 | 
						|
	[151]	= { "munlock" },
 | 
						|
	[152]	= { "mlockall" },
 | 
						|
	[153]	= { "munlockall" },
 | 
						|
	[154]	= { "sched_setparam" },
 | 
						|
	[155]	= { "sched_getparam" },
 | 
						|
	[156]	= { "sched_setscheduler" },
 | 
						|
	[157]	= { "sched_getscheduler" },
 | 
						|
	[158]	= { "sched_yield" },
 | 
						|
	[159]	= { "sched_get_priority_max" },
 | 
						|
	[160]	= { "sched_get_priority_min" },
 | 
						|
	[161]	= { "sched_rr_get_interval" },
 | 
						|
	[162]	= { "nanosleep",		0x000044	},
 | 
						|
	[163]	= { "mremap" },
 | 
						|
	[164]	= { "setresuid" },
 | 
						|
	[165]	= { "getresuid" },
 | 
						|
	[166]	= { "vm86" },
 | 
						|
	[167]	= { "query_module" },
 | 
						|
	[168]	= { "poll" },
 | 
						|
	[169]	= { "nfsservctl" },
 | 
						|
	[170]	= { "setresgid" },
 | 
						|
	[171]	= { "getresgid" },
 | 
						|
	[172]	= { "prctl",			0x333331	},
 | 
						|
	[173]	= { "rt_sigreturn",		0xffffff	},
 | 
						|
	[174]	= { "rt_sigaction",		0x001441	},
 | 
						|
	[175]	= { "rt_sigprocmask",		0x001441	},
 | 
						|
	[176]	= { "rt_sigpending",		0x000014	},
 | 
						|
	[177]	= { "rt_sigtimedwait",		0x001444	},
 | 
						|
	[178]	= { "rt_sigqueueinfo",		0x000411	},
 | 
						|
	[179]	= { "rt_sigsuspend",		0x000014	},
 | 
						|
	[180]	= { "pread",			0x003341	},
 | 
						|
	[181]	= { "pwrite",			0x003341	},
 | 
						|
	[182]	= { "chown",			0x000115	},
 | 
						|
	[183]	= { "getcwd" },
 | 
						|
	[184]	= { "capget" },
 | 
						|
	[185]	= { "capset" },
 | 
						|
	[186]	= { "sigaltstack" },
 | 
						|
	[187]	= { "sendfile" },
 | 
						|
	[188]	= { "getpmsg" },
 | 
						|
	[189]	= { "putpmsg" },
 | 
						|
	[190]	= { "vfork",			0xffffff	},
 | 
						|
	[191]	= { "ugetrlimit" },
 | 
						|
	[192]	= { "mmap2",			0x313314	},
 | 
						|
	[193]	= { "truncate64" },
 | 
						|
	[194]	= { "ftruncate64" },
 | 
						|
	[195]	= { "stat64",			0x000045	},
 | 
						|
	[196]	= { "lstat64",			0x000045	},
 | 
						|
	[197]	= { "fstat64",			0x000041	},
 | 
						|
	[198]	= { "lchown32" },
 | 
						|
	[199]	= { "getuid32",			0xffffff	},
 | 
						|
	[200]	= { "getgid32",			0xffffff	},
 | 
						|
	[201]	= { "geteuid32",		0xffffff	},
 | 
						|
	[202]	= { "getegid32",		0xffffff	},
 | 
						|
	[203]	= { "setreuid32" },
 | 
						|
	[204]	= { "setregid32" },
 | 
						|
	[205]	= { "getgroups32" },
 | 
						|
	[206]	= { "setgroups32" },
 | 
						|
	[207]	= { "fchown32" },
 | 
						|
	[208]	= { "setresuid32" },
 | 
						|
	[209]	= { "getresuid32" },
 | 
						|
	[210]	= { "setresgid32" },
 | 
						|
	[211]	= { "getresgid32" },
 | 
						|
	[212]	= { "chown32" },
 | 
						|
	[213]	= { "setuid32" },
 | 
						|
	[214]	= { "setgid32" },
 | 
						|
	[215]	= { "setfsuid32" },
 | 
						|
	[216]	= { "setfsgid32" },
 | 
						|
	[217]	= { "pivot_root" },
 | 
						|
	[218]	= { "mincore" },
 | 
						|
	[219]	= { "madvise" },
 | 
						|
	[220]	= { "getdents64" },
 | 
						|
	[221]	= { "fcntl64" },
 | 
						|
	[223]	= { "security" },
 | 
						|
	[224]	= { "gettid" },
 | 
						|
	[225]	= { "readahead" },
 | 
						|
	[226]	= { "setxattr" },
 | 
						|
	[227]	= { "lsetxattr" },
 | 
						|
	[228]	= { "fsetxattr" },
 | 
						|
	[229]	= { "getxattr" },
 | 
						|
	[230]	= { "lgetxattr" },
 | 
						|
	[231]	= { "fgetxattr" },
 | 
						|
	[232]	= { "listxattr" },
 | 
						|
	[233]	= { "llistxattr" },
 | 
						|
	[234]	= { "flistxattr" },
 | 
						|
	[235]	= { "removexattr" },
 | 
						|
	[236]	= { "lremovexattr" },
 | 
						|
	[237]	= { "fremovexattr" },
 | 
						|
	[238]	= { "tkill" },
 | 
						|
	[239]	= { "sendfile64" },
 | 
						|
	[240]	= { "futex" },
 | 
						|
	[241]	= { "sched_setaffinity" },
 | 
						|
	[242]	= { "sched_getaffinity" },
 | 
						|
	[243]	= { "set_thread_area" },
 | 
						|
	[244]	= { "get_thread_area" },
 | 
						|
	[245]	= { "io_setup" },
 | 
						|
	[246]	= { "io_destroy" },
 | 
						|
	[247]	= { "io_getevents" },
 | 
						|
	[248]	= { "io_submit" },
 | 
						|
	[249]	= { "io_cancel" },
 | 
						|
	[250]	= { "fadvise64" },
 | 
						|
	[252]	= { "exit_group",		0x000001	},
 | 
						|
	[253]	= { "lookup_dcookie" },
 | 
						|
	[254]	= { "epoll_create" },
 | 
						|
	[255]	= { "epoll_ctl" },
 | 
						|
	[256]	= { "epoll_wait" },
 | 
						|
	[257]	= { "remap_file_pages" },
 | 
						|
	[258]	= { "set_tid_address" },
 | 
						|
	[259]	= { "timer_create" },
 | 
						|
	[260]	= { "timer_settime" },
 | 
						|
	[261]	= { "timer_gettime" },
 | 
						|
	[262]	= { "timer_getoverrun" },
 | 
						|
	[263]	= { "timer_delete" },
 | 
						|
	[264]	= { "clock_settime" },
 | 
						|
	[265]	= { "clock_gettime" },
 | 
						|
	[266]	= { "clock_getres" },
 | 
						|
	[267]	= { "clock_nanosleep" },
 | 
						|
	[268]	= { "statfs64" },
 | 
						|
	[269]	= { "fstatfs64" },
 | 
						|
	[270]	= { "tgkill" },
 | 
						|
	[271]	= { "utimes" },
 | 
						|
	[272]	= { "fadvise64_64" },
 | 
						|
	[273]	= { "vserver" },
 | 
						|
	[274]	= { "mbind" },
 | 
						|
	[275]	= { "get_mempolicy" },
 | 
						|
	[276]	= { "set_mempolicy" },
 | 
						|
	[277]	= { "mq_open" },
 | 
						|
	[278]	= { "mq_unlink" },
 | 
						|
	[279]	= { "mq_timedsend" },
 | 
						|
	[280]	= { "mq_timedreceive" },
 | 
						|
	[281]	= { "mq_notify" },
 | 
						|
	[282]	= { "mq_getsetattr" },
 | 
						|
	[283]	= { "sys_kexec_load" },
 | 
						|
};
 | 
						|
 | 
						|
asmlinkage void do_syscall_trace(int leaving)
 | 
						|
{
 | 
						|
#if 0
 | 
						|
	unsigned long *argp;
 | 
						|
	const char *name;
 | 
						|
	unsigned argmask;
 | 
						|
	char buffer[16];
 | 
						|
 | 
						|
	if (!kstrace)
 | 
						|
		return;
 | 
						|
 | 
						|
	if (!current->mm)
 | 
						|
		return;
 | 
						|
 | 
						|
	if (__frame->gr7 == __NR_close)
 | 
						|
		return;
 | 
						|
 | 
						|
#if 0
 | 
						|
	if (__frame->gr7 != __NR_mmap2 &&
 | 
						|
	    __frame->gr7 != __NR_vfork &&
 | 
						|
	    __frame->gr7 != __NR_execve &&
 | 
						|
	    __frame->gr7 != __NR_exit)
 | 
						|
		return;
 | 
						|
#endif
 | 
						|
 | 
						|
	argmask = 0;
 | 
						|
	name = NULL;
 | 
						|
	if (__frame->gr7 < NR_syscalls) {
 | 
						|
		name = __syscall_name_table[__frame->gr7].name;
 | 
						|
		argmask = __syscall_name_table[__frame->gr7].argmask;
 | 
						|
	}
 | 
						|
	if (!name) {
 | 
						|
		sprintf(buffer, "sys_%lx", __frame->gr7);
 | 
						|
		name = buffer;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!leaving) {
 | 
						|
		if (!argmask) {
 | 
						|
			printk(KERN_CRIT "[%d] %s(%lx,%lx,%lx,%lx,%lx,%lx)\n",
 | 
						|
			       current->pid,
 | 
						|
			       name,
 | 
						|
			       __frame->gr8,
 | 
						|
			       __frame->gr9,
 | 
						|
			       __frame->gr10,
 | 
						|
			       __frame->gr11,
 | 
						|
			       __frame->gr12,
 | 
						|
			       __frame->gr13);
 | 
						|
		}
 | 
						|
		else if (argmask == 0xffffff) {
 | 
						|
			printk(KERN_CRIT "[%d] %s()\n",
 | 
						|
			       current->pid,
 | 
						|
			       name);
 | 
						|
		}
 | 
						|
		else {
 | 
						|
			printk(KERN_CRIT "[%d] %s(",
 | 
						|
			       current->pid,
 | 
						|
			       name);
 | 
						|
 | 
						|
			argp = &__frame->gr8;
 | 
						|
 | 
						|
			do {
 | 
						|
				switch (argmask & 0xf) {
 | 
						|
				case 1:
 | 
						|
					printk("%ld", (long) *argp);
 | 
						|
					break;
 | 
						|
				case 2:
 | 
						|
					printk("%lo", *argp);
 | 
						|
					break;
 | 
						|
				case 3:
 | 
						|
					printk("%lx", *argp);
 | 
						|
					break;
 | 
						|
				case 4:
 | 
						|
					printk("%p", (void *) *argp);
 | 
						|
					break;
 | 
						|
				case 5:
 | 
						|
					printk("\"%s\"", (char *) *argp);
 | 
						|
					break;
 | 
						|
				}
 | 
						|
 | 
						|
				argp++;
 | 
						|
				argmask >>= 4;
 | 
						|
				if (argmask)
 | 
						|
					printk(",");
 | 
						|
 | 
						|
			} while (argmask);
 | 
						|
 | 
						|
			printk(")\n");
 | 
						|
		}
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		if ((int)__frame->gr8 > -4096 && (int)__frame->gr8 < 4096)
 | 
						|
			printk(KERN_CRIT "[%d] %s() = %ld\n", current->pid, name, __frame->gr8);
 | 
						|
		else
 | 
						|
			printk(KERN_CRIT "[%d] %s() = %lx\n", current->pid, name, __frame->gr8);
 | 
						|
	}
 | 
						|
	return;
 | 
						|
#endif
 | 
						|
 | 
						|
	if (!test_thread_flag(TIF_SYSCALL_TRACE))
 | 
						|
		return;
 | 
						|
 | 
						|
	if (!(current->ptrace & PT_PTRACED))
 | 
						|
		return;
 | 
						|
 | 
						|
	/* we need to indicate entry or exit to strace */
 | 
						|
	if (leaving)
 | 
						|
		__frame->__status |= REG__STATUS_SYSC_EXIT;
 | 
						|
	else
 | 
						|
		__frame->__status |= REG__STATUS_SYSC_ENTRY;
 | 
						|
 | 
						|
	ptrace_notify(SIGTRAP);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * 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;
 | 
						|
	}
 | 
						|
}
 |