Commit 5ead99fa by Jakub Jelinek Committed by Jakub Jelinek

re PR target/43139 (ICE in output_operand)

	PR target/43139
	* config/i386/i386.c (ix86_delegitimize_address): Delegitimize all
	GOTOFF relocs, even when the base reg isn't pic pointer.

	* gcc.dg/pr43139.c: New test.

From-SVN: r157011
parent 2a8e30fb
2010-02-23 Jakub Jelinek <jakub@redhat.com>
PR target/43139
* config/i386/i386.c (ix86_delegitimize_address): Delegitimize all
GOTOFF relocs, even when the base reg isn't pic pointer.
2010-02-23 Michael Matz <matz@suse.de>
PR debug/43077
......
......@@ -10883,6 +10883,9 @@ static rtx
ix86_delegitimize_address (rtx x)
{
rtx orig_x = delegitimize_mem_from_attrs (x);
/* addend is NULL or some rtx if x is something+GOTOFF where
something doesn't include the PIC register. */
rtx addend = NULL_RTX;
/* reg_addend is NULL or a multiple of some register. */
rtx reg_addend = NULL_RTX;
/* const_addend is NULL or a const_int. */
......@@ -10921,14 +10924,13 @@ ix86_delegitimize_address (rtx x)
else if (ix86_pic_register_p (XEXP (reg_addend, 1)))
reg_addend = XEXP (reg_addend, 0);
else
return orig_x;
if (!REG_P (reg_addend)
&& GET_CODE (reg_addend) != MULT
&& GET_CODE (reg_addend) != ASHIFT)
return orig_x;
{
reg_addend = NULL_RTX;
addend = XEXP (x, 0);
}
}
else
return orig_x;
addend = XEXP (x, 0);
x = XEXP (XEXP (x, 1), 0);
if (GET_CODE (x) == PLUS
......@@ -10939,7 +10941,7 @@ ix86_delegitimize_address (rtx x)
}
if (GET_CODE (x) == UNSPEC
&& ((XINT (x, 1) == UNSPEC_GOT && MEM_P (orig_x))
&& ((XINT (x, 1) == UNSPEC_GOT && MEM_P (orig_x) && !addend)
|| (XINT (x, 1) == UNSPEC_GOTOFF && !MEM_P (orig_x))))
result = XVECEXP (x, 0, 0);
......@@ -10954,6 +10956,22 @@ ix86_delegitimize_address (rtx x)
result = gen_rtx_CONST (Pmode, gen_rtx_PLUS (Pmode, result, const_addend));
if (reg_addend)
result = gen_rtx_PLUS (Pmode, reg_addend, result);
if (addend)
{
/* If the rest of original X doesn't involve the PIC register, add
addend and subtract pic_offset_table_rtx. This can happen e.g.
for code like:
leal (%ebx, %ecx, 4), %ecx
...
movl foo@GOTOFF(%ecx), %edx
in which case we return (%ecx - %ebx) + foo. */
if (pic_offset_table_rtx)
result = gen_rtx_PLUS (Pmode, gen_rtx_MINUS (Pmode, copy_rtx (addend),
pic_offset_table_rtx),
result);
else
return orig_x;
}
return result;
}
......
2010-02-23 Jakub Jelinek <jakub@redhat.com>
PR target/43139
* gcc.dg/pr43139.c: New test.
PR debug/43077
* gcc.dg/guality/pr43077-1.c: New test.
......
/* PR target/43139 */
/* { dg-do compile { target fpic } } */
/* { dg-options "-g -O2 -fpic" } */
typedef double T1[10];
typedef double T2[10][10];
typedef int T3[10];
void __attribute__((noinline))
fn1 (void)
{
asm volatile ("" : : : "memory");
}
void __attribute__((noinline))
fn2 (int x, ...)
{
asm volatile ("" : : "r" (x) : "memory");
}
static void
bar (double v, double w, double x, double y, double z)
{
double a;
if (v / w < 200.0)
{
a = x + (y - x) * __builtin_exp (-v / w);
fn2 (0);
fn2 (1, a * 20.2 / z, z);
fn1 ();
}
}
static void
baz (T2 u, T2 v, T2 t, T2 x, T1 y, T3 z, double q, int j, int k)
{
int i = z[k];
if (u[i][j] > 0.0)
bar (q, x[i][j], v[i][j], t[i][j], y[i]);
}
static T2 a, b, c, d;
static T1 e;
static T3 f;
void __attribute__((noinline))
test (int j, int k, double q)
{
baz (a, b, c, d, e, f, q, j, k);
}
int
main (void)
{
d[0][6] = 1.0;
a[0][6] = 2.0;
test (6, 7, 400.0);
return 0;
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment