Fix for LWG Issue 2415: Inconsistency between unique_ptr and shared_ptr
git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@236953 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
928735abf1
commit
0ad232a882
@ -1639,7 +1639,7 @@ template <class _Traits, class _Tp>
|
||||
struct __rebind_alloc_helper
|
||||
{
|
||||
#ifndef _LIBCPP_HAS_NO_TEMPLATE_ALIASES
|
||||
typedef typename _Traits::template rebind_alloc<_Tp> type;
|
||||
typedef typename _Traits::template rebind_alloc<_Tp> type;
|
||||
#else
|
||||
typedef typename _Traits::template rebind_alloc<_Tp>::other type;
|
||||
#endif
|
||||
@ -4277,9 +4277,16 @@ shared_ptr<_Tp>::shared_ptr(unique_ptr<_Yp, _Dp> __r,
|
||||
>::type)
|
||||
: __ptr_(__r.get())
|
||||
{
|
||||
typedef __shared_ptr_pointer<_Yp*, _Dp, allocator<_Yp> > _CntrlBlk;
|
||||
__cntrl_ = new _CntrlBlk(__r.get(), __r.get_deleter(), allocator<_Yp>());
|
||||
__enable_weak_this(__r.get());
|
||||
#if _LIBCPP_STD_VER > 11
|
||||
if (__ptr_ == nullptr)
|
||||
__cntrl_ = nullptr;
|
||||
else
|
||||
#endif
|
||||
{
|
||||
typedef __shared_ptr_pointer<_Yp*, _Dp, allocator<_Yp> > _CntrlBlk;
|
||||
__cntrl_ = new _CntrlBlk(__r.get(), __r.get_deleter(), allocator<_Yp>());
|
||||
__enable_weak_this(__r.get());
|
||||
}
|
||||
__r.release();
|
||||
}
|
||||
|
||||
@ -4299,11 +4306,18 @@ shared_ptr<_Tp>::shared_ptr(unique_ptr<_Yp, _Dp> __r,
|
||||
>::type)
|
||||
: __ptr_(__r.get())
|
||||
{
|
||||
typedef __shared_ptr_pointer<_Yp*,
|
||||
reference_wrapper<typename remove_reference<_Dp>::type>,
|
||||
allocator<_Yp> > _CntrlBlk;
|
||||
__cntrl_ = new _CntrlBlk(__r.get(), ref(__r.get_deleter()), allocator<_Yp>());
|
||||
__enable_weak_this(__r.get());
|
||||
#if _LIBCPP_STD_VER > 11
|
||||
if (__ptr_ == nullptr)
|
||||
__cntrl_ = nullptr;
|
||||
else
|
||||
#endif
|
||||
{
|
||||
typedef __shared_ptr_pointer<_Yp*,
|
||||
reference_wrapper<typename remove_reference<_Dp>::type>,
|
||||
allocator<_Yp> > _CntrlBlk;
|
||||
__cntrl_ = new _CntrlBlk(__r.get(), ref(__r.get_deleter()), allocator<_Yp>());
|
||||
__enable_weak_this(__r.get());
|
||||
}
|
||||
__r.release();
|
||||
}
|
||||
|
||||
|
@ -67,7 +67,7 @@ int main()
|
||||
pB = std::move(pA);
|
||||
assert(B::count == 0);
|
||||
assert(A::count == 0);
|
||||
assert(pB.use_count() == 1);
|
||||
// assert(pB.use_count() == 1); // no longer true due to LWG 2415
|
||||
assert(pA.get() == 0);
|
||||
assert(pB.get() == ptrA);
|
||||
}
|
||||
@ -101,7 +101,7 @@ int main()
|
||||
pB = std::move(pA);
|
||||
assert(B::count == 0);
|
||||
assert(A::count == 0);
|
||||
assert(pB.use_count() == 1);
|
||||
// assert(pB.use_count() == 1); // no longer true due to LWG 2415
|
||||
assert(pA.get() == 0);
|
||||
assert(pB.get() == ptrA);
|
||||
}
|
||||
|
@ -58,6 +58,9 @@ int A::count = 0;
|
||||
void fn ( const std::shared_ptr<int> &) {}
|
||||
void fn ( const std::shared_ptr<B> &) { assert (false); }
|
||||
|
||||
template <typename T>
|
||||
void assert_deleter ( T * ) { assert(false); }
|
||||
|
||||
int main()
|
||||
{
|
||||
{
|
||||
@ -100,4 +103,13 @@ int main()
|
||||
throw_next = false;
|
||||
fn(std::unique_ptr<int>(new int));
|
||||
}
|
||||
|
||||
#if __cplusplus >= 201402L
|
||||
// LWG 2415
|
||||
{
|
||||
std::unique_ptr<int, void (*)(int*)> p(nullptr, assert_deleter<int>);
|
||||
std::shared_ptr<int> p2(std::move(p)); // should not call deleter when going out of scope
|
||||
}
|
||||
#endif
|
||||
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user