blob: a81ab99d72451f86d88c934a964f4f863495dba4 [file] [log] [blame]
/* { dg-do compile }
{ dg-options "-O2 -Wall -Wno-stringop-truncation -ftrack-macro-expansion=0" } */
#define NULL (void*)0
const char a[] = { 'a', 'b', 'c', 'd' };
const char b[] = { 'a', '\0', 'c', '\0', 'e' };
#define CONCAT(a, b) a ## b
#define CAT(a, b) CONCAT (a, b)
typedef struct FILE FILE;
extern FILE *fp;
extern char *d;
extern const char *s;
extern int n;
#define T(func, ...) \
__attribute__ ((noipa)) void \
CAT (test_ ## func, __LINE__) (void) \
{ \
sink (0, __builtin_ ## func (__VA_ARGS__), d, s, n); \
} typedef void dummy_type
void sink (void*, ...);
// Exercise string functions.
T (index, a, 'x'); // { dg-warning "missing terminating nul" "index" }
T (index, a, *s); // { dg-warning "missing terminating nul" "index" }
T (index, b, '0');
T (index, b + 1, '1');
T (index, b + 2, '2');
T (index, b + 3, '3');
T (index, b + 4, '4'); // { dg-warning "missing terminating nul" "index" }
T (rindex, a, 'x'); // { dg-warning "missing terminating nul" "rindex" }
T (rindex, a, *s); // { dg-warning "missing terminating nul" "rindex" }
T (rindex, b, '0');
T (rindex, b + 1, '1');
T (rindex, b + 2, '2');
T (rindex, b + 3, '3');
T (rindex, b + 4, '4'); // { dg-warning "missing terminating nul" "rindex" }
T (stpcpy, d, a); // { dg-warning "missing terminating nul" "stpcpy" }
T (stpncpy, d, a, 4);
T (stpncpy, d, a, 5); // { dg-warning "missing terminating nul" "stpncpy" }
T (stpncpy, d, a, n);
T (stpncpy, d, a + n, 4);
T (stpncpy, d, a + n, 5); // { dg-warning "missing terminating nul" "stpncpy" }
T (stpncpy, d, b, 4);
T (stpncpy, d, b, 5);
T (stpncpy, d, b, n);
T (stpncpy, d, b + 1, 4);
T (stpncpy, d, b + 1, 5);
T (stpncpy, d, b + 1, n);
T (stpncpy, d, b + 3, 4);
T (stpncpy, d, b + 3, 5);
T (stpncpy, d, b + 3, n);
T (stpncpy, d, b + 4, 1);
T (stpncpy, d, b + 4, 2); // { dg-warning "missing terminating nul" "stpncpy" }
T (stpncpy, d, b + 4, n);
/* The following might be worth warning about since it's only safe with
n < 4. */
T (stpncpy, d, b + n, 5);
T (strcasecmp, a, "ab"); // { dg-warning "missing terminating nul" "strcasecmp" }
T (strcasecmp, a, s); // { dg-warning "missing terminating nul" "strcasecmp" }
T (strcasecmp, a, b); // { dg-warning "missing terminating nul" "strcasecmp" }
T (strcasecmp, b, b + 1);
T (strcasecmp, b, b + 2);
T (strcasecmp, b, b + 3);
T (strcasecmp, b, b + 4); // { dg-warning "missing terminating nul" "strcasecmp" }
T (strcat, d, a); // { dg-warning "missing terminating nul" "strcat" }
T (strncat, d, a, 4);
T (strncat, d, a, 5); // { dg-warning "missing terminating nul" "strncat" }
T (strncat, d, a, n);
T (strncat, d, b, n);
T (strncat, d, b + 1, n);
T (strncat, d, b + 2, n);
T (strncat, d, b + 3, n);
T (strncat, d, b + 4, 0);
T (strncat, d, b + 4, 1);
T (strncat, d, b + 4, 2); // { dg-warning "missing terminating nul" "strncat" }
/* The following should probably trigger a warning since it's only safe
when n < 2, makes little sense with n == 0, and not much more with
n == 1. */
T (strncat, d, b + 4, n); // { dg-warning "missing terminating nul" "strncat" { xfail *-*-* } }
T (strchr, a, 'x'); // { dg-warning "missing terminating nul" "strchr" }
T (strchr, a, *s); // { dg-warning "missing terminating nul" "strchr" }
T (strcmp, a, "ab"); // { dg-warning "missing terminating nul" "strcmp" }
T (strcmp, "bc", a); // { dg-warning "missing terminating nul" "strcmp" }
T (strcmp, a, s); // { dg-warning "missing terminating nul" "strcmp" }
T (strcmp, s, a); // { dg-warning "missing terminating nul" "strcmp" }
T (strcmp, a, b); // { dg-warning "missing terminating nul" "strcmp" }
/* Even though most likely safe in reality because b[1] is nul,
the following is strictly undefined because a is not a string.
The warning is not issued because GCC folds the call to (int)*a. */
T (strcmp, a, b + 1); // { dg-warning "missing terminating nul" "bug" { xfail *-*-* } }
T (strncmp, a, "ab", 4);
T (strncmp, "bc", a, 4);
T (strncmp, a, a, 4);
T (strncmp, a, s, 4);
T (strncmp, s, a, 4);
/* The warning below is not issued because GCC folds strncmp calls with
the same arguments to zero before it checks for the missing nul. */
T (strncmp, a, a, 5); // { dg-warning "missing terminating nul" "pr92624" { xfail *-*-*} }
T (strncmp, a, s, 5); // { dg-warning "missing terminating nul" "strcmp" }
T (strncmp, s, a, 5); // { dg-warning "missing terminating nul" "strcmp" }
T (strcpy, d, a); // { dg-warning "missing terminating nul" "strcpy" }
T (strcspn, a, s); // { dg-warning "missing terminating nul" "strcspn" }
T (strcspn, s, a); // { dg-warning "missing terminating nul" "strcspn" }
T (strspn, a, s); // { dg-warning "missing terminating nul" "strcspn" }
T (strspn, s, a); // { dg-warning "missing terminating nul" "strcspn" }
T (strdup, a); // { dg-warning "missing terminating nul" "strdup" }
T (strndup, a, 4);
T (strndup, a, 5); // { dg-warning "missing terminating nul" "strndup" }
T (strndup, b + 3, 2);
T (strndup, b + 4, 1);
T (strndup, b + 4, 2); // { dg-warning "missing terminating nul" "strndup" }
T (strlen, a); // { dg-warning "missing terminating nul" "strlen" }
T (strnlen, a, 4);
T (strnlen, a, 5); // { dg-warning "specified bound 5 exceeds the size 4 of unterminated array" "strnlen" }
T (strnlen, a, n);
T (strpbrk, s, a); // { dg-warning "missing terminating nul" "strpbrk" }
T (strrchr, a, 'x'); // { dg-warning "missing terminating nul" "strrchr" }
T (strrchr, a, *s); // { dg-warning "missing terminating nul" "strrchr" }
T (strstr, a, "cde"); // { dg-warning "missing terminating nul" "strstr" }
T (strstr, a, s); // { dg-warning "missing terminating nul" "strstr" }
// Exercise a few string checking functions.
T (__stpcpy_chk, d, a, -1); // { dg-warning "missing terminating nul" "stpcpy" }
T (__stpncpy_chk, d, a, 4, -1);
T (__stpncpy_chk, d, a, 5, -1); // { dg-warning "missing terminating nul" "stpncpy_chk" }
T (__stpncpy_chk, d, a, n, -1);
T (__stpncpy_chk, d, a + n, 4, -1);
T (__stpncpy_chk, d, a + n, 5, -1); // { dg-warning "missing terminating nul" "stpncpy_chk" }
T (__stpncpy_chk, d, b, 4, -1);
T (__stpncpy_chk, d, b, 5, -1);
T (__stpncpy_chk, d, b, n, -1);
T (__stpncpy_chk, d, b + 1, 4, -1);
T (__stpncpy_chk, d, b + 1, 5, -1);
T (__stpncpy_chk, d, b + 1, n, -1);
T (__stpncpy_chk, d, b + 3, 4, -1);
T (__stpncpy_chk, d, b + 3, 5, -1);
T (__stpncpy_chk, d, b + 3, n, -1);
T (__stpncpy_chk, d, b + 4, 1, -1);
T (__stpncpy_chk, d, b + 4, 2, -1); // { dg-warning "missing terminating nul" "stpncpy_chk" }
T (__stpncpy_chk, d, b + 4, n, -1);
T (__strncat_chk, d, a, 4, -1);
T (__strncat_chk, d, a, 5, -1); // { dg-warning "missing terminating nul" "strncat_chk" }
T (__strncat_chk, d, a, n, -1);
T (__strncat_chk, d, a + n, 4, -1);
T (__strncat_chk, d, a + n, 5, -1); // { dg-warning "missing terminating nul" "strncat_chk" }
T (__strncat_chk, d, b, 4, -1);
T (__strncat_chk, d, b, 5, -1);
T (__strncat_chk, d, b, n, -1);
T (__strncat_chk, d, b + 1, 4, -1);
T (__strncat_chk, d, b + 1, 5, -1);
T (__strncat_chk, d, b + 1, n, -1);
T (__strncat_chk, d, b + 3, 4, -1);
T (__strncat_chk, d, b + 3, 5, -1);
T (__strncat_chk, d, b + 3, n, -1);
T (__strncat_chk, d, b + 4, 1, -1);
T (__strncat_chk, d, b + 4, 2, -1); // { dg-warning "missing terminating nul" "strncat_chk" }
T (__strncat_chk, d, b + 4, n, -1);
T (__strncpy_chk, d, a, 4, -1);
T (__strncpy_chk, d, a, 5, -1); // { dg-warning "missing terminating nul" "strncpy_chk" }
T (__strncpy_chk, d, a, n, -1);
T (__strncpy_chk, d, a + n, 4, -1);
T (__strncpy_chk, d, a + n, 5, -1); // { dg-warning "missing terminating nul" "strncpy_chk" }
T (__strncpy_chk, d, b, 4, -1);
T (__strncpy_chk, d, b, 5, -1);
T (__strncpy_chk, d, b, n, -1);
T (__strncpy_chk, d, b + 1, 4, -1);
T (__strncpy_chk, d, b + 1, 5, -1);
T (__strncpy_chk, d, b + 1, n, -1);
T (__strncpy_chk, d, b + 3, 4, -1);
T (__strncpy_chk, d, b + 3, 5, -1);
T (__strncpy_chk, d, b + 3, n, -1);
T (__strncpy_chk, d, b + 4, 1, -1);
T (__strncpy_chk, d, b + 4, 2, -1); // { dg-warning "missing terminating nul" "strncpy" }
T (__strncpy_chk, d, b + 4, n, -1);
// Exercise some stdio functions.
T (printf, a); // { dg-warning "unterminated format string" "printf" }
T (printf, "%s", a); // { dg-warning "not a nul-terminated string" "printf" }
T (sprintf, d, "%s", a); // { dg-warning "not a nul-terminated string" "sprintf" }
T (snprintf, d, n, "%s", a); // { dg-warning "not a nul-terminated string" "sprintf" }
T (__sprintf_chk, d, 0, -1, "%s", a); // { dg-warning "not a nul-terminated string" "sprintf" }
T (__snprintf_chk, d, n, 0, -1, "%s", a); // { dg-warning "not a nul-terminated string" "sprintf" }
T (fputs, a, fp); // { dg-warning "missing terminating nul" "fputs" }
T (fputs_unlocked, a, fp); // { dg-warning "missing terminating nul" "fputs_unlocked" }
T (puts, a); // { dg-warning "missing terminating nul" "puts" }
T (puts_unlocked, a); // { dg-warning "missing terminating nul" "puts_unlocked" }
// Exerise exec functions.
T (execl, a, s, NULL); // { dg-warning "missing terminating nul" "execl" }
T (execl, a, s, NULL); // { dg-warning "missing terminating nul" "execl" }
T (execle, a, s, NULL, NULL); // { dg-warning "missing terminating nul" "execl" }
T (execlp, a, s, NULL); // { dg-warning "missing terminating nul" "execl" }
T (execv, a, &d); // { dg-warning "missing terminating nul" "execl" }
T (execve, a, &d, &d); // { dg-warning "missing terminating nul" "execl" }
T (execvp, a, &d); // { dg-warning "missing terminating nul" "execl" }
T (gettext, a); // { dg-warning "missing terminating nul" "gettext" }
T (strfmon, d, n, a); // { dg-warning "unterminated format string" "strfmon" }