blob: a38679fa8883f197543d0dd6ebabd8f0f3ee1b6b [file] [log] [blame]
/* PR tree-optimization/86552 - missing warning for reading past the end
of non-string arrays
Exercise non-string detection in sprintf.
{ dg-do compile }
{ dg-options "-O2 -Wno-array-bounds -Wall -ftrack-macro-expansion=0" } */
#include "range.h"
typedef __WCHAR_TYPE__ wchar_t;
extern int sprintf (char*, const char*, ...);
extern char *dst;
int i0 = 0;
int i1 = 1;
void sink (int, ...);
#define CONCAT(a, b) a ## b
#define CAT(a, b) CONCAT(a, b)
#define T(fmt, ...) \
sink (sprintf (dst, fmt, __VA_ARGS__))
const char a[5] = "12345"; /* { dg-message "declared here" } */
const char b[6] = "123456"; /* { dg-message "declared here" } */
const char a2[][3] = {
"", "1", "12", "123", "123\000" /* { dg-warning "initializer-string for array of 'char' is too long" } */
};
void test_narrow (void)
{
/* Verify that precision suppresses the warning when it's less
than the size of the array. */
T ("%.0s%.1s%.2s%.3s%.4s%.5s", a, a, a, a, a, a);
T ("%s", a); /* { dg-warning ".%s. directive argument is not a nul-terminated string" } */
T ("%.6s", a); /* { dg-warning ".%.6s. directive argument is not a nul-terminated string" } */
/* Exercise conditional expressions involving strings and non-strings. */
const char *s0 = i0 < 0 ? a2[0] : a2[3];
T ("%s", s0); /* { dg-warning ".%s. directive argument is not a nul-terminated string" } */
s0 = i0 < 0 ? "123456" : a2[4];
T ("%s", s0); /* { dg-warning ".%s. directive argument is not a nul-terminated string" } */
const char *s1 = i0 < 0 ? a2[3] : a2[0];
T ("%s", s1); /* { dg-warning ".%s. directive argument is not a nul-terminated string" } */
const char *s2 = i0 < 0 ? a2[3] : a2[4];
T ("%s", s2); /* { dg-warning ".%s. directive argument is not a nul-terminated string" } */
s0 = i0 < 0 ? a : b;
T ("%.5s", s0);
/* Verify that the warning triggers even if precision prevents
reading past the end of one of the non-terminated arrays but
not the other. */
T ("%.6s", s0); /* { dg-warning ".%.6s. directive argument is not a nul-terminated string" } */
s0 = i0 < 0 ? b : a;
T ("%.7s", s0); /* { dg-warning ".%.7s. directive argument is not a nul-terminated string" } */
/* Verify that at -Wformat-overflow=1 the lower bound of precision
given by a range is used to determine whether or not to warn. */
int r = SR (4, 5);
T ("%.*s", r, a);
T ("%.*s", r, b);
r = SR (5, 6);
T ("%.*s", r, a);
T ("%.*s", r, b);
r = SR (6, 7);
T ("%.*s", r, a); /* { dg-warning ".%.\\\*s. directive argument is not a nul-terminated string" } */
T ("%.*s", r, b);
}
const wchar_t wa[5] = L"12345"; /* { dg-message "declared here" } */
void test_wide (void)
{
T ("%.0ls%.1ls%.2ls%.3ls%.4ls%.5ls", wa, wa, wa, wa, wa, wa);
T ("%ls", wa); /* { dg-warning ".%ls. directive argument is not a nul-terminated string" } */
T ("%.6ls", wa); /* { dg-warning ".%.6ls. directive argument is not a nul-terminated string" } */
}