diff options
| author | Marcin Swiderski <marcin.sfider@gmail.com> | 2010-10-04 03:38:22 +0000 | 
|---|---|---|
| committer | Marcin Swiderski <marcin.sfider@gmail.com> | 2010-10-04 03:38:22 +0000 | 
| commit | 87b1bb6fd2dd39eb98964da7ed3e998a4614cf1b (patch) | |
| tree | e7e4023a86ebaec8e104e7962b9c3a27cbcf18bc | |
| parent | 7a98e5b4e59e50a690e6eb2afe5d89bf8fd4bd10 (diff) | |
| download | bcm5719-llvm-87b1bb6fd2dd39eb98964da7ed3e998a4614cf1b.tar.gz bcm5719-llvm-87b1bb6fd2dd39eb98964da7ed3e998a4614cf1b.zip  | |
Added support for C++ initializers in CFG.
llvm-svn: 115493
| -rw-r--r-- | clang/include/clang/Analysis/CFG.h | 6 | ||||
| -rw-r--r-- | clang/lib/Analysis/CFG.cpp | 40 | ||||
| -rw-r--r-- | clang/test/Analysis/initializers-cfg-output.cpp | 89 | 
3 files changed, 128 insertions, 7 deletions
diff --git a/clang/include/clang/Analysis/CFG.h b/clang/include/clang/Analysis/CFG.h index 3960f51704e..f74b58551ef 100644 --- a/clang/include/clang/Analysis/CFG.h +++ b/clang/include/clang/Analysis/CFG.h @@ -432,7 +432,11 @@ public:    void appendStmt(Stmt* Statement, BumpVectorContext &C, bool asLValue) {      Elements.push_back(CFGStmt(Statement, asLValue), C);    } -   + +  void appendInitializer(CXXBaseOrMemberInitializer *I, BumpVectorContext& C) { +    Elements.push_back(CFGInitializer(I), C); +  } +    // Destructors must be inserted in reversed order. So insertion is in two    // steps. First we prepare space for some number of elements, then we insert    // the elements beginning at the last position in prepared space. diff --git a/clang/lib/Analysis/CFG.cpp b/clang/lib/Analysis/CFG.cpp index 047568a32a8..4eac9ae351b 100644 --- a/clang/lib/Analysis/CFG.cpp +++ b/clang/lib/Analysis/CFG.cpp @@ -304,7 +304,7 @@ private:    CFGBlock *addStmt(Stmt *S) {      return Visit(S, AddStmtChoice::AlwaysAdd);    } - +  CFGBlock *addInitializer(CXXBaseOrMemberInitializer *I);    void addAutomaticObjDtors(LocalScope::const_iterator B,                              LocalScope::const_iterator E, Stmt* S); @@ -322,6 +322,9 @@ private:                    AddStmtChoice asc = AddStmtChoice::AlwaysAdd) {      B->appendStmt(S, cfg->getBumpVectorContext(), asc.asLValue());    } +  void appendInitializer(CFGBlock *B, CXXBaseOrMemberInitializer *I) { +    B->appendInitializer(I, cfg->getBumpVectorContext()); +  }    void insertAutomaticObjDtors(CFGBlock* Blk, CFGBlock::iterator I,      LocalScope::const_iterator B, LocalScope::const_iterator E, Stmt* S); @@ -410,14 +413,19 @@ CFG* CFGBuilder::buildCFG(const Decl *D, Stmt* Statement, ASTContext* C,    if (badCFG)      return NULL; -  if (B) -    Succ = B; - +  // For C++ constructor add initializers to CFG.    if (const CXXConstructorDecl *CD = dyn_cast_or_null<CXXConstructorDecl>(D)) { -    // FIXME: Add code for base initializers and member initializers. -    (void)CD; +    for (CXXConstructorDecl::init_const_reverse_iterator I = CD->init_rbegin(), +        E = CD->init_rend(); I != E; ++I) { +      B = addInitializer(*I); +      if (badCFG) +        return NULL; +    }    } +  if (B) +    Succ = B; +    // Backpatch the gotos whose label -> block mappings we didn't know when we    // encountered them.    for (BackpatchBlocksTy::iterator I = BackpatchBlocks.begin(), @@ -466,6 +474,26 @@ CFGBlock* CFGBuilder::createBlock(bool add_successor) {    return B;  } +/// addInitializer - Add C++ base or member initializer element to CFG. +CFGBlock *CFGBuilder::addInitializer(CXXBaseOrMemberInitializer *I) { +  if (!BuildOpts.AddInitializers) +    return Block; + +  autoCreateBlock(); +  appendInitializer(Block, I); + +  if (Expr *Init = I->getInit()) { +    AddStmtChoice::Kind K = AddStmtChoice::NotAlwaysAdd; +    if (FieldDecl *FD = I->getMember()) +      if (FD->getType()->isReferenceType()) +        K = AddStmtChoice::AsLValueNotAlwaysAdd; + +    return Visit(Init, AddStmtChoice(K)); +  } +   +  return Block; +} +  /// addAutomaticObjDtors - Add to current block automatic objects destructors  /// for objects in range of local scope positions. Use S as trigger statement  /// for destructors. diff --git a/clang/test/Analysis/initializers-cfg-output.cpp b/clang/test/Analysis/initializers-cfg-output.cpp new file mode 100644 index 00000000000..0396c47c101 --- /dev/null +++ b/clang/test/Analysis/initializers-cfg-output.cpp @@ -0,0 +1,89 @@ +// RUN: %clang_cc1 -analyze -cfg-dump -cfg-add-initializers %s 2>&1 | FileCheck %s +// XPASS: * + +class A { +public: +  A() {} +  A(int i) {} +}; + +class B : public virtual A { +public: +  B() {} +  B(int i) : A(i) {} +}; + +class C : public virtual A { +public: +  C() {} +  C(int i) : A(i) {} +}; + +class TestOrder : public C, public B, public A { +  int i; +  int& r; +public: +  TestOrder(); +}; + +TestOrder::TestOrder() +  : r(i), B(), i(), C() { +  A a; +} + +class TestControlFlow { +  int x, y, z; +public: +  TestControlFlow(bool b); +}; + +TestControlFlow::TestControlFlow(bool b) +  : y(b ? 0 : 1) +  , x(0) +  , z(y) { +  int v; +} + +// CHECK:  [ B2 (ENTRY) ] +// CHECK:     Predecessors (0): +// CHECK:     Successors (1): B1 +// CHECK:  [ B1 ] +// CHECK:       1: A() (Base initializer) +// CHECK:       2: C() (Base initializer) +// CHECK:       3: B() (Base initializer) +// CHECK:       4: A() (Base initializer) +// CHECK:       5: i(/*implicit*/int()) (Member initializer) +// CHECK:       6: r(this->i) (Member initializer) +// CHECK:       7: A a; +// CHECK:     Predecessors (1): B2 +// CHECK:     Successors (1): B0 +// CHECK:  [ B0 (EXIT) ] +// CHECK:     Predecessors (1): B1 +// CHECK:     Successors (0): +// CHECK:  [ B5 (ENTRY) ] +// CHECK:     Predecessors (0): +// CHECK:     Successors (1): B4 +// CHECK:  [ B1 ] +// CHECK:       1: [B4.2] ? [B2.1] : [B3.1] +// CHECK:       2: y([B1.1]) (Member initializer) +// CHECK:       3: z(this->y) (Member initializer) +// CHECK:       4: int v; +// CHECK:     Predecessors (2): B2 B3 +// CHECK:     Successors (1): B0 +// CHECK:  [ B2 ] +// CHECK:       1: 0 +// CHECK:     Predecessors (1): B4 +// CHECK:     Successors (1): B1 +// CHECK:  [ B3 ] +// CHECK:       1: 1 +// CHECK:     Predecessors (1): B4 +// CHECK:     Successors (1): B1 +// CHECK:  [ B4 ] +// CHECK:       1: x(0) (Member initializer) +// CHECK:       2: b +// CHECK:       T: [B4.2] ? ... : ... +// CHECK:     Predecessors (1): B5 +// CHECK:     Successors (2): B2 B3 +// CHECK:  [ B0 (EXIT) ] +// CHECK:     Predecessors (1): B1 +// CHECK:     Successors (0):  | 

