blob: a2117bc80eac89ce3e489091704ed1c383f0c3a7 [file] [log] [blame]
// RUN: %clang_analyze_cc1 -analyzer-checker=core,debug.ExprInspection -std=c++11 -verify %s
// RUN: %clang_analyze_cc1 -analyzer-checker=core,debug.ExprInspection -std=c++17 -verify %s
void clang_analyzer_eval(bool);
struct S{
static int CtorInvocationCount;
static int DtorInvocationCount;
S(){CtorInvocationCount++;}
~S(){DtorInvocationCount++;}
};
int S::CtorInvocationCount = 0;
int S::DtorInvocationCount = 0;
void zeroSizeArrayStack() {
S::CtorInvocationCount = 0;
S arr[0];
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
}
void zeroSizeMultidimensionalArrayStack() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
{
S arr[2][0];
S arr2[0][2];
S arr3[0][2][2];
S arr4[2][2][0];
S arr5[2][0][2];
}
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}
void zeroSizeArrayStackInLambda() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
[]{
S arr[0];
}();
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}
void zeroSizeArrayHeap() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
auto *arr = new S[0];
delete[] arr;
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}
void zeroSizeMultidimensionalArrayHeap() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
auto *arr = new S[2][0];
delete[] arr;
auto *arr2 = new S[0][2];
delete[] arr2;
auto *arr3 = new S[0][2][2];
delete[] arr3;
auto *arr4 = new S[2][2][0];
delete[] arr4;
auto *arr5 = new S[2][0][2];
delete[] arr5;
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}
#if __cplusplus >= 201703L
void zeroSizeArrayBinding() {
S::CtorInvocationCount = 0;
S arr[0];
// Note: This is an error in gcc but a warning in clang.
// In MSVC the declaration of 'S arr[0]' is already an error
// and it doesn't recognize this syntax as a structured binding.
auto [] = arr; //expected-warning{{ISO C++17 does not allow a decomposition group to be empty}}
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
}
#endif
void zeroSizeArrayLambdaCapture() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
S arr[0];
auto l = [arr]{};
[arr]{}();
//FIXME: These should be TRUE. We should avoid calling the destructor
// of the temporary that is materialized as the lambda.
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}} expected-warning{{FALSE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}} expected-warning{{FALSE}}
}
// FIXME: Report a warning if the standard is at least C++17.
#if __cplusplus < 201703L
void zeroSizeArrayLambdaCaptureUndefined1() {
S arr[0];
int n;
auto l = [arr, n]{
int x = n; //expected-warning{{Assigned value is garbage or undefined}}
(void) x;
};
l();
}
#endif
void zeroSizeArrayLambdaCaptureUndefined2() {
S arr[0];
int n;
[arr, n]{
int x = n; //expected-warning{{Assigned value is garbage or undefined}}
(void) x;
}();
}
struct Wrapper{
S arr[0];
};
void zeroSizeArrayMember() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
{
Wrapper W;
}
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}
void zeroSizeArrayMemberCopyMove() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
{
Wrapper W;
Wrapper W2 = W;
Wrapper W3 = (Wrapper&&) W2;
}
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}
struct MultiWrapper{
S arr[2][0];
};
void zeroSizeMultidimensionalArrayMember() {
S::CtorInvocationCount = 0;
S::DtorInvocationCount = 0;
{
MultiWrapper MW;
}
clang_analyzer_eval(S::CtorInvocationCount == 0); //expected-warning{{TRUE}}
clang_analyzer_eval(S::DtorInvocationCount == 0); //expected-warning{{TRUE}}
}