mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-26 15:36:09 +00:00

Some AST nodes which stands for implicit initialization is shared. The analyzer will do the same evaluation on the same nodes resulting in the same state. The analyzer will "cache out", i.e. it thinks that it visited an already existing node in the exploded graph. This is not true in this case and we lose coverage. Since these nodes do not really require any processing from the analyzer we just omit them from the CFG. Differential Revision: https://reviews.llvm.org/D71371
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) {}
|