aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorHans-Peter Nilsson <hp@axis.com>2011-06-17 02:18:24 +0000
committerHans-Peter Nilsson <hp@gcc.gnu.org>2011-06-17 02:18:24 +0000
commit9a2091cd0a703e199c2c50225930071d00675a32 (patch)
tree691aeafddb0a87acbd9aac366c7ea95648cda6d0
parenta26335b56fa55e06467e54d2683b156943028c30 (diff)
downloadgcc-9a2091cd0a703e199c2c50225930071d00675a32.zip
gcc-9a2091cd0a703e199c2c50225930071d00675a32.tar.gz
gcc-9a2091cd0a703e199c2c50225930071d00675a32.tar.bz2
re PR rtl-optimization/48542 (unchanged variables in code which calls setjmp may be clobbered (including the return-address))
PR rtl-optimization/48542 * reload.c (find_equiv_reg): Stop looking when finding a setjmp-type call. * reload1.c (reload_as_needed): Invalidate all reload registers when crossing a setjmp-type call. From-SVN: r175132
-rw-r--r--gcc/ChangeLog8
-rw-r--r--gcc/reload.c9
-rw-r--r--gcc/reload1.c7
3 files changed, 24 insertions, 0 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog
index c1620d2..26c1f81 100644
--- a/gcc/ChangeLog
+++ b/gcc/ChangeLog
@@ -1,3 +1,11 @@
+2011-06-17 Hans-Peter Nilsson <hp@axis.com>
+
+ PR rtl-optimization/48542
+ * reload.c (find_equiv_reg): Stop looking when finding a
+ setjmp-type call.
+ * reload1.c (reload_as_needed): Invalidate all reload
+ registers when crossing a setjmp-type call.
+
2011-06-16 Jeff Law <law@redhat.com>
* tree-ssa-threadupdate.c (struct redirection_data): New field
diff --git a/gcc/reload.c b/gcc/reload.c
index 27d5787..3ad46b9 100644
--- a/gcc/reload.c
+++ b/gcc/reload.c
@@ -6791,6 +6791,15 @@ find_equiv_reg (rtx goal, rtx insn, enum reg_class rclass, int other,
|| num > PARAM_VALUE (PARAM_MAX_RELOAD_SEARCH_INSNS))
return 0;
+ /* Don't reuse register contents from before a setjmp-type
+ function call; on the second return (from the longjmp) it
+ might have been clobbered by a later reuse. It doesn't
+ seem worthwhile to actually go and see if it is actually
+ reused even if that information would be readily available;
+ just don't reuse it across the setjmp call. */
+ if (CALL_P (p) && find_reg_note (p, REG_SETJMP, NULL_RTX))
+ return 0;
+
if (NONJUMP_INSN_P (p)
/* If we don't want spill regs ... */
&& (! (reload_reg_p != 0
diff --git a/gcc/reload1.c b/gcc/reload1.c
index e65503b..a87e6ad 100644
--- a/gcc/reload1.c
+++ b/gcc/reload1.c
@@ -4844,6 +4844,13 @@ reload_as_needed (int live_known)
{
AND_COMPL_HARD_REG_SET (reg_reloaded_valid, call_used_reg_set);
AND_COMPL_HARD_REG_SET (reg_reloaded_valid, reg_reloaded_call_part_clobbered);
+
+ /* If this is a call to a setjmp-type function, we must not
+ reuse any reload reg contents across the call; that will
+ just be clobbered by other uses of the register in later
+ code, before the longjmp. */
+ if (find_reg_note (insn, REG_SETJMP, NULL_RTX))
+ CLEAR_HARD_REG_SET (reg_reloaded_valid);
}
}