101 lines
4.2 KiB
C++
101 lines
4.2 KiB
C++
|
// RUN: %clang_cc1 -Wunsafe-buffer-usage -fsafe-buffer-usage-suggestions \
|
||
|
// RUN: -std=c++20 -verify=expected %s
|
||
|
// RUN: %clang_cc1 -Wunsafe-buffer-usage -fsafe-buffer-usage-suggestions \
|
||
|
// RUN: -mllvm -debug-only=SafeBuffers \
|
||
|
// RUN: -std=c++20 -verify=expected,debug %s
|
||
|
|
||
|
// A generic -debug would also enable our notes. This is probably fine.
|
||
|
//
|
||
|
// RUN: %clang_cc1 -Wunsafe-buffer-usage -fsafe-buffer-usage-suggestions \
|
||
|
// RUN: -std=c++20 -mllvm -debug \
|
||
|
// RUN: -verify=expected,debug %s
|
||
|
|
||
|
// This test file checks the behavior under the assumption that no fixits
|
||
|
// were emitted for the test cases. If -Wunsafe-buffer-usage is improved
|
||
|
// to support these cases (thus failing the test), the test should be changed
|
||
|
// to showcase a different unsupported example.
|
||
|
//
|
||
|
// RUN: %clang_cc1 -Wunsafe-buffer-usage -fsafe-buffer-usage-suggestions \
|
||
|
// RUN: -mllvm -debug-only=SafeBuffers \
|
||
|
// RUN: -std=c++20 -fdiagnostics-parseable-fixits %s \
|
||
|
// RUN: 2>&1 | FileCheck %s
|
||
|
// CHECK-NOT: fix-it:
|
||
|
|
||
|
// This debugging facility is only available in debug builds.
|
||
|
//
|
||
|
// REQUIRES: asserts
|
||
|
|
||
|
void foo() {
|
||
|
int *x = new int[10]; // expected-warning{{'x' is an unsafe pointer used for buffer access}}
|
||
|
x[5] = 10; // expected-note{{used in buffer access here}}
|
||
|
int z = x[-1]; // expected-note{{used in buffer access here}} \
|
||
|
// debug-note{{safe buffers debug: gadget 'ULCArraySubscript' refused to produce a fix}}
|
||
|
}
|
||
|
|
||
|
void failed_decl() {
|
||
|
int a[10]; // expected-warning{{'a' is an unsafe buffer that does not perform bounds checks}} \
|
||
|
// debug-note{{safe buffers debug: failed to produce fixit for declaration 'a' : not a pointer}}
|
||
|
|
||
|
for (int i = 0; i < 10; i++) {
|
||
|
a[i] = i; // expected-note{{used in buffer access here}}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void failed_multiple_decl() {
|
||
|
int *a = new int[4], b; // expected-warning{{'a' is an unsafe pointer used for buffer access}} \
|
||
|
// debug-note{{safe buffers debug: failed to produce fixit for declaration 'a' : multiple VarDecls}}
|
||
|
a[4] = 3; // expected-note{{used in buffer access here}}
|
||
|
}
|
||
|
|
||
|
void failed_param_var_decl(int *a =new int[3]) { // expected-warning{{'a' is an unsafe pointer used for buffer access}} \
|
||
|
// debug-note{{safe buffers debug: failed to produce fixit for declaration 'a' : has default arg}}
|
||
|
a[4] = 6; // expected-note{{used in buffer access here}}
|
||
|
}
|
||
|
|
||
|
void unclaimed_use() {
|
||
|
int *a = new int[3]; // expected-warning{{'a' is an unsafe pointer used for buffer access}}
|
||
|
a[2] = 9; // expected-note{{used in buffer access here}}
|
||
|
int *b = a++; // expected-note{{used in pointer arithmetic here}} \
|
||
|
// debug-note{{safe buffers debug: failed to produce fixit for 'a' : has an unclaimed use}}
|
||
|
}
|
||
|
|
||
|
void implied_unclaimed_var(int *b) { // expected-warning{{'b' is an unsafe pointer used for buffer access}}
|
||
|
int *a = new int[3]; // expected-warning{{'a' is an unsafe pointer used for buffer access}}
|
||
|
a[4] = 7; // expected-note{{used in buffer access here}}
|
||
|
a = b; // debug-note{{safe buffers debug: gadget 'PointerAssignment' refused to produce a fix}}
|
||
|
b++; // expected-note{{used in pointer arithmetic here}} \
|
||
|
// debug-note{{safe buffers debug: failed to produce fixit for 'b' : has an unclaimed use}}
|
||
|
}
|
||
|
|
||
|
int *a = new int[3]; // expected-warning{{'a' is an unsafe pointer used for buffer access}} \
|
||
|
// debug-note{{safe buffers debug: failed to produce fixit for 'a' : neither local nor a parameter}}
|
||
|
void test_globals() {
|
||
|
a[7] = 4; // expected-note{{used in buffer access here}}
|
||
|
}
|
||
|
|
||
|
void test_decomp_decl() {
|
||
|
int a[2] = {1, 2};
|
||
|
auto [x, y] = a;
|
||
|
x = 9;
|
||
|
}
|
||
|
|
||
|
void test_claim_use_multiple() {
|
||
|
int *a = new int[8]; // expected-warning{{'a' is an unsafe pointer used for buffer access}}
|
||
|
a[6] = 9; // expected-note{{used in buffer access here}}
|
||
|
a++; // expected-note{{used in pointer arithmetic here}} \
|
||
|
// debug-note{{safe buffers debug: failed to produce fixit for 'a' : has an unclaimed use}}
|
||
|
}
|
||
|
|
||
|
struct S
|
||
|
{
|
||
|
int *x;
|
||
|
};
|
||
|
|
||
|
S f() { return S{new int[4]}; }
|
||
|
|
||
|
void test_struct_claim_use() {
|
||
|
auto [x] = f();
|
||
|
x[6] = 8; // expected-warning{{unsafe buffer access}}
|
||
|
x++; // expected-warning{{unsafe pointer arithmetic}}
|
||
|
}
|