// RUN: %clang_cc1 -std=c++20 -Wunsafe-buffer-usage -fsafe-buffer-usage-suggestions -verify %s namespace std { class type_info { }; } void local_assign_both_span() { int tmp; int* p = new int[10]; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' to 'std::span' to propagate bounds information between them}} tmp = p[4]; // expected-note{{used in buffer access here}} int* q = new int[10]; // expected-warning{{'q' is an unsafe pointer used for buffer access}} expected-note{{change type of 'q' to 'std::span' to preserve bounds information, and change 'p' to 'std::span' to propagate bounds information between them}} tmp = q[4]; // expected-note{{used in buffer access here}} q = p; } void local_assign_rhs_span() { int tmp; int* p = new int[10]; int* q = new int[10]; // expected-warning{{'q' is an unsafe pointer used for buffer access}} tmp = q[4]; // expected-note{{used in buffer access here}} p = q; // FIXME: we do not fix `p = q` here as the `.data()` fix-it is not generally correct } void local_assign_no_span() { int tmp; int* p = new int[10]; int* q = new int[10]; p = q; } void local_assign_lhs_span() { int tmp; int* p = new int[10]; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' to 'std::span' to propagate bounds information between them}} tmp = p[4]; // expected-note{{used in buffer access here}} int* q = new int[10]; p = q; } void lhs_span_multi_assign() { int *a = new int[2]; int *b = a; int *c = b; int *d = c; // expected-warning{{'d' is an unsafe pointer used for buffer access}} expected-note{{change type of 'd' to 'std::span' to preserve bounds information, and change 'c', 'b', and 'a' to 'std::span' to propagate bounds information between them}} int tmp = d[2]; // expected-note{{used in buffer access here}} } void rhs_span() { int *x = new int[3]; int *y; // expected-warning{{'y' is an unsafe pointer used for buffer access}} y[5] = 10; // expected-note{{used in buffer access here}} x = y; // FIXME: we do not fix `x = y` here as the `.data()` fix-it is not generally correct } void rhs_span1() { int *q = new int[12]; int *p = q; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' and 'r' to 'std::span' to propagate bounds information between them}} p[5] = 10; // expected-note{{used in buffer access here}} int *r = q; // expected-warning{{'r' is an unsafe pointer used for buffer access}} expected-note{{change type of 'r' to 'std::span' to preserve bounds information, and change 'p' and 'q' to 'std::span' to propagate bounds information between them}} r[10] = 5; // expected-note{{used in buffer access here}} } void rhs_span2() { int *q = new int[6]; int *p = q; // expected-warning{{'p' is an unsafe pointer used for buffer access}} p[5] = 10; // expected-note{{used in buffer access here}} int *r = q; // FIXME: we do not fix `int *r = q` here as the `.data()` fix-it is not generally correct } void test_grouping() { int *z = new int[8]; int tmp; int *y = new int[10]; // expected-warning{{'y' is an unsafe pointer used for buffer access}} tmp = y[5]; // expected-note{{used in buffer access here}} int *x = new int[10]; x = y; // FIXME: we do not fix `x = y` here as the `.data()` fix-it is not generally correct int *w = z; } void test_grouping1() { int tmp; int *y = new int[10]; // expected-warning{{'y' is an unsafe pointer used for buffer access}} tmp = y[5]; // expected-note{{used in buffer access here}} int *x = new int[10]; x = y; // FIXME: we do not fix `x = y` here as the `.data()` fix-it is not generally correct int *w = new int[10]; // expected-warning{{'w' is an unsafe pointer used for buffer access}} tmp = w[5]; // expected-note{{used in buffer access here}} int *z = new int[10]; z = w; // FIXME: we do not fix `z = w` here as the `.data()` fix-it is not generally correct } void foo1a() { int *r = new int[7]; int *p = new int[4]; // expected-warning{{'p' is an unsafe pointer used for buffer access}} p = r; int tmp = p[9]; // expected-note{{used in buffer access here}} int *q; q = r; // FIXME: we do not fix `q = r` here as the `.data()` fix-it is not generally correct } void foo1b() { int *r = new int[7]; int *p = new int[4]; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'r' and 'q' to 'std::span' to propagate bounds information between them}} p = r; int tmp = p[9]; // expected-note{{used in buffer access here}} int *q; // expected-warning{{'q' is an unsafe pointer used for buffer access}} expected-note{{change type of 'q' to 'std::span' to preserve bounds information, and change 'p' and 'r' to 'std::span' to propagate bounds information between them}} q = r; tmp = q[9]; // expected-note{{used in buffer access here}} } void foo1c() { int *r = new int[7]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} int *p = new int[4]; p = r; // FIXME: we do not fix `p = r` here as the `.data()` fix-it is not generally correct int tmp = r[9]; // expected-note{{used in buffer access here}} int *q; // expected-warning{{'q' is an unsafe pointer used for buffer access}} q = r; // FIXME: we do not fix `q = r` here as the `.data()` fix-it is not generally correct tmp = q[9]; // expected-note{{used in buffer access here}} } void foo2a() { int *r = new int[7]; int *p = new int[5]; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' and 'r' to 'std::span' to propagate bounds information between them}} int *q = new int[4]; p = q; int tmp = p[8]; // expected-note{{used in buffer access here}} q = r; } void foo2b() { int *r = new int[7]; int *p = new int[5]; int *q = new int[4]; // expected-warning{{'q' is an unsafe pointer used for buffer access}} p = q; // FIXME: we do not fix `p = q` here as the `.data()` fix-it is not generally correct int tmp = q[8]; // expected-note{{used in buffer access here}} q = r; } void foo2c() { int *r = new int[7]; int *p = new int[5]; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' and 'r' to 'std::span' to propagate bounds information between them}} int *q = new int[4]; // expected-warning{{'q' is an unsafe pointer used for buffer access}} expected-note{{change type of 'q' to 'std::span' to preserve bounds information, and change 'p' and 'r' to 'std::span' to propagate bounds information between them}} p = q; int tmp = p[8]; // expected-note{{used in buffer access here}} q = r; tmp = q[8]; // expected-note{{used in buffer access here}} } void foo3a() { int *r = new int[7]; int *p = new int[5]; // expected-warning{{'p' is an unsafe pointer used for buffer access}} int *q = new int[4]; q = p; // FIXME: we do not fix `q = p` here as the `.data()` fix-it is not generally correct int tmp = p[8]; // expected-note{{used in buffer access here}} q = r; } void foo3b() { int *r = new int[7]; int *p = new int[5]; int *q = new int[4]; // expected-warning{{'q' is an unsafe pointer used for buffer access}} //expected-note{{change type of 'q' to 'std::span' to preserve bounds information, and change 'r' and 'p' to 'std::span' to propagate bounds information between them}} q = p; int tmp = q[8]; // expected-note{{used in buffer access here}} q = r; } void test_crash() { int *r = new int[8]; int *q = r; int *p; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' and 'r' to 'std::span' to propagate bounds information between them}} p = q; int tmp = p[9]; // expected-note{{used in buffer access here}} } void foo_uuc() { int *ptr; int *local; // expected-warning{{'local' is an unsafe pointer used for buffer access}} local = ptr; local++; // expected-note{{used in pointer arithmetic here}} (local = ptr) += 5; // expected-warning{{unsafe pointer arithmetic}} } void check_rhs_fix() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} // expected-note{{change type of 'r' to 'std::span' to preserve bounds information, and change 'x' to 'std::span' to propagate bounds information between them}} int *x; r[7] = 9; // expected-note{{used in buffer access here}} r = x; } void check_rhs_nofix() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} int *x; // expected-warning{{'x' is an unsafe pointer used for buffer access}} r[7] = 9; // expected-note{{used in buffer access here}} r = x; x++; // expected-note{{used in pointer arithmetic here}} } void check_rhs_nofix_order() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} int *x; // expected-warning{{'x' is an unsafe pointer used for buffer access}} x++; // expected-note{{used in pointer arithmetic here}} r[7] = 9; // expected-note{{used in buffer access here}} r = x; } void check_rhs_nofix_order1() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} r[7] = 9; // expected-note{{used in buffer access here}} int *x; // expected-warning{{'x' is an unsafe pointer used for buffer access}} x++; // expected-note{{used in pointer arithmetic here}} r = x; } void check_rhs_nofix_order2() { int *x; // expected-warning{{'x' is an unsafe pointer used for buffer access}} int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} r[7] = 9; // expected-note{{used in buffer access here}} x++; // expected-note{{used in pointer arithmetic here}} r = x; } void check_rhs_nofix_order3() { int *x; // expected-warning{{'x' is an unsafe pointer used for buffer access}} int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} r = x; r[7] = 9; // expected-note{{used in buffer access here}} x++; // expected-note{{used in pointer arithmetic here}} } void check_rhs_nofix_order4() { int *x; // expected-warning{{'x' is an unsafe pointer used for buffer access}} int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} r[7] = 9; // expected-note{{used in buffer access here}} r = x; x++; // expected-note{{used in pointer arithmetic here}} } void no_unhandled_lhs() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} // expected-note{{change type of 'r' to 'std::span' to preserve bounds information, and change 'x' to 'std::span' to propagate bounds information between them}} r[7] = 9; // expected-note{{used in buffer access here}} int *x; r = x; } const std::type_info unhandled_lhs() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} r[7] = 9; // expected-note{{used in buffer access here}} int *x; r = x; return typeid(*r); } const std::type_info unhandled_rhs() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} r[7] = 9; // expected-note{{used in buffer access here}} int *x; r = x; return typeid(*x); } void test_negative_index() { int *x = new int[4]; // expected-warning{{'x' is an unsafe pointer used for buffer access}} int *p; // expected-warning{{'p' is an unsafe pointer used for buffer access}} p = &x[1]; // expected-note{{used in buffer access here}} p[-1] = 9; // expected-note{{used in buffer access here}} } void test_unfixable() { int *r = new int[8]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} int *x; // expected-warning{{'x' is an unsafe pointer used for buffer access}} x[7] = 9; // expected-note{{used in buffer access here}} r = x; r++; // expected-note{{used in pointer arithmetic here}} } void test_cyclic_deps() { int *r = new int[10]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} expected-note{{change type of 'r' to 'std::span' to preserve bounds information, and change 'p' and 'q' to 'std::span' to propagate bounds information between them}} int *q; q = r; int *p; p = q; r[3] = 9; // expected-note{{used in buffer access here}} r = p; } void test_cyclic_deps_a() { int *r = new int[10]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} int *q; q = r; int *p; // expected-warning{{'p' is an unsafe pointer used for buffer access}} p = q; r[3] = 9; // expected-note{{used in buffer access here}} r = p; p++; // expected-note{{used in pointer arithmetic here}} } void test_cyclic_deps1() { int *r = new int[10]; int *q; q = r; int *p; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' and 'r' to 'std::span' to propagate bounds information between them}} p = q; p[3] = 9; // expected-note{{used in buffer access here}} r = p; } void test_cyclic_deps2() { int *r = new int[10]; int *q; // expected-warning{{'q' is an unsafe pointer used for buffer access}} expected-note{{change type of 'q' to 'std::span' to preserve bounds information, and change 'r' and 'p' to 'std::span' to propagate bounds information between them}} q = r; int *p; p = q; q[3] = 9; // expected-note{{used in buffer access here}} r = p; } void test_cyclic_deps3() { int *r = new int[10]; int *q; // expected-warning{{'q' is an unsafe pointer used for buffer access}} expected-note{{change type of 'q' to 'std::span' to preserve bounds information, and change 'r' and 'p' to 'std::span' to propagate bounds information between them}} q = r; int *p; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'q' and 'r' to 'std::span' to propagate bounds information between them}} p = q; q[3] = 9; // expected-note{{used in buffer access here}} p[4] = 7; // expected-note{{used in buffer access here}} r = p; } void test_cyclic_deps4() { int *r = new int[10]; // expected-warning{{'r' is an unsafe pointer used for buffer access}} expected-note{{change type of 'r' to 'std::span' to preserve bounds information, and change 'p' and 'q' to 'std::span' to propagate bounds information between them}} int *q; // expected-warning{{'q' is an unsafe pointer used for buffer access}} expected-note{{change type of 'q' to 'std::span' to preserve bounds information, and change 'r' and 'p' to 'std::span' to propagate bounds information between them}} q = r; int *p; // expected-warning{{'p' is an unsafe pointer used for buffer access}} expected-note{{change type of 'p' to 'std::span' to preserve bounds information, and change 'r' and 'q' to 'std::span' to propagate bounds information between them}} p = q; q[3] = 9; // expected-note{{used in buffer access here}} p[4] = 7; // expected-note{{used in buffer access here}} r[1] = 5; // expected-note{{used in buffer access here}} r = p; }