forked from mirrors/linux
		
	x86/arch_prctl/64: Rename do_arch_prctl() to do_arch_prctl_64()
In order to introduce new arch_prctls that are not 64 bit only, rename the existing 64 bit implementation to do_arch_prctl_64(). Also rename the second argument of that function from 'addr' to 'arg2', because it will no longer always be an address. Signed-off-by: Kyle Huey <khuey@kylehuey.com> Reviewed-by: Andy Lutomirski <luto@kernel.org> Cc: Grzegorz Andrejczuk <grzegorz.andrejczuk@intel.com> Cc: kvm@vger.kernel.org Cc: Radim Krčmář <rkrcmar@redhat.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Dave Hansen <dave.hansen@linux.intel.com> Cc: Andi Kleen <andi@firstfloor.org> Cc: linux-kselftest@vger.kernel.org Cc: Nadav Amit <nadav.amit@gmail.com> Cc: Robert O'Callahan <robert@ocallahan.org> Cc: Richard Weinberger <richard@nod.at> Cc: "Rafael J. Wysocki" <rafael.j.wysocki@intel.com> Cc: Borislav Petkov <bp@suse.de> Cc: Len Brown <len.brown@intel.com> Cc: Shuah Khan <shuah@kernel.org> Cc: user-mode-linux-devel@lists.sourceforge.net Cc: Jeff Dike <jdike@addtoit.com> Cc: Alexander Viro <viro@zeniv.linux.org.uk> Cc: user-mode-linux-user@lists.sourceforge.net Cc: David Matlack <dmatlack@google.com> Cc: Boris Ostrovsky <boris.ostrovsky@oracle.com> Cc: Dmitry Safonov <dsafonov@virtuozzo.com> Cc: linux-fsdevel@vger.kernel.org Cc: Paolo Bonzini <pbonzini@redhat.com> Link: http://lkml.kernel.org/r/20170320081628.18952-5-khuey@kylehuey.com Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
This commit is contained in:
		
							parent
							
								
									ff3f097eef
								
							
						
					
					
						commit
						17a6e1b8e8
					
				
					 6 changed files with 32 additions and 31 deletions
				
			
		| 
						 | 
					@ -303,7 +303,7 @@ extern void maybe_sigio_broken(int fd, int read);
 | 
				
			||||||
extern void sigio_broken(int fd, int read);
 | 
					extern void sigio_broken(int fd, int read);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* prctl.c */
 | 
					/* prctl.c */
 | 
				
			||||||
extern int os_arch_prctl(int pid, int option, unsigned long *addr);
 | 
					extern int os_arch_prctl(int pid, int option, unsigned long *arg2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* tty.c */
 | 
					/* tty.c */
 | 
				
			||||||
extern int get_pty(void);
 | 
					extern int get_pty(void);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -9,6 +9,7 @@ void syscall_init(void);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef CONFIG_X86_64
 | 
					#ifdef CONFIG_X86_64
 | 
				
			||||||
void entry_SYSCALL_64(void);
 | 
					void entry_SYSCALL_64(void);
 | 
				
			||||||
 | 
					long do_arch_prctl_64(struct task_struct *task, int option, unsigned long arg2);
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef CONFIG_X86_32
 | 
					#ifdef CONFIG_X86_32
 | 
				
			||||||
| 
						 | 
					@ -30,6 +31,4 @@ void x86_report_nx(void);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
extern int reboot_force;
 | 
					extern int reboot_force;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
long do_arch_prctl(struct task_struct *task, int option, unsigned long addr);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#endif /* _ASM_X86_PROTO_H */
 | 
					#endif /* _ASM_X86_PROTO_H */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -205,7 +205,7 @@ int copy_thread_tls(unsigned long clone_flags, unsigned long sp,
 | 
				
			||||||
				(struct user_desc __user *)tls, 0);
 | 
									(struct user_desc __user *)tls, 0);
 | 
				
			||||||
		else
 | 
							else
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
			err = do_arch_prctl(p, ARCH_SET_FS, tls);
 | 
								err = do_arch_prctl_64(p, ARCH_SET_FS, tls);
 | 
				
			||||||
		if (err)
 | 
							if (err)
 | 
				
			||||||
			goto out;
 | 
								goto out;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -548,7 +548,7 @@ static long prctl_map_vdso(const struct vdso_image *image, unsigned long addr)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
