https://gcc.gnu.org/bugzilla/show_bug.cgi?id=65956

            Bug ID: 65956
           Summary: [5/6 Regression] Another ARM overaligned arg passing
                    issue
           Product: gcc
           Version: 5.1.0
            Status: UNCONFIRMED
          Severity: normal
          Priority: P3
         Component: target
          Assignee: unassigned at gcc dot gnu.org
          Reporter: jakub at gcc dot gnu.org
  Target Milestone: ---

The following testcase is miscompiled at -O2 on armhfp, starting with r221348,
and in this case r221795 didn't fix it.  Just hope it is the same thing that
breaks armhfp profiledbootstrap.  The difference on the testcase is just:
        mov     r0, r5
-       ldr     r1, [sp, #24]
+       ldr     r2, [sp, #24]
        bl      fn1
which means that we pass the second argument to fn1 in a wrong register,
supposedly because the type is void * aligned to long long alignment.

struct A { char *a; int b; long long c; };
char v[3];

__attribute__((noinline, noclone)) void
fn1 (char *x, char *y)
{
  if (x != &v[1] || y != &v[2])
    __builtin_abort ();
  v[1]++;
}

__attribute__((noinline, noclone)) int
fn2 (char *x)
{
  asm volatile ("" : "+g" (x) : : "memory");
  return x == &v[0];
}

__attribute__((noinline, noclone)) void
fn3 (const char *x)
{
  if (x[0] != 0)
    __builtin_abort ();
}

static struct A
foo (const char *x, struct A y, struct A z)
{
  struct A r = { 0, 0, 0 };
  if (y.b && z.b)
    {
      if (fn2 (y.a) && fn2 (z.a))
        switch (x[0])
          {
          case '|':
            break;
          default:
            fn3 (x);
          }
      fn1 (y.a, z.a);
    }
  return r;
}

__attribute__((noinline, noclone)) int
bar (int x, struct A *y)
{
  switch (x)
    {
    case 219:
      foo ("+", y[-2], y[0]);
    case 220:
      foo ("-", y[-2], y[0]);
    }
}

int
main ()
{
  struct A a[3] = { { &v[1], 1, 1LL }, { &v[0], 0, 0LL }, { &v[2], 2, 2LL } };
  bar (220, a + 2);
  if (v[1] != 1)
    __builtin_abort ();
  return 0;
}

Reply via email to