1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249
|
// 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}}
}
|