long do_arch_prctl(struct task_struct *task, int option, unsigned long addr)
 | 
					long do_arch_prctl_64(struct task_struct *task, int option, unsigned long arg2)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int ret = 0;
 | 
						int ret = 0;
 | 
				
			||||||
	int doit = task == current;
 | 
						int doit = task == current;
 | 
				
			||||||
| 
						 | 
					@ -556,62 +556,64 @@ long do_arch_prctl(struct task_struct *task, int option, unsigned long addr)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	switch (option) {
 | 
						switch (option) {
 | 
				
			||||||
	case ARCH_SET_GS:
 | 
						case ARCH_SET_GS:
 | 
				
			||||||
		if (addr >= TASK_SIZE_MAX)
 | 
							if (arg2 >= TASK_SIZE_MAX)
 | 
				
			||||||
			return -EPERM;
 | 
								return -EPERM;
 | 
				
			||||||
		cpu = get_cpu();
 | 
							cpu = get_cpu();
 | 
				
			||||||
		task->thread.gsindex = 0;
 | 
							task->thread.gsindex = 0;
 | 
				
			||||||
		task->thread.gsbase = addr;
 | 
							task->thread.gsbase = arg2;
 | 
				
			||||||
		if (doit) {
 | 
							if (doit) {
 | 
				
			||||||
			load_gs_index(0);
 | 
								load_gs_index(0);
 | 
				
			||||||
			ret = wrmsrl_safe(MSR_KERNEL_GS_BASE, addr);
 | 
								ret = wrmsrl_safe(MSR_KERNEL_GS_BASE, arg2);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		put_cpu();
 | 
							put_cpu();
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
	case ARCH_SET_FS:
 | 
						case ARCH_SET_FS:
 | 
				
			||||||
		/* Not strictly needed for fs, but do it for symmetry
 | 
							/* Not strictly needed for fs, but do it for symmetry
 | 
				
			||||||
		   with gs */
 | 
							   with gs */
 | 
				
			||||||
		if (addr >= TASK_SIZE_MAX)
 | 
							if (arg2 >= TASK_SIZE_MAX)
 | 
				
			||||||
			return -EPERM;
 | 
								return -EPERM;
 | 
				
			||||||
		cpu = get_cpu();
 | 
							cpu = get_cpu();
 | 
				
			||||||
		task->thread.fsindex = 0;
 | 
							task->thread.fsindex = 0;
 | 
				
			||||||
		task->thread.fsbase = addr;
 | 
							task->thread.fsbase = arg2;
 | 
				
			||||||
		if (doit) {
 | 
							if (doit) {
 | 
				
			||||||
			/* set the selector to 0 to not confuse __switch_to */
 | 
								/* set the selector to 0 to not confuse __switch_to */
 | 
				
			||||||
			loadsegment(fs, 0);
 | 
								loadsegment(fs, 0);
 | 
				
			||||||
			ret = wrmsrl_safe(MSR_FS_BASE, addr);
 | 
								ret = wrmsrl_safe(MSR_FS_BASE, arg2);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		put_cpu();
 | 
							put_cpu();
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
	case ARCH_GET_FS: {
 | 
						case ARCH_GET_FS: {
 | 
				
			||||||
		unsigned long base;
 | 
							unsigned long base;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (doit)
 | 
							if (doit)
 | 
				
			||||||
			rdmsrl(MSR_FS_BASE, base);
 | 
								rdmsrl(MSR_FS_BASE, base);
 | 
				
			||||||
		else
 | 
							else
 | 
				
			||||||
			base = task->thread.fsbase;
 | 
								base = task->thread.fsbase;
 | 
				
			||||||
		ret = put_user(base, (unsigned long __user *)addr);
 | 
							ret = put_user(base, (unsigned long __user *)arg2);
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	case ARCH_GET_GS: {
 | 
						case ARCH_GET_GS: {
 | 
				
			||||||
		unsigned long base;
 | 
							unsigned long base;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (doit)
 | 
							if (doit)
 | 
				
			||||||
			rdmsrl(MSR_KERNEL_GS_BASE, base);
 | 
								rdmsrl(MSR_KERNEL_GS_BASE, base);
 | 
				
			||||||
		else
 | 
							else
 | 
				
			||||||
			base = task->thread.gsbase;
 | 
								base = task->thread.gsbase;
 | 
				
			||||||
		ret = put_user(base, (unsigned long __user *)addr);
 | 
							ret = put_user(base, (unsigned long __user *)arg2);
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef CONFIG_CHECKPOINT_RESTORE
 | 
					#ifdef CONFIG_CHECKPOINT_RESTORE
 | 
				
			||||||
# ifdef CONFIG_X86_X32_ABI
 | 
					# ifdef CONFIG_X86_X32_ABI
 | 
				
			||||||
	case ARCH_MAP_VDSO_X32:
 | 
						case ARCH_MAP_VDSO_X32:
 | 
				
			||||||
		return prctl_map_vdso(&vdso_image_x32, addr);
 | 
							return prctl_map_vdso(&vdso_image_x32, arg2);
 | 
				
			||||||
# endif
 | 
					# endif
 | 
				
			||||||
# if defined CONFIG_X86_32 || defined CONFIG_IA32_EMULATION
 | 
					# if defined CONFIG_X86_32 || defined CONFIG_IA32_EMULATION
 | 
				
			||||||
	case ARCH_MAP_VDSO_32:
 | 
						case ARCH_MAP_VDSO_32:
 | 
				
			||||||
		return prctl_map_vdso(&vdso_image_32, addr);
 | 
							return prctl_map_vdso(&vdso_image_32, arg2);
 | 
				
			||||||
# endif
 | 
					# endif
 | 
				
			||||||
	case ARCH_MAP_VDSO_64:
 | 
						case ARCH_MAP_VDSO_64:
 | 
				
			||||||
		return prctl_map_vdso(&vdso_image_64, addr);
 | 
							return prctl_map_vdso(&vdso_image_64, arg2);
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	default:
 | 
						default:
 | 
				
			||||||
| 
						 | 
					@ -622,9 +624,9 @@ long do_arch_prctl(struct task_struct *task, int option, unsigned long addr)
 | 
				
			||||||
	return ret;
 | 
						return ret;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SYSCALL_DEFINE2(arch_prctl, int, option, unsigned long, addr)
 | 
					SYSCALL_DEFINE2(arch_prctl, int, option, unsigned long, arg2)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return do_arch_prctl(current, option, addr);
 | 
						return do_arch_prctl_64(current, option, arg2);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
unsigned long KSTK_ESP(struct task_struct *task)
 | 
					unsigned long KSTK_ESP(struct task_struct *task)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -396,12 +396,12 @@ static int putreg(struct task_struct *child,
 | 
				
			||||||
		if (value >= TASK_SIZE_MAX)
 | 
							if (value >= TASK_SIZE_MAX)
 | 
				
			||||||
			return -EIO;
 | 
								return -EIO;
 | 
				
			||||||
		/*
 | 
							/*
 | 
				
			||||||
		 * When changing the segment base, use do_arch_prctl
 | 
							 * When changing the segment base, use do_arch_prctl_64
 | 
				
			||||||
		 * to set either thread.fs or thread.fsindex and the
 | 
							 * to set either thread.fs or thread.fsindex and the
 | 
				
			||||||
		 * corresponding GDT slot.
 | 
							 * corresponding GDT slot.
 | 
				
			||||||
		 */
 | 
							 */
 | 
				
			||||||
		if (child->thread.fsbase != value)
 | 
							if (child->thread.fsbase != value)
 | 
				
			||||||
			return do_arch_prctl(child, ARCH_SET_FS, value);
 | 
								return do_arch_prctl_64(child, ARCH_SET_FS, value);
 | 
				
			||||||
		return 0;
 | 
							return 0;
 | 
				
			||||||
	case offsetof(struct user_regs_struct,gs_base):
 | 
						case offsetof(struct user_regs_struct,gs_base):
 | 
				
			||||||
		/*
 | 
							/*
 | 
				
			||||||
| 
						 | 
					@ -410,7 +410,7 @@ static int putreg(struct task_struct *child,
 | 
				
			||||||
		if (value >= TASK_SIZE_MAX)
 | 
							if (value >= TASK_SIZE_MAX)
 | 
				
			||||||
			return -EIO;
 | 
								return -EIO;
 | 
				
			||||||
		if (child->thread.gsbase != value)
 | 
							if (child->thread.gsbase != value)
 | 
				
			||||||
			return do_arch_prctl(child, ARCH_SET_GS, value);
 | 
								return do_arch_prctl_64(child, ARCH_SET_GS, value);
 | 
				
			||||||
		return 0;
 | 
							return 0;
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					@ -869,7 +869,7 @@ long arch_ptrace(struct task_struct *child, long request,
 | 
				
			||||||
		   Works just like arch_prctl, except that the arguments
 | 
							   Works just like arch_prctl, except that the arguments
 | 
				
			||||||
		   are reversed. */
 | 
							   are reversed. */
 | 
				
			||||||
	case PTRACE_ARCH_PRCTL:
 | 
						case PTRACE_ARCH_PRCTL:
 | 
				
			||||||
		ret = do_arch_prctl(child, data, addr);
 | 
							ret = do_arch_prctl_64(child, data, addr);
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -6,7 +6,7 @@
 | 
				
			||||||
#include <sys/ptrace.h>
 | 
					#include <sys/ptrace.h>
 | 
				
			||||||
#include <asm/ptrace.h>
 | 
					#include <asm/ptrace.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int os_arch_prctl(int pid, int option, unsigned long *addr)
 | 
					int os_arch_prctl(int pid, int option, unsigned long *arg2)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return ptrace(PTRACE_ARCH_PRCTL, pid, (unsigned long) addr, option);
 | 
						return ptrace(PTRACE_ARCH_PRCTL, pid, (unsigned long) arg2, option);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -12,10 +12,10 @@
 | 
				
			||||||
#include <asm/prctl.h> /* XXX This should get the constants from libc */
 | 
					#include <asm/prctl.h> /* XXX This should get the constants from libc */
 | 
				
			||||||
#include <os.h>
 | 
					#include <os.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
long arch_prctl(struct task_struct *task, int option
 | 
					long arch_prctl(struct task_struct *task, int option)
 | 
				
			||||||
		unsigned long __user *addr)
 | 
							unsigned long __user *arg2)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	unsigned long *ptr = addr, tmp;
 | 
						unsigned long *ptr = arg2, tmp;
 | 
				
			||||||
	long ret;
 | 
						long ret;
 | 
				
			||||||
	int pid = task->mm->context.id.u.pid;
 | 
						int pid = task->mm->context.id.u.pid;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -65,19 +65,19 @@ long arch_prctl(struct task_struct *task, int option
 | 
				
			||||||
		ret = save_registers(pid, ¤t->thread.regs.regs);
 | 
							ret = save_registers(pid, ¤t->thread.regs.regs);
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
	case ARCH_GET_FS:
 | 
						case ARCH_GET_FS:
 | 
				
			||||||
		ret = put_user(tmp, addr);
 | 
							ret = put_user(tmp, arg2);
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
	case ARCH_GET_GS:
 | 
						case ARCH_GET_GS:
 | 
				
			||||||
		ret = put_user(tmp, addr);
 | 
							ret = put_user(tmp, arg2);
 | 
				
			||||||
		break;
 | 
							break;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return ret;
 | 
						return ret;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SYSCALL_DEFINE2(arch_prctl, int, option, unsigned long, addr)
 | 
					SYSCALL_DEFINE2(arch_prctl, int, option, unsigned long, arg2)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	return arch_prctl(current, option, (unsigned long __user *) addr);
 | 
						return arch_prctl(current, option, (unsigned long __user *) arg2);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void arch_switch_to(struct task_struct *to)
 | 
					void arch_switch_to(struct task_struct *to)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue