2012-05-18 15:32:08 -07:00
|
|
|
/* Copyright (C) 2001-2012 Free Software Foundation, Inc.
|
2009-07-30 21:42:27 -07:00
|
|
|
This file is part of the GNU C Library.
|
|
|
|
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Lesser General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
The GNU C Library 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. See the GNU
|
|
|
|
Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
2012-02-09 23:18:22 +00:00
|
|
|
License along with the GNU C Library; if not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
2009-07-30 21:42:27 -07:00
|
|
|
|
|
|
|
#include <sysdep.h>
|
|
|
|
#include <jmpbuf-offsets.h>
|
|
|
|
#include <asm-syntax.h>
|
2012-05-25 13:31:57 -07:00
|
|
|
#include <stap-probe.h>
|
2009-07-30 21:42:27 -07:00
|
|
|
|
2012-05-15 16:43:11 -07:00
|
|
|
#include <sigaltstack-offsets.h>
|
|
|
|
|
2009-07-30 21:42:27 -07:00
|
|
|
.section .rodata.str1.1,"aMS",@progbits,1
|
|
|
|
.type longjmp_msg,@object
|
|
|
|
longjmp_msg:
|
|
|
|
.string "longjmp causes uninitialized stack frame"
|
|
|
|
.size longjmp_msg, .-longjmp_msg
|
|
|
|
|
|
|
|
|
|
|
|
//#define __longjmp ____longjmp_chk
|
|
|
|
|
|
|
|
#ifdef PIC
|
2012-05-15 16:43:11 -07:00
|
|
|
# define CALL_FAIL sub $8, %RSP_LP; \
|
2011-04-07 16:23:52 -04:00
|
|
|
cfi_remember_state; \
|
|
|
|
cfi_def_cfa_offset(16); \
|
2012-05-15 16:43:11 -07:00
|
|
|
lea longjmp_msg(%rip), %RDI_LP; \
|
2011-09-15 12:26:08 -07:00
|
|
|
call HIDDEN_JUMPTARGET(__fortify_fail); \
|
2011-04-07 16:23:52 -04:00
|
|
|
nop; \
|
|
|
|
cfi_restore_state
|
2009-07-30 21:42:27 -07:00
|
|
|
#else
|
2012-05-15 16:43:11 -07:00
|
|
|
# define CALL_FAIL sub $8, %RSP_LP; \
|
2011-04-07 16:23:52 -04:00
|
|
|
cfi_remember_state; \
|
|
|
|
cfi_def_cfa_offset(16); \
|
2012-05-15 16:43:11 -07:00
|
|
|
mov $longjmp_msg, %RDI_LP; \
|
2011-09-15 12:26:08 -07:00
|
|
|
call HIDDEN_JUMPTARGET(__fortify_fail); \
|
2011-04-07 16:23:52 -04:00
|
|
|
nop; \
|
|
|
|
cfi_restore_state
|
2009-07-30 21:42:27 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Jump to the position specified by ENV, causing the
|
|
|
|
setjmp call there to return VAL, or 1 if VAL is 0.
|
|
|
|
void __longjmp (__jmp_buf env, int val). */
|
|
|
|
.text
|
|
|
|
ENTRY(____longjmp_chk)
|
|
|
|
/* Restore registers. */
|
2012-05-15 16:43:11 -07:00
|
|
|
mov (JB_RSP*8)(%rdi), %R8_LP
|
2012-05-18 15:32:08 -07:00
|
|
|
mov (JB_RBP*8)(%rdi),%R9_LP
|
2012-05-15 16:43:11 -07:00
|
|
|
mov (JB_PC*8)(%rdi), %RDX_LP
|
2009-07-30 21:42:27 -07:00
|
|
|
#ifdef PTR_DEMANGLE
|
2012-05-15 16:43:11 -07:00
|
|
|
PTR_DEMANGLE (%R8_LP)
|
2012-05-18 15:32:08 -07:00
|
|
|
PTR_DEMANGLE (%R9_LP)
|
2012-05-15 16:43:11 -07:00
|
|
|
PTR_DEMANGLE (%RDX_LP)
|
2012-05-18 15:32:08 -07:00
|
|
|
# ifdef __ILP32__
|
|
|
|
/* We ignored the high bits of the %rbp value because only the low
|
|
|
|
bits are mangled. But we cannot presume that %rbp is being used
|
|
|
|
as a pointer and truncate it, so recover the high bits. */
|
|
|
|
movl (JB_RBP*8 + 4)(%rdi), %eax
|
|
|
|
shlq $32, %rax
|
|
|
|
orq %rax, %r9
|
|
|
|
# endif
|
2009-07-30 21:42:27 -07:00
|
|
|
#endif
|
|
|
|
|
2012-05-15 16:43:11 -07:00
|
|
|
cmp %R8_LP, %RSP_LP
|
2009-07-31 17:27:38 -07:00
|
|
|
jbe .Lok
|
|
|
|
|
2009-07-30 21:42:27 -07:00
|
|
|
/* Save function parameters. */
|
|
|
|
movq %rdi, %r10
|
2010-09-08 15:49:50 -07:00
|
|
|
cfi_register (%rdi, %r10)
|
|
|
|
movl %esi, %ebx
|
|
|
|
cfi_register (%rsi, %rbx)
|
2009-07-30 21:42:27 -07:00
|
|
|
|
|
|
|
xorl %edi, %edi
|
2012-05-15 16:43:11 -07:00
|
|
|
lea -sizeSS(%rsp), %RSI_LP
|
2009-07-30 21:42:27 -07:00
|
|
|
movl $__NR_sigaltstack, %eax
|
|
|
|
syscall
|
2009-07-31 17:27:38 -07:00
|
|
|
/* Without working sigaltstack we cannot perform the test. */
|
2009-07-30 21:42:27 -07:00
|
|
|
testl %eax, %eax
|
2009-07-31 17:27:38 -07:00
|
|
|
jne .Lok2
|
2012-05-15 16:43:11 -07:00
|
|
|
testl $1, (-sizeSS + oSS_FLAGS)(%rsp)
|
2009-07-31 17:27:38 -07:00
|
|
|
jz .Lfail
|
2009-07-30 21:42:27 -07:00
|
|
|
|
2012-05-15 16:43:11 -07:00
|
|
|
mov (-sizeSS + oSS_SP)(%rsp), %RAX_LP
|
|
|
|
add (-sizeSS + oSS_SIZE)(%rsp), %RAX_LP
|
|
|
|
sub %R8_LP, %RAX_LP
|
|
|
|
cmp (-sizeSS + oSS_SIZE)(%rsp), %RAX_LP
|
2009-07-31 17:27:38 -07:00
|
|
|
jae .Lok2
|
|
|
|
|
|
|
|
.Lfail: CALL_FAIL
|
|
|
|
|
|
|
|
.Lok2: movq %r10, %rdi
|
2010-09-08 15:49:50 -07:00
|
|
|
cfi_restore (%rdi)
|
|
|
|
movl %ebx, %esi
|
|
|
|
cfi_restore (%rsi)
|
2009-07-31 17:27:38 -07:00
|
|
|
|
2012-05-25 13:31:57 -07:00
|
|
|
.Lok:
|
|
|
|
LIBC_PROBE (longjmp, 3, LP_SIZE@%RDI_LP, -4@%esi, LP_SIZE@%RDX_LP)
|
|
|
|
/* We add unwind information for the target here. */
|
2009-07-31 17:27:38 -07:00
|
|
|
cfi_def_cfa(%rdi, 0)
|
2009-07-30 21:42:27 -07:00
|
|
|
cfi_register(%rsp,%r8)
|
|
|
|
cfi_register(%rbp,%r9)
|
|
|
|
cfi_register(%rip,%rdx)
|
|
|
|
cfi_offset(%rbx,JB_RBX*8)
|
|
|
|
cfi_offset(%r12,JB_R12*8)
|
|
|
|
cfi_offset(%r13,JB_R13*8)
|
|
|
|
cfi_offset(%r14,JB_R14*8)
|
|
|
|
cfi_offset(%r15,JB_R15*8)
|
2010-09-08 15:49:50 -07:00
|
|
|
movq (JB_RBX*8)(%rdi), %rbx
|
|
|
|
movq (JB_R12*8)(%rdi), %r12
|
|
|
|
movq (JB_R13*8)(%rdi), %r13
|
|
|
|
movq (JB_R14*8)(%rdi), %r14
|
|
|
|
movq (JB_R15*8)(%rdi), %r15
|
2009-07-30 21:42:27 -07:00
|
|
|
/* Set return value for setjmp. */
|
2009-07-31 17:27:38 -07:00
|
|
|
movl %esi, %eax
|
2012-05-15 16:43:11 -07:00
|
|
|
mov %R8_LP, %RSP_LP
|
2009-07-31 17:27:38 -07:00
|
|
|
movq %r9,%rbp
|
2012-05-25 13:31:57 -07:00
|
|
|
LIBC_PROBE (longjmp_target, 3,
|
|
|
|
LP_SIZE@%RDI_LP, -4@%eax, LP_SIZE@%RDX_LP)
|
2009-07-30 21:42:27 -07:00
|
|
|
jmpq *%rdx
|
2009-08-01 14:18:58 -07:00
|
|
|
END (____longjmp_chk)
|