//===----------------------------------------------------------------------===// // // 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 // //===----------------------------------------------------------------------===// // // shared_ptr // template shared_ptr(Y* p, D d); #include #include #include "test_macros.h" #include "deleter_types.h" struct A { static int count; A() {++count;} A(const A&) {++count;} ~A() {--count;} }; int A::count = 0; struct bad_ty { }; struct bad_deleter { void operator()(bad_ty) { } }; struct no_move_deleter { no_move_deleter(no_move_deleter const&) = delete; no_move_deleter(no_move_deleter &&) = delete; void operator()(int*) { } }; static_assert(!std::is_move_constructible::value, ""); struct Base { }; struct Derived : Base { }; template class MoveDeleter { MoveDeleter(); MoveDeleter(MoveDeleter const&); public: MoveDeleter(MoveDeleter&&) {}; explicit MoveDeleter(int) {} void operator()(T *ptr) { delete ptr; } }; int main(int, char**) { { A* ptr = new A; std::shared_ptr p(ptr, test_deleter(3)); assert(A::count == 1); assert(p.use_count() == 1); assert(p.get() == ptr); assert(test_deleter::count == 1); assert(test_deleter::dealloc_count == 0); #ifndef TEST_HAS_NO_RTTI test_deleter* d = std::get_deleter >(p); assert(d); assert(d->state() == 3); #endif } assert(A::count == 0); assert(test_deleter::count == 0); assert(test_deleter::dealloc_count == 1); { A const* ptr = new A; std::shared_ptr p(ptr, test_deleter(3)); assert(p.get() == ptr); } { // Make sure we can't construct with: // a) a deleter that doesn't have an operator ()(int*) // b) a deleter that doesn't have a move constructor. static_assert(!std::is_constructible, int*, bad_deleter>::value, ""); static_assert(!std::is_constructible, int*, no_move_deleter>::value, ""); // Make sure that we can construct a shared_ptr where the element type and pointer type // aren't "convertible" but are "compatible". static_assert(!std::is_constructible, Base[4], test_deleter >::value, ""); } #if TEST_STD_VER >= 11 { MoveDeleter d(0); std::shared_ptr p0(new int, std::move(d)); std::shared_ptr p1(nullptr, std::move(d)); } #endif // TEST_STD_VER >= 11 return 0; }