ia32_signal.c 15.9 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
 *  linux/arch/x86_64/ia32/ia32_signal.c
 *
 *  Copyright (C) 1991, 1992  Linus Torvalds
 *
 *  1997-11-28  Modified for POSIX.1b signals by Richard Henderson
 *  2000-06-20  Pentium III FXSR, SSE support by Gareth Hughes
 *  2000-12-*   x86-64 compatibility mode signal handling by Andi Kleen
 */

#include <linux/sched.h>
#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/ptrace.h>
#include <linux/unistd.h>
#include <linux/stddef.h>
#include <linux/personality.h>
#include <linux/compat.h>
23
#include <linux/binfmts.h>
L
Linus Torvalds 已提交
24 25 26 27 28 29 30 31 32 33
#include <asm/ucontext.h>
#include <asm/uaccess.h>
#include <asm/i387.h>
#include <asm/ia32.h>
#include <asm/ptrace.h>
#include <asm/ia32_unistd.h>
#include <asm/user32.h>
#include <asm/sigcontext32.h>
#include <asm/fpu32.h>
#include <asm/proto.h>
R
Roland McGrath 已提交
34
#include <asm/vdso.h>
L
Linus Torvalds 已提交
35 36 37 38 39 40 41 42 43 44 45

#define DEBUG_SIG 0

#define _BLOCKABLE (~(sigmask(SIGKILL) | sigmask(SIGSTOP)))

asmlinkage int do_signal(struct pt_regs *regs, sigset_t *oldset);
void signal_fault(struct pt_regs *regs, void __user *frame, char *where);

int copy_siginfo_to_user32(compat_siginfo_t __user *to, siginfo_t *from)
{
	int err;
46 47

	if (!access_ok(VERIFY_WRITE, to, sizeof(compat_siginfo_t)))
L
Linus Torvalds 已提交
48 49 50 51 52 53 54 55 56
		return -EFAULT;

	/* If you change siginfo_t structure, please make sure that
	   this code is fixed accordingly.
	   It should never copy any pad contained in the structure
	   to avoid security leaks, but must copy the generic
	   3 ints plus the relevant union member.  */
	err = __put_user(from->si_signo, &to->si_signo);
	err |= __put_user(from->si_errno, &to->si_errno);
57
	err |= __put_user((short)from->si_code, &to->si_code);
L
Linus Torvalds 已提交
58 59 60

	if (from->si_code < 0) {
		err |= __put_user(from->si_pid, &to->si_pid);
61 62
		err |= __put_user(from->si_uid, &to->si_uid);
		err |= __put_user(ptr_to_compat(from->si_ptr), &to->si_ptr);
L
Linus Torvalds 已提交
63
	} else {
64 65 66 67 68 69
		/*
		 * First 32bits of unions are always present:
		 * si_pid === si_band === si_tid === si_addr(LS half)
		 */
		err |= __put_user(from->_sifields._pad[0],
				  &to->_sifields._pad[0]);
L
Linus Torvalds 已提交
70 71 72 73 74 75 76 77 78 79 80 81 82
		switch (from->si_code >> 16) {
		case __SI_FAULT >> 16:
			break;
		case __SI_CHLD >> 16:
			err |= __put_user(from->si_utime, &to->si_utime);
			err |= __put_user(from->si_stime, &to->si_stime);
			err |= __put_user(from->si_status, &to->si_status);
			/* FALL THROUGH */
		default:
		case __SI_KILL >> 16:
			err |= __put_user(from->si_uid, &to->si_uid);
			break;
		case __SI_POLL >> 16:
83
			err |= __put_user(from->si_fd, &to->si_fd);
L
Linus Torvalds 已提交
84 85
			break;
		case __SI_TIMER >> 16:
86
			err |= __put_user(from->si_overrun, &to->si_overrun);
L
Linus Torvalds 已提交
87
			err |= __put_user(ptr_to_compat(from->si_ptr),
88
					  &to->si_ptr);
L
Linus Torvalds 已提交
89
			break;
90 91
			 /* This is not generated by the kernel as of now.  */
		case __SI_RT >> 16:
L
Linus Torvalds 已提交
92 93 94 95 96 97 98 99 100 101 102 103 104
		case __SI_MESGQ >> 16:
			err |= __put_user(from->si_uid, &to->si_uid);
			err |= __put_user(from->si_int, &to->si_int);
			break;
		}
	}
	return err;
}

