blob: ad2dd296a9a0ad0382cc15fad279b7ac9f185531 [file] [log] [blame]
/* { dg-do run } */
/* { dg-options "-O2" } */
/* { dg-require-effective-target alloca } */
typedef __SIZE_TYPE__ size_t;
extern void abort (void);
extern void exit (int);
extern void *malloc (size_t);
extern void *calloc (size_t, size_t);
extern void *alloca (size_t);
extern void *memcpy (void *, const void *, size_t);
extern void *memset (void *, int, size_t);
extern char *strcpy (char *, const char *);
struct A
{
char a[10];
int b;
char c[10];
} y, w[4];
extern char exta[];
extern char extb[30];
extern struct A extc[];
struct A zerol[0];
void
__attribute__ ((noinline))
test1 (void *q, int x)
{
struct A a;
void *p = &a.a[3], *r;
char var[x + 10];
struct A vara[x + 10];
if (x < 0)
r = &a.a[9];
else
r = &a.c[1];
if (__builtin_object_size (p, 1) != sizeof (a.a) - 3)
abort ();
if (__builtin_object_size (&a.c[9], 1)
!= sizeof (a.c) - 9)
abort ();
if (__builtin_object_size (q, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (r, 1) != sizeof (a.c) - 1)
abort ();
if (x < 6)
r = &w[2].a[1];
else
r = &a.a[6];
if (__builtin_object_size (&y, 1) != sizeof (y))
abort ();
if (__builtin_object_size (w, 1) != sizeof (w))
abort ();
if (__builtin_object_size (&y.b, 1) != sizeof (a.b))
abort ();
if (__builtin_object_size (r, 1) != sizeof (a.a) - 1)
abort ();
if (x < 20)
r = malloc (30);
else
r = calloc (2, 16);
/* We may duplicate this test onto the two exit paths. On one path
the size will be 32, the other it will be 30. If we don't duplicate
this test, then the size will be 32. */
if (__builtin_object_size (r, 1) != 2 * 16
&& __builtin_object_size (r, 1) != 30)
abort ();
if (x < 20)
r = malloc (30);
else
r = calloc (2, 14);
if (__builtin_object_size (r, 1) != 30)
abort ();
if (x < 30)
r = malloc (sizeof (a));
else
r = &a.a[3];
if (__builtin_object_size (r, 1) != sizeof (a))
abort ();
r = memcpy (r, "a", 2);
if (__builtin_object_size (r, 1) != sizeof (a))
abort ();
r = memcpy (r + 2, "b", 2) + 2;
if (__builtin_object_size (r, 1) != sizeof (a) - 4)
abort ();
r = &a.a[4];
r = memset (r, 'a', 2);
if (__builtin_object_size (r, 1) != sizeof (a.a) - 4)
abort ();
r = memset (r + 2, 'b', 2) + 2;
if (__builtin_object_size (r, 1) != sizeof (a.a) - 8)
abort ();
r = &a.a[1];
r = strcpy (r, "ab");
if (__builtin_object_size (r, 1) != sizeof (a.a) - 1)
abort ();
r = strcpy (r + 2, "cd") + 2;
if (__builtin_object_size (r, 1) != sizeof (a.a) - 5)
abort ();
if (__builtin_object_size (exta, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (exta + 10, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&exta[5], 1) != (size_t) -1)
abort ();
if (__builtin_object_size (extb, 1) != sizeof (extb))
abort ();
if (__builtin_object_size (extb + 10, 1) != sizeof (extb) - 10)
abort ();
if (__builtin_object_size (&extb[5], 1) != sizeof (extb) - 5)
abort ();
if (__builtin_object_size (extc, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (extc + 10, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&extc[5], 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&extc->a, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&(extc + 10)->b, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&extc[5].c[3], 1) != (size_t) -1)
abort ();
if (__builtin_object_size (var, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (var + 10, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&var[5], 1) != (size_t) -1)
abort ();
if (__builtin_object_size (vara, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (vara + 10, 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&vara[5], 1) != (size_t) -1)
abort ();
if (__builtin_object_size (&vara[0].a, 1) != sizeof (vara[0].a))
abort ();
if (__builtin_object_size (&vara[10].a[0], 1) != sizeof (vara[0].a))
abort ();
if (__builtin_object_size (&vara[5].a[4], 1) != sizeof (vara[0].a) - 4)
abort ();
if (__builtin_object_size (&vara[5].b, 1) != sizeof (vara[0].b))
abort ();
if (__builtin_object_size (&vara[7].c[7], 1) != sizeof (vara[0].c) - 7)
abort ();
if (__builtin_object_size (zerol, 1) != 0)
abort ();
if (__builtin_object_size (&zerol, 1) != 0)
abort ();
if (__builtin_object_size (&zerol[0], 1) != 0)
abort ();
if (__builtin_object_size (zerol[0].a, 1) != 0)
abort ();
if (__builtin_object_size (&zerol[0].a[0], 1) != 0)
abort ();
if (__builtin_object_size (&zerol[0].b, 1) != 0)
abort ();
if (__builtin_object_size ("abcdefg", 1) != sizeof ("abcdefg"))
abort ();
if (__builtin_object_size ("abcd\0efg", 1) != sizeof ("abcd\0efg"))
abort ();
if (__builtin_object_size (&"abcd\0efg", 1) != sizeof ("abcd\0efg"))
abort ();
if (__builtin_object_size (&"abcd\0efg"[0], 1) != sizeof ("abcd\0efg"))
abort ();
if (__builtin_object_size (&"abcd\0efg"[4], 1) != sizeof ("abcd\0efg") - 4)
abort ();
if (__builtin_object_size ("abcd\0efg" + 5, 1) != sizeof ("abcd\0efg") - 5)
abort ();
if (__builtin_object_size (L"abcdefg", 1) != sizeof (L"abcdefg"))
abort ();
r = (char *) L"abcd\0efg";
if (__builtin_object_size (r + 2, 1) != sizeof (L"abcd\0efg") - 2)
abort ();
}
size_t l1 = 1;
void
__attribute__ ((noinline))
test2 (void)
{
struct B { char buf1[10]; char buf2[10]; } a;
char *r, buf3[20];
int i;
if (sizeof (a) != 20)
return;
r = buf3;
for (i = 0; i < 4; ++i)
{
if (i == l1 - 1)
r = &a.buf1[1];
else if (i == l1)
r = &a.buf2[7];
else if (i == l1 + 1)
r = &buf3[5];
else if (i == l1 + 2)
r = &a.buf1[9];
}
if (__builtin_object_size (r, 1) != sizeof (buf3))
abort ();
r = &buf3[20];
for (i = 0; i < 4; ++i)
{
if (i == l1 - 1)
r = &a.buf1[7];
else if (i == l1)
r = &a.buf2[7];
else if (i == l1 + 1)
r = &buf3[5];
else if (i == l1 + 2)
r = &a.buf1[9];
}
if (__builtin_object_size (r, 1) != sizeof (buf3) - 5)
abort ();
r += 8;
if (__builtin_object_size (r, 1) != sizeof (buf3) - 13)
abort ();
if (__builtin_object_size (r + 6, 1) != sizeof (buf3) - 19)
abort ();
}
void
__attribute__ ((noinline))
test3 (void)
{
char buf4[10];
struct B { struct A a[2]; struct A b; char c[4]; char d; double e;
_Complex double f; } x;
double y;
_Complex double z;
double *dp;
if (__builtin_object_size (buf4, 1) != sizeof (buf4))
abort ();
if (__builtin_object_size (&buf4, 1) != sizeof (buf4))
abort ();
if (__builtin_object_size (&buf4[0], 1) != sizeof (buf4))
abort ();
if (__builtin_object_size (&buf4[1], 1) != sizeof (buf4) - 1)
abort ();
if (__builtin_object_size (&x, 1) != sizeof (x))
abort ();
if (__builtin_object_size (&x.a, 1) != sizeof (x.a))
abort ();
if (__builtin_object_size (&x.a[0], 1) != sizeof (x.a))
abort ();
if (__builtin_object_size (&x.a[0].a, 1) != sizeof (x.a[0].a))
abort ();
if (__builtin_object_size (&x.a[0].a[0], 1) != sizeof (x.a[0].a))
abort ();
if (__builtin_object_size (&x.a[0].a[3], 1) != sizeof (x.a[0].a) - 3)
abort ();
if (__builtin_object_size (&x.a[0].b, 1) != sizeof (x.a[0].b))
abort ();
if (__builtin_object_size (&x.a[1].c, 1) != sizeof (x.a[1].c))
abort ();
if (__builtin_object_size (&x.a[1].c[0], 1) != sizeof (x.a[1].c))
abort ();
if (__builtin_object_size (&x.a[1].c[3], 1) != sizeof (x.a[1].c) - 3)
abort ();
if (__builtin_object_size (&x.b, 1) != sizeof (x.b))
abort ();
if (__builtin_object_size (&x.b.a, 1) != sizeof (x.b.a))
abort ();
if (__builtin_object_size (&x.b.a[0], 1) != sizeof (x.b.a))
abort ();
if (__builtin_object_size (&x.b.a[3], 1) != sizeof (x.b.a) - 3)
abort ();
if (__builtin_object_size (&x.b.b, 1) != sizeof (x.b.b))
abort ();
if (__builtin_object_size (&x.b.c, 1) != sizeof (x.b.c))
abort ();
if (__builtin_object_size (&x.b.c[0], 1) != sizeof (x.b.c))
abort ();
if (__builtin_object_size (&x.b.c[3], 1) != sizeof (x.b.c) - 3)
abort ();
if (__builtin_object_size (&x.c, 1) != sizeof (x.c))
abort ();
if (__builtin_object_size (&x.c[0], 1) != sizeof (x.c))
abort ();
if (__builtin_object_size (&x.c[1], 1) != sizeof (x.c) - 1)
abort ();
if (__builtin_object_size (&x.d, 1) != sizeof (x.d))
abort ();
if (__builtin_object_size (&x.e, 1) != sizeof (x.e))
abort ();
if (__builtin_object_size (&x.f, 1) != sizeof (x.f))
abort ();
dp = &__real__ x.f;
if (__builtin_object_size (dp, 1) != sizeof (x.f) / 2)
abort ();
dp = &__imag__ x.f;
if (__builtin_object_size (dp, 1) != sizeof (x.f) / 2)
abort ();
dp = &y;
if (__builtin_object_size (dp, 1) != sizeof (y))
abort ();
if (__builtin_object_size (&z, 1) != sizeof (z))
abort ();
dp = &__real__ z;
if (__builtin_object_size (dp, 1) != sizeof (z) / 2)
abort ();
dp = &__imag__ z;
if (__builtin_object_size (dp, 1) != sizeof (z) / 2)
abort ();
}
struct S { unsigned int a; };
char *
__attribute__ ((noinline))
test4 (char *x, int y)
{
register int i;
struct A *p;
for (i = 0; i < y; i++)
{
p = (struct A *) x;
x = (char *) &p[1];
if (__builtin_object_size (p, 1) != (size_t) -1)
abort ();
}
return x;
}
void
__attribute__ ((noinline))
test5 (size_t x)
{
struct T { char buf[64]; char buf2[64]; } t;
char *p = &t.buf[8];
size_t i;
for (i = 0; i < x; ++i)
p = p + 4;
if (__builtin_object_size (p, 1) != sizeof (t.buf) - 8)
abort ();
memset (p, ' ', sizeof (t.buf) - 8 - 4 * 4);
}
void
__attribute__ ((noinline))
test6 (void)
{
char buf[64];
struct T { char buf[64]; char buf2[64]; } t;
char *p = &buf[64], *q = &t.buf[64];
if (__builtin_object_size (p + 64, 1) != 0)
abort ();
if (__builtin_object_size (q + 0, 1) != 0)
abort ();
if (__builtin_object_size (q + 64, 1) != 0)
abort ();
}
void
__attribute__ ((noinline))
test7 (void)
{
struct T { char buf[10]; char buf2[10]; } t;
char *p = &t.buf2[-4];
char *q = &t.buf2[0];
if (__builtin_object_size (p, 1) != 0)
abort ();
if (__builtin_object_size (q, 1) != sizeof (t.buf2))
abort ();
q = &t.buf[10];
if (__builtin_object_size (q, 1) != 0)
abort ();
q = &t.buf[11];
if (__builtin_object_size (q, 1) != 0)
abort ();
p = &t.buf[-4];
if (__builtin_object_size (p, 1) != 0)
abort ();
}
int
main (void)
{
struct S s[10];
__asm ("" : "=r" (l1) : "0" (l1));
test1 (main, 6);
test2 ();
test3 ();
test4 ((char *) s, 10);
test5 (4);
test6 ();
test7 ();
exit (0);
}