]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
aarch64: fix return address access with pac [PR94891][PR94791]
authorSzabolcs Nagy <szabolcs.nagy@arm.com>
Tue, 2 Jun 2020 15:44:41 +0000 (16:44 +0100)
committerSzabolcs Nagy <szabolcs.nagy@arm.com>
Thu, 16 Jul 2020 12:46:37 +0000 (13:46 +0100)
This is a big hammer fix for __builtin_return_address (PR target/94891)
returning signed addresses (sometimes, depending on wether lr happens
to be signed or not at the time of call which depends on optimizations),
and similarly -pg may pass signed return address to _mcount
(PR target/94791).

At the time of return address expansion we don't know if it's signed or
not so it is done unconditionally.

2020-07-13  Szabolcs Nagy  <szabolcs.nagy@arm.com>

gcc/ChangeLog:

PR target/94891
PR target/94791
* config/aarch64/aarch64-protos.h (aarch64_return_addr_rtx): Declare.
* config/aarch64/aarch64.c (aarch64_return_addr_rtx): New.
(aarch64_return_addr): Use aarch64_return_addr_rtx.
* config/aarch64/aarch64.h (PROFILE_HOOK): Likewise.

(cherry picked from commit 463a54e5d4956143f81c1f23b91cbd2d93855741)

gcc/config/aarch64/aarch64-protos.h
gcc/config/aarch64/aarch64.c
gcc/config/aarch64/aarch64.h

index b9bfb28127570f1cf9ddb063d71b9fd6e22e4667..af2a17f0bf3b50a306b14d8c0aa431269f54ef2e 100644 (file)
@@ -492,6 +492,7 @@ int aarch64_vec_fpconst_pow_of_2 (rtx);
 rtx aarch64_eh_return_handler_rtx (void);
 rtx aarch64_mask_from_zextract_ops (rtx, rtx);
 const char *aarch64_output_move_struct (rtx *operands);
+rtx aarch64_return_addr_rtx (void);
 rtx aarch64_return_addr (int, rtx);
 rtx aarch64_simd_gen_const_vector_dup (machine_mode, HOST_WIDE_INT);
 bool aarch64_simd_mem_operand_p (rtx);
index 35b5b2fd3a652b6aa1c6956c8e02858c3a167696..4bb16d125c643377b06d00c416ac582667ba2416 100644 (file)
@@ -8425,6 +8425,24 @@ aarch64_initial_elimination_offset (unsigned from, unsigned to)
   return cfun->machine->frame.frame_size;
 }
 
+
+/* Get return address without mangling.  */
+
+rtx
+aarch64_return_addr_rtx (void)
+{
+  rtx val = get_hard_reg_initial_val (Pmode, LR_REGNUM);
+  /* Note: aarch64_return_address_signing_enabled only
+     works after cfun->machine->frame.laid_out is set,
+     so here we don't know if the return address will
+     be signed or not.  */
+  rtx lr = gen_rtx_REG (Pmode, LR_REGNUM);
+  emit_move_insn (lr, val);
+  emit_insn (GEN_FCN (CODE_FOR_xpaclri) ());
+  return lr;
+}
+
+
 /* Implement RETURN_ADDR_RTX.  We do not support moving back to a
    previous frame.  */
 
@@ -8433,7 +8451,7 @@ aarch64_return_addr (int count, rtx frame ATTRIBUTE_UNUSED)
 {
   if (count != 0)
     return const0_rtx;
-  return get_hard_reg_initial_val (Pmode, LR_REGNUM);
+  return aarch64_return_addr_rtx ();
 }
 
 
index 772a97296880688c13a3421a3d1c822444cb4a33..0e9b79e23f05f34cfee5fdccf5792af077a95798 100644 (file)
@@ -968,7 +968,7 @@ typedef struct
 #define PROFILE_HOOK(LABEL)                                            \
   {                                                                    \
     rtx fun, lr;                                                       \
-    lr = get_hard_reg_initial_val (Pmode, LR_REGNUM);                  \
+    lr = aarch64_return_addr_rtx ();                                   \
     fun = gen_rtx_SYMBOL_REF (Pmode, MCOUNT_NAME);                     \
     emit_library_call (fun, LCT_NORMAL, VOIDmode, lr, Pmode);          \
   }