This patch adds the run-time CFG branch that would skip initialization of virtual base classes depending on whether the constructor is called from a superclass constructor or not. Previously the Static Analyzer was already skipping virtual base-class initializers in such constructors, but it wasn't skipping their arguments and their potential side effects, which was causing pr41300 (and was generally incorrect). The previous skipping behavior is now replaced with a hard assertion that we're not even getting there due to how our CFG works. The new CFG element is under a CFG build option so that not to break other consumers of the CFG by this change. Static Analyzer support for this change is implemented. Differential Revision: https://reviews.llvm.org/D61816 llvm-svn: 361681
296 lines
10 KiB
C++
296 lines
10 KiB
C++
// 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: /*implicit*/(int)0
|
|
// CHECK-NEXT: 8: i([B1.7]) (Member initializer)
|
|
// CHECK-NEXT: 9: this
|
|
// CHECK-NEXT: 10: [B1.9]->i
|
|
// CHECK-NEXT: 11: r([B1.10]) (Member initializer)
|
|
// WARNINGS-NEXT: 12: (CXXConstructExpr, class A)
|
|
// ANALYZER-NEXT: 12: (CXXConstructExpr, [B1.13], class A)
|
|
// CHECK-NEXT: 13: 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) {}
|