llvm/clang/test/Analysis/ctu-main.cpp

// 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}}
}