hjl / source-git / glibc

Forked from source-git/glibc 4 years ago
Clone

Blame sysdeps/unix/sysv/linux/aarch64/setcontext.S

Packit 6c4009
/* Set current context.
Packit 6c4009
Packit 6c4009
   Copyright (C) 2009-2018 Free Software Foundation, Inc.
Packit 6c4009
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public License as
Packit 6c4009
   published by the Free Software Foundation; either version 2.1 of the
Packit 6c4009
   License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <sysdep.h>
Packit 6c4009
#include "ucontext_i.h"
Packit 6c4009
#include "ucontext-internal.h"
Packit 6c4009
Packit 6c4009
/*  int __setcontext (const ucontext_t *ucp)
Packit 6c4009
Packit 6c4009
  Restores the machine context in UCP and thereby resumes execution
Packit 6c4009
  in that context.
Packit 6c4009
Packit 6c4009
  This implementation is intended to be used for *synchronous* context
Packit 6c4009
  switches only.  Therefore, it does not have to restore anything
Packit 6c4009
  other than the PRESERVED state.  */
Packit 6c4009
Packit 6c4009
	.text
Packit 6c4009
Packit 6c4009
ENTRY (__setcontext)
Packit 6c4009
	DELOUSE (0)
Packit 6c4009
	/* Save a copy of UCP.  */
Packit 6c4009
	mov	x9, x0
Packit 6c4009
Packit 6c4009
	/* Set the signal mask with
Packit 6c4009
	   rt_sigprocmask (SIG_SETMASK, mask, NULL, _NSIG/8).  */
Packit 6c4009
	mov	x0, #SIG_SETMASK
Packit 6c4009
	add	x1, x9, #UCONTEXT_SIGMASK
Packit 6c4009
	mov	x2, #0
Packit 6c4009
	mov	x3, #_NSIG8
Packit 6c4009
	mov	x8, SYS_ify (rt_sigprocmask)
Packit 6c4009
	svc	0
Packit 6c4009
	cbz	x0, 1f
Packit 6c4009
	b	C_SYMBOL_NAME (__syscall_error)
Packit 6c4009
1:
Packit 6c4009
	/* Restore the general purpose registers.  */
Packit 6c4009
	mov	x0, x9
Packit 6c4009
	cfi_def_cfa (x0, 0)
Packit 6c4009
	cfi_offset (x18, oX0 + 18 * SZREG)
Packit 6c4009
	cfi_offset (x19, oX0 + 19 * SZREG)
Packit 6c4009
	cfi_offset (x20, oX0 + 20 * SZREG)
Packit 6c4009
	cfi_offset (x21, oX0 + 21 * SZREG)
Packit 6c4009
	cfi_offset (x22, oX0 + 22 * SZREG)
Packit 6c4009
	cfi_offset (x23, oX0 + 23 * SZREG)
Packit 6c4009
	cfi_offset (x24, oX0 + 24 * SZREG)
Packit 6c4009
	cfi_offset (x25, oX0 + 25 * SZREG)
Packit 6c4009
	cfi_offset (x26, oX0 + 26 * SZREG)
Packit 6c4009
	cfi_offset (x27, oX0 + 27 * SZREG)
Packit 6c4009
	cfi_offset (x28, oX0 + 28 * SZREG)
Packit 6c4009
	cfi_offset (x29, oX0 + 29 * SZREG)
Packit 6c4009
	cfi_offset (x30, oX0 + 30 * SZREG)
Packit 6c4009
Packit 6c4009
	cfi_offset ( d8, oV0 + 8 * SZVREG)
Packit 6c4009
	cfi_offset ( d9, oV0 + 9 * SZVREG)
Packit 6c4009
	cfi_offset (d10, oV0 + 10 * SZVREG)
Packit 6c4009
	cfi_offset (d11, oV0 + 11 * SZVREG)
Packit 6c4009
	cfi_offset (d12, oV0 + 12 * SZVREG)
