// RUN: %clang_analyze_cc1 -std=c++11 -analyzer-checker=debug.DumpCFG -analyzer-config cfg-rich-constructors=false %s 2>&1 | FileCheck -check-prefixes=CHECK,WARNINGS %s // RUN: %clang_analyze_cc1 -std=c++11 -analyzer-checker=debug.DumpCFG -analyzer-config cfg-rich-constructors=true %s 2>&1 | FileCheck -check-prefixes=CHECK,ANALYZER %s // This file tests how we construct two different flavors of the Clang CFG - // the CFG used by the Sema analysis-based warnings and the CFG used by the // static analyzer. The difference in the behavior is checked via FileCheck // prefixes (WARNINGS and ANALYZER respectively). When introducing new analyzer // flags, no new run lines should be added - just these flags would go to the // respective line depending on where is it turned on and where is it turned // off. Feel free to add tests that test only one of the CFG flavors if you're // not sure how the other flavor is supposed to work in your case. class A { public: // CHECK: A() // CHECK: [B1 (ENTRY)] // CHECK-NEXT: Succs (1): B0 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (1): B1 A() {} // CHECK: A(int i) // CHECK: [B1 (ENTRY)] // CHECK-NEXT: Succs (1): B0 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (1): B1 A(int i) {} }; class B : public virtual A { public: // CHECK: B() // CHECK: [B3 (ENTRY)] // CHECK-NEXT: Succs (1): B2 // CHECK: [B1] // WARNINGS-NEXT: 1: (CXXConstructExpr, class A) // ANALYZER-NEXT: 1: (CXXConstructExpr, A() (Base initializer), class A) // CHECK-NEXT: 2: A([B1.1]) (Base initializer) // CHECK-NEXT: Preds (1): B2 // CHECK-NEXT: Succs (1): B0 // CHECK: [B2] // CHECK-NEXT: T: (See if most derived ctor has already initialized vbases) // CHECK-NEXT: Preds (1): B3 // CHECK-NEXT: Succs (2): B0 B1 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (2): B1 B2 B() {} // CHECK: B(int i) // CHECK: [B3 (ENTRY)] // CHECK-NEXT: Succs (1): B2 // CHECK: [B1] // CHECK-NEXT: 1: i // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, LValueToRValue, int) // WARNINGS-NEXT: 3: [B1.2] (CXXConstructExpr, class A) // ANALYZER-NEXT: 3: [B1.2] (CXXConstructExpr, A([B1.2]) (Base initializer), class A) // CHECK-NEXT: 4: A([B1.3]) (Base initializer) // CHECK-NEXT: Preds (1): B2 // CHECK-NEXT: Succs (1): B0 // CHECK: [B2] // CHECK-NEXT: T: (See if most derived ctor has already initialized vbases) // CHECK-NEXT: Preds (1): B3 // CHECK-NEXT: Succs (2): B0 B1 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (2): B1 B2 B(int i) : A(i) {} }; class C : public virtual A { public: // CHECK: C() // CHECK: [B3 (ENTRY)] // CHECK-NEXT: Succs (1): B2 // CHECK: [B1] // WARNINGS-NEXT: 1: (CXXConstructExpr, class A) // ANALYZER-NEXT: 1: (CXXConstructExpr, A() (Base initializer), class A) // CHECK-NEXT: 2: A([B1.1]) (Base initializer) // CHECK-NEXT: Preds (1): B2 // CHECK-NEXT: Succs (1): B0 // CHECK: [B2] // CHECK-NEXT: T: (See if most derived ctor has already initialized vbases) // CHECK-NEXT: Preds (1): B3 // CHECK-NEXT: Succs (2): B0 B1 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (2): B1 B2 C() {} // CHECK: C(int i) // CHECK: [B3 (ENTRY)] // CHECK-NEXT: Succs (1): B2 // CHECK: [B1] // CHECK-NEXT: 1: i // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, LValueToRValue, int) // WARNINGS-NEXT: 3: [B1.2] (CXXConstructExpr, class A) // ANALYZER-NEXT: 3: [B1.2] (CXXConstructExpr, A([B1.2]) (Base initializer), class A) // CHECK-NEXT: 4: A([B1.3]) (Base initializer) // CHECK-NEXT: Preds (1): B2 // CHECK-NEXT: Succs (1): B0 // CHECK: [B2] // CHECK-NEXT: T: (See if most derived ctor has already initialized vbases) // CHECK-NEXT: Preds (1): B3 // CHECK-NEXT: Succs (2): B0 B1 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (2): B1 B2 C(int i) : A(i) {} }; class TestOrder : public C, public B, public A { int i; int& r; public: TestOrder(); }; // CHECK: TestOrder::TestOrder() // CHECK: [B4 (ENTRY)] // CHECK-NEXT: Succs (1): B3 // CHECK: [B1] // WARNINGS-NEXT: 1: (CXXConstructExpr, class C) // ANALYZER-NEXT: 1: (CXXConstructExpr, C() (Base initializer), class C) // CHECK-NEXT: 2: C([B1.1]) (Base initializer) // WARNINGS-NEXT: 3: (CXXConstructExpr, class B) // ANALYZER-NEXT: 3: (CXXConstructExpr, B() (Base initializer), class B) // CHECK-NEXT: 4: B([B1.3]) (Base initializer) // WARNINGS-NEXT: 5: (CXXConstructExpr, class A) // ANALYZER-NEXT: 5: (CXXConstructExpr, A() (Base initializer), class A) // CHECK-NEXT: 6: A([B1.5]) (Base initializer) // CHECK-NEXT: 7: i(/*implicit*/(int)0) (Member initializer) // CHECK-NEXT: 8: this // CHECK-NEXT: 9: [B1.8]->i // CHECK-NEXT: 10: r([B1.9]) (Member initializer) // WARNINGS-NEXT: 11: (CXXConstructExpr, class A) // ANALYZER-NEXT: 11: (CXXConstructExpr, [B1.12], class A) // CHECK-NEXT: 12: A a; // CHECK-NEXT: Preds (2): B2 B3 // CHECK-NEXT: Succs (1): B0 // CHECK: [B2] // WARNINGS-NEXT: 1: (CXXConstructExpr, class A) // ANALYZER-NEXT: 1: (CXXConstructExpr, A() (Base initializer), class A) // CHECK-NEXT: 2: A([B2.1]) (Base initializer) // CHECK-NEXT: Preds (1): B3 // CHECK-NEXT: Succs (1): B1 // CHECK: [B3] // CHECK-NEXT: T: (See if most derived ctor has already initialized vbases) // CHECK-NEXT: Preds (1): B4 // CHECK-NEXT: Succs (2): B1 B2 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (1): B1 TestOrder::TestOrder() : r(i), B(), i(), C() { A a; } class TestControlFlow { int x, y, z; public: TestControlFlow(bool b); }; // CHECK: TestControlFlow::TestControlFlow(bool b) // CHECK: [B5 (ENTRY)] // CHECK-NEXT: Succs (1): B4 // CHECK: [B1] // CHECK-NEXT: 1: [B4.4] ? [B2.1] : [B3.1] // CHECK-NEXT: 2: y([B1.1]) (Member initializer) // CHECK-NEXT: 3: this // CHECK-NEXT: 4: [B1.3]->y // CHECK-NEXT: 5: [B1.4] (ImplicitCastExpr, LValueToRValue, int) // CHECK-NEXT: 6: z([B1.5]) (Member initializer) // CHECK-NEXT: 7: int v; // CHECK-NEXT: Preds (2): B2 B3 // CHECK-NEXT: Succs (1): B0 // CHECK: [B2] // CHECK-NEXT: 1: 0 // CHECK-NEXT: Preds (1): B4 // CHECK-NEXT: Succs (1): B1 // CHECK: [B3] // CHECK-NEXT: 1: 1 // CHECK-NEXT: Preds (1): B4 // CHECK-NEXT: Succs (1): B1 // CHECK: [B4] // CHECK-NEXT: 1: 0 // CHECK-NEXT: 2: x([B4.1]) (Member initializer) // CHECK-NEXT: 3: b // CHECK-NEXT: 4: [B4.3] (ImplicitCastExpr, LValueToRValue, _Bool) // CHECK-NEXT: T: [B4.4] ? ... : ... // CHECK-NEXT: Preds (1): B5 // CHECK-NEXT: Succs (2): B2 B3 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (1): B1 TestControlFlow::TestControlFlow(bool b) : y(b ? 0 : 1) , x(0) , z(y) { int v; } class TestDelegating { int x, z; public: // CHECK: TestDelegating() // CHECK: [B2 (ENTRY)] // CHECK-NEXT: Succs (1): B1 // CHECK: [B1] // CHECK-NEXT: 1: 2 // CHECK-NEXT: 2: 3 // WARNINGS-NEXT: 3: [B1.1], [B1.2] (CXXConstructExpr, class TestDelegating) // ANALYZER-NEXT: 3: [B1.1], [B1.2] (CXXConstructExpr, TestDelegating([B1.1], [B1.2]) (Delegating initializer), class TestDelegating) // CHECK-NEXT: 4: TestDelegating([B1.3]) (Delegating initializer) // CHECK-NEXT: Preds (1): B2 // CHECK-NEXT: Succs (1): B0 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (1): B1 TestDelegating() : TestDelegating(2, 3) {} // CHECK: TestDelegating(int x, int z) // CHECK: [B2 (ENTRY)] // CHECK-NEXT: Succs (1): B1 // CHECK: [B1] // CHECK-NEXT: 1: x // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, LValueToRValue, int) // CHECK-NEXT: 3: x([B1.2]) (Member initializer) // CHECK-NEXT: 4: z // CHECK-NEXT: 5: [B1.4] (ImplicitCastExpr, LValueToRValue, int) // CHECK-NEXT: 6: z([B1.5]) (Member initializer) // CHECK-NEXT: Preds (1): B2 // CHECK-NEXT: Succs (1): B0 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (1): B1 TestDelegating(int x, int z) : x(x), z(z) {} }; class TestMoreControlFlow : public virtual A { A a; public: TestMoreControlFlow(bool coin); }; // CHECK: TestMoreControlFlow::TestMoreControlFlow(bool coin) // CHECK: [B10 (ENTRY)] // CHECK-NEXT: Succs (1): B9 // CHECK: [B1] // CHECK-NEXT: 1: [B4.2] ? [B2.1] : [B3.1] // WARNINGS-NEXT: 2: [B1.1] (CXXConstructExpr, class A) // ANALYZER-NEXT: 2: [B1.1] (CXXConstructExpr, a([B1.1]) (Member initializer), class A) // CHECK-NEXT: 3: a([B1.2]) (Member initializer) // CHECK-NEXT: Preds (2): B2 B3 // CHECK-NEXT: Succs (1): B0 // CHECK: [B2] // CHECK-NEXT: 1: 3 // CHECK-NEXT: Preds (1): B4 // CHECK-NEXT: Succs (1): B1 // CHECK: [B3] // CHECK-NEXT: 1: 4 // CHECK-NEXT: Preds (1): B4 // CHECK-NEXT: Succs (1): B1 // CHECK: [B4] // CHECK-NEXT: 1: coin // CHECK-NEXT: 2: [B4.1] (ImplicitCastExpr, LValueToRValue, _Bool) // CHECK-NEXT: T: [B4.2] ? ... : ... // CHECK-NEXT: Preds (2): B5 B9 // CHECK-NEXT: Succs (2): B2 B3 // CHECK: [B5] // CHECK-NEXT: 1: [B8.2] ? [B6.1] : [B7.1] // WARNINGS-NEXT: 2: [B5.1] (CXXConstructExpr, class A) // ANALYZER-NEXT: 2: [B5.1] (CXXConstructExpr, A([B5.1]) (Base initializer), class A) // CHECK-NEXT: 3: A([B5.2]) (Base initializer) // CHECK-NEXT: Preds (2): B6 B7 // CHECK-NEXT: Succs (1): B4 // CHECK: [B6] // CHECK-NEXT: 1: 1 // CHECK-NEXT: Preds (1): B8 // CHECK-NEXT: Succs (1): B5 // CHECK: [B7] // CHECK-NEXT: 1: 2 // CHECK-NEXT: Preds (1): B8 // CHECK-NEXT: Succs (1): B5 // CHECK: [B8] // CHECK-NEXT: 1: coin // CHECK-NEXT: 2: [B8.1] (ImplicitCastExpr, LValueToRValue, _Bool) // CHECK-NEXT: T: [B8.2] ? ... : ... // CHECK-NEXT: Preds (1): B9 // CHECK-NEXT: Succs (2): B6 B7 // CHECK: [B9] // CHECK-NEXT: T: (See if most derived ctor has already initialized vbases) // CHECK-NEXT: Preds (1): B10 // CHECK-NEXT: Succs (2): B4 B8 // CHECK: [B0 (EXIT)] // CHECK-NEXT: Preds (1): B1 TestMoreControlFlow::TestMoreControlFlow(bool coin) : A(coin ? 1 : 2), a(coin ? 3 : 4) {}