compat_signal.c 5.0 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 * Copyright 2010 Tilera Corporation. All Rights Reserved.
 *
 *   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, version 2.
 *
 *   This program is distributed in the hope that it will be useful, but
 *   WITHOUT ANY WARRANTY; without even the implied warranty of
 *   MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE, GOOD TITLE or
 *   NON INFRINGEMENT.  See the GNU General Public License for
 *   more details.
 */

#include <linux/sched.h>
16
#include <linux/sched/task_stack.h>
17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
#include <linux/mm.h>
#include <linux/smp.h>
#include <linux/kernel.h>
#include <linux/signal.h>
#include <linux/errno.h>
#include <linux/wait.h>
#include <linux/unistd.h>
#include <linux/stddef.h>
#include <linux/personality.h>
#include <linux/suspend.h>
#include <linux/ptrace.h>
#include <linux/elf.h>
#include <linux/compat.h>
#include <linux/syscalls.h>
#include <linux/uaccess.h>
#include <asm/processor.h>
#include <asm/ucontext.h>
#include <asm/sigframe.h>
35
#include <asm/syscalls.h>
36
#include <asm/vdso.h>
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
#include <arch/interrupts.h>

struct compat_ucontext {
	compat_ulong_t	  uc_flags;
	compat_uptr_t     uc_link;
	struct compat_sigaltstack	  uc_stack;
	struct sigcontext uc_mcontext;
	sigset_t	  uc_sigmask;	/* mask last for extensibility */
};

struct compat_rt_sigframe {
	unsigned char save_area[C_ABI_SAVE_AREA_SIZE]; /* caller save area */
	struct compat_siginfo info;
	struct compat_ucontext uc;
};

53
/* The assembly shim for this function arranges to ignore the return value. */
54
long compat_sys_rt_sigreturn(void)
55
{
56
	struct pt_regs *regs = current_pt_regs();
57 58 59 60 61 62 63 64 65
	struct compat_rt_sigframe __user *frame =
		(struct compat_rt_sigframe __user *) compat_ptr(regs->sp);
	sigset_t set;

	if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
		goto badframe;
	if (__copy_from_user(&set, &frame->uc.uc_sigmask, sizeof(set)))
		goto badframe;

66
	set_current_blocked(&set);
67

68
	if (restore_sigcontext(regs, &frame->uc.uc_mcontext))
69 70
		goto badframe;

A
Al Viro 已提交
71
	if (compat_restore_altstack(&frame->uc.uc_stack))
72 73
		goto badframe;

74
	return 0;
75 76

badframe:
77
	signal_fault("bad sigreturn frame", regs, frame, 0);
78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
	return 0;
}

/*
 * Determine which stack to use..
 */
static inline void __user *compat_get_sigframe(struct k_sigaction *ka,
					       struct pt_regs *regs,
					       size_t frame_size)
{
	unsigned long sp;

	/* Default to using normal stack */
	sp = (unsigned long)compat_ptr(regs->sp);

	/*
	 * If we are on the alternate signal stack and would overflow
	 * it, don't.  Return an always-bogus address instead so we
	 * will die with SIGSEGV.
	 */
	if (on_sig_stack(sp) && !likely(on_sig_stack(sp - frame_size)))
99
		return (void __user __force *)-1UL;
100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115

	/* This is the X/Open sanctioned signal stack switching.  */
	if (ka->sa.sa_flags & SA_ONSTACK) {
		if (sas_ss_flags(sp) == 0)
			sp = current->sas_ss_sp + current->sas_ss_size;
	}

	sp -= frame_size;
	/*
	 * Align the stack pointer according to the TILE ABI,
	 * i.e. so that on function entry (sp & 15) == 0.
	 */
	sp &= -16UL;
	return (void __user *) sp;
}

116 117
int compat_setup_rt_frame(struct ksignal *ksig, sigset_t *set,
			  struct pt_regs *regs)
118 119 120
{
	unsigned long restorer;
	struct compat_rt_sigframe __user *frame;
121
	int err = 0, sig = ksig->sig;
122

123
	frame = compat_get_sigframe(&ksig->ka, regs, sizeof(*frame));
124 125

	if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
126
		goto err;
127 128

	/* Always write at least the signal number for the stack backtracer. */
129
	if (ksig->ka.sa.sa_flags & SA_SIGINFO) {
130
		/* At sigreturn time, restore the callee-save registers too. */
131
		err |= copy_siginfo_to_user32(&frame->info, &ksig->info);
132 133
		regs->flags |= PT_FLAGS_RESTORE_REGS;
	} else {
134
		err |= __put_user(ksig->info.si_signo, &frame->info.si_signo);
135 136 137 138 139 140
	}

	/* Create the ucontext.  */
	err |= __clear_user(&frame->save_area, sizeof(frame->save_area));
	err |= __put_user(0, &frame->uc.uc_flags);
	err |= __put_user(0, &frame->uc.uc_link);
A
Al Viro 已提交
141
	err |= __compat_save_altstack(&frame->uc.uc_stack, regs->sp);
142 143 144
	err |= setup_sigcontext(&frame->uc.uc_mcontext, regs);
	err |= __copy_to_user(&frame->uc.uc_sigmask, set, sizeof(*set));
	if (err)
145
		goto err;
146

147
	restorer = VDSO_SYM(&__vdso_rt_sigreturn);
148 149
	if (ksig->ka.sa.sa_flags & SA_RESTORER)
		restorer = ptr_to_compat_reg(ksig->ka.sa.sa_restorer);
150 151 152 153 154

	/*
	 * Set up registers for signal handler.
	 * Registers that we don't modify keep the value they had from
	 * user-space at the time we took the signal.
155 156
	 * We always pass siginfo and mcontext, regardless of SA_SIGINFO,
	 * since some things rely on this (e.g. glibc's debug/segfault.c).
157
	 */
158
	regs->pc = ptr_to_compat_reg(ksig->ka.sa.sa_handler);
159 160 161
	regs->ex1 = PL_ICS_EX1(USER_PL, 1); /* set crit sec in handler */
	regs->sp = ptr_to_compat_reg(frame);
	regs->lr = restorer;
162
	regs->regs[0] = (unsigned long) sig;
163 164 165
	regs->regs[1] = ptr_to_compat_reg(&frame->info);
	regs->regs[2] = ptr_to_compat_reg(&frame->uc);
	regs->flags |= PT_FLAGS_CALLER_SAVES;
166 167
	return 0;

168 169 170
err:
	trace_unhandled_signal("bad sigreturn frame", regs,
			      (unsigned long)frame, SIGSEGV);
171 172
	return -EFAULT;
}