// 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 eagerly-assume=false \
// RUN: -analyzer-config experimental-enable-naive-ctu-analysis=true \
// RUN: -analyzer-config ctu-dir=%t/ctudir \
// RUN: -analyzer-config ctu-phase1-inlining=none \
// RUN: -verify=newctu %s
// Simulate the behavior of the previous CTU implementation by inlining all
// functions during the first phase. This way, the second phase is a noop.
// RUN: %clang_analyze_cc1 -std=c++14 -triple x86_64-pc-linux-gnu \
// RUN: -analyzer-checker=core,debug.ExprInspection \
// RUN: -analyzer-config eagerly-assume=false \
// RUN: -analyzer-config experimental-enable-naive-ctu-analysis=true \
// RUN: -analyzer-config ctu-dir=%t/ctudir \
// RUN: -analyzer-config ctu-phase1-inlining=all \
// RUN: -verify=oldctu %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 S extNonConstS;
struct NonTrivialS {
int a;
// User declaring a dtor makes it non-trivial.
~NonTrivialS();
};
extern const NonTrivialS extNTS;
extern const int extHere;
const int extHere = 6;
struct A {
static const int a;
};
struct SC {
const int a;
};
extern const SC extSC;
struct ST {
static const struct SC sc;
};
struct SCNest {
struct SCN {
const int a;
} scn;
};
extern SCNest extSCN;
extern const SCNest::SCN extSubSCN;
struct SCC {
SCC(int c);
const int a;
};
extern SCC extSCC;
union U {
const int a;
const unsigned int b;
};
extern const U extU;
void test_virtual_functions(mycls* obj) {
// The dynamic type is known.
clang_analyzer_eval(mycls().fvcl(1) == 8); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(derived().fvcl(1) == 9); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
// We cannot decide about the dynamic type.
clang_analyzer_eval(obj->fvcl(1) == 8); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} ctu, stu
// oldctu-warning@-2{{TRUE}}
// oldctu-warning@-3{{UNKNOWN}}
}
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); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(f(4) == 3); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(f(5) == 3); // newctu-warning{{FALSE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{FALSE}}
clang_analyzer_eval(g(4) == 6); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(h(2) == 8); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(myns::fns(2) == 9); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(myns::embed_ns::fens(2) == -1); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(mycls().fcl(1) == 6); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(mycls::fscl(1) == 7); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(myns::embed_cls().fecl(1) == -6); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(mycls::embed_cls2().fecl2(0) == -11); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(chns::chf1(4) == 12); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(fun_using_anon_struct(8) == 8); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
clang_analyzer_eval(other_macro_diag(1) == 1); // newctu-warning{{TRUE}} ctu
// newctu-warning@-1{{UNKNOWN}} stu
// oldctu-warning@-2{{TRUE}}
// newctu-warning@Inputs/ctu-other.cpp:93{{REACHABLE}}
// oldctu-warning@Inputs/ctu-other.cpp:93{{REACHABLE}}
MACRODIAG(); // newctu-warning{{REACHABLE}}
// oldctu-warning@-1{{REACHABLE}}
// FIXME we should report an UNKNOWN as well for all external variables!
clang_analyzer_eval(extInt == 2); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(intns::extInt == 3); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(extS.a == 4); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(extNonConstS.a == 4); // newctu-warning{{UNKNOWN}}
// oldctu-warning@-1{{UNKNOWN}}
// Do not import non-trivial classes' initializers.
clang_analyzer_eval(extNTS.a == 4); // newctu-warning{{UNKNOWN}}
// oldctu-warning@-1{{UNKNOWN}}
clang_analyzer_eval(extHere == 6); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(A::a == 3); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(extSC.a == 8); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(ST::sc.a == 2); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
// clang_analyzer_eval(extSCN.scn.a == 9); // TODO
clang_analyzer_eval(extSubSCN.a == 1); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
// clang_analyzer_eval(extSCC.a == 7); // TODO
clang_analyzer_eval(extU.a == 4); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(TestAnonUnionUSR::Test == 5); // newctu-warning{{TRUE}}
// oldctu-warning@-1{{TRUE}}
clang_analyzer_eval(testImportOfIncompleteDefaultParmDuringImport(9) == 9);
// newctu-warning@-1{{TRUE}} ctu
// newctu-warning@-2{{UNKNOWN}} stu
// oldctu-warning@-3{{TRUE}}
clang_analyzer_eval(testImportOfDelegateConstructor(10) == 10);
// newctu-warning@-1{{TRUE}} ctu
// newctu-warning@-2{{UNKNOWN}} stu
// oldctu-warning@-3{{TRUE}}
}