// RUN: %clang_cc1 -std=c++17 %s -triple x86_64-linux-gnu -emit-llvm -o - | FileCheck %s --implicit-check-not='call{{.*}}dtor'
// RUN: %clang_cc1 -std=c++23 %s -triple x86_64-linux-gnu -emit-llvm -o - | FileCheck %s --check-prefixes=CHECK-CXX23,CHECK-CXX23-NEXT,CHECK-CXX23-LABEL
namespace std {
typedef decltype(sizeof(int)) size_t;
template <class E>
struct initializer_list {
const E *begin;
size_t size;
initializer_list() : begin(nullptr), size(0) {}
};
template <typename E>
struct list {
list() {}
~list() {}
E *begin();
E *end();
const E *begin() const;
const E *end() const;
};
template <typename E>
struct vector {
vector() {}
vector(std::initializer_list<E>) {}
~vector() {}
E *begin();
E *end();
const E *begin() const;
const E *end() const;
};
template <typename T>
struct lock_guard {
lock_guard(T) {}
~lock_guard() {}
};
struct mutex {};
} // namespace std
void then();
struct dtor {
~dtor();
};
dtor ctor();
auto &&lambda = [a = {ctor()}] {};
// CHECK-LABEL: define
// CHECK: call {{.*}}ctor
// CHECK: call {{.*}}atexit{{.*}}global_array_dtor
// CHECK-LABEL: define{{.*}}global_array_dtor
// CHECK: call {{.*}}dtor
// [lifetime extension occurs if the object was obtained by]
// -- a temporary materialization conversion
// CHECK-LABEL: ref_binding
void ref_binding() {
// CHECK: call {{.*}}ctor
auto &&x = ctor();
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// -- ( expression )
// CHECK-LABEL: parens
void parens() {
// CHECK: call {{.*}}ctor
auto &&x = ctor();
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// -- subscripting of an array
// CHECK-LABEL: array_subscript_1
void array_subscript_1() {
using T = dtor[1];
// CHECK: call {{.*}}ctor
auto &&x = T{ctor()}[0];
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// CHECK-LABEL: array_subscript_2
void array_subscript_2() {
using T = dtor[1];
// CHECK: call {{.*}}ctor
auto &&x = ((dtor*)T{ctor()})[0];
// CHECK: call {{.*}}dtor
// CHECK: call {{.*}}then
then();
// CHECK: }
}
struct with_member { dtor d; ~with_member(); };
struct with_ref_member { dtor &&d; ~with_ref_member(); };
// -- a class member access using the . operator [...]
// CHECK-LABEL: member_access_1
void member_access_1() {
// CHECK: call {{.*}}ctor
auto &&x = with_member{ctor()}.d;
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}with_member
// CHECK: }
}
// CHECK-LABEL: member_access_2
void member_access_2() {
// CHECK: call {{.*}}ctor
auto &&x = with_ref_member{ctor()}.d;
// CHECK: call {{.*}}with_ref_member
// CHECK: call {{.*}}dtor
// CHECK: call {{.*}}then
then();
// CHECK: }
}
// CHECK-LABEL: member_access_3
void member_access_3() {
// CHECK: call {{.*}}ctor
auto &&x = (&(const with_member&)with_member{ctor()})->d;
// CHECK: call {{.*}}with_member
// CHECK: call {{.*}}then
then();
// CHECK: }
}
// -- a pointer-to-member operation using the .* operator [...]
// CHECK-LABEL: member_ptr_access_1
void member_ptr_access_1() {
// CHECK: call {{.*}}ctor
auto &&x = with_member{ctor()}.*&with_member::d;
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}with_member
// CHECK: }
}
// CHECK-LABEL: member_ptr_access_2
void member_ptr_access_2() {
// CHECK: call {{.*}}ctor
auto &&x = (&(const with_member&)with_member{ctor()})->*&with_member::d;
// CHECK: call {{.*}}with_member
// CHECK: call {{.*}}then
then();
// CHECK: }
}
// -- a [named] cast [...]
// CHECK-LABEL: static_cast
void test_static_cast() {
// CHECK: call {{.*}}ctor
auto &&x = static_cast<dtor&&>(ctor());
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// CHECK-LABEL: const_cast
void test_const_cast() {
// CHECK: call {{.*}}ctor
auto &&x = const_cast<dtor&&>(ctor());
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// CHECK-LABEL: reinterpret_cast
void test_reinterpret_cast() {
// CHECK: call {{.*}}ctor
auto &&x = reinterpret_cast<dtor&&>(static_cast<dtor&&>(ctor()));
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// CHECK-LABEL: dynamic_cast
void test_dynamic_cast() {
// CHECK: call {{.*}}ctor
auto &&x = dynamic_cast<dtor&&>(ctor());
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// -- [explicit cast notation is defined in terms of the above]
// CHECK-LABEL: c_style_cast
void c_style_cast() {
// CHECK: call {{.*}}ctor
auto &&x = (dtor&&)ctor();
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// CHECK-LABEL: function_style_cast
void function_style_cast() {
// CHECK: call {{.*}}ctor
using R = dtor&&;
auto &&x = R(ctor());
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// -- a conditional operator
// CHECK-LABEL: conditional
void conditional(bool b) {
// CHECK: call {{.*}}ctor
// CHECK: call {{.*}}ctor
auto &&x = b ? (dtor&&)ctor() : (dtor&&)ctor();
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: call {{.*}}dtor
// CHECK: }
}
// -- a comma expression
// CHECK-LABEL: comma
void comma() {
// CHECK: call {{.*}}ctor
auto &&x = (true, (dtor&&)ctor());
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// This applies recursively: if an object is lifetime-extended and contains a
// reference, the referent is also extended.
// CHECK-LABEL: init_capture_ref
void init_capture_ref() {
// CHECK: call {{.*}}ctor
auto x = [&a = (const dtor&)ctor()] {};
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// CHECK-LABEL: init_capture_ref_indirect
void init_capture_ref_indirect() {
// CHECK: call {{.*}}ctor
auto x = [&a = (const dtor&)ctor()] {};
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
// CHECK-LABEL: init_capture_init_list
void init_capture_init_list() {
// CHECK: call {{.*}}ctor
auto x = [a = {ctor()}] {};
// CHECK: call {{.*}}then
then();
// CHECK: call {{.*}}dtor
// CHECK: }
}
namespace P2718R0 {
namespace basic {
template <typename E> using T2 = std::list<E>;
template <typename E> const T2<E> &f1_temp(const T2<E> &t) { return t; }
template <typename E> const T2<E> &f2_temp(T2<E> t) { return t; }
template <typename E> T2<E> g_temp() { return T2<E>{}; }
template <typename E>
void foo_dependent_context1() {
// CHECK-CXX23: void @_ZN7P2718R05basic22foo_dependent_context1IiEEvv()
// CHECK-CXX23: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
for (auto e : f1_temp(g_temp<E>())) {} // OK, lifetime of return value of g() extended
}
template <typename E>
void foo_dependent_context2() {
// CHECK-CXX23: void @_ZN7P2718R05basic22foo_dependent_context2IiEEvv()
// CHECK-CXX23-NEXT: entry:
// CHECK-CXX23-NEXT: call void @_ZN7P2718R05basic6g_tempIiEESt4listIT_Ev(
// CHECK-CXX23-NEXT: call {{.*}} @_ZN7P2718R05basic7f2_tempIiEERKSt4listIT_ES4_(
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
// CHECK-CXX23: call {{.*}} @_ZNKSt4listIiE5beginEv(
// CHECK-CXX23: call {{.*}} @_ZNKSt4listIiE3endEv(
for (auto e : f2_temp(g_temp<E>())) {} // undefined behavior
}
template void foo_dependent_context1<int>();
template void foo_dependent_context2<int>();
} // namespace basic
namespace discard_value_expression {
template <typename T>
void f_dependent_context1() {
std::vector<T> v = { 42, 17, 13 };
std::mutex m;
// CHECK-CXX23: void @_ZN7P2718R024discard_value_expression20f_dependent_context1IiEEvv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt10lock_guardISt5mutexED1Ev(
for (T x : std::lock_guard<std::mutex>(m), v) // lock released in C++ 2023
std::lock_guard<std::mutex> guard(m); // OK in C++ 2023, now deadlocks
}
template <typename T>
void f_dependent_context2() {
std::vector<T> v = { 42, 17, 13 };
std::mutex m;
// CHECK-CXX23: void @_ZN7P2718R024discard_value_expression20f_dependent_context2IiEEvv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt10lock_guardISt5mutexED1Ev(
for (T x : (void)std::lock_guard<std::mutex>(m), v) // lock released in C++ 2023
std::lock_guard<std::mutex> guard(m); // OK in C++ 2023, now deadlocks
}
template <typename T>
void f_dependent_context3() {
std::vector<T> v = { 42, 17, 13 };
std::mutex m;
// CHECK-CXX23: void @_ZN7P2718R024discard_value_expression20f_dependent_context3IiEEvv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt10lock_guardISt5mutexED1Ev(
for (T x : static_cast<void>(std::lock_guard<std::mutex>(m)), v) // lock released in C++ 2023
std::lock_guard<std::mutex> guard(m); // OK in C++ 2023, now deadlocks
}
template void f_dependent_context1<int>();
template void f_dependent_context2<int>();
template void f_dependent_context3<int>();
} // namespace discard_value_expression
namespace member_call {
template <typename T>
struct ListWrapper {
std::list<T> list;
ListWrapper() {}
~ListWrapper() {}
const T *begin() const { return list.begin(); }
const T *end() const { return list.end(); }
ListWrapper& r() { return *this; }
ListWrapper g() { return ListWrapper(); }
};
template <typename E>
ListWrapper<E> g_temp() { return ListWrapper<E>{}; }
template <typename T>
void member_call_dependent_context() {
// CHECK-CXX23: void @_ZN7P2718R011member_call29member_call_dependent_contextIiEEvv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
for (auto e : g_temp<T>().r().g().r().g().r().g()) {}
}
template void member_call_dependent_context<int>();
} // namespace member_call
namespace default_arg {
template <typename T>
struct DefaultArg {
DefaultArg() {}
DefaultArg(int) {}
~DefaultArg() {}
};
template <typename T>
struct C2 : public std::list<T> {
C2() {}
C2(int, const C2 &, const DefaultArg<T> &Default = DefaultArg<T>{}) {}
};
template <typename T>
std::list<T> temp_foo(const std::list<T>&, const DefaultArg<T> &Default = DefaultArg<T>{}) {
return std::list<T>{};
}
template <typename T>
void default_arg_dependent_context1() {
// CHECK-CXX23: void @_ZN7P2718R011default_arg30default_arg_dependent_context1IiEEvv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
for (auto e : temp_foo(std::list<T>{})) {}
}
template <typename T>
void default_arg_dependent_context2() {
// CHECK-CXX23: void @_ZN7P2718R011default_arg30default_arg_dependent_context2IiEEvv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
for (auto e : temp_foo(temp_foo(std::list<T>{}))) {}
}
template <typename T>
void default_arg_dependent_context3() {
// CHECK-CXX23: void @_ZN7P2718R011default_arg30default_arg_dependent_context3IiEEvv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg2C2IiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg2C2IiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg2C2IiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg2C2IiED1Ev(
for (auto e : C2<T>(0, C2<T>(0, C2<T>(0, C2<T>())))) {}
}
template void default_arg_dependent_context1<int>();
template void default_arg_dependent_context2<int>();
template void default_arg_dependent_context3<int>();
} // namespace default_arg
namespace basic {
using T = std::list<int>;
const T& f1(const T& t) { return t; }
const T& f2(T t) { return t; }
T g() { return T{}; }
void foo1() {
// CHECK-CXX23: void @_ZN7P2718R05basic4foo1Ev()
// CHECK-CXX23: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
for (auto e : f1(g())) {} // OK, lifetime of return value of g() extended
}
void foo2() {
// CHECK-CXX23: void @_ZN7P2718R05basic4foo2Ev()
// CHECK-CXX23-NEXT: call void @_ZN7P2718R05basic1gEv(
// CHECK-CXX23-NEXT: call {{.*}} @_ZN7P2718R05basic2f2ESt4listIiE(
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
for (auto e : f2(g())) {} // undefined behavior
}
} // namespace basic
namespace discard_value_expression {
void f1() {
std::vector<int> v = { 42, 17, 13 };
std::mutex m;
// CHECK-CXX23: void @_ZN7P2718R024discard_value_expression2f1Ev()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt10lock_guardISt5mutexED1Ev(
for (int x : std::lock_guard<std::mutex>(m), v) // lock released in C++ 2023
std::lock_guard<std::mutex> guard(m); // OK in C++ 2023, now deadlocks
}
void f2() {
std::vector<int> v = { 42, 17, 13 };
std::mutex m;
// CHECK-CXX23: void @_ZN7P2718R024discard_value_expression2f2Ev()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt10lock_guardISt5mutexED1Ev(
for (int x : (void)std::lock_guard<std::mutex>(m), v) // lock released in C++ 2023
std::lock_guard<std::mutex> guard(m); // OK in C++ 2023, now deadlocks
}
void f3() {
std::vector<int> v = { 42, 17, 13 };
std::mutex m;
// CHECK-CXX23: void @_ZN7P2718R024discard_value_expression2f3Ev()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt10lock_guardISt5mutexED1Ev(
for (int x : static_cast<void>(std::lock_guard<std::mutex>(m)), v) // lock released in C++ 2023
std::lock_guard<std::mutex> guard(m); // OK in C++ 2023, now deadlocks
}
} // namespace discard_value_expression
namespace member_call {
using A = ListWrapper<int>;
A g() { return A(); }
const A &f1(const A &t) { return t; }
void member_call() {
// CHECK-CXX23: void @_ZN7P2718R011member_call11member_callEv()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011member_call11ListWrapperIiED1Ev(
for (auto e : g().r().g().r().g().r().g()) {}
}
} // namespace member_call
namespace default_arg {
using A = std::list<int>;
using DefaultA = DefaultArg<int>;
struct C : public A {
C() {}
C(int, const C &, const DefaultA & = DefaultA()) {}
};
A foo(const A&, const DefaultA &Default = DefaultA()) {
return A();
}
int (&some_func(const A & = A{}))[3];
void default_arg1() {
// CHECK-CXX23: void @_ZN7P2718R011default_arg12default_arg1Ev()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
for (auto e : some_func()) {}
}
void default_arg2() {
// CHECK-CXX23: void @_ZN7P2718R011default_arg12default_arg2Ev()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZNSt4listIiED1Ev(
for (auto e : some_func(foo(foo(A())))) {}
}
void default_arg3() {
// CHECK-CXX23: void @_ZN7P2718R011default_arg12default_arg3Ev()
// CHECK-CXX23-LABEL: for.cond.cleanup:
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg1CD1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg1CD1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg1CD1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg10DefaultArgIiED1Ev(
// CHECK-CXX23-NEXT: call void @_ZN7P2718R011default_arg1CD1Ev(
for (auto e : C(0, C(0, C(0, C())))) {}
}
} // namespace default_arg
} // namespace P2718R0