| Index: base/memory/scoped_ptr.h
|
| diff --git a/base/memory/scoped_ptr.h b/base/memory/scoped_ptr.h
|
| index 869bdfdbe92f6d8ab1f086c4b05afd9f1b7d207a..b3d5a35d593c44a4b5a85e93c6c9f9a419599bdf 100644
|
| --- a/base/memory/scoped_ptr.h
|
| +++ b/base/memory/scoped_ptr.h
|
| @@ -86,6 +86,7 @@
|
|
|
| #include <iosfwd>
|
| #include <memory>
|
| +#include <type_traits>
|
| #include <utility>
|
|
|
| #include "base/basictypes.h"
|
| @@ -243,8 +244,8 @@ class scoped_ptr {
|
|
|
| public:
|
| // The element and deleter types.
|
| - typedef T element_type;
|
| - typedef D deleter_type;
|
| + using element_type = T;
|
| + using deleter_type = D;
|
|
|
| // Constructor. Defaults to initializing with nullptr.
|
| scoped_ptr() : impl_(nullptr) {}
|
| @@ -258,35 +259,78 @@ class scoped_ptr {
|
| // Constructor. Allows construction from a nullptr.
|
| scoped_ptr(std::nullptr_t) : impl_(nullptr) {}
|
|
|
| - // Constructor. Allows construction from a scoped_ptr rvalue for a
|
| + // Move constructor.
|
| + //
|
| + // IMPLEMENTATION NOTE: Clang requires a move constructor to be defined (and
|
| + // not just the conversion constructor) in order to warn on pessimizing moves.
|
| + // The requirements for the move constructor are specified in C++11
|
| + // 20.7.1.2.1.15-17, which has some subtleties around reference deleters. As
|
| + // we don't support reference (or move-only) deleters, the post conditions are
|
| + // trivially true: we always copy construct the deleter from other's deleter.
|
| + scoped_ptr(scoped_ptr&& other) : impl_(&other.impl_) {}
|
| +
|
| + // Conversion constructor. Allows construction from a scoped_ptr rvalue for a
|
| // convertible type and deleter.
|
| //
|
| - // IMPLEMENTATION NOTE: C++11 unique_ptr<> keeps this constructor distinct
|
| - // from the normal move constructor. By C++11 20.7.1.2.1.21, this constructor
|
| - // has different post-conditions if D is a reference type. Since this
|
| - // implementation does not support deleters with reference type,
|
| - // we do not need a separate move constructor allowing us to avoid one
|
| - // use of SFINAE. You only need to care about this if you modify the
|
| - // implementation of scoped_ptr.
|
| - template <typename U, typename V>
|
| - scoped_ptr(scoped_ptr<U, V>&& other)
|
| - : impl_(&other.impl_) {
|
| - COMPILE_ASSERT(!base::is_array<U>::value, U_cannot_be_an_array);
|
| + // IMPLEMENTATION NOTE: C++ 20.7.1.2.1.19 requires this constructor to only
|
| + // participate in overload resolution if all the following are true:
|
| + // - U is implicitly convertible to T: this is important for 2 reasons:
|
| + // 1. So type traits don't incorrectly return true, e.g.
|
| + // std::is_convertible<scoped_ptr<Base>, scoped_ptr<Derived>>::value
|
| + // should be false.
|
| + // 2. To make sure code like this compiles:
|
| + // void F(scoped_ptr<int>);
|
| + // void F(scoped_ptr<Base>);
|
| + // // Ambiguous since both conversion constructors match.
|
| + // F(scoped_ptr<Derived>());
|
| + // - U is not an array type: to prevent conversions from scoped_ptr<T[]> to
|
| + // scoped_ptr<T>.
|
| + // - D is a reference type and E is the same type, or D is not a reference
|
| + // type and E is implicitly convertible to D: again, we don't support
|
| + // reference deleters, so we only worry about the latter requirement.
|
| + template <typename U,
|
| + typename E,
|
| + typename std::enable_if<!std::is_array<U>::value &&
|
| + std::is_convertible<U*, T*>::value &&
|
| + std::is_convertible<E, D>::value>::type* =
|
| + nullptr>
|
| + scoped_ptr(scoped_ptr<U, E>&& other)
|
| + : impl_(&other.impl_) {}
|
| +
|
| + // operator=.
|
| + //
|
| + // IMPLEMENTATION NOTE: Unlike the move constructor, Clang does not appear to
|
| + // require a move assignment operator to trigger the pessimizing move warning:
|
| + // in this case, the warning triggers when moving a temporary. For consistency
|
| + // with the move constructor, we define it anyway. C++11 20.7.1.2.3.1-3
|
| + // defines several requirements around this: like the move constructor, the
|
| + // requirements are simplified by the fact that we don't support move-only or
|
| + // reference deleters.
|
| + scoped_ptr& operator=(scoped_ptr&& rhs) {
|
| + impl_.TakeState(&rhs.impl_);
|
| + return *this;
|
| }
|
|
|
| // operator=. Allows assignment from a scoped_ptr rvalue for a convertible
|
| // type and deleter.
|
| //
|
| // IMPLEMENTATION NOTE: C++11 unique_ptr<> keeps this operator= distinct from
|
| - // the normal move assignment operator. By C++11 20.7.1.2.3.4, this templated
|
| - // form has different requirements on for move-only Deleters. Since this
|
| - // implementation does not support move-only Deleters, we do not need a
|
| - // separate move assignment operator allowing us to avoid one use of SFINAE.
|
| - // You only need to care about this if you modify the implementation of
|
| - // scoped_ptr.
|
| - template <typename U, typename V>
|
| - scoped_ptr& operator=(scoped_ptr<U, V>&& rhs) {
|
| - COMPILE_ASSERT(!base::is_array<U>::value, U_cannot_be_an_array);
|
| + // the normal move assignment operator. C++11 20.7.1.2.3.4-7 contains the
|
| + // requirement for this operator, but like the conversion constructor, the
|
| + // requirements are greatly simplified by not supporting move-only or
|
| + // reference deleters.
|
| + template <typename U,
|
| + typename E,
|
| + typename std::enable_if<!std::is_array<U>::value &&
|
| + std::is_convertible<U*, T*>::value &&
|
| + // Note that this really should be
|
| + // std::is_assignable, but <type_traits>
|
| + // appears to be missing this on some
|
| + // platforms. This is close enough (though
|
| + // it's not the same).
|
| + std::is_convertible<D, E>::value>::type* =
|
| + nullptr>
|
| + scoped_ptr& operator=(scoped_ptr<U, E>&& rhs) {
|
| impl_.TakeState(&rhs.impl_);
|
| return *this;
|
| }
|
| @@ -362,8 +406,8 @@ class scoped_ptr<T[], D> {
|
|
|
| public:
|
| // The element and deleter types.
|
| - typedef T element_type;
|
| - typedef D deleter_type;
|
| + using element_type = T;
|
| + using deleter_type = D;
|
|
|
| // Constructor. Defaults to initializing with nullptr.
|
| scoped_ptr() : impl_(nullptr) {}
|
|
|