Mercurial > hg > CbC > CbC_llvm
view clang/test/Analysis/cfg-rich-constructors.cpp @ 236:c4bab56944e8 llvm-original
LLVM 16
author | kono |
---|---|
date | Wed, 09 Nov 2022 17:45:10 +0900 |
parents | 79ff65ed7e25 |
children |
line wrap: on
line source
// RUN: %clang_analyze_cc1 -analyzer-checker=debug.DumpCFG -triple x86_64-apple-darwin12 -std=c++11 -w %s > %t 2>&1 // RUN: FileCheck --input-file=%t -check-prefixes=CHECK,CXX11,CXX11-ELIDE %s // RUN: %clang_analyze_cc1 -analyzer-checker=debug.DumpCFG -triple x86_64-apple-darwin12 -std=c++17 -w %s > %t 2>&1 // RUN: FileCheck --input-file=%t -check-prefixes=CHECK,CXX17 %s // RUN: %clang_analyze_cc1 -analyzer-checker=debug.DumpCFG -triple x86_64-apple-darwin12 -std=c++11 -w -analyzer-config elide-constructors=false %s > %t 2>&1 // RUN: FileCheck --input-file=%t -check-prefixes=CHECK,CXX11,CXX11-NOELIDE %s // RUN: %clang_analyze_cc1 -analyzer-checker=debug.DumpCFG -triple x86_64-apple-darwin12 -std=c++17 -w -analyzer-config elide-constructors=false %s > %t 2>&1 // RUN: FileCheck --input-file=%t -check-prefixes=CHECK,CXX17 %s class C { public: C(); C(C *); C(int, int); static C get(); operator bool() const; }; 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], 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], C) // CHECK-NEXT: 4: new C([B1.3]) // CHECK-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], 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], 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], C) // CHECK-NEXT: 7: new ([B1.4]) C([B1.6]) void operatorPlacementNewWithConstructorWithinPlacementArgument() { new (new C()) C(); } } // namespace operator_new namespace decl_stmt { // CHECK: void simpleVariable() // CHECK: 1: (CXXConstructExpr, [B1.2], C) // CHECK-NEXT: 2: C c; void simpleVariable() { C c; } // CHECK: void simpleVariableWithBraces() // CHECK: 1: {} (CXXConstructExpr, [B1.2], C) // CHECK-NEXT: 2: C c{}; void simpleVariableWithBraces() { C c{}; } // CHECK: void simpleVariableWithConstructorArgument() // CHECK: 1: 0 // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, C *) // CHECK-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], 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], C) // CHECK-NEXT: 3: new C([B1.2]) // CHECK-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.5], 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], C) // CHECK-NEXT: 3: new C([B1.2]) // CHECK-NEXT: 4: {[B1.3]} (CXXConstructExpr, [B1.5], 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, C (*)(void)) // CXX11-ELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.5]) // CXX11-NOELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4]) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], C) // CXX11-NEXT: 6: C c = C::get(); // CXX17-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4]) // CXX17-NEXT: 4: C c = C::get(); void simpleVariableInitializedByValue() { C c = C::get(); } // 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. // CHECK: void simpleVariableWithTernaryOperator(bool coin) // CHECK: [B1] // CXX11-NEXT: 1: [B4.2] ? [B2.5] : [B3.6] // CXX11-NEXT: 2: [B1.1] // CXX11-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], 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); // CHECK: [B2] // CHECK-NEXT: 1: C::get // CHECK-NEXT: 2: [B2.1] (ImplicitCastExpr, FunctionToPointerDecay, C (*)(void)) // CXX11-ELIDE-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B2.4], [B2.5]) // CXX11-NOELIDE-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B2.4]) // CXX11-NEXT: 4: [B2.3] // CXX11-ELIDE-NEXT: 5: [B2.4] (CXXConstructExpr, [B1.2], [B1.3], C) // CXX11-NOELIDE-NEXT: 5: [B2.4] (CXXConstructExpr, [B1.2], C) // CXX17-NEXT: 3: [B2.2]() // CHECK: [B3] // CHECK-NEXT: 1: 0 // CHECK-NEXT: 2: [B3.1] (ImplicitCastExpr, NullToPointer, C *) // CXX11-ELIDE-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.5], [B3.6], C) // CXX11-NOELIDE-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.5], C) // CXX11-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CXX11-NEXT: 5: [B3.4] // CXX11-ELIDE-NEXT: 6: [B3.5] (CXXConstructExpr, [B1.2], [B1.3], C) // CXX11-NOELIDE-NEXT: 6: [B3.5] (CXXConstructExpr, [B1.2], C) // CXX17-NEXT: 3: [B3.2] (CXXConstructExpr, C) // CXX17-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, C) // 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); } // CHECK: void simpleVariableWithElidableCopy() // CHECK: 1: 0 // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, C *) // CXX11-ELIDE-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], [B1.6], C) // CXX11-NOELIDE-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], C) // CXX17-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], C) // CHECK-NEXT: 4: C([B1.3]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CXX11-NEXT: 5: [B1.4] // CXX11-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.7], C) // CXX11-NEXT: 7: C c = C(0); // CXX17-NEXT: 5: C c = C(0); void simpleVariableWithElidableCopy() { C c = C(0); } // CHECK: void referenceVariableWithConstructor() // CHECK: 1: 0 // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, C *) // CHECK-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], const C) // CHECK-NEXT: 4: [B1.3] // CHECK-NEXT: 5: const C &c(0); void referenceVariableWithConstructor() { const C &c(0); } // CHECK: void referenceVariableWithInitializer() // CHECK: 1: C() (CXXConstructExpr, [B1.3], C) // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const C) // CHECK-NEXT: 3: [B1.2] // CHECK-NEXT: 4: const C &c = C(); void referenceVariableWithInitializer() { const C &c = C(); } // CHECK: void referenceVariableWithTernaryOperator(bool coin) // CHECK: [B1] // CXX11-NEXT: 1: [B4.2] ? [B2.5] : [B3.6] // CXX17-NEXT: 1: [B4.2] ? [B2.3] : [B3.4] // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const 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, C (*)(void)) // CXX11-ELIDE-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B2.4], [B2.5]) // CXX11-NOELIDE-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B2.4]) // CXX11-NEXT: 4: [B2.3] // CXX11-NEXT: 5: [B2.4] (CXXConstructExpr, [B1.3], C) // CXX17-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B1.3]) // CHECK: [B3] // CHECK-NEXT: 1: 0 // CHECK-NEXT: 2: [B3.1] (ImplicitCastExpr, NullToPointer, C *) // CXX11-ELIDE-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.5], [B3.6], C) // CXX11-NOELIDE-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.5], C) // CXX11-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CXX11-NEXT: 5: [B3.4] // CXX11-NEXT: 6: [B3.5] (CXXConstructExpr, [B1.3], C) // CXX17-NEXT: 3: [B3.2] (CXXConstructExpr, [B1.3], C) // CXX17-NEXT: 4: C([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, C) // 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); } } // end namespace decl_stmt namespace ctor_initializers { class D: public C { C c1; public: // CHECK: D() // CHECK: 1: (CXXConstructExpr, C() (Base initializer), C) // CHECK-NEXT: 2: C([B1.1]) (Base initializer) // CHECK-NEXT: 3: CFGNewAllocator(C *) // CHECK-NEXT: 4: (CXXConstructExpr, [B1.5], C) // CHECK-NEXT: 5: new C([B1.4]) // CHECK-NEXT: 6: [B1.5] (CXXConstructExpr, c1([B1.5]) (Member initializer), 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() {} // 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. // CHECK: D(double) // CHECK: 1: C::get // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, C (*)(void)) // CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4]) // CHECK-NEXT: 4: [B1.3] // CHECK-NEXT: 5: [B1.4] (CXXConstructExpr, C([B1.4]) (Base initializer), 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, C (*)(void)) // CXX11-ELIDE-NEXT: 10: [B1.9]() (CXXRecordTypedCall, [B1.11], [B1.12]) // CXX11-NOELIDE-NEXT: 10: [B1.9]() (CXXRecordTypedCall, [B1.11]) // CXX11-NEXT: 11: [B1.10] // CXX11-NEXT: 12: [B1.11] (CXXConstructExpr, [B1.13], C) // CXX11-NEXT: 13: new C([B1.12]) // CXX11-NEXT: 14: [B1.13] (CXXConstructExpr, c1([B1.13]) (Member initializer), 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), C) // CXX17-NEXT: 13: c1([B1.12]) (Member initializer) D(double): C(C::get()), c1(new C(C::get())) {} }; // 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, E (*)( // CXX11-ELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.6], [B1.7]) // CXX11-NOELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.6]) // CXX11-NEXT: 4: [B1.3] (BindTemporary) // CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const E) // CXX11-NEXT: 6: [B1.5] // CXX11-NEXT: 7: [B1.6] (CXXConstructExpr, e([B1.6]) (Member initializer), E) // CXX11-NEXT: 8: e([B1.7]) (Member initializer) // CXX11-NEXT: 9: ~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: ~E() (Temporary object destructor) F(): e(E::get()) {} }; } // end namespace ctor_initializers namespace return_stmt_without_dtor { // CHECK: C returnVariable() // CHECK: 1: (CXXConstructExpr, [B1.2], C) // CHECK-NEXT: 2: C c; // CHECK-NEXT: 3: c // CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, C) // CHECK-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], C) // CHECK-NEXT: 6: return [B1.5]; C returnVariable() { C c; return c; } // CHECK: C returnEmptyBraces() // CHECK: 1: {} (CXXConstructExpr, [B1.2], C) // CHECK-NEXT: 2: return [B1.1]; C returnEmptyBraces() { return {}; } // CHECK: C returnBracesWithOperatorNew() // CHECK: 1: CFGNewAllocator(C *) // CHECK-NEXT: 2: (CXXConstructExpr, [B1.3], C) // CHECK-NEXT: 3: new C([B1.2]) // CHECK-NEXT: 4: {[B1.3]} (CXXConstructExpr, [B1.5], 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], C) // CHECK-NEXT: 4: return [B1.3]; C returnBracesWithMultipleItems() { return {123, 456}; } // CHECK: C returnTemporary() // CXX11-ELIDE: 1: C() (CXXConstructExpr, [B1.2], [B1.3], C) // CXX11-NOELIDE: 1: C() (CXXConstructExpr, [B1.2], C) // CXX11-NEXT: 2: [B1.1] // CXX11-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.4], C) // CXX11-NEXT: 4: return [B1.3]; // CXX17: 1: C() (CXXConstructExpr, [B1.2], C) // CXX17-NEXT: 2: return [B1.1]; C returnTemporary() { return C(); } // CHECK: C returnTemporaryWithArgument() // CHECK: 1: nullptr // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NullToPointer, C *) // CXX11-ELIDE-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], [B1.6], C) // CXX11-NOELIDE-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], C) // CXX17-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], C) // CHECK-NEXT: 4: C([B1.3]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CXX11-NEXT: 5: [B1.4] // CXX11-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.7], C) // CXX11-NEXT: 7: return [B1.6]; // CXX17-NEXT: 5: return [B1.4]; C returnTemporaryWithArgument() { return C(nullptr); } // CHECK: C returnTemporaryConstructedByFunction() // CHECK: 1: C::get // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, C (*)(void)) // CXX11-ELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.5]) // CXX11-NOELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4]) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], C) // CXX11-NEXT: 6: return [B1.5]; // CXX17-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4]) // CXX17-NEXT: 4: return [B1.3]; C returnTemporaryConstructedByFunction() { return C::get(); } // CHECK: C returnChainOfCopies() // CHECK: 1: C::get // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, C (*)(void)) // CXX11-ELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.5]) // CXX11-NOELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4]) // CXX11-NEXT: 4: [B1.3] // CXX11-ELIDE-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.7], [B1.8], C) // CXX11-NOELIDE-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.7], C) // CXX11-NEXT: 6: C([B1.5]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CXX11-NEXT: 7: [B1.6] // CXX11-NEXT: 8: [B1.7] (CXXConstructExpr, [B1.9], C) // CXX11-NEXT: 9: return [B1.8]; // CXX17-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5]) // CXX17-NEXT: 4: C([B1.3]) (CXXFunctionalCastExpr, NoOp, C) // CXX17-NEXT: 5: return [B1.4]; C returnChainOfCopies() { return C(C::get()); } } // end namespace return_stmt_without_dtor namespace return_stmt_with_dtor { class D { public: D(); ~D(); }; // CHECK: D returnTemporary() // CXX11-ELIDE: 1: D() (CXXConstructExpr, [B1.2], [B1.4], [B1.5], D) // CXX11-NOELIDE: 1: D() (CXXConstructExpr, [B1.2], [B1.4], D) // CXX11-NEXT: 2: [B1.1] (BindTemporary) // CXX11-NEXT: 3: [B1.2] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.7], D) // CXX11-NEXT: 6: ~D() (Temporary object destructor) // CXX11-NEXT: 7: return [B1.5]; // CXX17: 1: D() (CXXConstructExpr, [B1.3], [B1.2], D) // CXX17-NEXT: 2: [B1.1] (BindTemporary) // CXX17-NEXT: 3: return [B1.2]; D returnTemporary() { return D(); } // CHECK: void returnByValueIntoVariable() // CHECK: 1: returnTemporary // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, D (*)(void)) // CXX11-ELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.6], [B1.7]) // CXX11-NOELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.6]) // CXX11-NEXT: 4: [B1.3] (BindTemporary) // CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 6: [B1.5] // CXX11-NEXT: 7: [B1.6] (CXXConstructExpr, [B1.8], D) // CXX11-NEXT: 8: D d = returnTemporary(); // CXX11-NEXT: 9: ~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: D d = returnTemporary(); void returnByValueIntoVariable() { D d = returnTemporary(); } } // 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, C) void simpleTemporary() { C(); } // CHECK: void temporaryInCondition() // CHECK: 1: C() (CXXConstructExpr, [B2.2], C) // CHECK-NEXT: 2: [B2.1] // CHECK-NEXT: 3: [B2.2] (ImplicitCastExpr, NoOp, const class C) // CHECK-NEXT: 4: [B2.3].operator bool // CHECK-NEXT: 5: [B2.3] // CHECK-NEXT: 6: [B2.5] (ImplicitCastExpr, UserDefinedConversion, _Bool) // CHECK-NEXT: T: if [B2.6] void temporaryInCondition() { if (C()); } // CHECK: void temporaryInConditionVariable() // CXX11-ELIDE: 1: C() (CXXConstructExpr, [B2.2], [B2.3], C) // CXX11-NOELIDE: 1: C() (CXXConstructExpr, [B2.2], C) // CXX11-NEXT: 2: [B2.1] // CXX11-NEXT: 3: [B2.2] (CXXConstructExpr, [B2.4], 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: 1: C() (CXXConstructExpr, [B2.2], 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: if [B2.7] void temporaryInConditionVariable() { if (C c = C()); } // CHECK: void temporaryInForLoopConditionVariable() // CHECK: [B2] // CXX11-ELIDE-NEXT: 1: C() (CXXConstructExpr, [B2.2], [B2.3], C) // CXX11-NOELIDE-NEXT: 1: C() (CXXConstructExpr, [B2.2], C) // CXX11-NEXT: 2: [B2.1] // CXX11-NEXT: 3: [B2.2] (CXXConstructExpr, [B2.4], 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], 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-ELIDE-NEXT: 1: C() (CXXConstructExpr, [B3.2], [B3.3], C) // CXX11-NOELIDE-NEXT: 1: C() (CXXConstructExpr, [B3.2], C) // CXX11-NEXT: 2: [B3.1] // CXX11-NEXT: 3: [B3.2] (CXXConstructExpr, [B3.4], C) // CXX11-NEXT: 4: C c1 = C(); // CXX17-NEXT: 1: C() (CXXConstructExpr, [B3.2], C) // CXX17-NEXT: 2: C c1 = C(); void temporaryInForLoopConditionVariable() { for (C c1 = C(); C c2 = C(); ); } // CHECK: void temporaryInWhileLoopConditionVariable() // CXX11-ELIDE: 1: C() (CXXConstructExpr, [B2.2], [B2.3], C) // CXX11-NOELIDE: 1: C() (CXXConstructExpr, [B2.2], C) // CXX11-NEXT: 2: [B2.1] // CXX11-NEXT: 3: [B2.2] (CXXConstructExpr, [B2.4], 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], 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()); } } // 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: D() (CXXConstructExpr, [B1.2], D) // CHECK-NEXT: 2: [B1.1] (BindTemporary) // CHECK-NEXT: 3: ~D() (Temporary object destructor) void simpleTemporary() { D(); } // CHECK: void temporaryInCondition() // CHECK: 1: D() (CXXConstructExpr, [B2.2], [B2.3], D) // CHECK-NEXT: 2: [B2.1] (BindTemporary) // CHECK-NEXT: 3: [B2.2] // CHECK-NEXT: 4: [B2.3] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D) // CHECK-NEXT: 5: [B2.4].operator bool // CHECK-NEXT: 6: [B2.4] // CHECK-NEXT: 7: [B2.6] (ImplicitCastExpr, UserDefinedConversion, _Bool) // CHECK-NEXT: 8: ~D() (Temporary object destructor) // CHECK-NEXT: T: if [B2.7] void temporaryInCondition() { if (D()); } // CHECK: void referenceVariableWithConstructor() // CHECK: 1: 0 // CHECK-NEXT: 2: [B1.1] (CXXConstructExpr, [B1.4], const D) // CHECK-NEXT: 3: [B1.2] (BindTemporary) // CHECK-NEXT: 4: [B1.3] // CHECK-NEXT: 5: const D &d(0); // CHECK-NEXT: 6: [B1.5].~D() (Implicit destructor) void referenceVariableWithConstructor() { const D &d(0); } // CHECK: void referenceVariableWithInitializer() // CHECK: 1: D() (CXXConstructExpr, [B1.4], D) // CHECK-NEXT: 2: [B1.1] (BindTemporary) // CHECK-NEXT: 3: [B1.2] (ImplicitCastExpr, NoOp, const D) // CHECK-NEXT: 4: [B1.3] // CHECK-NEXT: 5: const D &d = D(); // CHECK-NEXT: 6: [B1.5].~D() (Implicit destructor) void referenceVariableWithInitializer() { const D &d = D(); } // CHECK: void referenceVariableWithTernaryOperator(bool coin) // CXX11: [B1] // CXX11-NEXT: 1: [B4.4].~D() (Implicit destructor) // CXX11: [B2] // CXX11-NEXT: 1: ~D() (Temporary object destructor) // CXX11: [B3] // CXX11-NEXT: 1: ~D() (Temporary object destructor) // CXX11: [B4] // CXX11-NEXT: 1: [B7.2] ? [B5.8] : [B6.8] // CXX11-NEXT: 2: [B4.1] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 3: [B4.2] // CXX11-NEXT: 4: const D &d = coin ? D::get() : D(0); // CXX11-NEXT: T: (Temp Dtor) [B6.3] // CXX11: [B5] // CXX11-NEXT: 1: D::get // CXX11-NEXT: 2: [B5.1] (ImplicitCastExpr, FunctionToPointerDecay, D (*)(void)) // CXX11-ELIDE-NEXT: 3: [B5.2]() (CXXRecordTypedCall, [B5.4], [B5.6], [B5.7]) // CXX11-NOELIDE-NEXT: 3: [B5.2]() (CXXRecordTypedCall, [B5.4], [B5.6]) // CXX11-NEXT: 4: [B5.3] (BindTemporary) // CXX11-NEXT: 5: [B5.4] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 6: [B5.5] // CXX11-NEXT: 7: [B5.6] (CXXConstructExpr, [B4.3], D) // CXX11-NEXT: 8: [B5.7] (BindTemporary) // CXX11: [B6] // CXX11-NEXT: 1: 0 // CXX11-ELIDE-NEXT: 2: [B6.1] (CXXConstructExpr, [B6.3], [B6.6], [B6.7], D) // CXX11-NOELIDE-NEXT: 2: [B6.1] (CXXConstructExpr, [B6.3], [B6.6], D) // CXX11-NEXT: 3: [B6.2] (BindTemporary) // CXX11-NEXT: 4: D([B6.3]) (CXXFunctionalCastExpr, ConstructorConversion, D) // CXX11-NEXT: 5: [B6.4] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 6: [B6.5] // CXX11-NEXT: 7: [B6.6] (CXXConstructExpr, [B4.3], 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 D) // CXX17-NEXT: 3: [B1.2] // CXX17-NEXT: 4: const D &d = coin ? D::get() : D(0); // CXX17-NEXT: 5: [B1.4].~D() (Implicit destructor) // CXX17: [B2] // CXX17-NEXT: 1: D::get // CXX17-NEXT: 2: [B2.1] (ImplicitCastExpr, FunctionToPointerDecay, D (*)(void)) // CXX17-NEXT: 3: [B2.2]() (CXXRecordTypedCall, [B1.3]) // CXX17-NEXT: 4: [B2.3] (BindTemporary) // CXX17: [B3] // CXX17-NEXT: 1: 0 // CXX17-NEXT: 2: [B3.1] (CXXConstructExpr, [B1.3], D) // CXX17-NEXT: 3: [B3.2] (BindTemporary) // CXX17-NEXT: 4: D([B3.3]) (CXXFunctionalCastExpr, ConstructorConversion, D) // CXX17: [B4] // CXX17-NEXT: 1: coin // CXX17-NEXT: 2: [B4.1] (ImplicitCastExpr, LValueToRValue, _Bool) // CXX17-NEXT: T: [B4.2] ? ... : ... void referenceVariableWithTernaryOperator(bool coin) { const D &d = coin ? D::get() : D(0); } // CHECK: void referenceWithFunctionalCast() // CHECK: 1: 1 // CHECK-NEXT: 2: [B1.1] (CXXConstructExpr, [B1.5], D) // CHECK-NEXT: 3: [B1.2] (BindTemporary) // CHECK-NEXT: 4: D([B1.3]) (CXXFunctionalCastExpr, ConstructorCon // CHECK-NEXT: 5: [B1.4] // CHECK-NEXT: 6: D &&d = D(1); // CHECK-NEXT: 7: [B1.6].~D() (Implicit destructor) void referenceWithFunctionalCast() { D &&d = D(1); } // Test the condition constructor, we don't care about branch constructors here. // CHECK: void constructorInTernaryCondition() // CXX11: 1: 1 // CXX11-NEXT: 2: [B7.1] (CXXConstructExpr, [B7.3], [B7.5], D) // CXX11-NEXT: 3: [B7.2] (BindTemporary) // CXX11-NEXT: 4: D([B7.3]) (CXXFunctionalCastExpr, ConstructorConversion, D) // CXX11-NEXT: 5: [B7.4] // CXX11-NEXT: 6: [B7.5] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D) // CXX11-NEXT: 7: [B7.6].operator bool // CXX11-NEXT: 8: [B7.6] // CXX11-NEXT: 9: [B7.8] (ImplicitCastExpr, UserDefinedConversion, _Bool) // CXX11-NEXT: T: [B7.9] ? ... : ... // CXX17: 1: 1 // CXX17-NEXT: 2: [B4.1] (CXXConstructExpr, [B4.3], [B4.5], D) // CXX17-NEXT: 3: [B4.2] (BindTemporary) // CXX17-NEXT: 4: D([B4.3]) (CXXFunctionalCastExpr, ConstructorConversion, D) // CXX17-NEXT: 5: [B4.4] // CXX17-NEXT: 6: [B4.5] (ImplicitCastExpr, NoOp, const class temporary_object_expr_with_dtors::D) // CXX17-NEXT: 7: [B4.6].operator bool // CXX17-NEXT: 8: [B4.6] // CXX17-NEXT: 9: [B4.8] (ImplicitCastExpr, UserDefinedConversion, _Bool) // CXX17-NEXT: T: [B4.9] ? ... : ... void constructorInTernaryCondition() { const D &d = D(1) ? D(2) : D(3); } } // end namespace temporary_object_expr_with_dtors namespace implicit_constructor_conversion { class A {}; A get(); class B { public: B(const A &); ~B() {} }; // CHECK: void implicitConstructionConversionFromTemporary() // CHECK: 1: A() (CXXConstructExpr, [B1.3], A) // CXX11-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const A) // CXX11-NEXT: 3: [B1.2] // CXX11-ELIDE-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.6], [B1.8], [B1.9], B) // CXX11-NOELIDE-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.6], [B1.8], B) // CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, ConstructorConversion, B) // CXX11-NEXT: 6: [B1.5] (BindTemporary) // CXX11-NEXT: 7: [B1.6] (ImplicitCastExpr, NoOp, const B) // CXX11-NEXT: 8: [B1.7] // CXX11-NEXT: 9: [B1.8] (CXXConstructExpr, [B1.10], B) // CXX11-NEXT: 10: B b = A(); // CXX11-NEXT: 11: ~B() (Temporary object destructor) // CXX11-NEXT: 12: [B1.10].~B() (Implicit destructor) // CXX17-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const A) // CXX17-NEXT: 3: [B1.2] // CXX17-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.6], B) // CXX17-NEXT: 5: [B1.4] (ImplicitCastExpr, ConstructorConversion, B) // CXX17-NEXT: 6: B b = A(); // CXX17-NEXT: 7: [B1.6].~B() (Implicit destructor) void implicitConstructionConversionFromTemporary() { B b = A(); } // CHECK: void implicitConstructionConversionFromFunctionValue() // CHECK: 1: get // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, A (*)(void)) // CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5]) // CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, const A) // CHECK-NEXT: 5: [B1.4] // CXX11-ELIDE-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.8], [B1.10], [B1.11], B) // CXX11-NOELIDE-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.8], [B1.10], B) // CXX11-NEXT: 7: [B1.6] (ImplicitCastExpr, ConstructorConversion, B) // CXX11-NEXT: 8: [B1.7] (BindTemporary) // CXX11-NEXT: 9: [B1.8] (ImplicitCastExpr, NoOp, const B) // CXX11-NEXT: 10: [B1.9] // CXX11-NEXT: 11: [B1.10] (CXXConstructExpr, [B1.12], B) // CXX11-NEXT: 12: B b = get(); // CXX11-NEXT: 13: ~B() (Temporary object destructor) // CXX11-NEXT: 14: [B1.12].~B() (Implicit destructor) // CXX17-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.8], B) // CXX17-NEXT: 7: [B1.6] (ImplicitCastExpr, ConstructorConversion, B) // CXX17-NEXT: 8: B b = get(); // CXX17-NEXT: 9: [B1.8].~B() (Implicit destructor) void implicitConstructionConversionFromFunctionValue() { B b = get(); } // CHECK: void implicitConstructionConversionFromTemporaryWithLifetimeExtension() // CHECK: 1: A() (CXXConstructExpr, [B1.3], A) // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const A) // CHECK-NEXT: 3: [B1.2] // CHECK-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.7], B) // CHECK-NEXT: 5: [B1.4] (ImplicitCastExpr, ConstructorConversion, B) // CHECK-NEXT: 6: [B1.5] (ImplicitCastExpr, NoOp, const B) // CHECK-NEXT: 7: [B1.6] // CHECK-NEXT: 8: const B &b = A(); // CHECK-NEXT: 9: [B1.8].~B() (Implicit destructor) void implicitConstructionConversionFromTemporaryWithLifetimeExtension() { const B &b = A(); } // CHECK: void implicitConstructionConversionFromFunctionValueWithLifetimeExtension() // CHECK: 1: get // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, A (*)(void)) // CHECK-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5]) // CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, const A) // CHECK-NEXT: 5: [B1.4] // CHECK-NEXT: 6: [B1.5] (CXXConstructExpr, [B1.9], B) // CHECK-NEXT: 7: [B1.6] (ImplicitCastExpr, ConstructorConversion, B) // CHECK-NEXT: 8: [B1.7] (ImplicitCastExpr, NoOp, const B) // CHECK-NEXT: 9: [B1.8] // CHECK-NEXT: 10: const B &b = get(); // CHECK-NEXT: 11: [B1.10].~B() (Implicit destructor) void implicitConstructionConversionFromFunctionValueWithLifetimeExtension() { const B &b = get(); // no-crash } } // end namespace implicit_constructor_conversion namespace argument_constructors { class D { public: D(); ~D(); }; class E { public: E(D d); E(D d1, D d2); }; void useC(C c); void useCByReference(const C &c); void useD(D d); void useDByReference(const D &d); void useCAndD(C c, D d); // CHECK: void passArgument() // CHECK: 1: useC // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(C)) // CXX11-ELIDE-NEXT: 3: C() (CXXConstructExpr, [B1.4], [B1.5], C) // CXX11-NOELIDE-NEXT: 3: C() (CXXConstructExpr, [B1.4], C) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6]+0, C) // CXX11-NEXT: 6: [B1.2]([B1.5]) // CXX17-NEXT: 3: C() (CXXConstructExpr, [B1.4]+0, C) // CXX17-NEXT: 4: [B1.2]([B1.3]) void passArgument() { useC(C()); } // CHECK: void passTwoArguments() // CHECK: [B1] // CHECK-NEXT: 1: useCAndD // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(C, D)) // CXX11-ELIDE-NEXT: 3: C() (CXXConstructExpr, [B1.4], [B1.5], C) // CXX11-NOELIDE-NEXT: 3: C() (CXXConstructExpr, [B1.4], C) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.12]+0, C) // CXX11-ELIDE-NEXT: 6: D() (CXXConstructExpr, [B1.7], [B1.9], [B1.10], D) // CXX11-NOELIDE-NEXT: 6: D() (CXXConstructExpr, [B1.7], [B1.9], D) // CXX11-NEXT: 7: [B1.6] (BindTemporary) // CXX11-NEXT: 8: [B1.7] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 9: [B1.8] // CXX11-NEXT: 10: [B1.9] (CXXConstructExpr, [B1.11], [B1.12]+1, D) // CXX11-NEXT: 11: [B1.10] (BindTemporary) // CXX11-NEXT: 12: [B1.2]([B1.5], [B1.11]) // CXX11-NEXT: 13: ~D() (Temporary object destructor) // CXX11-NEXT: 14: ~D() (Temporary object destructor) // CXX17-NEXT: 3: C() (CXXConstructExpr, [B1.6]+0, C) // CXX17-NEXT: 4: D() (CXXConstructExpr, [B1.5], [B1.6]+1, D) // CXX17-NEXT: 5: [B1.4] (BindTemporary) // CXX17-NEXT: 6: [B1.2]([B1.3], [B1.5]) // CXX17-NEXT: 7: ~D() (Temporary object destructor) void passTwoArguments() { useCAndD(C(), D()); } // CHECK: void passArgumentByReference() // CHECK: 1: useCByReference // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(const C &)) // CHECK-NEXT: 3: C() (CXXConstructExpr, [B1.5], C) // CHECK-NEXT: 4: [B1.3] (ImplicitCastExpr, NoOp, const C) // CHECK-NEXT: 5: [B1.4] // CHECK-NEXT: 6: [B1.2]([B1.5]) void passArgumentByReference() { useCByReference(C()); } // CHECK: void passArgumentWithDestructor() // CHECK: 1: useD // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(D)) // CXX11-ELIDE-NEXT: 3: D() (CXXConstructExpr, [B1.4], [B1.6], [B1.7], D) // CXX11-NOELIDE-NEXT: 3: D() (CXXConstructExpr, [B1.4], [B1.6], D) // CXX11-NEXT: 4: [B1.3] (BindTemporary) // CXX11-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 6: [B1.5] // CXX11-NEXT: 7: [B1.6] (CXXConstructExpr, [B1.8], [B1.9]+0, D) // CXX11-NEXT: 8: [B1.7] (BindTemporary) // CXX11-NEXT: 9: [B1.2]([B1.8]) // CXX11-NEXT: 10: ~D() (Temporary object destructor) // CXX11-NEXT: 11: ~D() (Temporary object destructor) // CXX17-NEXT: 3: D() (CXXConstructExpr, [B1.4], [B1.5]+0, D) // CXX17-NEXT: 4: [B1.3] (BindTemporary) // CXX17-NEXT: 5: [B1.2]([B1.4]) // CXX17-NEXT: 6: ~D() (Temporary object destructor) void passArgumentWithDestructor() { useD(D()); } // CHECK: void passArgumentWithDestructorByReference() // CHECK: 1: useDByReference // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, void (*)(const D &)) // CHECK-NEXT: 3: D() (CXXConstructExpr, [B1.4], [B1.6], D) // CHECK-NEXT: 4: [B1.3] (BindTemporary) // CHECK-NEXT: 5: [B1.4] (ImplicitCastExpr, NoOp, const D) // CHECK-NEXT: 6: [B1.5] // CHECK-NEXT: 7: [B1.2]([B1.6]) // CHECK-NEXT: 8: ~D() (Temporary object destructor) void passArgumentWithDestructorByReference() { useDByReference(D()); } // CHECK: void passArgumentIntoAnotherConstructor() // CXX11-ELIDE: 1: D() (CXXConstructExpr, [B1.2], [B1.4], [B1.5], D) // CXX11-NOELIDE: 1: D() (CXXConstructExpr, [B1.2], [B1.4], D) // CXX11-NEXT: 2: [B1.1] (BindTemporary) // CXX11-NEXT: 3: [B1.2] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], [B1.7]+0, D) // CXX11-NEXT: 6: [B1.5] (BindTemporary) // CXX11-ELIDE-NEXT: 7: [B1.6] (CXXConstructExpr, [B1.9], [B1.10], E) // CXX11-NOELIDE-NEXT: 7: [B1.6] (CXXConstructExpr, [B1.9], E) // CXX11-NEXT: 8: E([B1.7]) (CXXFunctionalCastExpr, ConstructorConversion, E) // CXX11-NEXT: 9: [B1.8] // CXX11-NEXT: 10: [B1.9] (CXXConstructExpr, [B1.11], E) // CXX11-NEXT: 11: E e = E(D()); // CXX11-NEXT: 12: ~D() (Temporary object destructor) // CXX11-NEXT: 13: ~D() (Temporary object destructor) // CXX17: 1: D() (CXXConstructExpr, [B1.2], [B1.3]+0, D) // CXX17-NEXT: 2: [B1.1] (BindTemporary) // CXX17-NEXT: 3: [B1.2] (CXXConstructExpr, [B1.5], E) // CXX17-NEXT: 4: E([B1.3]) (CXXFunctionalCastExpr, ConstructorConversion, E) // CXX17-NEXT: 5: E e = E(D()); // CXX17-NEXT: 6: ~D() (Temporary object destructor) void passArgumentIntoAnotherConstructor() { E e = E(D()); } // CHECK: void passTwoArgumentsIntoAnotherConstructor() // CXX11-ELIDE: 1: D() (CXXConstructExpr, [B1.2], [B1.4], [B1.5], D) // CXX11-NOELIDE: 1: D() (CXXConstructExpr, [B1.2], [B1.4], D) // CXX11-NEXT: 2: [B1.1] (BindTemporary) // CXX11-NEXT: 3: [B1.2] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: [B1.4] (CXXConstructExpr, [B1.6], [B1.13]+0, D) // CXX11-NEXT: 6: [B1.5] (BindTemporary) // CXX11-ELIDE-NEXT: 7: D() (CXXConstructExpr, [B1.8], [B1.10], [B1.11], D) // CXX11-NOELIDE-NEXT: 7: D() (CXXConstructExpr, [B1.8], [B1.10], D) // CXX11-NEXT: 8: [B1.7] (BindTemporary) // CXX11-NEXT: 9: [B1.8] (ImplicitCastExpr, NoOp, const D) // CXX11-NEXT: 10: [B1.9] // CXX11-NEXT: 11: [B1.10] (CXXConstructExpr, [B1.12], [B1.13]+1, D) // CXX11-NEXT: 12: [B1.11] (BindTemporary) // CXX11-NEXT: 13: E([B1.6], [B1.12]) (CXXConstructExpr, E) // CXX11-NEXT: 14: ~D() (Temporary object destructor) // CXX11-NEXT: 15: ~D() (Temporary object destructor) // CXX11-NEXT: 16: ~D() (Temporary object destructor) // CXX11-NEXT: 17: ~D() (Temporary object destructor) // CXX17: 1: D() (CXXConstructExpr, [B1.2], [B1.5]+0, D) // CXX17-NEXT: 2: [B1.1] (BindTemporary) // CXX17-NEXT: 3: D() (CXXConstructExpr, [B1.4], [B1.5]+1, D) // CXX17-NEXT: 4: [B1.3] (BindTemporary) // CXX17-NEXT: 5: E([B1.2], [B1.4]) (CXXConstructExpr, E) // CXX17-NEXT: 6: ~D() (Temporary object destructor) // CXX17-NEXT: 7: ~D() (Temporary object destructor) void passTwoArgumentsIntoAnotherConstructor() { E(D(), D()); } } // end namespace argument_constructors namespace copy_elision_with_extra_arguments { class C { public: C(); C(const C &c, int x = 0); }; // CHECK: void testCopyElisionWhenCopyConstructorHasExtraArguments() // CHECK: [B1] // CXX11-ELIDE-NEXT: 1: C() (CXXConstructExpr, [B1.3], [B1.5], C) // CXX11-NOELIDE-NEXT: 1: C() (CXXConstructExpr, [B1.3], C) // CXX11-NEXT: 2: [B1.1] (ImplicitCastExpr, NoOp, const C) // CXX11-NEXT: 3: [B1.2] // CXX11-NEXT: 4: // CXX11-NEXT: 5: [B1.3] (CXXConstructExpr, [B1.6], C) // CXX11-NEXT: 6: C c = C(); // CXX17-NEXT: 1: C() (CXXConstructExpr, [B1.2], C) // CXX17-NEXT: 2: C c = C(); void testCopyElisionWhenCopyConstructorHasExtraArguments() { C c = C(); } } // namespace copy_elision_with_extra_arguments namespace operators { class C { public: C(int); C &operator+(C Other); }; // FIXME: Find construction context for the this-argument of the operator. // CHECK: void testOperators() // CHECK: [B1] // CHECK-NEXT: 1: operator+ // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, C &(*)(C)) // CHECK-NEXT: 3: 1 // CHECK-NEXT: 4: [B1.3] (CXXConstructExpr, [B1.6], C) // CHECK-NEXT: 5: C([B1.4]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CHECK-NEXT: 6: [B1.5] // CHECK-NEXT: 7: 2 // CXX11-ELIDE-NEXT: 8: [B1.7] (CXXConstructExpr, [B1.10], [B1.11], C) // CXX11-NOELIDE-NEXT: 8: [B1.7] (CXXConstructExpr, [B1.10], C) // CXX11-NEXT: 9: C([B1.8]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CXX11-NEXT: 10: [B1.9] // CXX11-NEXT: 11: [B1.10] (CXXConstructExpr, [B1.12]+1, C) // CXX11-NEXT: 12: [B1.6] + [B1.11] (OperatorCall) // CXX17-NEXT: 8: [B1.7] (CXXConstructExpr, [B1.10]+1, C) // CXX17-NEXT: 9: C([B1.8]) (CXXFunctionalCastExpr, ConstructorConversion, C) // CXX17-NEXT: 10: [B1.6] + [B1.9] (OperatorCall) void testOperators() { C(1) + C(2); } } // namespace operators namespace variadic_function_arguments { class C { public: C(int); }; int variadic(...); // This code is quite exotic, so let's not test the CFG for it, // but only make sure we don't crash. void testCrashOnVariadicArgument() { C c(variadic(0 ? c : 0)); // no-crash } } // namespace variadic_function_arguments // CHECK: void testTransparentInitListExprs() // CHECK: [B1] // CHECK-NEXT: 1: getC // CHECK-NEXT: 2: [B1.1] (ImplicitCastExpr, FunctionToPointerDecay, C (*)(void)) // CXX11-ELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4], [B1.5]) // CXX11-NOELIDE-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.4]) // CXX11-NEXT: 4: [B1.3] // CXX11-NEXT: 5: {[B1.4]} (CXXConstructExpr, [B1.6], C) // CXX11-NEXT: 6: C c{getC()}; // CXX17-NEXT: 3: [B1.2]() (CXXRecordTypedCall, [B1.5]) // CXX17-NEXT: 4: {[B1.3]} // CXX17-NEXT: 5: C c{getC()}; namespace transparent_init_list_exprs { class C {}; C getC(); void testTransparentInitListExprs() { C c{getC()}; } } // namespace transparent_init_list_exprs