mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-17 08:36:41 +00:00

- Use Strategy to determine whether to fix a parameter - Fix the `Strategy` construction so that only variables on the graph are assigned the `std::span` strategy Reviewed by: t-rasmud (Rashmi Mudduluru), NoQ (Artem Dergachev) Differential revision: https://reviews.llvm.org/D157441
346 lines
16 KiB
C++
346 lines
16 KiB
C++
// 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;
|
|
}
|