/* PR tree-optimization/102238 - missing -Wrestrict on sprintf formatting a struct member into enclosing object { dg-do compile } { dg-options "-O2 -Wall -Wno-format-overflow" } */ extern int sprintf (char*, const char*, ...); extern void sink (void*, ...); struct A { char a[4]; }; struct B { struct A a1, a2; }; extern struct B eb; enum { B_a2_a_off = __builtin_offsetof (struct B, a2.a) }; void test_warn_src_decl_plus (void) { { char *s = (char*)&eb + B_a2_a_off; char *d = eb.a2.a; sprintf (d, "%s", s); // { dg-warning "overlaps" } } { // If strlen (s) > 0 there is overlap with a[1]. char *s = (char*)&eb + B_a2_a_off + 1; char *d = eb.a2.a; sprintf (d, "%s", s); // { dg-warning "may overlap" } } { // strlen (s) must be at most 1 so there can be no overlap with a. char *s = (char*)&eb + B_a2_a_off + 2; char *d = eb.a2.a; sprintf (d, "%s", s); // { dg-bogus "-Wrestrict" } } { // strlen (s) must be at most 0 so there can be no overlap with a. char *s = (char*)&eb + B_a2_a_off + 3; char *d = eb.a2.a; sprintf (d, "%s", s); // { dg-bogus "-Wrestrict" } } } void test_warn_src_ptr_plus (struct B *p) { { char *s = (char*)p + B_a2_a_off; char *d = p->a2.a; sprintf (d, "%s", s); // { dg-warning "overlaps" } } { // If strlen (s) > 0 there is overlap with a[1]. char *s = (char*)p + B_a2_a_off + 1; char *d = p->a2.a; sprintf (d, "%s", s); // { dg-warning "may overlap" } } { // strlen (s) must be at most 1 so there can be no overlap with a. char *s = (char*)p + B_a2_a_off + 2; char *d = p->a2.a; sprintf (d, "%s", s); // { dg-bogus "-Wrestrict" } } { // strlen (s) must be at most 0 so there can be no overlap with a. char *s = (char*)p + B_a2_a_off + 3; char *d = p->a2.a; sprintf (d, "%s", s); // { dg-bogus "-Wrestrict" } } } void test_warn_dst_decl_plus (void) { { char *s = eb.a2.a; char *d = (char*)&eb + B_a2_a_off; sprintf (d, "%s", s); // { dg-warning "overlaps" } } { // If strlen (a) > 0 there is overlap with a[1]. char *s = eb.a2.a; char *d = (char*)&eb + B_a2_a_off + 1; sprintf (d, "%s", s); // { dg-warning "may overlap" } } { // If strlen (a) > 1 there is overlap with a[2]. char *s = eb.a2.a; char *d = (char*)&eb + B_a2_a_off + 2; sprintf (d, "%s", s); // { dg-warning "may overlap" } } { // If strlen (a) > 2 there is overlap with a[3]. char *s = eb.a2.a; char *d = (char*)&eb + B_a2_a_off + 3; sprintf (d, "%s", s); // { dg-warning "may overlap" } } } void test_warn_dst_ptr_plus (struct B *p) { { char *s = p->a2.a; char *d = (char*)p + B_a2_a_off; sprintf (d, "%s", s); // { dg-warning "overlaps" } } { // If strlen (a) > 0 there is overlap with a[1]. char *s = p->a2.a; char *d = (char*)p + B_a2_a_off + 1; sprintf (d, "%s", s); // { dg-warning "may overlap" } } { // If strlen (a) > 1 there is overlap with a[2]. char *s = p->a2.a; char *d = (char*)p + B_a2_a_off + 2; sprintf (d, "%s", s); // { dg-warning "may overlap" } } { // If strlen (a) > 2 there is overlap with a[3]. char *s = p->a2.a; char *d = (char*)p + B_a2_a_off + 3; sprintf (d, "%s", s); // { dg-warning "may overlap" } } }