1997-04-14 10:11:12 +08:00
|
|
|
/* Assembler macros for ARM.
|
2013-01-03 03:01:50 +08:00
|
|
|
Copyright (C) 1997-2013 Free Software Foundation, Inc.
|
1997-04-14 10:11:12 +08:00
|
|
|
This file is part of the GNU C Library.
|
|
|
|
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
2001-07-06 12:56:23 +08:00
|
|
|
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.
|
1997-04-14 10:11:12 +08:00
|
|
|
|
|
|
|
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
|
2001-07-06 12:56:23 +08:00
|
|
|
Lesser General Public License for more details.
|
1997-04-14 10:11:12 +08:00
|
|
|
|
2001-07-06 12:56:23 +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/>. */
|
1997-04-14 10:11:12 +08:00
|
|
|
|
|
|
|
#include <sysdeps/generic/sysdep.h>
|
2011-02-17 01:30:13 +08:00
|
|
|
#include <features.h>
|
1997-04-14 10:11:12 +08:00
|
|
|
|
2004-12-05 05:20:17 +08:00
|
|
|
#if (!defined (__ARM_ARCH_2__) && !defined (__ARM_ARCH_3__) \
|
|
|
|
&& !defined (__ARM_ARCH_3M__) && !defined (__ARM_ARCH_4__))
|
|
|
|
# define __USE_BX__
|
|
|
|
#endif
|
|
|
|
|
1998-05-29 18:20:59 +08:00
|
|
|
#ifdef __ASSEMBLER__
|
1997-04-14 10:11:12 +08:00
|
|
|
|
|
|
|
/* Syntactic details of assembler. */
|
|
|
|
|
1998-05-20 00:11:52 +08:00
|
|
|
#define ALIGNARG(log2) log2
|
1997-04-14 10:11:12 +08:00
|
|
|
#define ASM_SIZE_DIRECTIVE(name) .size name,.-name
|
|
|
|
|
1998-03-18 22:29:30 +08:00
|
|
|
#define PLTJMP(_x) _x##(PLT)
|
|
|
|
|
1997-11-18 10:39:35 +08:00
|
|
|
/* APCS-32 doesn't preserve the condition codes across function call. */
|
|
|
|
#ifdef __APCS_32__
|
1997-04-14 10:11:12 +08:00
|
|
|
#define LOADREGS(cond, base, reglist...)\
|
|
|
|
ldm##cond base,reglist
|
2004-12-05 05:20:17 +08:00
|
|
|
#ifdef __USE_BX__
|
|
|
|
#define RETINSTR(cond, reg) \
|
|
|
|
bx##cond reg
|
2003-09-18 02:09:36 +08:00
|
|
|
#define DO_RET(_reg) \
|
|
|
|
bx _reg
|
|
|
|
#else
|
2004-12-05 05:20:17 +08:00
|
|
|
#define RETINSTR(cond, reg) \
|
|
|
|
mov##cond pc, reg
|
2003-09-18 02:09:36 +08:00
|
|
|
#define DO_RET(_reg) \
|
|
|
|
mov pc, _reg
|
|
|
|
#endif
|
1997-11-18 10:39:35 +08:00
|
|
|
#else /* APCS-26 */
|
1997-04-14 10:11:12 +08:00
|
|
|
#define LOADREGS(cond, base, reglist...)\
|
|
|
|
ldm##cond base,reglist^
|
2004-12-05 05:20:17 +08:00
|
|
|
#define RETINSTR(cond, reg) \
|
|
|
|
mov##cond##s pc, reg
|
2003-09-18 02:09:36 +08:00
|
|
|
#define DO_RET(_reg) \
|
|
|
|
movs pc, _reg
|
1997-04-14 10:11:12 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Define an entry point visible from C. */
|
2013-02-28 15:04:17 +08:00
|
|
|
#define ENTRY(name) \
|
|
|
|
.globl C_SYMBOL_NAME(name); \
|
|
|
|
.type C_SYMBOL_NAME(name),%function; \
|
|
|
|
.align ALIGNARG(4); \
|
|
|
|
C_LABEL(name) \
|
|
|
|
CFI_SECTIONS; \
|
|
|
|
cfi_startproc; \
|
|
|
|
CALL_MCOUNT
|
|
|
|
|
|
|
|
#define CFI_SECTIONS \
|
|
|
|
.cfi_sections .debug_frame
|
2013-02-28 08:36:47 +08:00
|
|
|
|
1997-04-14 10:11:12 +08:00
|
|
|
#undef END
|
2013-02-28 15:04:17 +08:00
|
|
|
#define END(name) \
|
|
|
|
cfi_endproc; \
|
|
|
|
ASM_SIZE_DIRECTIVE(name)
|
1997-04-14 10:11:12 +08:00
|
|
|
|
|
|
|
/* If compiled for profiling, call `mcount' at the start of each function. */
|
|
|
|
#ifdef PROF
|
2012-04-22 01:03:39 +08:00
|
|
|
/* Call __gnu_mcount_nc if GCC >= 4.4. */
|
|
|
|
#if __GNUC_PREREQ(4,4)
|
2013-02-28 15:04:17 +08:00
|
|
|
#define CALL_MCOUNT \
|
|
|
|
str lr,[sp, #-4]!; \
|
|
|
|
cfi_adjust_cfa_offset (4); \
|
|
|
|
cfi_rel_offset (lr, 0); \
|
|
|
|
bl PLTJMP(mcount); \
|
|
|
|
cfi_adjust_cfa_offset (-4); \
|
|
|
|
cfi_restore (lr)
|
2011-02-17 01:30:13 +08:00
|
|
|
#else /* else call _mcount */
|
2013-02-28 15:04:17 +08:00
|
|
|
#define CALL_MCOUNT \
|
|
|
|
str lr,[sp, #-4]!; \
|
|
|
|
cfi_adjust_cfa_offset (4); \
|
|
|
|
cfi_rel_offset (lr, 0); \
|
|
|
|
bl PLTJMP(mcount); \
|
|
|
|
ldr lr, [sp], #4; \
|
|
|
|
cfi_adjust_cfa_offset (-4); \
|
|
|
|
cfi_restore (lr)
|
2011-02-17 01:30:13 +08:00
|
|
|
#endif
|
1997-04-14 10:11:12 +08:00
|
|
|
#else
|
|
|
|
#define CALL_MCOUNT /* Do nothing. */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Since C identifiers are not normally prefixed with an underscore
|
|
|
|
on this system, the asm identifier `syscall_error' intrudes on the
|
|
|
|
C name space. Make sure we use an innocuous name. */
|
|
|
|
#define syscall_error __syscall_error
|
2012-04-22 01:03:39 +08:00
|
|
|
#if __GNUC_PREREQ(4,4)
|
2011-02-17 01:30:13 +08:00
|
|
|
#define mcount __gnu_mcount_nc
|
|
|
|
#else
|
1997-04-14 10:11:12 +08:00
|
|
|
#define mcount _mcount
|
|
|
|
#endif
|
|
|
|
|
2009-10-23 03:39:47 +08:00
|
|
|
/* Tag_ABI_align8_preserved: This code preserves 8-byte
|
|
|
|
alignment in any callee. */
|
|
|
|
.eabi_attribute 25, 1
|
|
|
|
/* Tag_ABI_align8_needed: This code may require 8-byte alignment from
|
|
|
|
the caller. */
|
|
|
|
.eabi_attribute 24, 1
|
|
|
|
|
2013-02-15 02:32:04 +08:00
|
|
|
/* The thumb2 encoding is reasonably complete. Unless suppressed, use it. */
|
|
|
|
.syntax unified
|
|
|
|
# if defined(__thumb2__) && !defined(NO_THUMB)
|
|
|
|
.thumb
|
|
|
|
#else
|
|
|
|
# undef __thumb__
|
|
|
|
# undef __thumb2__
|
|
|
|
.arm
|
|
|
|
# endif
|
|
|
|
|
2013-02-14 13:21:39 +08:00
|
|
|
/* Load or store to/from a pc-relative EXPR into/from R, using T. */
|
|
|
|
# ifdef __thumb2__
|
|
|
|
# define LDST_PCREL(OP, R, T, EXPR) \
|
|
|
|
ldr T, 98f; \
|
|
|
|
.subsection 2; \
|
|
|
|
98: .word EXPR - 99f - PC_OFS; \
|
|
|
|
.previous; \
|
|
|
|
99: add T, T, pc; \
|
|
|
|
OP R, [T]
|
|
|
|
# else
|
|
|
|
# define LDST_PCREL(OP, R, T, EXPR) \
|
|
|
|
ldr T, 98f; \
|
|
|
|
.subsection 2; \
|
|
|
|
98: .word EXPR - 99f - PC_OFS; \
|
|
|
|
.previous; \
|
|
|
|
99: OP R, [pc, T]
|
|
|
|
# endif
|
2013-02-15 01:46:56 +08:00
|
|
|
|
|
|
|
/* Cope with negative memory offsets, which thumb can't encode.
|
|
|
|
Use NEGOFF_ADJ_BASE to (conditionally) alter the base register,
|
|
|
|
and then NEGOFF_OFF1 to use 0 for thumb and the offset for arm,
|
|
|
|
or NEGOFF_OFF2 to use A-B for thumb and A for arm. */
|
|
|
|
# ifdef __thumb2__
|
|
|
|
# define NEGOFF_ADJ_BASE(R, OFF) add R, R, $OFF
|
|
|
|
# define NEGOFF_ADJ_BASE2(D, S, OFF) add D, S, $OFF
|
|
|
|
# define NEGOFF_OFF1(R, OFF) [R]
|
|
|
|
# define NEGOFF_OFF2(R, OFFA, OFFB) [R, $((OFFA) - (OFFB))]
|
|
|
|
# else
|
|
|
|
# define NEGOFF_ADJ_BASE(R, OFF)
|
|
|
|
# define NEGOFF_ADJ_BASE2(D, S, OFF) mov D, S
|
|
|
|
# define NEGOFF_OFF1(R, OFF) [R, $OFF]
|
|
|
|
# define NEGOFF_OFF2(R, OFFA, OFFB) [R, $OFFA]
|
|
|
|
# endif
|
2013-02-14 12:10:45 +08:00
|
|
|
|
|
|
|
/* Helper to get the TLS base pointer. The interface is that TMP is a
|
|
|
|
register that may be used to hold the LR, if necessary. TMP may be
|
|
|
|
LR itself to indicate that LR need not be saved. The base pointer
|
|
|
|
is returned in R0. Only R0 and TMP are modified.
|
|
|
|
|
|
|
|
At this generic level we have no tricks to pull. Call the ABI routine. */
|
|
|
|
# define GET_TLS(TMP) \
|
|
|
|
push { r1, r2, r3, lr }; \
|
|
|
|
cfi_remember_state; \
|
|
|
|
cfi_adjust_cfa_offset (16); \
|
|
|
|
cfi_rel_offset (r1, 0); \
|
|
|
|
cfi_rel_offset (r2, 4); \
|
|
|
|
cfi_rel_offset (r3, 8); \
|
|
|
|
cfi_rel_offset (lr, 12); \
|
|
|
|
bl __aeabi_read_tp; \
|
|
|
|
pop { r1, r2, r3, lr }; \
|
|
|
|
cfi_restore_state
|
|
|
|
|
1998-05-29 18:20:59 +08:00
|
|
|
#endif /* __ASSEMBLER__ */
|
2013-02-13 09:15:52 +08:00
|
|
|
|
|
|
|
/* This number is the offset from the pc at the current location. */
|
2013-02-15 02:32:04 +08:00
|
|
|
#ifdef __thumb__
|
2013-02-13 09:15:52 +08:00
|
|
|
# define PC_OFS 4
|
|
|
|
#else
|
|
|
|
# define PC_OFS 8
|
|
|
|
#endif
|