171 lines
5.0 KiB
C++
171 lines
5.0 KiB
C++
|
// RUN: rm -rf %t && mkdir %t
|
||
|
// RUN: mkdir -p %t/ctudir
|
||
|
// RUN: %clang_cc1 -std=c++14 -triple x86_64-pc-linux-gnu \
|
||
|
// RUN: -emit-pch -o %t/ctudir/ctu-other.cpp.ast %S/Inputs/ctu-other.cpp
|
||
|
// RUN: %clang_cc1 -std=c++14 -triple x86_64-pc-linux-gnu \
|
||
|
// RUN: -emit-pch -o %t/ctudir/ctu-chain.cpp.ast %S/Inputs/ctu-chain.cpp
|
||
|
// RUN: cp %S/Inputs/ctu-other.cpp.externalDefMap.ast-dump.txt %t/ctudir/externalDefMap.txt
|
||
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple x86_64-pc-linux-gnu \
|
||
|
// RUN: -analyzer-checker=core,debug.ExprInspection \
|
||
|
// RUN: -analyzer-config experimental-enable-naive-ctu-analysis=true \
|
||
|
// RUN: -analyzer-config ctu-dir=%t/ctudir \
|
||
|
// RUN: -verify %s
|
||
|
// RUN: %clang_analyze_cc1 -std=c++14 -triple x86_64-pc-linux-gnu \
|
||
|
// RUN: -analyzer-checker=core,debug.ExprInspection \
|
||
|
// RUN: -analyzer-config experimental-enable-naive-ctu-analysis=true \
|
||
|
// RUN: -analyzer-config ctu-dir=%t/ctudir \
|
||
|
// RUN: -analyzer-config display-ctu-progress=true 2>&1 %s | FileCheck %s
|
||
|
|
||
|
// CHECK: CTU loaded AST file: {{.*}}ctu-other.cpp.ast
|
||
|
// CHECK: CTU loaded AST file: {{.*}}ctu-chain.cpp.ast
|
||
|
|
||
|
#include "ctu-hdr.h"
|
||
|
|
||
|
void clang_analyzer_eval(int);
|
||
|
|
||
|
int f(int);
|
||
|
int g(int);
|
||
|
int h(int);
|
||
|
|
||
|
int callback_to_main(int x) { return x + 1; }
|
||
|
|
||
|
namespace myns {
|
||
|
int fns(int x);
|
||
|
|
||
|
namespace embed_ns {
|
||
|
int fens(int x);
|
||
|
}
|
||
|
|
||
|
class embed_cls {
|
||
|
public:
|
||
|
int fecl(int x);
|
||
|
};
|
||
|
}
|
||
|
|
||
|
class mycls {
|
||
|
public:
|
||
|
int fcl(int x);
|
||
|
virtual int fvcl(int x);
|
||
|
static int fscl(int x);
|
||
|
|
||
|
class embed_cls2 {
|
||
|
public:
|
||
|
int fecl2(int x);
|
||
|
};
|
||
|
};
|
||
|
|
||
|
class derived : public mycls {
|
||
|
public:
|
||
|
virtual int fvcl(int x) override;
|
||
|
};
|
||
|
|
||
|
namespace chns {
|
||
|
int chf1(int x);
|
||
|
}
|
||
|
|
||
|
int fun_using_anon_struct(int);
|
||
|
int other_macro_diag(int);
|
||
|
|
||
|
extern const int extInt;
|
||
|
namespace intns {
|
||
|
extern const int extInt;
|
||
|
}
|
||
|
struct S {
|
||
|
int a;
|
||
|
};
|
||
|
extern const S extS;
|
||
|
extern const int extHere;
|
||
|
const int extHere = 6;
|
||
|
struct A {
|
||
|
static const int a;
|
||
|
};
|
||
|
struct SC {
|
||
|
const int a;
|
||
|
};
|
||
|
extern SC extSC;
|
||
|
struct ST {
|
||
|
static struct SC sc;
|
||
|
};
|
||
|
struct SCNest {
|
||
|
struct SCN {
|
||
|
const int a;
|
||
|
} scn;
|
||
|
};
|
||
|
extern SCNest extSCN;
|
||
|
extern SCNest::SCN extSubSCN;
|
||
|
struct SCC {
|
||
|
SCC(int c);
|
||
|
const int a;
|
||
|
};
|
||
|
extern SCC extSCC;
|
||
|
union U {
|
||
|
const int a;
|
||
|
const unsigned int b;
|
||
|
};
|
||
|
extern U extU;
|
||
|
|
||
|
void test_virtual_functions(mycls* obj) {
|
||
|
// The dynamic type is known.
|
||
|
clang_analyzer_eval(mycls().fvcl(1) == 8); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(derived().fvcl(1) == 9); // expected-warning{{TRUE}}
|
||
|
// We cannot decide about the dynamic type.
|
||
|
clang_analyzer_eval(obj->fvcl(1) == 8); // expected-warning{{FALSE}} expected-warning{{TRUE}}
|
||
|
}
|
||
|
|
||
|
class TestAnonUnionUSR {
|
||
|
public:
|
||
|
inline float f(int value) {
|
||
|
union {
|
||
|
float f;
|
||
|
int i;
|
||
|
};
|
||
|
i = value;
|
||
|
return f;
|
||
|
}
|
||
|
static const int Test;
|
||
|
};
|
||
|
|
||
|
extern int testImportOfIncompleteDefaultParmDuringImport(int);
|
||
|
|
||
|
extern int testImportOfDelegateConstructor(int);
|
||
|
|
||
|
int main() {
|
||
|
clang_analyzer_eval(f(3) == 2); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(f(4) == 3); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(f(5) == 3); // expected-warning{{FALSE}}
|
||
|
clang_analyzer_eval(g(4) == 6); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(h(2) == 8); // expected-warning{{TRUE}}
|
||
|
|
||
|
clang_analyzer_eval(myns::fns(2) == 9); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(myns::embed_ns::fens(2) == -1); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(mycls().fcl(1) == 6); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(mycls::fscl(1) == 7); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(myns::embed_cls().fecl(1) == -6); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(mycls::embed_cls2().fecl2(0) == -11); // expected-warning{{TRUE}}
|
||
|
|
||
|
clang_analyzer_eval(chns::chf1(4) == 12); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(fun_using_anon_struct(8) == 8); // expected-warning{{TRUE}}
|
||
|
|
||
|
clang_analyzer_eval(other_macro_diag(1) == 1); // expected-warning{{TRUE}}
|
||
|
// expected-warning@Inputs/ctu-other.cpp:93{{REACHABLE}}
|
||
|
MACRODIAG(); // expected-warning{{REACHABLE}}
|
||
|
|
||
|
clang_analyzer_eval(extInt == 2); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(intns::extInt == 3); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(extS.a == 4); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(extHere == 6); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(A::a == 3); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(extSC.a == 8); // expected-warning{{TRUE}}
|
||
|
clang_analyzer_eval(ST::sc.a == 2); // expected-warning{{TRUE}}
|
||
|
// clang_analyzer_eval(extSCN.scn.a == 9); // TODO
|
||
|
clang_analyzer_eval(extSubSCN.a == 1); // expected-warning{{TRUE}}
|
||
|
// clang_analyzer_eval(extSCC.a == 7); // TODO
|
||
|
clang_analyzer_eval(extU.a == 4); // expected-warning{{TRUE}}
|
||
|
|
||
|
clang_analyzer_eval(TestAnonUnionUSR::Test == 5); // expected-warning{{TRUE}}
|
||
|
|
||
|
clang_analyzer_eval(testImportOfIncompleteDefaultParmDuringImport(9) == 9); // expected-warning{{TRUE}}
|
||
|
|
||
|
clang_analyzer_eval(testImportOfDelegateConstructor(10) == 10); // expected-warning{{TRUE}}
|
||
|
}
|