2014-01-01 19:03:15 +08:00
|
|
|
/* Copyright (C) 2009-2014 Free Software Foundation, Inc.
|
2009-05-17 06:20:23 +08: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.
|
|
|
|
|
2011-03-21 23:40:37 +08:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
2012-03-10 07:56:38 +08:00
|
|
|
License along with the GNU C Library. If not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
2009-05-17 06:20:23 +08:00
|
|
|
|
2009-08-07 00:53:09 +08:00
|
|
|
#include <sysdep.h>
|
2009-05-17 06:20:23 +08: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
|
|
|
|
.text
|
|
|
|
|
|
|
|
#define __longjmp ____longjmp_chk
|
|
|
|
|
|
|
|
#ifdef PIC
|
|
|
|
# define CALL_FAIL \
|
|
|
|
ldr sl, .L_GOT; \
|
2010-03-27 02:12:56 +08:00
|
|
|
cfi_undefined (sl); \
|
2009-05-17 06:20:23 +08:00
|
|
|
.L_GOT_OFF: \
|
|
|
|
add sl, pc, sl; \
|
|
|
|
ldr r0, .Lstr; \
|
|
|
|
add r0, sl, r0; \
|
|
|
|
B PLTJMP(HIDDEN_JUMPTARGET(__fortify_fail)); \
|
|
|
|
.L_GOT: \
|
|
|
|
.word _GLOBAL_OFFSET_TABLE_-(.L_GOT_OFF+8); \
|
|
|
|
.Lstr: \
|
|
|
|
.word longjmp_msg(GOTOFF);
|
|
|
|
#else
|
|
|
|
# define CALL_FAIL \
|
|
|
|
ldr r0, .Lstr; \
|
|
|
|
B HIDDEN_JUMPTARGET(__fortify_fail); \
|
|
|
|
.Lstr: \
|
|
|
|
.word longjmp_msg;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define CHECK_SP(reg) \
|
2010-03-27 02:12:56 +08:00
|
|
|
cfi_remember_state; \
|
2009-08-07 00:53:09 +08:00
|
|
|
cmp sp, reg; \
|
2009-05-19 03:48:54 +08:00
|
|
|
bls .Lok; \
|
2013-02-15 13:00:14 +08:00
|
|
|
push { r7 }; \
|
2011-04-02 00:44:42 +08:00
|
|
|
cfi_adjust_cfa_offset (4); \
|
|
|
|
cfi_rel_offset (r7, 0); \
|
2009-08-07 00:53:09 +08:00
|
|
|
mov r5, r0; \
|
2010-03-27 02:12:56 +08:00
|
|
|
cfi_undefined (r5); \
|
2009-08-07 00:53:09 +08:00
|
|
|
mov r7, #SYS_ify(sigaltstack); \
|
|
|
|
mov r0, #0; \
|
2011-04-02 00:44:42 +08:00
|
|
|
sub sp, sp, #12; /* == sizeof (stack_t) */ \
|
|
|
|
cfi_adjust_cfa_offset (12); \
|
2010-03-27 02:12:56 +08:00
|
|
|
cfi_remember_state; \
|
2009-08-07 00:53:09 +08:00
|
|
|
mov r1, sp; \
|
|
|
|
swi #0; \
|
|
|
|
cmp r0, #0; \
|
|
|
|
bne .Lok2; \
|
|
|
|
ldr r1, [sp, #4]; \
|
|
|
|
tst r1, #1; \
|
|
|
|
beq .Lfail; \
|
|
|
|
ldr r2, [sp, #0]; \
|
|
|
|
ldr r3, [sp, #8]; \
|
|
|
|
add r2, r2, r3; \
|
|
|
|
sub r2, r2, reg; \
|
|
|
|
cmp r2, r3; \
|
|
|
|
bhi .Lok2; \
|
|
|
|
.Lfail: \
|
2011-04-02 00:44:42 +08:00
|
|
|
add sp, sp, #12; \
|
|
|
|
cfi_adjust_cfa_offset (-12); \
|
2013-02-15 13:00:14 +08:00
|
|
|
pop { r7 }; \
|
2011-04-02 00:44:42 +08:00
|
|
|
cfi_adjust_cfa_offset (-4); \
|
|
|
|
cfi_restore (r7); \
|
2009-05-17 06:20:23 +08:00
|
|
|
CALL_FAIL \
|
2010-03-27 02:12:56 +08:00
|
|
|
cfi_restore_state; \
|
2009-08-07 00:53:09 +08:00
|
|
|
.Lok2: \
|
|
|
|
mov r0, r5; \
|
2010-03-27 02:12:56 +08:00
|
|
|
cfi_restore_state; \
|
2009-05-17 06:20:23 +08:00
|
|
|
.Lok:
|
|
|
|
|
|
|
|
#include <__longjmp.S>
|