int copy_siginfo_from_user32(siginfo_t *to, compat_siginfo_t __user *from)
{
	int err;
	u32 ptr32;
105 106

	if (!access_ok(VERIFY_READ, from, sizeof(compat_siginfo_t)))
L
Linus Torvalds 已提交
107 108 109 110 111 112 113 114 115 116 117 118 119 120
		return -EFAULT;

	err = __get_user(to->si_signo, &from->si_signo);
	err |= __get_user(to->si_errno, &from->si_errno);
	err |= __get_user(to->si_code, &from->si_code);

	err |= __get_user(to->si_pid, &from->si_pid);
	err |= __get_user(to->si_uid, &from->si_uid);
	err |= __get_user(ptr32, &from->si_ptr);
	to->si_ptr = compat_ptr(ptr32);

	return err;
}

121
asmlinkage long sys32_sigsuspend(int history0, int history1, old_sigset_t mask)
L
Linus Torvalds 已提交
122 123 124
{
	mask &= _BLOCKABLE;
	spin_lock_irq(&current->sighand->siglock);
A
Andi Kleen 已提交
125
	current->saved_sigmask = current->blocked;
L
Linus Torvalds 已提交
126 127 128 129
	siginitset(&current->blocked, mask);
	recalc_sigpending();
	spin_unlock_irq(&current->sighand->siglock);

A
Andi Kleen 已提交
130 131 132 133
	current->state = TASK_INTERRUPTIBLE;
	schedule();
	set_thread_flag(TIF_RESTORE_SIGMASK);
	return -ERESTARTNOHAND;
L
Linus Torvalds 已提交
134 135
}

136 137 138
asmlinkage long sys32_sigaltstack(const stack_ia32_t __user *uss_ptr,
				  stack_ia32_t __user *uoss_ptr,
				  struct pt_regs *regs)
L
Linus Torvalds 已提交
139
{
140
	stack_t uss, uoss;
L
Linus Torvalds 已提交
141
	int ret;
142 143 144
	mm_segment_t seg;

	if (uss_ptr) {
L
Linus Torvalds 已提交
145
		u32 ptr;
146 147 148

		memset(&uss, 0, sizeof(stack_t));
		if (!access_ok(VERIFY_READ, uss_ptr, sizeof(stack_ia32_t)) ||
L
Linus Torvalds 已提交
149 150 151 152 153 154
			    __get_user(ptr, &uss_ptr->ss_sp) ||
			    __get_user(uss.ss_flags, &uss_ptr->ss_flags) ||
			    __get_user(uss.ss_size, &uss_ptr->ss_size))
			return -EFAULT;
		uss.ss_sp = compat_ptr(ptr);
	}
155 156
	seg = get_fs();
	set_fs(KERNEL_DS);
157
	ret = do_sigaltstack(uss_ptr ? &uss : NULL, &uoss, regs->sp);
158
	set_fs(seg);
L
Linus Torvalds 已提交
159
	if (ret >= 0 && uoss_ptr)  {
160
		if (!access_ok(VERIFY_WRITE, uoss_ptr, sizeof(stack_ia32_t)) ||
L
Linus Torvalds 已提交
161 162 163 164
		    __put_user(ptr_to_compat(uoss.ss_sp), &uoss_ptr->ss_sp) ||
		    __put_user(uoss.ss_flags, &uoss_ptr->ss_flags) ||
		    __put_user(uoss.ss_size, &uoss_ptr->ss_size))
			ret = -EFAULT;
165 166
	}
	return ret;
L
Linus Torvalds 已提交
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197
}

