blob: 44e5bd36127d16b934160ee93b66ce520125b73d [file] [log] [blame]
/* PR tree-optimization/83776: missing -Warray-bounds indexing past the end
of a string literal
Test to exercise warnings for computations of otherwise in-bounds indices
into strings that temporarily exceed the bounds of the string.
{ dg-do compile }
{ dg-options "-O2 -Warray-bounds=2 -ftrack-macro-expansion=0" } */
#include "range.h"
#define MAX DIFF_MAX
#define MIN DIFF_MIN
typedef __WCHAR_TYPE__ wchar_t;
void sink (int, ...);
#define T(expr) sink (0, expr)
void test_narrow (void)
{
int i = SR (1, 2);
const char *p0 = "12";
const char *p1 = p0 + i; /* points at '2' or beyond */
const char *p2 = p1 + i; /* points at '\0' or beyond */
const char *p3 = p2 + i; /* points just past the end */
const char *p4 = p3 + i; /* invalid */
T (p0[-1]); /* { dg-warning "array subscript \(-1|\[0-9\]+) is outside array bounds of .char\\\[3]." } */
T (p0[0]);
T (p0[1]);
T (p0[2]);
T (p0[3]); /* { dg-warning "array subscript 3 is outside array bounds of .char\\\[3]." } */
T (&p0[-1]); /* { dg-warning "array subscript \(-1|\[0-9\]+) is \(above|below|outside\) array bounds of .char\\\[3]." } */
T (&p0[0]);
T (&p0[1]);
T (&p0[2]);
T (&p0[3]);
T (&p0[4]); /* { dg-warning "array subscript 4 is \(above|outside\) array bounds of .char\\\[3]." } */
T (p1[-3]); /* { dg-warning "array subscript \\\[-2, -1] is outside array bounds of .char\\\[3]." } */
T (p1[-2]);
T (p1[-1]);
T (p1[ 0]);
T (p1[ 1]);
T (p1[ 2]); /* { dg-warning "array subscript 3 is outside array bounds of .char\\\[3]." } */
T (p1[ 3]); /* { dg-warning "array subscript \\\[4, 5] is outside array bounds of .char\\\[3]." } */
T (&p1[-3]); /* { dg-warning "array subscript \\\[-2, -1] is outside array bounds of .char\\\[3]." "bug" { xfail *-*-* } } */
T (&p1[-2]);
T (&p1[-1]);
T (&p1[ 0]);
T (&p1[ 1]);
T (&p1[ 2]);
T (&p1[ 3]); /* { dg-warning "array subscript \\\[4, 6] is outside array bounds of .char\\\[3]." "bug" { xfail *-*-* } } */
T (p2[-4]); /* { dg-warning "subscript \\\[-2, -1\\\] is outside array bounds of .char\\\[3]." } */
T (p2[-3]);
T (p2[-2]);
T (p2[-1]);
T (p2[ 0]);
/* Even though the lower bound of p3's offsets is in bounds, in order
to subtract 4 from p3 and get a dereferenceable pointer its value
would have to be out-of-bounds. */
T (p3[-4]); /* { dg-warning "array subscript -1 is outside array bounds of .char\\\[3]." } */
T (p3[-3]);
T (p3[-2]);
T (p3[-1]);
T (p3[ 0]); /* { dg-warning "array subscript 3 is outside array bounds of .char\\\[3]." } */
T (p4[-4]); /* { dg-warning "intermediate array offset 4 is outside array bounds of .char\\\[3]." } */
T (p4[-3]); /* { dg-warning "intermediate array offset 4 is outside array bounds of .char\\\[3]." } */
T (p4[-2]); /* { dg-warning "intermediate array offset 4 is outside array bounds of .char\\\[3]." } */
/* The final subscripts below are invalid. */
T (p4[-1]); /* { dg-warning "array subscript 3 is outside array bounds of .char\\\[3]." } */
T (p4[ 0]); /* { dg-warning "array subscript \\\[4, 5] is outside array bounds of .char\\\[3]." } */
}
void test_narrow_vflow (void)
{
int i = SR (DIFF_MAX - 2, DIFF_MAX);
int j = SR (1, DIFF_MAX);
const char *p0 = "123";
const char *p1 = p0 + i; /* points at '2' or beyond */
const char *p2 = p1 + i; /* points at '\0' or beyond */
const char *p3 = p2 + i; /* points just past the end */
const char *p4 = p3 + i; /* invalid */
}
void test_wide (void)
{
int i = SR (1, 2);
const wchar_t *p0 = L"123";
const wchar_t *p1 = p0 + i; /* points at L'2' or beyond */
const wchar_t *p2 = p1 + i; /* points at L'3' or beyond */
const wchar_t *p3 = p2 + i; /* points at L'\0' or beyond */
const wchar_t *p4 = p3 + i; /* points just past the end */
const wchar_t *p5 = p4 + i; /* invalid */
T (p0[0]);
T (p0[1]);
T (p0[2]);
T (p0[3]);
T (p0[4]); /* { dg-warning "array subscript 4 is outside array bounds of .\[a-z \]+\\\[4]." } */
T (p1[-1]);
T (p1[ 0]);
T (p1[ 1]);
T (p1[ 2]);
T (p1[ 3]); /* { dg-warning "array subscript 4 is outside array bounds of .\[a-z \]+\\\[4]." } */
T (&p1[-1]);
T (&p1[ 0]);
T (&p1[ 1]);
T (&p1[ 2]);
T (&p1[ 3]);
T (&p1[ 4]); /* { dg-warning "array subscript \\\[5, 6] is outside array bounds of .\[a-z \]+\\\[4]." "bug" { xfail *-*-* } } */
T (p2[-5]); /* { dg-warning "array subscript \\\[-3, -1] is outside array bounds of .\[a-z \]+\\\[4]." } */
T (p2[-4]);
T (p2[-3]);
T (p2[-2]);
T (p2[-1]);
T (p2[ 0]);
/* Even though the lower bound of p3's offsets is in bounds, in order
to subtract 5 from p3 and get a dereferenceable pointer its value
would have to be out-of-bounds. */
T (p3[-5]); /* { dg-warning "array subscript \\\[-2, -1\\\] is outside array bounds of .\[a-z \]+\\\[4]." } */
T (p3[-4]);
T (p3[-3]);
T (p3[-2]);
T (p3[-1]);
T (p3[ 0]);
T (p3[ 1]); /* { dg-warning "array subscript 4 is outside array bounds of .\[a-z \]+\\\[4]." } */
T (p4[-5]); /* { dg-warning "array subscript -1 is outside array bounds of .\[a-z \]+\\\[4]." } */
T (p4[-4]);
T (p4[-3]);
T (p4[-2]);
T (p4[-1]);
T (p4[ 0]); /* { dg-warning "array subscript 4 is outside array bounds of .\[a-z \]+\\\[4]." } */
}