2018-02-15 11:13:36 +08:00
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=debug.DumpCFG -triple x86_64-apple-darwin12 -analyzer-config cfg-temporary-dtors=true -std=c++11 -w %s > %t 2>&1
|
2018-03-23 05:37:39 +08:00
|
|
|
// RUN: FileCheck --input-file=%t -check-prefixes=CHECK,CXX11 %s
|
|
|
|
// RUN: %clang_analyze_cc1 -analyzer-checker=debug.DumpCFG -triple x86_64-apple-darwin12 -analyzer-config cfg-temporary-dtors=true -std=c++17 -w %s > %t 2>&1
|
|
|
|
// RUN: FileCheck --input-file=%t -check-prefixes=CHECK,CXX17 %s
|
[CFG] Add extra context to C++ constructor statement elements.
This patch adds a new CFGStmt sub-class, CFGConstructor, which replaces
the regular CFGStmt with CXXConstructExpr in it whenever the CFG has additional
information to provide regarding what sort of object is being constructed.
It is useful for figuring out what memory is initialized in client of the
CFG such as the Static Analyzer, which do not operate by recursive AST
traversal, but instead rely on the CFG to provide all the information when they
need it. Otherwise, the statement that triggers the construction and defines
what memory is being initialized would normally occur after the
construct-expression, and the client would need to peek to the next CFG element
or use statement parent map to understand the necessary facts about
the construct-expression.
As a proof of concept, CFGConstructors are added for new-expressions
and the respective test cases are provided to demonstrate how it works.
For now, the only additional data contained in the CFGConstructor element is
the "trigger statement", such as new-expression, which is the parent of the
constructor. It will be significantly expanded in later commits. The additional
data is organized as an auxiliary structure - the "construction context",
which is allocated separately from the CFGElement.
Differential Revision: https://reviews.llvm.org/D42672
llvm-svn: 324668
2018-02-09 06:58:15 +08:00
|
|
|
|
|
|
|
class C {
|
|
|
|
public:
|
|
|
|
C();
|
|
|
|
C(C *);
|
2018-02-13 06:36:36 +08:00
|
|
|
C(int, int);
|
2018-02-10 09:55:23 +08:00
|
|
|
|
|
|
|
static C get();
|
2018-02-15 11:13:36 +08:00
|
|
|
operator bool() const;
|
[CFG] Add extra context to C++ constructor statement elements.
This patch adds a new CFGStmt sub-class, CFGConstructor, which replaces
the regular CFGStmt with CXXConstructExpr in it whenever the CFG has additional
information to provide regarding what sort of object is being constructed.
It is useful for figuring out what memory is initialized in client of the
CFG such as the Static Analyzer, which do not operate by recursive AST
traversal, but instead rely on the CFG to provide all the information when they
need it. Otherwise, the statement that triggers the construction and defines
what memory is being initialized would normally occur after the
construct-expression, and the client would need to peek to the next CFG element
or use statement parent map to understand the necessary facts about
the construct-expression.
As a proof of concept, CFGConstructors are added for new-expressions
and the respective test cases are provided to demonstrate how it works.
For now, the only additional data contained in the CFGConstructor element is
the "trigger statement", such as new-expression, which is the parent of the
constructor. It will be significantly expanded in later commits. The additional
data is organized as an auxiliary structure - the "construction context",
which is allocated separately from the CFGElement.
Differential Revision: https://reviews.llvm.org/D42672
llvm-svn: 324668
2018-02-09 06:58:15 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef __typeof(sizeof(int)) size_t;
|
|
|
|
void *operator new(size_t size, void *placement);
|
|
|
|
|
|
|
|
namespace operator_new {
|
|
|
|
|
|
|
|
// CHECK: void operatorNewWithConstructor()
|
|
|
|
// CHECK: 1: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 2: (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CHECK-NEXT: 3: new C([B1.2])
|
|
|
|
void operatorNewWithConstructor() {
|
|
|
|
new C();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void operatorNewWithConstructorWithOperatorNewWithContstructor()
|
|
|
|
// CHECK: 1: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 2: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 3: (CXXConstructExpr, [B1.4], class C)
|
|
|
|
// CHECK-NEXT: 4: new C([B1.3])
|
|
|
|
// CHECK-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], class C)
|
|
|
|
// CHECK-NEXT: 6: new C([B1.5])
|
|
|
|
void operatorNewWithConstructorWithOperatorNewWithContstructor() {
|
|
|
|
new C(new C());
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void operatorPlacementNewWithConstructorWithinPlacementArgument()
|
|
|
|
// CHECK: 1: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 2: (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CHECK-NEXT: 3: new C([B1.2])
|
|
|
|
// CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, BitCast, void *)
|
|
|
|
// CHECK-NEXT: 5: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 6: (CXXConstructExpr, [B1.7], class C)
|
|
|
|
// CHECK-NEXT: 7: new ([B1.4]) C([B1.6])
|
|
|
|
void operatorPlacementNewWithConstructorWithinPlacementArgument() {
|
|
|
|
new (new C()) C();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace operator_new
|
2018-02-10 09:55:23 +08:00
|
|
|
|
|
|
|
namespace decl_stmt {
|
|
|
|
|
|
|
|
// CHECK: void simpleVariable()
|
|
|
|
// CHECK: 1: (CXXConstructExpr, [B1.2], class C)
|
|
|
|
// CHECK-NEXT: 2: C c;
|
|
|
|
void simpleVariable() {
|
|
|
|
C c;
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void simpleVariableWithBraces()
|
|
|
|
// CHECK: 1: {} (CXXConstructExpr, [B1.2], class C)
|
|
|
|
// CHECK-NEXT: 2: C c{};
|
|
|
|
void simpleVariableWithBraces() {
|
|
|
|
C c{};
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void simpleVariableWithConstructorArgument()
|
|
|
|
// CHECK: 1: 0
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, class C *)
|
|
|
|
// CHECK-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], class C)
|
|
|
|
// CHECK-NEXT: 4: C c(0);
|
|
|
|
void simpleVariableWithConstructorArgument() {
|
|
|
|
C c(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void simpleVariableWithOperatorNewInConstructorArgument()
|
|
|
|
// CHECK: 1: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 2: (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CHECK-NEXT: 3: new C([B1.2])
|
|
|
|
// CHECK-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.5], class C)
|
|
|
|
// CHECK-NEXT: 5: C c(new C());
|
|
|
|
void simpleVariableWithOperatorNewInConstructorArgument() {
|
|
|
|
C c(new C());
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void simpleVariableWithOperatorNewInBraces()
|
|
|
|
// CHECK: 1: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 2: (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CHECK-NEXT: 3: new C([B1.2])
|
|
|
|
// CHECK-NEXT: 4: {[B1.3]} (CXXConstructExpr, [B1.5], class C)
|
|
|
|
// CHECK-NEXT: 5: C c{new C()};
|
|
|
|
void simpleVariableWithOperatorNewInBraces() {
|
|
|
|
C c{new C()};
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void simpleVariableInitializedByValue()
|
|
|
|
// CHECK: 1: C::get
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class C (*)(void))
|
2018-03-13 07:12:40 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4])
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 4: [B1.3]
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], class C)
|
|
|
|
// CXX11-NEXT: 6: C c = C::get();
|
|
|
|
// CXX17-NEXT: 4: C c = C::get();
|
2018-02-10 09:55:23 +08:00
|
|
|
void simpleVariableInitializedByValue() {
|
|
|
|
C c = C::get();
|
|
|
|
}
|
|
|
|
|
2018-03-31 03:21:18 +08:00
|
|
|
// FIXME: Find construction contexts for both branches in C++17.
|
|
|
|
// Note that once it gets detected, the test for the get() branch would not
|
|
|
|
// fail, because FileCheck allows partial matches.
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: void simpleVariableWithTernaryOperator(bool coin)
|
|
|
|
// CHECK: [B1]
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 1: [B4.2] ? [B2.5] : [B3.6]
|
|
|
|
// CXX11-NEXT: 2: [B1.1]
|
|
|
|
// CXX11-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], class C)
|
|
|
|
// CXX11-NEXT: 4: C c = coin ? C::get() : C(0);
|
|
|
|
// CXX17-NEXT: 1: [B4.2] ? [B2.3] : [B3.4]
|
|
|
|
// CXX17-NEXT: 2: C c = coin ? C::get() : C(0);
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: [B2]
|
|
|
|
// CHECK-NEXT: 1: C::get
|
|
|
|
// CHECK-NEXT: 2: [B2.1] (ImplicitCastExpr, FunctionToPointerDecay, class C (*)(void))
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B2.4])
|
|
|
|
// CXX11-NEXT: 4: [B2.3]
|
|
|
|
// CXX11-NEXT: 5: [B2.4] (CXXConstructExpr, [B1.2], class C)
|
2018-03-31 03:21:18 +08:00
|
|
|
// CXX17-NEXT: 3: [B2.2]()
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: [B3]
|
|
|
|
// CHECK-NEXT: 1: 0
|
|
|
|
// CHECK-NEXT: 2: [B3.1] (ImplicitCastExpr, NullToPointer, class C *)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.5], class C)
|
|
|
|
// CXX11-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, class C)
|
|
|
|
// CXX11-NEXT: 5: [B3.4]
|
|
|
|
// CXX11-NEXT: 6: [B3.5] (CXXConstructExpr, [B1.2], class C)
|
2018-03-31 03:21:18 +08:00
|
|
|
// CXX17-NEXT: 3: [B3.2] (CXXConstructExpr, class C)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX17-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, class C)
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: [B4]
|
|
|
|
// CHECK-NEXT: 1: coin
|
|
|
|
// CHECK-NEXT: 2: [B4.1] (ImplicitCastExpr, LValueToRValue, _Bool)
|
|
|
|
// CHECK-NEXT: T: [B4.2] ? ... : ...
|
|
|
|
void simpleVariableWithTernaryOperator(bool coin) {
|
|
|
|
C c = coin ? C::get() : C(0);
|
|
|
|
}
|
|
|
|
|
2018-02-24 10:05:11 +08:00
|
|
|
// CHECK: void simpleVariableWithElidableCopy()
|
|
|
|
// CHECK: 1: 0
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, class C *)
|
|
|
|
// CHECK-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], class C)
|
|
|
|
// CHECK-NEXT: 4: C([B1.3]) (CXXFunctionalCastExpr, ConstructorConversion, class C)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 5: [B1.4]
|
|
|
|
// CXX11-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.7], class C)
|
|
|
|
// CXX11-NEXT: 7: C c = C(0);
|
|
|
|
// CXX17-NEXT: 5: C c = C(0);
|
2018-02-24 10:05:11 +08:00
|
|
|
void simpleVariableWithElidableCopy() {
|
|
|
|
C c = C(0);
|
|
|
|
}
|
|
|
|
|
2018-02-10 09:55:23 +08:00
|
|
|
// CHECK: void referenceVariableWithConstructor()
|
|
|
|
// CHECK: 1: 0
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, class C *)
|
2018-02-24 10:00:30 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], const class C)
|
2018-02-10 09:55:23 +08:00
|
|
|
// CHECK-NEXT: 4: [B1.3]
|
|
|
|
// CHECK-NEXT: 5: const C &c(0);
|
|
|
|
void referenceVariableWithConstructor() {
|
|
|
|
const C &c(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void referenceVariableWithInitializer()
|
2018-02-24 10:07:50 +08:00
|
|
|
// CHECK: 1: C() (CXXConstructExpr, [B1.3], class C)
|
2018-02-10 09:55:23 +08:00
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CHECK-NEXT: 3: [B1.2]
|
|
|
|
// CHECK-NEXT: 4: const C &c = C();
|
|
|
|
void referenceVariableWithInitializer() {
|
|
|
|
const C &c = C();
|
|
|
|
}
|
|
|
|
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: void referenceVariableWithTernaryOperator(bool coin)
|
|
|
|
// CHECK: [B1]
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 1: [B4.2] ? [B2.5] : [B3.6]
|
|
|
|
// CXX17-NEXT: 1: [B4.2] ? [B2.3] : [B3.4]
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CHECK-NEXT: 3: [B1.2]
|
|
|
|
// CHECK-NEXT: 4: const C &c = coin ? C::get() : C(0);
|
|
|
|
// CHECK: [B2]
|
|
|
|
// CHECK-NEXT: 1: C::get
|
|
|
|
// CHECK-NEXT: 2: [B2.1] (ImplicitCastExpr, FunctionToPointerDecay, class C (*)(void))
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B2.4])
|
|
|
|
// CXX11-NEXT: 4: [B2.3]
|
|
|
|
// CXX11-NEXT: 5: [B2.4] (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CXX17-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B1.3])
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: [B3]
|
|
|
|
// CHECK-NEXT: 1: 0
|
|
|
|
// CHECK-NEXT: 2: [B3.1] (ImplicitCastExpr, NullToPointer, class C *)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.5], class C)
|
|
|
|
// CXX11-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, class C)
|
|
|
|
// CXX11-NEXT: 5: [B3.4]
|
|
|
|
// CXX11-NEXT: 6: [B3.5] (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CXX17-NEXT: 3: [B3.2] (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CXX17-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, class C)
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: [B4]
|
|
|
|
// CHECK-NEXT: 1: coin
|
|
|
|
// CHECK-NEXT: 2: [B4.1] (ImplicitCastExpr, LValueToRValue, _Bool)
|
|
|
|
// CHECK-NEXT: T: [B4.2] ? ... : ...
|
|
|
|
void referenceVariableWithTernaryOperator(bool coin) {
|
|
|
|
const C &c = coin ? C::get() : C(0);
|
|
|
|
}
|
|
|
|
|
2018-02-10 09:55:23 +08:00
|
|
|
} // end namespace decl_stmt
|
2018-02-10 10:18:04 +08:00
|
|
|
|
|
|
|
namespace ctor_initializers {
|
|
|
|
|
|
|
|
class D: public C {
|
|
|
|
C c1;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
// CHECK: D()
|
|
|
|
// CHECK: 1: (CXXConstructExpr, C() (Base initializer), class C)
|
|
|
|
// CHECK-NEXT: 2: C([B1.1]) (Base initializer)
|
|
|
|
// CHECK-NEXT: 3: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 4: (CXXConstructExpr, [B1.5], class C)
|
|
|
|
// CHECK-NEXT: 5: new C([B1.4])
|
|
|
|
// CHECK-NEXT: 6: [B1.5] (CXXConstructExpr, c1([B1.5]) (Member initializer), class C)
|
|
|
|
// CHECK-NEXT: 7: c1([B1.6]) (Member initializer)
|
|
|
|
D(): C(), c1(new C()) {}
|
|
|
|
|
|
|
|
// CHECK: D(int)
|
|
|
|
// CHECK: 1: (CXXConstructExpr, D() (Delegating initializer), class ctor_initializers::D)
|
|
|
|
// CHECK-NEXT: 2: D([B1.1]) (Delegating initializer)
|
|
|
|
D(int): D() {}
|
2018-02-10 10:46:14 +08:00
|
|
|
|
2018-03-23 05:37:39 +08:00
|
|
|
// FIXME: Why is CXXRecordTypedCall not present in C++17? Note that once it gets
|
|
|
|
// detected the test would not fail, because FileCheck allows partial matches.
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK: D(double)
|
|
|
|
// CHECK: 1: C::get
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class C (*)(void))
|
2018-03-13 07:12:40 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4])
|
2018-02-10 10:46:14 +08:00
|
|
|
// CHECK-NEXT: 4: [B1.3]
|
|
|
|
// CHECK-NEXT: 5: [B1.4] (CXXConstructExpr, C([B1.4]) (Base initializer), class C)
|
|
|
|
// CHECK-NEXT: 6: C([B1.5]) (Base initializer)
|
|
|
|
// CHECK-NEXT: 7: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 8: C::get
|
|
|
|
// CHECK-NEXT: 9: [B1.8] (ImplicitCastExpr, FunctionToPointerDecay, class C (*)(void))
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 10: [B1.9]() (CXXRecordTypedCall, [B1.11])
|
|
|
|
// CXX11-NEXT: 11: [B1.10]
|
|
|
|
// CXX11-NEXT: 12: [B1.11] (CXXConstructExpr, [B1.13], class C)
|
|
|
|
// CXX11-NEXT: 13: new C([B1.12])
|
|
|
|
// CXX11-NEXT: 14: [B1.13] (CXXConstructExpr, c1([B1.13]) (Member initializer), class C)
|
|
|
|
// CXX11-NEXT: 15: c1([B1.14]) (Member initializer)
|
|
|
|
// CXX17-NEXT: 10: [B1.9]()
|
|
|
|
// CXX17-NEXT: 11: new C([B1.10])
|
|
|
|
// CXX17-NEXT: 12: [B1.11] (CXXConstructExpr, c1([B1.11]) (Member initializer), class C)
|
|
|
|
// CXX17-NEXT: 13: c1([B1.12]) (Member initializer)
|
2018-02-10 10:46:14 +08:00
|
|
|
D(double): C(C::get()), c1(new C(C::get())) {}
|
2018-02-10 10:18:04 +08:00
|
|
|
};
|
|
|
|
|
2018-03-23 06:02:38 +08:00
|
|
|
// Let's see if initializers work well for fields with destructors.
|
|
|
|
class E {
|
|
|
|
public:
|
|
|
|
static E get();
|
|
|
|
~E();
|
|
|
|
};
|
|
|
|
|
|
|
|
class F {
|
|
|
|
E e;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// FIXME: There should be no temporary destructor in C++17.
|
|
|
|
// CHECK: F()
|
|
|
|
// CHECK: 1: E::get
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class ctor_initializers::E (*)(
|
|
|
|
// CXX11-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.6])
|
|
|
|
// CXX11-NEXT: 4: [B1.3] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const class ctor_initializers::E)
|
|
|
|
// CXX11-NEXT: 6: [B1.5]
|
|
|
|
// CXX11-NEXT: 7: [B1.6] (CXXConstructExpr, e([B1.6]) (Member initializer), class ctor_initializers
|
|
|
|
// CXX11-NEXT: 8: e([B1.7]) (Member initializer)
|
|
|
|
// CXX11-NEXT: 9: ~ctor_initializers::E() (Temporary object destructor)
|
|
|
|
// CXX17-NEXT: 3: [B1.2]() (CXXRecordTypedCall, e([B1.4]) (Member initializer), [B1.4])
|
|
|
|
// CXX17-NEXT: 4: [B1.3] (BindTemporary)
|
|
|
|
// CXX17-NEXT: 5: e([B1.4]) (Member initializer)
|
|
|
|
// CXX17-NEXT: 6: ~ctor_initializers::E() (Temporary object destructor)
|
|
|
|
F(): e(E::get()) {}
|
|
|
|
};
|
2018-02-10 10:18:04 +08:00
|
|
|
} // end namespace ctor_initializers
|
2018-02-13 06:36:36 +08:00
|
|
|
|
2018-02-15 11:13:36 +08:00
|
|
|
namespace return_stmt_without_dtor {
|
2018-02-13 06:36:36 +08:00
|
|
|
|
|
|
|
// CHECK: C returnVariable()
|
|
|
|
// CHECK: 1: (CXXConstructExpr, [B1.2], class C)
|
|
|
|
// CHECK-NEXT: 2: C c;
|
|
|
|
// CHECK-NEXT: 3: c
|
|
|
|
// CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, class C)
|
|
|
|
// CHECK-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], class C)
|
|
|
|
// CHECK-NEXT: 6: return [B1.5];
|
|
|
|
C returnVariable() {
|
|
|
|
C c;
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: C returnEmptyBraces()
|
|
|
|
// CHECK: 1: {} (CXXConstructExpr, [B1.2], class C)
|
|
|
|
// CHECK-NEXT: 2: return [B1.1];
|
|
|
|
C returnEmptyBraces() {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: C returnBracesWithOperatorNew()
|
|
|
|
// CHECK: 1: CFGNewAllocator(C *)
|
|
|
|
// CHECK-NEXT: 2: (CXXConstructExpr, [B1.3], class C)
|
|
|
|
// CHECK-NEXT: 3: new C([B1.2])
|
|
|
|
// CHECK-NEXT: 4: {[B1.3]} (CXXConstructExpr, [B1.5], class C)
|
|
|
|
// CHECK-NEXT: 5: return [B1.4];
|
|
|
|
C returnBracesWithOperatorNew() {
|
|
|
|
return {new C()};
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: C returnBracesWithMultipleItems()
|
|
|
|
// CHECK: 1: 123
|
|
|
|
// CHECK-NEXT: 2: 456
|
|
|
|
// CHECK-NEXT: 3: {[B1.1], [B1.2]} (CXXConstructExpr, [B1.4], class C)
|
|
|
|
// CHECK-NEXT: 4: return [B1.3];
|
|
|
|
C returnBracesWithMultipleItems() {
|
|
|
|
return {123, 456};
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: C returnTemporary()
|
2018-02-24 10:00:30 +08:00
|
|
|
// CHECK: 1: C() (CXXConstructExpr, [B1.2], class C)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 2: [B1.1]
|
|
|
|
// CXX11-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], class C)
|
|
|
|
// CXX11-NEXT: 4: return [B1.3];
|
|
|
|
// CXX17-NEXT: 2: return [B1.1];
|
2018-02-13 06:36:36 +08:00
|
|
|
C returnTemporary() {
|
|
|
|
return C();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: C returnTemporaryWithArgument()
|
|
|
|
// CHECK: 1: nullptr
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, class C *)
|
2018-02-24 10:05:11 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], class C)
|
2018-02-13 06:36:36 +08:00
|
|
|
// CHECK-NEXT: 4: C([B1.3]) (CXXFunctionalCastExpr, ConstructorConversion, class C)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 5: [B1.4]
|
|
|
|
// CXX11-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.7], class C)
|
|
|
|
// CXX11-NEXT: 7: return [B1.6];
|
|
|
|
// CXX17-NEXT: 5: return [B1.4];
|
|
|
|
|
2018-02-13 06:36:36 +08:00
|
|
|
C returnTemporaryWithArgument() {
|
|
|
|
return C(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: C returnTemporaryConstructedByFunction()
|
|
|
|
// CHECK: 1: C::get
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class C (*)(void))
|
2018-03-13 07:12:40 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4])
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 4: [B1.3]
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], class C)
|
|
|
|
// CXX11-NEXT: 6: return [B1.5];
|
|
|
|
// CXX17-NEXT: 4: return [B1.3];
|
2018-02-13 06:36:36 +08:00
|
|
|
C returnTemporaryConstructedByFunction() {
|
|
|
|
return C::get();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: C returnChainOfCopies()
|
|
|
|
// CHECK: 1: C::get
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class C (*)(void))
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4])
|
|
|
|
// CXX11-NEXT: 4: [B1.3]
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.7], class C)
|
|
|
|
// CXX11-NEXT: 6: C([B1.5]) (CXXFunctionalCastExpr, ConstructorConversion, class C)
|
|
|
|
// CXX11-NEXT: 7: [B1.6]
|
|
|
|
// CXX11-NEXT: 8: [B1.7] (CXXConstructExpr, [B1.9], class C)
|
|
|
|
// CXX11-NEXT: 9: return [B1.8];
|
|
|
|
// CXX17-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5])
|
|
|
|
// CXX17-NEXT: 4: C([B1.3]) (CXXFunctionalCastExpr, NoOp, class C)
|
|
|
|
// CXX17-NEXT: 5: return [B1.4];
|
2018-02-13 06:36:36 +08:00
|
|
|
C returnChainOfCopies() {
|
|
|
|
return C(C::get());
|
|
|
|
}
|
|
|
|
|
2018-02-15 11:13:36 +08:00
|
|
|
} // end namespace return_stmt_without_dtor
|
|
|
|
|
|
|
|
namespace return_stmt_with_dtor {
|
|
|
|
|
|
|
|
class D {
|
|
|
|
public:
|
|
|
|
D();
|
|
|
|
~D();
|
|
|
|
};
|
|
|
|
|
2018-03-23 05:37:39 +08:00
|
|
|
// FIXME: There should be no temporary destructor in C++17.
|
2018-02-15 11:13:36 +08:00
|
|
|
// CHECK: return_stmt_with_dtor::D returnTemporary()
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11: 1: return_stmt_with_dtor::D() (CXXConstructExpr, [B1.2], [B1.4], class return_stmt_with_dtor::D)
|
|
|
|
// CXX11-NEXT: 2: [B1.1] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 3: [B1.2] (ImplicitCastExpr, NoOp, const class return_stmt_with_dtor::D)
|
|
|
|
// CXX11-NEXT: 4: [B1.3]
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.7], class return_stmt_with_dtor::D)
|
|
|
|
// CXX11-NEXT: 6: ~return_stmt_with_dtor::D() (Temporary object destructor)
|
|
|
|
// CXX11-NEXT: 7: return [B1.5];
|
|
|
|
// CXX17: 1: return_stmt_with_dtor::D() (CXXConstructExpr, [B1.4], [B1.2], class return_stmt_w
|
|
|
|
// CXX17-NEXT: 2: [B1.1] (BindTemporary)
|
|
|
|
// CXX17-NEXT: 3: ~return_stmt_with_dtor::D() (Temporary object destructor)
|
|
|
|
// CXX17-NEXT: 4: return [B1.2];
|
2018-02-15 11:13:36 +08:00
|
|
|
D returnTemporary() {
|
|
|
|
return D();
|
|
|
|
}
|
|
|
|
|
2018-03-23 05:37:39 +08:00
|
|
|
// FIXME: There should be no temporary destructor in C++17.
|
|
|
|
// CHECK: void returnByValueIntoVariable()
|
|
|
|
// CHECK: 1: returnTemporary
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class return_stmt_with_dtor::D (*)(void))
|
|
|
|
// CXX11-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.6])
|
|
|
|
// CXX11-NEXT: 4: [B1.3] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const class return_stmt_with_dtor::D)
|
|
|
|
// CXX11-NEXT: 6: [B1.5]
|
|
|
|
// CXX11-NEXT: 7: [B1.6] (CXXConstructExpr, [B1.8], class return_stmt_with_dtor::D)
|
|
|
|
// CXX11-NEXT: 8: return_stmt_with_dtor::D d = returnTemporary();
|
|
|
|
// CXX11-NEXT: 9: ~return_stmt_with_dtor::D() (Temporary object destructor)
|
|
|
|
// CXX11-NEXT: 10: [B1.8].~D() (Implicit destructor)
|
|
|
|
// CXX17-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5], [B1.4])
|
|
|
|
// CXX17-NEXT: 4: [B1.3] (BindTemporary)
|
|
|
|
// CXX17-NEXT: 5: return_stmt_with_dtor::D d = returnTemporary();
|
|
|
|
// CXX17-NEXT: 6: ~return_stmt_with_dtor::D() (Temporary object destructor)
|
|
|
|
// CXX17-NEXT: 7: [B1.5].~D() (Implicit destructor)
|
|
|
|
void returnByValueIntoVariable() {
|
|
|
|
D d = returnTemporary();
|
|
|
|
}
|
|
|
|
|
2018-02-15 11:13:36 +08:00
|
|
|
} // end namespace return_stmt_with_dtor
|
|
|
|
|
|
|
|
namespace temporary_object_expr_without_dtors {
|
|
|
|
|
|
|
|
// TODO: Should provide construction context for the constructor,
|
|
|
|
// even if there is no specific trigger statement here.
|
|
|
|
// CHECK: void simpleTemporary()
|
|
|
|
// CHECK 1: C() (CXXConstructExpr, class C)
|
|
|
|
void simpleTemporary() {
|
|
|
|
C();
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Should provide construction context for the constructor,
|
|
|
|
// CHECK: void temporaryInCondition()
|
|
|
|
// CHECK: 1: C() (CXXConstructExpr, class C)
|
|
|
|
// CHECK-NEXT: 2: [B2.1] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CHECK-NEXT: 3: [B2.2].operator bool
|
|
|
|
// CHECK-NEXT: 4: [B2.2]
|
|
|
|
// CHECK-NEXT: 5: [B2.4] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CHECK-NEXT: T: if [B2.5]
|
|
|
|
void temporaryInCondition() {
|
|
|
|
if (C());
|
|
|
|
}
|
|
|
|
|
2018-04-20 07:30:15 +08:00
|
|
|
// CHECK: void temporaryInConditionVariable()
|
|
|
|
// CHECK: 1: C() (CXXConstructExpr, [B2.2], class C)
|
|
|
|
// CXX11-NEXT: 2: [B2.1]
|
|
|
|
// CXX11-NEXT: 3: [B2.2] (CXXConstructExpr, [B2.4], class C)
|
|
|
|
// CXX11-NEXT: 4: C c = C();
|
|
|
|
// CXX11-NEXT: 5: c
|
|
|
|
// CXX11-NEXT: 6: [B2.5] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CXX11-NEXT: 7: [B2.6].operator bool
|
|
|
|
// CXX11-NEXT: 8: [B2.6]
|
|
|
|
// CXX11-NEXT: 9: [B2.8] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX11-NEXT: T: if [B2.9]
|
|
|
|
// CXX17-NEXT: 2: C c = C();
|
|
|
|
// CXX17-NEXT: 3: c
|
|
|
|
// CXX17-NEXT: 4: [B2.3] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CXX17-NEXT: 5: [B2.4].operator bool
|
|
|
|
// CXX17-NEXT: 6: [B2.4]
|
|
|
|
// CXX17-NEXT: 7: [B2.6] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX17-NEXT: T: if [B2.7]
|
|
|
|
void temporaryInConditionVariable() {
|
|
|
|
if (C c = C());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// CHECK: void temporaryInForLoopConditionVariable()
|
|
|
|
// CHECK: [B2]
|
|
|
|
// CXX11-NEXT: 1: C() (CXXConstructExpr, [B2.2], class C)
|
|
|
|
// CXX11-NEXT: 2: [B2.1]
|
|
|
|
// CXX11-NEXT: 3: [B2.2] (CXXConstructExpr, [B2.4], class C)
|
|
|
|
// CXX11-NEXT: 4: C c2 = C();
|
|
|
|
// CXX11-NEXT: 5: c2
|
|
|
|
// CXX11-NEXT: 6: [B2.5] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CXX11-NEXT: 7: [B2.6].operator bool
|
|
|
|
// CXX11-NEXT: 8: [B2.6]
|
|
|
|
// CXX11-NEXT: 9: [B2.8] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX11-NEXT: T: for (...; [B2.9]; )
|
|
|
|
// CXX17-NEXT: 1: C() (CXXConstructExpr, [B2.2], class C)
|
|
|
|
// CXX17-NEXT: 2: C c2 = C();
|
|
|
|
// CXX17-NEXT: 3: c2
|
|
|
|
// CXX17-NEXT: 4: [B2.3] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CXX17-NEXT: 5: [B2.4].operator bool
|
|
|
|
// CXX17-NEXT: 6: [B2.4]
|
|
|
|
// CXX17-NEXT: 7: [B2.6] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX17-NEXT: T: for (...; [B2.7]; )
|
|
|
|
// CHECK: [B3]
|
|
|
|
// CXX11-NEXT: 1: C() (CXXConstructExpr, [B3.2], class C)
|
|
|
|
// CXX11-NEXT: 2: [B3.1]
|
|
|
|
// CXX11-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.4], class C)
|
|
|
|
// CXX11-NEXT: 4: C c1 = C();
|
|
|
|
// CXX17-NEXT: 1: C() (CXXConstructExpr, [B3.2], class C)
|
|
|
|
// CXX17-NEXT: 2: C c1 = C();
|
|
|
|
void temporaryInForLoopConditionVariable() {
|
|
|
|
for (C c1 = C(); C c2 = C(); );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// FIXME: Find construction context for the loop condition variable.
|
|
|
|
// CHECK: void temporaryInWhileLoopConditionVariable()
|
|
|
|
// CXX11: 1: C() (CXXConstructExpr, [B2.2], class C)
|
|
|
|
// CXX11-NEXT: 2: [B2.1]
|
|
|
|
// CXX11-NEXT: 3: [B2.2] (CXXConstructExpr, [B2.4], class C)
|
|
|
|
// CXX11-NEXT: 4: C c = C();
|
|
|
|
// CXX11-NEXT: 5: c
|
|
|
|
// CXX11-NEXT: 6: [B2.5] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CXX11-NEXT: 7: [B2.6].operator bool
|
|
|
|
// CXX11-NEXT: 8: [B2.6]
|
|
|
|
// CXX11-NEXT: 9: [B2.8] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX11-NEXT: T: while [B2.9]
|
|
|
|
// CXX17: 1: C() (CXXConstructExpr, [B2.2], class C)
|
|
|
|
// CXX17-NEXT: 2: C c = C();
|
|
|
|
// CXX17-NEXT: 3: c
|
|
|
|
// CXX17-NEXT: 4: [B2.3] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CXX17-NEXT: 5: [B2.4].operator bool
|
|
|
|
// CXX17-NEXT: 6: [B2.4]
|
|
|
|
// CXX17-NEXT: 7: [B2.6] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX17-NEXT: T: while [B2.7]
|
|
|
|
void temporaryInWhileLoopConditionVariable() {
|
|
|
|
while (C c = C());
|
|
|
|
}
|
|
|
|
|
2018-02-15 11:13:36 +08:00
|
|
|
} // end namespace temporary_object_expr_without_dtors
|
|
|
|
|
|
|
|
namespace temporary_object_expr_with_dtors {
|
|
|
|
|
|
|
|
class D {
|
|
|
|
public:
|
|
|
|
D();
|
|
|
|
D(int);
|
|
|
|
~D();
|
|
|
|
|
|
|
|
static D get();
|
|
|
|
|
|
|
|
operator bool() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
// CHECK: void simpleTemporary()
|
|
|
|
// CHECK: 1: temporary_object_expr_with_dtors::D() (CXXConstructExpr, [B1.2], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (BindTemporary)
|
|
|
|
// CHECK-NEXT: 3: ~temporary_object_expr_with_dtors::D() (Temporary object destructor)
|
|
|
|
void simpleTemporary() {
|
|
|
|
D();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void temporaryInCondition()
|
|
|
|
// CHECK: 1: temporary_object_expr_with_dtors::D() (CXXConstructExpr, [B2.2], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CHECK-NEXT: 2: [B2.1] (BindTemporary)
|
|
|
|
// CHECK-NEXT: 3: [B2.2] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CHECK-NEXT: 4: [B2.3].operator bool
|
|
|
|
// CHECK-NEXT: 5: [B2.3]
|
|
|
|
// CHECK-NEXT: 6: [B2.5] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CHECK-NEXT: 7: ~temporary_object_expr_with_dtors::D() (Temporary object destructor)
|
|
|
|
// CHECK-NEXT: T: if [B2.6]
|
|
|
|
void temporaryInCondition() {
|
|
|
|
if (D());
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void referenceVariableWithConstructor()
|
|
|
|
// CHECK: 1: 0
|
2018-02-24 10:00:30 +08:00
|
|
|
// CHECK-NEXT: 2: [B1.1] (CXXConstructExpr, [B1.3], [B1.4], const class temporary_object_expr_with_dtors::D)
|
2018-02-15 11:13:36 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2] (BindTemporary)
|
|
|
|
// CHECK-NEXT: 4: [B1.3]
|
|
|
|
// CHECK-NEXT: 5: const temporary_object_expr_with_dtors::D &d(0);
|
|
|
|
// CHECK-NEXT: 6: [B1.5].~D() (Implicit destructor)
|
|
|
|
void referenceVariableWithConstructor() {
|
|
|
|
const D &d(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void referenceVariableWithInitializer()
|
2018-02-24 10:07:50 +08:00
|
|
|
// CHECK: 1: temporary_object_expr_with_dtors::D() (CXXConstructExpr, [B1.2], [B1.4], class temporary_object_expr_with_dtors::D)
|
2018-02-15 11:13:36 +08:00
|
|
|
// CHECK-NEXT: 2: [B1.1] (BindTemporary)
|
|
|
|
// CHECK-NEXT: 3: [B1.2] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CHECK-NEXT: 4: [B1.3]
|
|
|
|
// CHECK-NEXT: 5: const temporary_object_expr_with_dtors::D &d = temporary_object_expr_with_dtors::D();
|
|
|
|
// CHECK-NEXT: 6: [B1.5].~D() (Implicit destructor)
|
|
|
|
void referenceVariableWithInitializer() {
|
|
|
|
const D &d = D();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void referenceVariableWithTernaryOperator(bool coin)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11: [B1]
|
|
|
|
// CXX11-NEXT: 1: [B4.4].~D() (Implicit destructor)
|
|
|
|
// CXX11: [B2]
|
|
|
|
// CXX11-NEXT: 1: ~temporary_object_expr_with_dtors::D() (Temporary object destructor)
|
|
|
|
// CXX11: [B3]
|
|
|
|
// CXX11-NEXT: 1: ~temporary_object_expr_with_dtors::D() (Temporary object destructor)
|
|
|
|
// CXX11: [B4]
|
|
|
|
// CXX11-NEXT: 1: [B7.2] ? [B5.8] : [B6.8]
|
|
|
|
// CXX11-NEXT: 2: [B4.1] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 3: [B4.2]
|
|
|
|
// CXX11-NEXT: 4: const temporary_object_expr_with_dtors::D &d = coin ? D::get() : temporary_object_expr_with_dtors::D(0);
|
|
|
|
// CXX11-NEXT: T: (Temp Dtor) [B6.3]
|
|
|
|
// CXX11: [B5]
|
|
|
|
// CXX11-NEXT: 1: D::get
|
|
|
|
// CXX11-NEXT: 2: [B5.1] (ImplicitCastExpr, FunctionToPointerDecay, class temporary_object_expr_with_dtors::D (*)(void))
|
|
|
|
// CXX11-NEXT: 3: [B5.2]() (CXXRecordTypedCall, [B5.4], [B5.6])
|
|
|
|
// CXX11-NEXT: 4: [B5.3] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 5: [B5.4] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 6: [B5.5]
|
|
|
|
// CXX11-NEXT: 7: [B5.6] (CXXConstructExpr, [B5.8], [B4.3], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 8: [B5.7] (BindTemporary)
|
|
|
|
// CXX11: [B6]
|
|
|
|
// CXX11-NEXT: 1: 0
|
|
|
|
// CXX11-NEXT: 2: [B6.1] (CXXConstructExpr, [B6.3], [B6.6], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 3: [B6.2] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 4: temporary_object_expr_with_dtors::D([B6.3]) (CXXFunctionalCastExpr, ConstructorConversion, class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 5: [B6.4] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 6: [B6.5]
|
|
|
|
// CXX11-NEXT: 7: [B6.6] (CXXConstructExpr, [B6.8], [B4.3], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 8: [B6.7] (BindTemporary)
|
|
|
|
// CXX11: [B7]
|
|
|
|
// CXX11-NEXT: 1: coin
|
|
|
|
// CXX11-NEXT: 2: [B7.1] (ImplicitCastExpr, LValueToRValue, _Bool)
|
|
|
|
// CXX11-NEXT: T: [B7.2] ? ... : ...
|
|
|
|
// CXX17: [B1]
|
|
|
|
// CXX17-NEXT: 1: [B4.2] ? [B2.4] : [B3.4]
|
|
|
|
// CXX17-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX17-NEXT: 3: [B1.2]
|
|
|
|
// CXX17-NEXT: 4: const temporary_object_expr_with_dtors::D &d = coin ? D::get() : temporary_object_expr_with_dtors::D(0);
|
|
|
|
// CXX17-NEXT: 5: [B1.4].~D() (Implicit destructor)
|
|
|
|
// CXX17: [B2]
|
|
|
|
// CXX17-NEXT: 1: D::get
|
|
|
|
// CXX17-NEXT: 2: [B2.1] (ImplicitCastExpr, FunctionToPointerDecay, class temporary_object_expr_with_dtors::D (*)(void))
|
|
|
|
// CXX17-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B2.4], [B1.3])
|
|
|
|
// CXX17-NEXT: 4: [B2.3] (BindTemporary)
|
|
|
|
// CXX17: [B3]
|
|
|
|
// CXX17-NEXT: 1: 0
|
|
|
|
// CXX17-NEXT: 2: [B3.1] (CXXConstructExpr, [B3.3], [B1.3], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX17-NEXT: 3: [B3.2] (BindTemporary)
|
|
|
|
// CXX17-NEXT: 4: temporary_object_expr_with_dtors::D([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX17: [B4]
|
|
|
|
// CXX17-NEXT: 1: coin
|
|
|
|
// CXX17-NEXT: 2: [B4.1] (ImplicitCastExpr, LValueToRValue, _Bool)
|
|
|
|
// CXX17-NEXT: T: [B4.2] ? ... : ...
|
2018-02-15 11:13:36 +08:00
|
|
|
void referenceVariableWithTernaryOperator(bool coin) {
|
|
|
|
const D &d = coin ? D::get() : D(0);
|
|
|
|
}
|
2018-02-24 10:05:11 +08:00
|
|
|
|
|
|
|
// CHECK: void referenceWithFunctionalCast()
|
|
|
|
// CHECK: 1: 1
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (CXXConstructExpr, [B1.3], [B1.5], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CHECK-NEXT: 3: [B1.2] (BindTemporary)
|
|
|
|
// CHECK-NEXT: 4: temporary_object_expr_with_dtors::D([B1.3]) (CXXFunctionalCastExpr, ConstructorCon
|
|
|
|
// CHECK-NEXT: 5: [B1.4]
|
|
|
|
// CHECK-NEXT: 6: temporary_object_expr_with_dtors::D &&d = temporary_object_expr_with_dtors::D(1);
|
|
|
|
// CHECK-NEXT: 7: [B1.6].~D() (Implicit destructor)
|
|
|
|
void referenceWithFunctionalCast() {
|
|
|
|
D &&d = D(1);
|
|
|
|
}
|
2018-02-24 11:10:15 +08:00
|
|
|
|
|
|
|
// Test the condition constructor, we don't care about branch constructors here.
|
|
|
|
// CHECK: void constructorInTernaryCondition()
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11: 1: 1
|
|
|
|
// CXX11-NEXT: 2: [B7.1] (CXXConstructExpr, [B7.3], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 3: [B7.2] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 4: temporary_object_expr_with_dtors::D([B7.3]) (CXXFunctionalCastExpr, ConstructorConversion, class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 5: [B7.4] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX11-NEXT: 6: [B7.5].operator bool
|
|
|
|
// CXX11-NEXT: 7: [B7.5]
|
|
|
|
// CXX11-NEXT: 8: [B7.7] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX11-NEXT: T: [B7.8] ? ... : ...
|
|
|
|
// CXX17: 1: 1
|
|
|
|
// CXX17-NEXT: 2: [B4.1] (CXXConstructExpr, [B4.3], class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX17-NEXT: 3: [B4.2] (BindTemporary)
|
|
|
|
// CXX17-NEXT: 4: temporary_object_expr_with_dtors::D([B4.3]) (CXXFunctionalCastExpr, ConstructorConversion, class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX17-NEXT: 5: [B4.4] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D)
|
|
|
|
// CXX17-NEXT: 6: [B4.5].operator bool
|
|
|
|
// CXX17-NEXT: 7: [B4.5]
|
|
|
|
// CXX17-NEXT: 8: [B4.7] (ImplicitCastExpr, UserDefinedConversion, _Bool)
|
|
|
|
// CXX17-NEXT: T: [B4.8] ? ... : ...
|
2018-02-24 11:10:15 +08:00
|
|
|
void constructorInTernaryCondition() {
|
|
|
|
const D &d = D(1) ? D(2) : D(3);
|
|
|
|
}
|
2018-03-01 09:09:24 +08:00
|
|
|
|
2018-02-15 11:13:36 +08:00
|
|
|
} // end namespace temporary_object_expr_with_dtors
|
2018-03-01 09:09:24 +08:00
|
|
|
|
|
|
|
namespace implicit_constructor_conversion {
|
|
|
|
|
|
|
|
class A {};
|
|
|
|
A get();
|
|
|
|
|
|
|
|
class B {
|
|
|
|
public:
|
|
|
|
B(const A &);
|
|
|
|
~B() {}
|
|
|
|
};
|
|
|
|
|
2018-03-09 09:39:59 +08:00
|
|
|
// CHECK: void implicitConstructionConversionFromTemporary()
|
|
|
|
// CHECK: 1: implicit_constructor_conversion::A() (CXXConstructExpr, [B1.3], class implicit_constructor_conversion::A)
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::A)
|
|
|
|
// CXX11-NEXT: 3: [B1.2]
|
|
|
|
// CXX11-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.6], [B1.8], class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, ConstructorConversion, class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 6: [B1.5] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 7: [B1.6] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 8: [B1.7]
|
|
|
|
// CXX11-NEXT: 9: [B1.8] (CXXConstructExpr, [B1.10], class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 10: implicit_constructor_conversion::B b = implicit_constructor_conversion::A();
|
|
|
|
// CXX11-NEXT: 11: ~implicit_constructor_conversion::B() (Temporary object destructor)
|
|
|
|
// CXX11-NEXT: 12: [B1.10].~B() (Implicit destructor)
|
|
|
|
// CXX17-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::A)
|
|
|
|
// CXX17-NEXT: 3: [B1.2]
|
|
|
|
// CXX17-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.6], class implicit_constructor_conversion::B)
|
|
|
|
// CXX17-NEXT: 5: [B1.4] (ImplicitCastExpr, ConstructorConversion, class implicit_constructor_conversion::B)
|
|
|
|
// CXX17-NEXT: 6: implicit_constructor_conversion::B b = implicit_constructor_conversion::A();
|
|
|
|
// CXX17-NEXT: 7: [B1.6].~B() (Implicit destructor)
|
2018-03-09 09:39:59 +08:00
|
|
|
void implicitConstructionConversionFromTemporary() {
|
|
|
|
B b = A();
|
|
|
|
}
|
|
|
|
|
2018-03-01 09:09:24 +08:00
|
|
|
// CHECK: void implicitConstructionConversionFromFunctionValue()
|
|
|
|
// CHECK: 1: get
|
2018-03-09 09:39:59 +08:00
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class implicit_constructor_conversion::A (*)(void))
|
2018-03-13 07:12:40 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5])
|
2018-03-09 09:39:59 +08:00
|
|
|
// CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::A)
|
|
|
|
// CHECK-NEXT: 5: [B1.4]
|
2018-03-23 05:37:39 +08:00
|
|
|
// CXX11-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.8], [B1.10], class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 7: [B1.6] (ImplicitCastExpr, ConstructorConversion, class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 8: [B1.7] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 9: [B1.8] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 10: [B1.9]
|
|
|
|
// CXX11-NEXT: 11: [B1.10] (CXXConstructExpr, [B1.12], class implicit_constructor_conversion::B)
|
|
|
|
// CXX11-NEXT: 12: implicit_constructor_conversion::B b = get();
|
|
|
|
// CXX11-NEXT: 13: ~implicit_constructor_conversion::B() (Temporary object destructor)
|
|
|
|
// CXX11-NEXT: 14: [B1.12].~B() (Implicit destructor)
|
|
|
|
// CXX17-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.8], class implicit_constructor_conversion::B)
|
|
|
|
// CXX17-NEXT: 7: [B1.6] (ImplicitCastExpr, ConstructorConversion, class implicit_constructor_conversion::B)
|
|
|
|
// CXX17-NEXT: 8: implicit_constructor_conversion::B b = get();
|
|
|
|
// CXX17-NEXT: 9: [B1.8].~B() (Implicit destructor)
|
2018-03-09 09:39:59 +08:00
|
|
|
void implicitConstructionConversionFromFunctionValue() {
|
|
|
|
B b = get();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void implicitConstructionConversionFromTemporaryWithLifetimeExtension()
|
|
|
|
// CHECK: 1: implicit_constructor_conversion::A() (CXXConstructExpr, [B1.3], class implicit_constructor_conversion::A)
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::A)
|
|
|
|
// CHECK-NEXT: 3: [B1.2]
|
|
|
|
// CHECK-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.7], class implicit_constructor_conversion::B)
|
|
|
|
// CHECK-NEXT: 5: [B1.4] (ImplicitCastExpr, ConstructorConversion, class implicit_constructor_conversion::B)
|
|
|
|
// CHECK-NEXT: 6: [B1.5] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::B)
|
|
|
|
// CHECK-NEXT: 7: [B1.6]
|
|
|
|
// CHECK-NEXT: 8: const implicit_constructor_conversion::B &b = implicit_constructor_conversion::A();
|
|
|
|
// CHECK-NEXT: 9: [B1.8].~B() (Implicit destructor)
|
|
|
|
void implicitConstructionConversionFromTemporaryWithLifetimeExtension() {
|
|
|
|
const B &b = A();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void implicitConstructionConversionFromFunctionValueWithLifetimeExtension()
|
|
|
|
// CHECK: 1: get
|
2018-03-01 09:09:24 +08:00
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, class implicit_constructor_conver
|
2018-03-13 07:12:40 +08:00
|
|
|
// CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5])
|
2018-03-01 09:09:24 +08:00
|
|
|
// CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::A)
|
|
|
|
// CHECK-NEXT: 5: [B1.4]
|
2018-03-09 09:39:59 +08:00
|
|
|
// CHECK-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.9], class implicit_constructor_conversion::B)
|
2018-03-01 09:09:24 +08:00
|
|
|
// CHECK-NEXT: 7: [B1.6] (ImplicitCastExpr, ConstructorConversion, class implicit_constructor_convers
|
|
|
|
// CHECK-NEXT: 8: [B1.7] (ImplicitCastExpr, NoOp, const class implicit_constructor_conversion::B)
|
|
|
|
// CHECK-NEXT: 9: [B1.8]
|
|
|
|
// CHECK-NEXT: 10: const implicit_constructor_conversion::B &b = get();
|
|
|
|
// CHECK-NEXT: 11: [B1.10].~B() (Implicit destructor)
|
2018-03-09 09:39:59 +08:00
|
|
|
void implicitConstructionConversionFromFunctionValueWithLifetimeExtension() {
|
2018-03-01 09:09:24 +08:00
|
|
|
const B &b = get(); // no-crash
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end namespace implicit_constructor_conversion
|
2018-04-20 07:09:22 +08:00
|
|
|
|
|
|
|
namespace argument_constructors {
|
|
|
|
class D {
|
|
|
|
public:
|
|
|
|
D();
|
|
|
|
~D();
|
|
|
|
};
|
|
|
|
|
|
|
|
void useC(C c);
|
|
|
|
void useCByReference(const C &c);
|
|
|
|
void useD(D d);
|
|
|
|
void useDByReference(const D &d);
|
|
|
|
|
|
|
|
// FIXME: Find construction context for the argument.
|
|
|
|
// CHECK: void passArgument()
|
|
|
|
// CHECK: 1: useC
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(class C))
|
|
|
|
// CXX11-NEXT: 3: C() (CXXConstructExpr, [B1.4], class C)
|
|
|
|
// CXX11-NEXT: 4: [B1.3]
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, class C)
|
|
|
|
// CXX11-NEXT: 6: [B1.2]([B1.5])
|
|
|
|
// CXX17-NEXT: 3: C() (CXXConstructExpr, class C)
|
|
|
|
// CXX17-NEXT: 4: [B1.2]([B1.3])
|
|
|
|
void passArgument() {
|
|
|
|
useC(C());
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void passArgumentByReference()
|
|
|
|
// CHECK: 1: useCByReference
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(const class C &))
|
|
|
|
// CHECK-NEXT: 3: C() (CXXConstructExpr, [B1.5], class C)
|
|
|
|
// CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, const class C)
|
|
|
|
// CHECK-NEXT: 5: [B1.4]
|
|
|
|
// CHECK-NEXT: 6: [B1.2]([B1.5])
|
|
|
|
void passArgumentByReference() {
|
|
|
|
useCByReference(C());
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Find construction context for the argument.
|
|
|
|
// CHECK: void passArgumentWithDestructor()
|
|
|
|
// CHECK: 1: useD
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(class argument_constructors::D))
|
|
|
|
// CXX11-NEXT: 3: argument_constructors::D() (CXXConstructExpr, [B1.4], [B1.6], class argument_constructors::D)
|
|
|
|
// CXX11-NEXT: 4: [B1.3] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const class argument_constructors::D)
|
|
|
|
// CXX11-NEXT: 6: [B1.5]
|
|
|
|
// CXX11-NEXT: 7: [B1.6] (CXXConstructExpr, class argument_constructors::D)
|
|
|
|
// CXX11-NEXT: 8: [B1.7] (BindTemporary)
|
|
|
|
// CXX11-NEXT: 9: [B1.2]([B1.8])
|
|
|
|
// CXX11-NEXT: 10: ~argument_constructors::D() (Temporary object destructor)
|
|
|
|
// CXX11-NEXT: 11: ~argument_constructors::D() (Temporary object destructor)
|
|
|
|
// CXX17-NEXT: 3: argument_constructors::D() (CXXConstructExpr, class argument_constructors::D)
|
|
|
|
// CXX17-NEXT: 4: [B1.3] (BindTemporary)
|
|
|
|
// CXX17-NEXT: 5: [B1.2]([B1.4])
|
|
|
|
// CXX17-NEXT: 6: ~argument_constructors::D() (Temporary object destructor)
|
|
|
|
void passArgumentWithDestructor() {
|
|
|
|
useD(D());
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: void passArgumentWithDestructorByReference()
|
|
|
|
// CHECK: 1: useDByReference
|
|
|
|
// CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(const class argumen
|
|
|
|
// CHECK-NEXT: 3: argument_constructors::D() (CXXConstructExpr, [B1.4], [B1.6], class argument_c
|
|
|
|
// CHECK-NEXT: 4: [B1.3] (BindTemporary)
|
|
|
|
// CHECK-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const class argument_constructors::D)
|
|
|
|
// CHECK-NEXT: 6: [B1.5]
|
|
|
|
// CHECK-NEXT: 7: [B1.2]([B1.6])
|
|
|
|
// CHECK-NEXT: 8: ~argument_constructors::D() (Temporary object destructor)
|
|
|
|
void passArgumentWithDestructorByReference() {
|
|
|
|
useDByReference(D());
|
|
|
|
}
|
|
|
|
} // end namespace argument_constructors
|