/*
 * Do a signal return; undo the signal stack.
 */

struct sigframe
{
	u32 pretcode;
	int sig;
	struct sigcontext_ia32 sc;
	struct _fpstate_ia32 fpstate;
	unsigned int extramask[_COMPAT_NSIG_WORDS-1];
	char retcode[8];
};

struct rt_sigframe
{
	u32 pretcode;
	int sig;
	u32 pinfo;
	u32 puc;
	compat_siginfo_t info;
	struct ucontext_ia32 uc;
	struct _fpstate_ia32 fpstate;
	char retcode[8];
};

#define COPY(x)		{ \
	unsigned int reg;			\
	err |= __get_user(reg, &sc->e ##x);	\
198
	regs->x = reg;			\
L
Linus Torvalds 已提交
199 200 201
}

#define RELOAD_SEG(seg,mask)						\
202
	{ unsigned int cur;						\
L
Linus Torvalds 已提交
203 204
	  unsigned short pre;						\
	  err |= __get_user(pre, &sc->seg);				\
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222
	  asm volatile("movl %%" #seg ",%0" : "=r" (cur));		\
	  pre |= mask;							\
	  if (pre != cur) loadsegment(seg, pre); }

static int ia32_restore_sigcontext(struct pt_regs *regs,
				   struct sigcontext_ia32 __user *sc,
				   unsigned int *peax)
{
	unsigned int tmpflags, gs, oldgs, err = 0;
	struct _fpstate_ia32 __user *buf;
	u32 tmp;

	/* Always make any pending restarted system calls return -EINTR */
	current_thread_info()->restart_block.fn = do_no_restart_syscall;

#if DEBUG_SIG
	printk(KERN_DEBUG "SIG restore_sigcontext: "
	       "sc=%p err(%x) eip(%x) cs(%x) flg(%x)\n",
223
	       sc, sc->err, sc->ip, sc->cs, sc->flags);
224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240
#endif

	/*
	 * Reload fs and gs if they have changed in the signal
	 * handler.  This does not handle long fs/gs base changes in
	 * the handler, but does not clobber them at least in the
	 * normal case.
	 */
	err |= __get_user(gs, &sc->gs);
	gs |= 3;
	asm("movl %%gs,%0" : "=r" (oldgs));
	if (gs != oldgs)
		load_gs_index(gs);

	RELOAD_SEG(fs, 3);
	RELOAD_SEG(ds, 3);
	RELOAD_SEG(es, 3);
L
Linus Torvalds 已提交
241 242 243

	COPY(di); COPY(si); COPY(bp); COPY(sp); COPY(bx);
	COPY(dx); COPY(cx); COPY(ip);
244 245 246 247 248 249 250 251
	/* Don't touch extended registers */

	err |= __get_user(regs->cs, &sc->cs);
	regs->cs |= 3;
	err |= __get_user(regs->ss, &sc->ss);
	regs->ss |= 3;

	err |= __get_user(tmpflags, &sc->eflags);
252
	regs->flags = (regs->flags & ~0x40DD5) | (tmpflags & 0x40DD5);
253
	/* disable syscall checks */
254
	regs->orig_ax = -1;
255 256 257 258 259 260 261 262 263

	err |= __get_user(tmp, &sc->fpstate);
	buf = compat_ptr(tmp);
	if (buf) {
		if (!access_ok(VERIFY_READ, buf, sizeof(*buf)))
			goto badframe;
		err |= restore_i387_ia32(current, buf, 0);
	} else {
		struct task_struct *me = current;
L
Linus Torvalds 已提交
264

265 266 267
		if (used_math()) {
			clear_fpu(me);
			clear_used_math();
L
Linus Torvalds 已提交
268 269 270
		}
	}

271 272 273
	err |= __get_user(tmp, &sc->eax);
	*peax = tmp;

L
Linus Torvalds 已提交
274 275 276 277 278 279 280 281
	return err;

badframe:
	return 1;
}

asmlinkage long sys32_sigreturn(struct pt_regs *regs)
{
282
	struct sigframe __user *frame = (struct sigframe __user *)(regs->sp-8);
L
Linus Torvalds 已提交
283
	sigset_t set;
284
	unsigned int ax;
L
Linus Torvalds 已提交
285 286 287 288 289

	if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
		goto badframe;
	if (__get_user(set.sig[0], &frame->sc.oldmask)
	    || (_COMPAT_NSIG_WORDS > 1
290 291
		&& __copy_from_user((((char *) &set.sig) + 4),
				    &frame->extramask,
L
Linus Torvalds 已提交
292 293 294 295 296 297 298 299
				    sizeof(frame->extramask))))
		goto badframe;

	sigdelsetmask(&set, ~_BLOCKABLE);
	spin_lock_irq(&current->sighand->siglock);
	current->blocked = set;
	recalc_sigpending();
	spin_unlock_irq(&current->sighand->siglock);
300

301
	if (ia32_restore_sigcontext(regs, &frame->sc, &ax))
L
Linus Torvalds 已提交
302
		goto badframe;
303
	return ax;
L
Linus Torvalds 已提交
304 305 306 307

badframe:
	signal_fault(regs, frame, "32bit sigreturn");
	return 0;
308
}
L
Linus Torvalds 已提交
309 310 311 312 313

asmlinkage long sys32_rt_sigreturn(struct pt_regs *regs)
{
	struct rt_sigframe __user *frame;
	sigset_t set;
314
	unsigned int ax;
L
Linus Torvalds 已提交
315 316
	struct pt_regs tregs;

317
	frame = (struct rt_sigframe __user *)(regs->sp - 4);
L
Linus Torvalds 已提交
318 319 320 321 322 323 324 325 326 327 328

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

	sigdelsetmask(&set, ~_BLOCKABLE);
	spin_lock_irq(&current->sighand->siglock);
	current->blocked = set;
	recalc_sigpending();
	spin_unlock_irq(&current->sighand->siglock);
329

330
	if (ia32_restore_sigcontext(regs, &frame->uc.uc_mcontext, &ax))
L
Linus Torvalds 已提交
331 332 333 334 335 336
		goto badframe;

	tregs = *regs;
	if (sys32_sigaltstack(&frame->uc.uc_stack, NULL, &tregs) == -EFAULT)
		goto badframe;

337
	return ax;
L
Linus Torvalds 已提交
338 339

badframe:
340
	signal_fault(regs, frame, "32bit rt sigreturn");
L
Linus Torvalds 已提交
341
	return 0;
342
}
L
Linus Torvalds 已提交
343 344 345 346 347

/*
 * Set up a signal frame.
 */

348 349 350
static int ia32_setup_sigcontext(struct sigcontext_ia32 __user *sc,
				 struct _fpstate_ia32 __user *fpstate,
				 struct pt_regs *regs, unsigned int mask)
L
Linus Torvalds 已提交
351 352 353 354 355 356 357 358 359 360 361 362 363
{
	int tmp, err = 0;

	tmp = 0;
	__asm__("movl %%gs,%0" : "=r"(tmp): "0"(tmp));
	err |= __put_user(tmp, (unsigned int __user *)&sc->gs);
	__asm__("movl %%fs,%0" : "=r"(tmp): "0"(tmp));
	err |= __put_user(tmp, (unsigned int __user *)&sc->fs);
	__asm__("movl %%ds,%0" : "=r"(tmp): "0"(tmp));
	err |= __put_user(tmp, (unsigned int __user *)&sc->ds);
	__asm__("movl %%es,%0" : "=r"(tmp): "0"(tmp));
	err |= __put_user(tmp, (unsigned int __user *)&sc->es);

364 365 366 367 368 369 370 371
	err |= __put_user((u32)regs->di, &sc->edi);
	err |= __put_user((u32)regs->si, &sc->esi);
	err |= __put_user((u32)regs->bp, &sc->ebp);
	err |= __put_user((u32)regs->sp, &sc->esp);
	err |= __put_user((u32)regs->bx, &sc->ebx);
	err |= __put_user((u32)regs->dx, &sc->edx);
	err |= __put_user((u32)regs->cx, &sc->ecx);
	err |= __put_user((u32)regs->ax, &sc->eax);
L
Linus Torvalds 已提交
372 373 374 375
	err |= __put_user((u32)regs->cs, &sc->cs);
	err |= __put_user((u32)regs->ss, &sc->ss);
	err |= __put_user(current->thread.trap_no, &sc->trapno);
	err |= __put_user(current->thread.error_code, &sc->err);
376 377 378
	err |= __put_user((u32)regs->ip, &sc->eip);
	err |= __put_user((u32)regs->flags, &sc->eflags);
	err |= __put_user((u32)regs->sp, &sc->esp_at_signal);
L
Linus Torvalds 已提交
379 380 381 382

	tmp = save_i387_ia32(current, fpstate, regs, 0);
	if (tmp < 0)
		err = -EFAULT;
383
	else {
L
Linus Torvalds 已提交
384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399
		clear_used_math();
		stts();
		err |= __put_user(ptr_to_compat(tmp ? fpstate : NULL),
					&sc->fpstate);
	}

	/* non-iBCS2 extensions.. */
	err |= __put_user(mask, &sc->oldmask);
	err |= __put_user(current->thread.cr2, &sc->cr2);

	return err;
}

/*
 * Determine which stack to use..
 */
400 401
static void __user *get_sigframe(struct k_sigaction *ka, struct pt_regs *regs,
				 size_t frame_size)
L
Linus Torvalds 已提交
402
{
403
	unsigned long sp;
L
Linus Torvalds 已提交
404 405

	/* Default to using normal stack */
406
	sp = regs->sp;
L
Linus Torvalds 已提交
407 408 409

	/* This is the X/Open sanctioned signal stack switching.  */
	if (ka->sa.sa_flags & SA_ONSTACK) {
410 411
		if (sas_ss_flags(sp) == 0)
			sp = current->sas_ss_sp + current->sas_ss_size;
L
Linus Torvalds 已提交
412 413 414 415 416
	}

	/* This is the legacy signal stack switching. */
	else if ((regs->ss & 0xffff) != __USER_DS &&
		!(ka->sa.sa_flags & SA_RESTORER) &&
417
		 ka->sa.sa_restorer)
418
		sp = (unsigned long) ka->sa.sa_restorer;
L
Linus Torvalds 已提交
419

420
	sp -= frame_size;
421 422
	/* Align the stack pointer according to the i386 ABI,
	 * i.e. so that on function entry ((sp + 4) & 15) == 0. */
423 424
	sp = ((sp + 4) & -16ul) - 4;
	return (void __user *) sp;
L
Linus Torvalds 已提交
425 426
}

427
int ia32_setup_frame(int sig, struct k_sigaction *ka,
428
		     compat_sigset_t *set, struct pt_regs *regs)
L
Linus Torvalds 已提交
429 430
{
	struct sigframe __user *frame;
431
	void __user *restorer;
L
Linus Torvalds 已提交
432 433
	int err = 0;

434 435 436 437 438 439 440 441 442 443 444 445 446
	/* copy_to_user optimizes that into a single 8 byte store */
	static const struct {
		u16 poplmovl;
		u32 val;
		u16 int80;
		u16 pad;
	} __attribute__((packed)) code = {
		0xb858,		 /* popl %eax ; movl $...,%eax */
		__NR_ia32_sigreturn,
		0x80cd,		/* int $0x80 */
		0,
	};

L
Linus Torvalds 已提交
447 448 449 450 451
	frame = get_sigframe(ka, regs, sizeof(*frame));

	if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
		goto give_sigsegv;

452
	err |= __put_user(sig, &frame->sig);
L
Linus Torvalds 已提交
453 454 455 456 457 458 459 460 461 462 463
	if (err)
		goto give_sigsegv;

	err |= ia32_setup_sigcontext(&frame->sc, &frame->fpstate, regs,
					set->sig[0]);
	if (err)
		goto give_sigsegv;

	if (_COMPAT_NSIG_WORDS > 1) {
		err |= __copy_to_user(frame->extramask, &set->sig[1],
				      sizeof(frame->extramask));
464 465
		if (err)
			goto give_sigsegv;
L
Linus Torvalds 已提交
466 467
	}

R
Roland McGrath 已提交
468
	if (ka->sa.sa_flags & SA_RESTORER) {
469
		restorer = ka->sa.sa_restorer;
R
Roland McGrath 已提交
470 471 472 473 474 475 476 477
	} else {
		/* Return stub is in 32bit vsyscall page */
		if (current->binfmt->hasvdso)
			restorer = VDSO32_SYMBOL(current->mm->context.vdso,
						 sigreturn);
		else
			restorer = (void *)&frame->retcode;
	}
478 479 480 481 482 483 484
	err |= __put_user(ptr_to_compat(restorer), &frame->pretcode);

	/*
	 * These are actually not used anymore, but left because some
	 * gdb versions depend on them as a marker.
	 */
	err |= __copy_to_user(frame->retcode, &code, 8);
L
Linus Torvalds 已提交
485 486 487 488
	if (err)
		goto give_sigsegv;

	/* Set up registers for signal handler */
489 490
	regs->sp = (unsigned long) frame;
	regs->ip = (unsigned long) ka->sa.sa_handler;
L
Linus Torvalds 已提交
491

492
	/* Make -mregparm=3 work */
493 494 495
	regs->ax = sig;
	regs->dx = 0;
	regs->cx = 0;
496

497 498
	asm volatile("movl %0,%%ds" :: "r" (__USER32_DS));
	asm volatile("movl %0,%%es" :: "r" (__USER32_DS));
L
Linus Torvalds 已提交
499

500 501
	regs->cs = __USER32_CS;
	regs->ss = __USER32_DS;
L
Linus Torvalds 已提交
502 503

	set_fs(USER_DS);
504
	regs->flags &= ~TF_MASK;
505 506
	if (test_thread_flag(TIF_SINGLESTEP))
		ptrace_notify(SIGTRAP);
L
Linus Torvalds 已提交
507 508

#if DEBUG_SIG
509
	printk(KERN_DEBUG "SIG deliver (%s:%d): sp=%p pc=%lx ra=%u\n",
510
	       current->comm, current->pid, frame, regs->ip, frame->pretcode);
L
Linus Torvalds 已提交
511 512
#endif

A
Andi Kleen 已提交
513
	return 0;
L
Linus Torvalds 已提交
514 515 516

give_sigsegv:
	force_sigsegv(sig, current);
A
Andi Kleen 已提交
517
	return -EFAULT;
L
Linus Torvalds 已提交
518 519
}

520
int ia32_setup_rt_frame(int sig, struct k_sigaction *ka, siginfo_t *info,
521
			compat_sigset_t *set, struct pt_regs *regs)
L
Linus Torvalds 已提交
522 523
{
	struct rt_sigframe __user *frame;
524
	struct exec_domain *ed = current_thread_info()->exec_domain;
R
Roland McGrath 已提交
525
	void __user *restorer;
L
Linus Torvalds 已提交
526 527
	int err = 0;

528 529 530 531 532 533 534 535 536 537 538 539 540 541
	/* __copy_to_user optimizes that into a single 8 byte store */
	static const struct {
		u8 movl;
		u32 val;
		u16 int80;
		u16 pad;
		u8  pad2;
	} __attribute__((packed)) code = {
		0xb8,
		__NR_ia32_rt_sigreturn,
		0x80cd,
		0,
	};

L
Linus Torvalds 已提交
542 543 544 545 546
	frame = get_sigframe(ka, regs, sizeof(*frame));

	if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
		goto give_sigsegv;

547 548
	err |= __put_user((ed && ed->signal_invmap && sig < 32
			   ? ed->signal_invmap[sig] : sig), &frame->sig);
L
Linus Torvalds 已提交
549 550 551 552 553 554 555 556 557 558
	err |= __put_user(ptr_to_compat(&frame->info), &frame->pinfo);
	err |= __put_user(ptr_to_compat(&frame->uc), &frame->puc);
	err |= copy_siginfo_to_user32(&frame->info, info);
	if (err)
		goto give_sigsegv;

	/* Create the ucontext.  */
	err |= __put_user(0, &frame->uc.uc_flags);
	err |= __put_user(0, &frame->uc.uc_link);
	err |= __put_user(current->sas_ss_sp, &frame->uc.uc_stack.ss_sp);
559
	err |= __put_user(sas_ss_flags(regs->sp),
L
Linus Torvalds 已提交
560 561 562
			  &frame->uc.uc_stack.ss_flags);
	err |= __put_user(current->sas_ss_size, &frame->uc.uc_stack.ss_size);
	err |= ia32_setup_sigcontext(&frame->uc.uc_mcontext, &frame->fpstate,
563
				     regs, set->sig[0]);
L
Linus Torvalds 已提交
564 565 566 567
	err |= __copy_to_user(&frame->uc.uc_sigmask, set, sizeof(*set));
	if (err)
		goto give_sigsegv;

568 569
	if (ka->sa.sa_flags & SA_RESTORER)
		restorer = ka->sa.sa_restorer;
R
Roland McGrath 已提交
570 571 572
	else
		restorer = VDSO32_SYMBOL(current->mm->context.vdso,
					 rt_sigreturn);
573
	err |= __put_user(ptr_to_compat(restorer), &frame->pretcode);
L
Linus Torvalds 已提交
574

575 576 577 578 579
	/*
	 * Not actually used anymore, but left because some gdb
	 * versions need it.
	 */
	err |= __copy_to_user(frame->retcode, &code, 8);
L
Linus Torvalds 已提交
580 581 582 583
	if (err)
		goto give_sigsegv;

	/* Set up registers for signal handler */
584 585
	regs->sp = (unsigned long) frame;
	regs->ip = (unsigned long) ka->sa.sa_handler;
L
Linus Torvalds 已提交
586

587
	/* Make -mregparm=3 work */
588 589 590
	regs->ax = sig;
	regs->dx = (unsigned long) &frame->info;
	regs->cx = (unsigned long) &frame->uc;
591

592
	/* Make -mregparm=3 work */
593 594 595
	regs->ax = sig;
	regs->dx = (unsigned long) &frame->info;
	regs->cx = (unsigned long) &frame->uc;
596

597 598 599 600 601
	asm volatile("movl %0,%%ds" :: "r" (__USER32_DS));
	asm volatile("movl %0,%%es" :: "r" (__USER32_DS));

	regs->cs = __USER32_CS;
	regs->ss = __USER32_DS;
L
Linus Torvalds 已提交
602 603

	set_fs(USER_DS);
604
	regs->flags &= ~TF_MASK;
605 606
	if (test_thread_flag(TIF_SINGLESTEP))
		ptrace_notify(SIGTRAP);
L
Linus Torvalds 已提交
607 608

#if DEBUG_SIG
609
	printk(KERN_DEBUG "SIG deliver (%s:%d): sp=%p pc=%lx ra=%u\n",
610
	       current->comm, current->pid, frame, regs->ip, frame->pretcode);
L
Linus Torvalds 已提交
611 612
#endif

A
Andi Kleen 已提交
613
	return 0;
L
Linus Torvalds 已提交
614 615 616

give_sigsegv:
	force_sigsegv(sig, current);
A
Andi Kleen 已提交
617
	return -EFAULT;
L
Linus Torvalds 已提交
618
}