blob: 25c792ee2e29d3ca7950ae281e4f972731517cf5 [file] [log] [blame]
/* Test exercising invalid calls to arithmetic overflow checking built-ins,
including PR c/71392 - SEGV calling integer overflow built-ins with a null
pointer, (issuing a warning for such invocations). */
/* { dg-do compile } */
/* { dg-additional-options "-Wnonnull" }
/* Verify that calls with fewer or more than 3 arguments to the generic
__builtin_op_overflow functions are rejected. */
#ifndef __cplusplus
#define bool _Bool
#endif
int
generic_0 (void)
{
int x = __builtin_add_overflow (); /* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow (); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow (); /* { dg-error "too few arguments to function" } */
x += __builtin_add_overflow_p (); /* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow_p (); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow_p (); /* { dg-error "too few arguments to function" } */
return x;
}
int
generic_1 (int a)
{
int x = __builtin_add_overflow (a); /* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow (a); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow (a); /* { dg-error "too few arguments to function" } */
/* Literal argument. */
x += __builtin_add_overflow (1); /* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow (2); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow (3); /* { dg-error "too few arguments to function" } */
return x;
}
int
generic_2 (int a, int b)
{
int x = __builtin_add_overflow (a, b);/* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow (a, b); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow (a, b); /* { dg-error "too few arguments to function" } */
x += __builtin_add_overflow (a, 1); /* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow (a, 2); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow (a, 3); /* { dg-error "too few arguments to function" } */
x += __builtin_add_overflow (4, b); /* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow (5, b); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow (6, b); /* { dg-error "too few arguments to function" } */
return x;
}
/* Verify that calls with the correct number of arguments to the generic
__builtin_op_overflow functions are accepted. */
int
generic_3 (int a, int b, int c)
{
int x = __builtin_add_overflow (a, b, &c);
x += __builtin_sub_overflow (a, b, &c);
x += __builtin_mul_overflow (a, b, &c);
x += __builtin_add_overflow (a, 1, &c);
x += __builtin_sub_overflow (a, 2, &c);
x += __builtin_mul_overflow (a, 3, &c);
x += __builtin_add_overflow (4, b, &c);
x += __builtin_sub_overflow (5, b, &c);
x += __builtin_mul_overflow (6, b, &c);
x += __builtin_add_overflow (7, 8, &c);
x += __builtin_sub_overflow (9, 10, &c);
x += __builtin_mul_overflow (11, 12, &c);
/* Verify that a null pointer to an integer is diagnosed. */
/* The following two are rejected due to c/71479 - error on
__builtin_add_overflow with bool or enum pointer as last argument.
x += __builtin_add_overflow (0, 0, (bool *)0);
enum E { e0 };
x += __builtin_add_overflow (0, 0, (enum E *)0); */
x += __builtin_sub_overflow (0, 0, (char *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_add_overflow (0, 0, (short *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_add_overflow (a, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_sub_overflow (a, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_mul_overflow (a, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_add_overflow (a, 1, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_sub_overflow (a, 2, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_mul_overflow (a, 3, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_add_overflow (4, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_sub_overflow (5, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_mul_overflow (6, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_add_overflow (7, 8, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_sub_overflow (9, 10, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_mul_overflow (11, 12, (int *)0); /* { dg-warning "argument 3 null" } */
return x;
}
int
generic_4 (int a, int b, int *c, int d)
{
int x = __builtin_add_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
x += __builtin_sub_overflow (a, b, c, d, d, d); /* { dg-error "too many arguments to function" } */
x += __builtin_sub_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
x += __builtin_mul_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
return x;
}
/* Verify that calls with fewer or more than 3 arguments to the type
specific forms of the __builtin_op_overflow functions are rejected. */
int
generic_wrong_type (int a, int b)
{
void *p = 0;
double d = 0;
int x = __builtin_add_overflow (a, b, p); /* { dg-error "does not have pointer to integral type" } */
x += __builtin_sub_overflow (a, b, &p); /* { dg-error "does not have pointer to integral type" } */
x += __builtin_mul_overflow (a, b, &d); /* { dg-error "does not have pointer to integral type" } */
/* Also verify literal arguments. */
x += __builtin_add_overflow (1, 1, p); /* { dg-error "does not have pointer to integral type" } */
x += __builtin_sub_overflow (1, 1, &p); /* { dg-error "does not have pointer to integral type" } */
x += __builtin_mul_overflow (1, 1, &d); /* { dg-error "does not have pointer to integral type" } */
return x;
}
/* Verify that calls with fewer than 2 or more than 3 arguments to
the typed __builtin_op_overflow functions are rejected. */
int
typed_0 (void)
{
int x = __builtin_add_overflow (); /* { dg-error "too few arguments to function" } */
x += __builtin_sub_overflow (); /* { dg-error "too few arguments to function" } */
x += __builtin_mul_overflow (); /* { dg-error "too few arguments to function" } */
return x;
}
int
typed_1 (int a)
{
int x = __builtin_sadd_overflow (a); /* { dg-error "too few arguments to function" } */
x += __builtin_ssub_overflow (a); /* { dg-error "too few arguments to function" } */
x += __builtin_smul_overflow (a); /* { dg-error "too few arguments to function" } */
return x;
}
int
typed_2 (int a, int b)
{
int x = __builtin_sadd_overflow (a, b); /* { dg-error "too few arguments to function" } */
x += __builtin_ssub_overflow (a, b); /* { dg-error "too few arguments to function" } */
x += __builtin_smul_overflow (a, b); /* { dg-error "too few arguments to function" } */
return x;
}
/* Exercise PR c/71392 - SEGV calling integer overflow built-ins with
a null pointer. Verify that calls with a null argument are diagnosed
with -Wnonnull. */
int
typed_3_null (int a, int b)
{
int x = 0;
x += __builtin_sadd_overflow (a, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_uadd_overflow (a, b, (unsigned *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_saddl_overflow (a, b, (long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_uaddl_overflow (a, b, (unsigned long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_saddll_overflow (a, b, (long long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_uaddll_overflow (a, b, (unsigned long long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_ssub_overflow (a, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_usub_overflow (a, b, (unsigned *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_ssubl_overflow (a, b, (long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_usubl_overflow (a, b, (unsigned long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_ssubll_overflow (a, b, (long long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_usubll_overflow (a, b, (unsigned long long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_smul_overflow (a, b, (int *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_umul_overflow (a, b, (unsigned *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_smull_overflow (a, b, (long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_umull_overflow (a, b, (unsigned long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_smulll_overflow (a, b, (long long *)0); /* { dg-warning "argument 3 null" } */
x += __builtin_umulll_overflow (a, b, (unsigned long long *)0); /* { dg-warning "argument 3 null" } */
return x;
}
int
typed_4 (int a, int b, int *c, int d)
{
int x = __builtin_sadd_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
x += __builtin_ssub_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
x += __builtin_smul_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
return x;
}
int
f2 (int a, int b, int *c, int d)
{
int x = __builtin_add_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
x += __builtin_sub_overflow (a, b, c, d, d, d); /* { dg-error "too many arguments to function" } */
x += __builtin_mul_overflow (a, b, c, d); /* { dg-error "too many arguments to function" } */
x += __builtin_add_overflow_p (a, b, d, d); /* { dg-error "too many arguments to function" } */
x += __builtin_sub_overflow_p (a, b, d, d, 1, d); /* { dg-error "too many arguments to function" } */
x += __builtin_mul_overflow_p (a, b, d, d); /* { dg-error "too many arguments to function" } */
return x;
}
enum E { e0 = 0, e1 = 1 };
int
f3 (float fa, int a, _Complex long int ca, double fb, void *pb, int b, enum E eb, bool bb, int *c)
{
int x = __builtin_add_overflow (fa, b, c); /* { dg-error "argument 1 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_sub_overflow (ca, b, c); /* { dg-error "argument 1 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_mul_overflow (a, fb, c); /* { dg-error "argument 2 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_add_overflow (a, pb, c); /* { dg-error "argument 2 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_sub_overflow (a, eb, c);
x += __builtin_mul_overflow (a, bb, c);
x += __builtin_add_overflow_p (fa, b, a); /* { dg-error "argument 1 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_sub_overflow_p (ca, b, eb); /* { dg-error "argument 1 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_mul_overflow_p (a, fb, bb); /* { dg-error "argument 2 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_add_overflow_p (a, pb, a); /* { dg-error "argument 2 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_sub_overflow_p (a, eb, eb); /* { dg-error "argument 3 in call to function\[^\n\r]*has enumerated type" } */
x += __builtin_mul_overflow_p (a, bb, bb); /* { dg-error "argument 3 in call to function\[^\n\r]*has boolean type" } */
x += __builtin_add_overflow_p (a, b, fa); /* { dg-error "argument 3 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_sub_overflow_p (a, b, ca); /* { dg-error "argument 3 in call to function\[^\n\r]*does not have integral type" } */
x += __builtin_mul_overflow_p (a, b, c); /* { dg-error "argument 3 in call to function\[^\n\r]*does not have integral type" } */
return x;
}
int
f4 (float *fp, double *dp, _Complex int *cp, enum E *ep, bool *bp, long long int *llp)
{
int x = __builtin_add_overflow (1, 2, fp); /* { dg-error "argument 3 in call to function\[^\n\r]*does not have pointer to integral type" } */
x += __builtin_sub_overflow (1, 2, dp); /* { dg-error "argument 3 in call to function\[^\n\r]*does not have pointer to integral type" } */
x += __builtin_mul_overflow (1, 2, cp); /* { dg-error "argument 3 in call to function\[^\n\r]*does not have pointer to integral type" } */
x += __builtin_add_overflow (1, 2, ep); /* { dg-error "argument 3 in call to function\[^\n\r]*has pointer to enumerated type" } */
x += __builtin_sub_overflow (1, 2, bp); /* { dg-error "argument 3 in call to function\[^\n\r]*has pointer to boolean type" } */
x += __builtin_mul_overflow (1, 2, llp);
return x;
}