//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // FIXME: In MSVC mode, even "std::function f(aref);" causes // allocations. // XFAIL: target=x86_64-pc-windows-msvc && stdlib=libc++ // UNSUPPORTED: c++03 // // class function // function(const function& f); // function(function&& f); // noexcept in C++20 #include #include #include #include #include #include "test_macros.h" #include "count_new.h" class A { int data_[10]; public: static int count; A() { ++count; for (int i = 0; i < 10; ++i) data_[i] = i; } A(const A&) {++count;} ~A() {--count;} int operator()(int i) const { for (int j = 0; j < 10; ++j) i += data_[j]; return i; } }; int A::count = 0; int g(int) {return 0;} int main(int, char**) { globalMemCounter.reset(); assert(globalMemCounter.checkOutstandingNewEq(0)); { std::function f = A(); assert(A::count == 1); assert(globalMemCounter.checkOutstandingNewEq(1)); RTTI_ASSERT(f.target()); RTTI_ASSERT(f.target() == 0); std::function f2 = f; assert(A::count == 2); assert(globalMemCounter.checkOutstandingNewEq(2)); RTTI_ASSERT(f2.target()); RTTI_ASSERT(f2.target() == 0); } assert(A::count == 0); assert(globalMemCounter.checkOutstandingNewEq(0)); { std::function f = g; assert(globalMemCounter.checkOutstandingNewEq(0)); RTTI_ASSERT(f.target()); RTTI_ASSERT(f.target() == 0); std::function f2 = f; assert(globalMemCounter.checkOutstandingNewEq(0)); RTTI_ASSERT(f2.target()); RTTI_ASSERT(f2.target() == 0); } assert(globalMemCounter.checkOutstandingNewEq(0)); { std::function f; assert(globalMemCounter.checkOutstandingNewEq(0)); RTTI_ASSERT(f.target() == 0); RTTI_ASSERT(f.target() == 0); std::function f2 = f; assert(globalMemCounter.checkOutstandingNewEq(0)); RTTI_ASSERT(f2.target() == 0); RTTI_ASSERT(f2.target() == 0); } { std::function f; assert(globalMemCounter.checkOutstandingNewEq(0)); RTTI_ASSERT(f.target() == 0); RTTI_ASSERT(f.target() == 0); assert(!f); std::function g = f; assert(globalMemCounter.checkOutstandingNewEq(0)); RTTI_ASSERT(g.target() == 0); RTTI_ASSERT(g.target() == 0); assert(!g); } #if TEST_STD_VER >= 11 assert(globalMemCounter.checkOutstandingNewEq(0)); { // Test rvalue references std::function f = A(); assert(A::count == 1); assert(globalMemCounter.checkOutstandingNewEq(1)); RTTI_ASSERT(f.target()); RTTI_ASSERT(f.target() == 0); LIBCPP_ASSERT_NOEXCEPT(std::function(std::move(f))); #if TEST_STD_VER > 17 ASSERT_NOEXCEPT(std::function(std::move(f))); #endif std::function f2 = std::move(f); assert(A::count == 1); assert(globalMemCounter.checkOutstandingNewEq(1)); RTTI_ASSERT(f2.target()); RTTI_ASSERT(f2.target() == 0); RTTI_ASSERT(f.target() == 0); RTTI_ASSERT(f.target() == 0); } assert(globalMemCounter.checkOutstandingNewEq(0)); { // Test that moving a function constructed from a reference wrapper // is done without allocating. DisableAllocationGuard g; using Ref = std::reference_wrapper; A a; Ref aref(a); std::function f(aref); assert(A::count == 1); RTTI_ASSERT(f.target() == nullptr); RTTI_ASSERT(f.target()); LIBCPP_ASSERT_NOEXCEPT(std::function(std::move(f))); #if TEST_STD_VER > 17 ASSERT_NOEXCEPT(std::function(std::move(f))); #endif std::function f2(std::move(f)); assert(A::count == 1); RTTI_ASSERT(f2.target() == nullptr); RTTI_ASSERT(f2.target()); #if defined(_LIBCPP_VERSION) RTTI_ASSERT(f.target()); // f is unchanged because the target is small #endif } { // Test that moving a function constructed from a function pointer // is done without allocating DisableAllocationGuard guard; using Ptr = int(*)(int); Ptr p = g; std::function f(p); RTTI_ASSERT(f.target() == nullptr); RTTI_ASSERT(f.target()); LIBCPP_ASSERT_NOEXCEPT(std::function(std::move(f))); #if TEST_STD_VER > 17 ASSERT_NOEXCEPT(std::function(std::move(f))); #endif std::function f2(std::move(f)); RTTI_ASSERT(f2.target() == nullptr); RTTI_ASSERT(f2.target()); #if defined(_LIBCPP_VERSION) RTTI_ASSERT(f.target()); // f is unchanged because the target is small #endif } #endif // TEST_STD_VER >= 11 return 0; }