Packit 6c4009
	cfi_offset (d13, oV0 + 13 * SZVREG)
Packit 6c4009
	cfi_offset (d14, oV0 + 14 * SZVREG)
Packit 6c4009
	cfi_offset (d15, oV0 + 15 * SZVREG)
Packit 6c4009
	ldp	x18, x19, [x0, oX0 + 18 * SZREG]
Packit 6c4009
	ldp	x20, x21, [x0, oX0 + 20 * SZREG]
Packit 6c4009
	ldp	x22, x23, [x0, oX0 + 22 * SZREG]
Packit 6c4009
	ldp	x24, x25, [x0, oX0 + 24 * SZREG]
Packit 6c4009
	ldp	x26, x27, [x0, oX0 + 26 * SZREG]
Packit 6c4009
	ldp	x28, x29, [x0, oX0 + 28 * SZREG]
Packit 6c4009
	ldr     x30,      [x0, oX0 + 30 * SZREG]
Packit 6c4009
	ldr     x2, [x0, oSP]
Packit 6c4009
	mov	sp, x2
Packit 6c4009
Packit 6c4009
	/* Check for FP SIMD context.  We don't support restoring
Packit 6c4009
	   contexts created by the kernel, so this context must have
Packit 6c4009
	   been created by getcontext.  Hence we can rely on the
Packit 6c4009
	   first extension block being the FP SIMD context.  */
Packit 6c4009
	add     x2, x0, #oEXTENSION
Packit 6c4009
Packit 6c4009
	mov	w3, #(FPSIMD_MAGIC & 0xffff)
Packit 6c4009
	movk	w3, #(FPSIMD_MAGIC >> 16), lsl #16
Packit 6c4009
	ldr	w1, [x2, #oHEAD + oMAGIC]
Packit 6c4009
	cmp	w1, w3
Packit 6c4009
	b.ne	2f
Packit 6c4009
Packit 6c4009
	/* Restore the FP SIMD context.  */
Packit 6c4009
	add	x3, x2, #oV0 + 8 * SZVREG
Packit 6c4009
	ldp	 q8,  q9, [x3], #2 * SZVREG
Packit 6c4009
	ldp	q10, q11, [x3], #2 * SZVREG
Packit 6c4009
	ldp	q12, q13, [x3], #2 * SZVREG
Packit 6c4009
	ldp	q14, q15, [x3], #2 * SZVREG
Packit 6c4009
Packit 6c4009
	add	x3, x2, oFPSR
Packit 6c4009
Packit 6c4009
	ldr	w4, [x3]
Packit 6c4009
	msr	fpsr, x4
Packit 6c4009
Packit 6c4009
	ldr	w4, [x3, oFPCR - oFPSR]
Packit 6c4009
	msr	fpcr, x4
Packit 6c4009
Packit 6c4009
2:
Packit 6c4009
	ldr     x16, [x0, oPC]
Packit 6c4009
	/* Restore arg registers.  */
Packit 6c4009
	ldp	x2, x3, [x0, oX0 + 2 * SZREG]
Packit 6c4009
	ldp	x4, x5, [x0, oX0 + 4 * SZREG]
Packit 6c4009
	ldp	x6, x7, [x0, oX0 + 6 * SZREG]
Packit 6c4009
	ldp	x0, x1, [x0, oX0 + 0 * SZREG]
Packit 6c4009
	/* Jump to the new pc value.  */
Packit 6c4009
	br	x16
Packit 6c4009
PSEUDO_END (__setcontext)
Packit 6c4009
weak_alias (__setcontext, setcontext)
Packit 6c4009
Packit 6c4009
ENTRY (__startcontext)
Packit 6c4009
	mov	x0, x19
Packit 6c4009
	cbnz	x0, __setcontext
Packit 6c4009
1:	b       HIDDEN_JUMPTARGET (exit)
Packit 6c4009
END (__startcontext)