295 lines
10 KiB
C++
295 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: 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) {}
|