/* 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; }