mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 05:37:11 +00:00
69d1f32ff7
- Bug 1268085 - Remove unused post barrier callbacks r=terrence (0ab13411c9) - Bug 1267699 - Move some public types to the right namespace; r=sfink (3d5008e610) - Bug 1267550 (part 1) - Rename MOZ_MUST_USE as MOZ_MUST_USE_TYPE. r=ehsan. (6f47375796) - Bug 1259021 - Rename Vector::extractRawBuffer to extractOrCopyRawBuffer r=Waldo (97ca94495b) - Bug 1259021 - Add Vector::extractRawBuffer method that doesn't copy the buffer r=Waldo (e58deec48f) - Bug 1265892 - Change Vector to use Impl::new_ consistently. r=Waldo (7a52d21b29) - Bug 1267912 - Rename nsNetUtil.inl as nsNetUtilInlines.h. r=valentin. (548a41b293) - Bug 1265690 part 1 - Mark StringBuffer methods WARN_UNUSED_RESULT, fix OOM issues. r=jonco (0d7e6837e3) - Bug 1265690 part 2 - Fix some more OOM issues in TypedObject code. r=jonco (b60902453e) - Bug 1263490 - Part 2: Add GetFirstDollarIndex intrinsic and use it inRegExpReplace. r=till (4ba19db8c4) - Bug 1263490 - Part 3: Inline GetFirstDollarIndex intrinsic. r=h4writer (e7d9b5d1cc) - Bug 1263490 - Part 4: Fold GetFirstDollarIndex into a integer constant. r=h4writer (3479c7d1af) - Bug 1267269 - Make MIRType an enum class. r=bbouvier (d580ef372a) - Bug 1259295 - BaldrMonkey: Postorder (r=luke) (6ef7a77663) - Bug 1254142: BaldrMonkey: make br_table yield (r=luke) (80e7635e58) - Bug 1263202 - BaldrMonkey: switch to arities on branches, calls and return (r=bbouvier) (f5a0358634) - Bug 1236358 - Improper reading of string16 in Pickle::ReadString16. r=jld (8370ba6a0b) - Bug 1263205 - BaldrMonkey: Update section headers for proposed spec changes (r=luke) (0def2e6bc2) - Bug 1263205 - BaldrMonkey: Update for proposed new section names (r=luke) (e57f0e3367) - Bug 1263205 - BaldrMonkey: Add 'form' field to types section (r=bbouvier) (794edc890f) - Bug 1259021 - Use in-place storage in AutoStableStringChars to avoid allocation for short strings r=jandem r=Waldo (ffb53cbcf4) - Bug 1267550 (part 2) - Rename MOZ_WARN_UNUSED_RESULT as MOZ_MUST_USE. r=froydnj. (47bc674b86) - Bug 1268518: Baldr: implement int32/int64 rotations; r=luke (0d5eedccce) - Bug 1255008: IonMonkey - Add a by default disabled flow sensitive alias analysis pass, r=jandem (521c585d75) - Bug 1266781: Baldr: implement proper checked truncations to integer types; r=sunfish (46078fb3d3) - Bug 1266781: Rename MTruncateToInt64 into MWasmTruncateInt64; r=sunfish (c7d7d1ac11) - Bug 1266781: Add new traps; r=luke (b7ed3d44e6) - Bug 1268024: Pass the atomic attribute down to EmitHeapAccess; r=luke (6195f7d7a3) - Bug 1268024: A few cleanups related to loads/stores; r=luke (88141e3a01) - Bug 1258312 - Make Pickle::Resize infallible r=jld (241ee9b60d) - Bug 1162772, part 1 - Allow CompartmentCreationOptions to store Secure Context state. r=jorendorff (ff666384cf) - Bug 1162772, part 2 - Expose whether SEC_FORCE_INHERIT_PRINCIPAL was dropped from an nsILoadInfo. r=bz (ada46f86bf) - Bug 1162772, part 3 - Add a getChannelResultPrincipalIfNotSandboxed method to nsIScriptSecurityManager. r=bz (5b1d9f6807) - Bug 1162772, part 4 - Implement nsGlobalWindow::IsSecureContext. r=bz (f392f439c9) - Bug 1162772, part 5 - Expose Window.isSecureContext to content. r=bz (e7296e2cf1) - Bug 1267509 - Make nsContentSecurityManager::IsURIPotentiallyTrustworthy act on an nsIPrincipal. r=bz (83de80350a) - Bug 1219098 - Use UniquePtr in UncompressedSourceCache, for it is good (r=jandem) (b68769c729) - Bug 1244279 - Part 1: Take a bit in ObjectElements::Flags to indicate whether the object is in the whole cell store buffer. r=terrence (968cf373f9) - Bug 1244279 - Part 0: Add a GC ubench for large arrays with both elements and properties. r=terrence (ec76b48323) - Bug 1255925 - Give a name to getters/setters and integer-named methods. r=efaust (f978cc6916) - Bug 888969 - Make the getPrototypeOf/setPrototypeOf traps scriptable. r=efaust, r=bholley (eb2325a9ea) - Bug 1267557 part 0 - Move JS poison constants to jsutil.h. r=jonco (65afc690d2) - Bug 1267557 part 1 - Also poison bytes allocated before the actual jitcode. r=nbp (70f0b327d3) - Bug 1267557 part 2 - Use different jitcode poison values. r=nbp (08008ab9dc) - Bug 1267557 part 3 - Define JS_SWEPT_CODE_PATTERN for mips. r=nbp (17e894d59d) - Bug 1267449 - Do not infinite loop in js_fputs; r=jimb (67f961b6cd) - Bug 1219098 - Reenable compression on large sources, but revert to uncompressed if decompression happens (r=jandem) (b44ee8d77d) - Bug 1267551 (part 1) - Use MOZ_MUST_USE more in jsnum.h. r=jonco. (d2476bf8f4) - Bug 1267551 (part 2) - Use MOZ_MUST_USE more in js/src/ds/. r=jonco. (4ff5d9aa88) - Bug 1267412 - Use MutableHandleValue instead of pointer-to-AutoValueVector; r=sfink (3f6dd284bb) - Bug 1266406 - Use EnumSet<AllocKind> to simplify GC sweeping phase information r=terrence (64811500e7) - Bug 1266457 - Update pointers in GC things in two phases when compacting r=terrence (f6f5bc4e4d) - Bug 1266457 - Simplify typed object trace hook r=terence (3b06c8d1e5) - Bug 1268541 - Compact arenas containing base shapes r=terrence (b458b92eea) - Bug 1268805 - Implement PrivateGCThingValue. (r=terrence) (deec9a83ae) - Bug 1268415: Initialize members in UpdatePointerTasks; r=jonco (6cb219005a) - Bug 1268501 - Release the GC lock periodically when releasing arenas on the backgound thread r=terrence (37f0997682) - Bug 1263572 - Wait for background sweeping to finish before checking base shapes r=terrence (354801a411) - Bug 1266887 - Store Rooted heads on the Zone; r=sfink (91c0101ee3) - Bug 1266402 - Add iteration to EnumSet<T> so that it can be used in range-based for loops r=Waldo (e9507a2524) - Bug 1266404 - Allow construction of an EnumSet<T> using an initializer list r=Waldo (1b6d340e99) - Bug 1254020 - Always compute theme scaling factor when per-monitor dpi aware, even if only a single display is currently present. r=emk (a00cda21f4) - Bug 1263525 - Add dedicated function for std_Array self-hosted intrinsic. r=efaust (449d8bb7eb) - Bug 1255925 - Change JSFunction::name to return a JSAtom. r=efaust (5ab396ce83) - Bug 888969 - Make our tree's sole implementation of nsIRemoteTagService.getRemoteObjectTag not depend upon the infallibility of [[GetPrototypeOf]] on the object provided to it. r=bz (f388f4bf1f) - Bug 1264896 - Kill off nsIRemoteTagService and do what it does, in its sole caller, in far-faster C++. r=billm (5ed3fb103d) - Bug 1268246 - Add a simple Poison class lifetime checker. r=froydnj (7b237bc70e) - Bug 1249496 - Don't apply dpi-based scaling for window titlebar dimensions when on a secondary display, because windows doesn't scale it. r=emk (64dd706dbc) - Bug 1164518 - Avoid unnecessary DB updates when caching Safe Browsing results. r=gcp (3cafd9a4df) - Bug 1264472 - Use nsRunnables in FIDO U2F. r=keeler (3aa9570132) - Bug 1236060 - Dispatch error should advance queue. r=smaug (74155b75dd) - Bug 1251697 part 1. Thread an ErrorResult reference through the worker XHR WorkerThreadProxySyncRunnable implementations. r=khuey (77804cbb7c) - Bug 1251697 part 2. Have WorkerThreadProxySyncRunnable hand the ErrorResult reference it holds to its ResponseRunnable so it can report exceptions on there instead of on a JSContext. r=khuey (355c9ee313) - Bug 1251697 part 3. Remove the JSContext argument of StopSyncLoopRunnable::MaybeSetException. r=khuey (010f5b1058) - Bug 1155328. r=smaug (e1f8dac304) - Bug 1265927: Move nsRunnable to mozilla::Runnable, CancelableRunnable to mozilla::CancelableRunnable. r=froydnj (f83bfcae02) - Bug 1239946 - Change test to return error on Speak. r=eeejay (1d402beb02) - Bug 1254378 - Update synth tests and introduce no voiceschanged test. r=smaug (f5823bb70e) - Bug 1251627. Fix XMLHttpRequest.send() to follow the spec better in terms of the exceptions it throws. r=khuey (cd0e321948) - Bug 1268868: [MSE] P1. Re-enable gap detection within a media segment. r=gerald (b8b8df4bc2) - Bug 1268868: [MSE] P2. Reset longest duration after keyframe is seen. r=gerald (2b1401465c) - Bug 1268868: [MSE] P3. Prevent crash should gap be detected in content. r=gerald (063d9376fc) - Bug 1254378 - Implement nsISynthVoiceRegistry.notifyVoicesChanged. r=smaug (4b63b1c360) - Bug 1266804 - Un-inline js::Unbox(); r=jorendorff (0f288b6173) - Bug 1268863 - Report ScriptSources that are only reachable via AsmJSModule (r=njn) (5ba40acb64) - bump version to 45.1b1 (1414db0ca8) - Bug 1262062 - remove old futex names. r=bbouvier (62662bdd2e) - memory: build fix after renaming MOZ_WARN_UNUSED_RESULT (7254dc8d53) - import from mozilla: - Bug 1268725 - BaldrMonkey: Refactor away the internal storage from ExprIter. r=luke (1931bd636f17) - Bug 1268725 - BaldrMonkey: Convert default arguments into explicit arguments. r=luke (c8a11b8b6bbd) (867ec715d6)
698 lines
23 KiB
C++
698 lines
23 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
/* Smart pointer managing sole ownership of a resource. */
|
|
|
|
#ifndef mozilla_UniquePtr_h
|
|
#define mozilla_UniquePtr_h
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/Attributes.h"
|
|
#include "mozilla/Compiler.h"
|
|
#include "mozilla/Move.h"
|
|
#include "mozilla/Pair.h"
|
|
#include "mozilla/TypeTraits.h"
|
|
|
|
namespace mozilla {
|
|
|
|
template<typename T> class DefaultDelete;
|
|
template<typename T, class D = DefaultDelete<T>> class UniquePtr;
|
|
|
|
} // namespace mozilla
|
|
|
|
namespace mozilla {
|
|
|
|
namespace detail {
|
|
|
|
struct HasPointerTypeHelper
|
|
{
|
|
template <class U> static double Test(...);
|
|
template <class U> static char Test(typename U::pointer* = 0);
|
|
};
|
|
|
|
template <class T>
|
|
class HasPointerType : public IntegralConstant<bool, sizeof(HasPointerTypeHelper::Test<T>(0)) == 1>
|
|
{
|
|
};
|
|
|
|
template <class T, class D, bool = HasPointerType<D>::value>
|
|
struct PointerTypeImpl
|
|
{
|
|
typedef typename D::pointer Type;
|
|
};
|
|
|
|
template <class T, class D>
|
|
struct PointerTypeImpl<T, D, false>
|
|
{
|
|
typedef T* Type;
|
|
};
|
|
|
|
template <class T, class D>
|
|
struct PointerType
|
|
{
|
|
typedef typename PointerTypeImpl<T, typename RemoveReference<D>::Type>::Type Type;
|
|
};
|
|
|
|
} // namespace detail
|
|
|
|
/**
|
|
* UniquePtr is a smart pointer that wholly owns a resource. Ownership may be
|
|
* transferred out of a UniquePtr through explicit action, but otherwise the
|
|
* resource is destroyed when the UniquePtr is destroyed.
|
|
*
|
|
* UniquePtr is similar to C++98's std::auto_ptr, but it improves upon auto_ptr
|
|
* in one crucial way: it's impossible to copy a UniquePtr. Copying an auto_ptr
|
|
* obviously *can't* copy ownership of its singly-owned resource. So what
|
|
* happens if you try to copy one? Bizarrely, ownership is implicitly
|
|
* *transferred*, preserving single ownership but breaking code that assumes a
|
|
* copy of an object is identical to the original. (This is why auto_ptr is
|
|
* prohibited in STL containers.)
|
|
*
|
|
* UniquePtr solves this problem by being *movable* rather than copyable.
|
|
* Instead of passing a |UniquePtr u| directly to the constructor or assignment
|
|
* operator, you pass |Move(u)|. In doing so you indicate that you're *moving*
|
|
* ownership out of |u|, into the target of the construction/assignment. After
|
|
* the transfer completes, |u| contains |nullptr| and may be safely destroyed.
|
|
* This preserves single ownership but also allows UniquePtr to be moved by
|
|
* algorithms that have been made move-safe. (Note: if |u| is instead a
|
|
* temporary expression, don't use |Move()|: just pass the expression, because
|
|
* it's already move-ready. For more information see Move.h.)
|
|
*
|
|
* UniquePtr is also better than std::auto_ptr in that the deletion operation is
|
|
* customizable. An optional second template parameter specifies a class that
|
|
* (through its operator()(T*)) implements the desired deletion policy. If no
|
|
* policy is specified, mozilla::DefaultDelete<T> is used -- which will either
|
|
* |delete| or |delete[]| the resource, depending whether the resource is an
|
|
* array. Custom deletion policies ideally should be empty classes (no member
|
|
* fields, no member fields in base classes, no virtual methods/inheritance),
|
|
* because then UniquePtr can be just as efficient as a raw pointer.
|
|
*
|
|
* Use of UniquePtr proceeds like so:
|
|
*
|
|
* UniquePtr<int> g1; // initializes to nullptr
|
|
* g1.reset(new int); // switch resources using reset()
|
|
* g1 = nullptr; // clears g1, deletes the int
|
|
*
|
|
* UniquePtr<int> g2(new int); // owns that int
|
|
* int* p = g2.release(); // g2 leaks its int -- still requires deletion
|
|
* delete p; // now freed
|
|
*
|
|
* struct S { int x; S(int x) : x(x) {} };
|
|
* UniquePtr<S> g3, g4(new S(5));
|
|
* g3 = Move(g4); // g3 owns the S, g4 cleared
|
|
* S* p = g3.get(); // g3 still owns |p|
|
|
* assert(g3->x == 5); // operator-> works (if .get() != nullptr)
|
|
* assert((*g3).x == 5); // also operator* (again, if not cleared)
|
|
* Swap(g3, g4); // g4 now owns the S, g3 cleared
|
|
* g3.swap(g4); // g3 now owns the S, g4 cleared
|
|
* UniquePtr<S> g5(Move(g3)); // g5 owns the S, g3 cleared
|
|
* g5.reset(); // deletes the S, g5 cleared
|
|
*
|
|
* struct FreePolicy { void operator()(void* p) { free(p); } };
|
|
* UniquePtr<int, FreePolicy> g6(static_cast<int*>(malloc(sizeof(int))));
|
|
* int* ptr = g6.get();
|
|
* g6 = nullptr; // calls free(ptr)
|
|
*
|
|
* Now, carefully note a few things you *can't* do:
|
|
*
|
|
* UniquePtr<int> b1;
|
|
* b1 = new int; // BAD: can only assign another UniquePtr
|
|
* int* ptr = b1; // BAD: no auto-conversion to pointer, use get()
|
|
*
|
|
* UniquePtr<int> b2(b1); // BAD: can't copy a UniquePtr
|
|
* UniquePtr<int> b3 = b1; // BAD: can't copy-assign a UniquePtr
|
|
*
|
|
* (Note that changing a UniquePtr to store a direct |new| expression is
|
|
* permitted, but usually you should use MakeUnique, defined at the end of this
|
|
* header.)
|
|
*
|
|
* A few miscellaneous notes:
|
|
*
|
|
* UniquePtr, when not instantiated for an array type, can be move-constructed
|
|
* and move-assigned, not only from itself but from "derived" UniquePtr<U, E>
|
|
* instantiations where U converts to T and E converts to D. If you want to use
|
|
* this, you're going to have to specify a deletion policy for both UniquePtr
|
|
* instantations, and T pretty much has to have a virtual destructor. In other
|
|
* words, this doesn't work:
|
|
*
|
|
* struct Base { virtual ~Base() {} };
|
|
* struct Derived : Base {};
|
|
*
|
|
* UniquePtr<Base> b1;
|
|
* // BAD: DefaultDelete<Base> and DefaultDelete<Derived> don't interconvert
|
|
* UniquePtr<Derived> d1(Move(b));
|
|
*
|
|
* UniquePtr<Base> b2;
|
|
* UniquePtr<Derived, DefaultDelete<Base>> d2(Move(b2)); // okay
|
|
*
|
|
* UniquePtr is specialized for array types. Specializing with an array type
|
|
* creates a smart-pointer version of that array -- not a pointer to such an
|
|
* array.
|
|
*
|
|
* UniquePtr<int[]> arr(new int[5]);
|
|
* arr[0] = 4;
|
|
*
|
|
* What else is different? Deletion of course uses |delete[]|. An operator[]
|
|
* is provided. Functionality that doesn't make sense for arrays is removed.
|
|
* The constructors and mutating methods only accept array pointers (not T*, U*
|
|
* that converts to T*, or UniquePtr<U[]> or UniquePtr<U>) or |nullptr|.
|
|
*
|
|
* It's perfectly okay for a function to return a UniquePtr. This transfers
|
|
* the UniquePtr's sole ownership of the data, to the fresh UniquePtr created
|
|
* in the calling function, that will then solely own that data. Such functions
|
|
* can return a local variable UniquePtr, |nullptr|, |UniquePtr(ptr)| where
|
|
* |ptr| is a |T*|, or a UniquePtr |Move()|'d from elsewhere.
|
|
*
|
|
* UniquePtr will commonly be a member of a class, with lifetime equivalent to
|
|
* that of that class. If you want to expose the related resource, you could
|
|
* expose a raw pointer via |get()|, but ownership of a raw pointer is
|
|
* inherently unclear. So it's better to expose a |const UniquePtr&| instead.
|
|
* This prohibits mutation but still allows use of |get()| when needed (but
|
|
* operator-> is preferred). Of course, you can only use this smart pointer as
|
|
* long as the enclosing class instance remains live -- no different than if you
|
|
* exposed the |get()| raw pointer.
|
|
*
|
|
* To pass a UniquePtr-managed resource as a pointer, use a |const UniquePtr&|
|
|
* argument. To specify an inout parameter (where the method may or may not
|
|
* take ownership of the resource, or reset it), or to specify an out parameter
|
|
* (where simply returning a |UniquePtr| isn't possible), use a |UniquePtr&|
|
|
* argument. To unconditionally transfer ownership of a UniquePtr
|
|
* into a method, use a |UniquePtr| argument. To conditionally transfer
|
|
* ownership of a resource into a method, should the method want it, use a
|
|
* |UniquePtr&&| argument.
|
|
*/
|
|
template<typename T, class D>
|
|
class UniquePtr
|
|
{
|
|
public:
|
|
typedef T ElementType;
|
|
typedef D DeleterType;
|
|
typedef typename detail::PointerType<T, DeleterType>::Type Pointer;
|
|
|
|
private:
|
|
Pair<Pointer, DeleterType> mTuple;
|
|
|
|
Pointer& ptr() { return mTuple.first(); }
|
|
const Pointer& ptr() const { return mTuple.first(); }
|
|
|
|
DeleterType& del() { return mTuple.second(); }
|
|
const DeleterType& del() const { return mTuple.second(); }
|
|
|
|
public:
|
|
/**
|
|
* Construct a UniquePtr containing |nullptr|.
|
|
*/
|
|
MOZ_CONSTEXPR UniquePtr()
|
|
: mTuple(static_cast<Pointer>(nullptr), DeleterType())
|
|
{
|
|
static_assert(!IsPointer<D>::value, "must provide a deleter instance");
|
|
static_assert(!IsReference<D>::value, "must provide a deleter instance");
|
|
}
|
|
|
|
/**
|
|
* Construct a UniquePtr containing |aPtr|.
|
|
*/
|
|
explicit UniquePtr(Pointer aPtr)
|
|
: mTuple(aPtr, DeleterType())
|
|
{
|
|
static_assert(!IsPointer<D>::value, "must provide a deleter instance");
|
|
static_assert(!IsReference<D>::value, "must provide a deleter instance");
|
|
}
|
|
|
|
UniquePtr(Pointer aPtr,
|
|
typename Conditional<IsReference<D>::value,
|
|
D,
|
|
const D&>::Type aD1)
|
|
: mTuple(aPtr, aD1)
|
|
{}
|
|
|
|
// If you encounter an error with MSVC10 about RemoveReference below, along
|
|
// the lines that "more than one partial specialization matches the template
|
|
// argument list": don't use UniquePtr<T, reference to function>! Ideally
|
|
// you should make deletion use the same function every time, using a
|
|
// deleter policy:
|
|
//
|
|
// // BAD, won't compile with MSVC10, deleter doesn't need to be a
|
|
// // variable at all
|
|
// typedef void (&FreeSignature)(void*);
|
|
// UniquePtr<int, FreeSignature> ptr((int*) malloc(sizeof(int)), free);
|
|
//
|
|
// // GOOD, compiles with MSVC10, deletion behavior statically known and
|
|
// // optimizable
|
|
// struct DeleteByFreeing
|
|
// {
|
|
// void operator()(void* aPtr) { free(aPtr); }
|
|
// };
|
|
//
|
|
// If deletion really, truly, must be a variable: you might be able to work
|
|
// around this with a deleter class that contains the function reference.
|
|
// But this workaround is untried and untested, because variable deletion
|
|
// behavior really isn't something you should use.
|
|
UniquePtr(Pointer aPtr,
|
|
typename RemoveReference<D>::Type&& aD2)
|
|
: mTuple(aPtr, Move(aD2))
|
|
{
|
|
static_assert(!IsReference<D>::value,
|
|
"rvalue deleter can't be stored by reference");
|
|
}
|
|
|
|
UniquePtr(UniquePtr&& aOther)
|
|
: mTuple(aOther.release(), Forward<DeleterType>(aOther.get_deleter()))
|
|
{}
|
|
|
|
MOZ_IMPLICIT
|
|
UniquePtr(decltype(nullptr))
|
|
: mTuple(nullptr, DeleterType())
|
|
{
|
|
static_assert(!IsPointer<D>::value, "must provide a deleter instance");
|
|
static_assert(!IsReference<D>::value, "must provide a deleter instance");
|
|
}
|
|
|
|
template<typename U, class E>
|
|
MOZ_IMPLICIT
|
|
UniquePtr(UniquePtr<U, E>&& aOther,
|
|
typename EnableIf<IsConvertible<typename UniquePtr<U, E>::Pointer,
|
|
Pointer>::value &&
|
|
!IsArray<U>::value &&
|
|
(IsReference<D>::value
|
|
? IsSame<D, E>::value
|
|
: IsConvertible<E, D>::value),
|
|
int>::Type aDummy = 0)
|
|
: mTuple(aOther.release(), Forward<E>(aOther.get_deleter()))
|
|
{
|
|
}
|
|
|
|
~UniquePtr() { reset(nullptr); }
|
|
|
|
UniquePtr& operator=(UniquePtr&& aOther)
|
|
{
|
|
reset(aOther.release());
|
|
get_deleter() = Forward<DeleterType>(aOther.get_deleter());
|
|
return *this;
|
|
}
|
|
|
|
template<typename U, typename E>
|
|
UniquePtr& operator=(UniquePtr<U, E>&& aOther)
|
|
{
|
|
static_assert(IsConvertible<typename UniquePtr<U, E>::Pointer,
|
|
Pointer>::value,
|
|
"incompatible UniquePtr pointees");
|
|
static_assert(!IsArray<U>::value,
|
|
"can't assign from UniquePtr holding an array");
|
|
|
|
reset(aOther.release());
|
|
get_deleter() = Forward<E>(aOther.get_deleter());
|
|
return *this;
|
|
}
|
|
|
|
UniquePtr& operator=(decltype(nullptr))
|
|
{
|
|
reset(nullptr);
|
|
return *this;
|
|
}
|
|
|
|
T& operator*() const { return *get(); }
|
|
Pointer operator->() const
|
|
{
|
|
MOZ_ASSERT(get(), "dereferencing a UniquePtr containing nullptr");
|
|
return get();
|
|
}
|
|
|
|
explicit operator bool() const { return get() != nullptr; }
|
|
|
|
Pointer get() const { return ptr(); }
|
|
|
|
DeleterType& get_deleter() { return del(); }
|
|
const DeleterType& get_deleter() const { return del(); }
|
|
|
|
MOZ_MUST_USE Pointer release()
|
|
{
|
|
Pointer p = ptr();
|
|
ptr() = nullptr;
|
|
return p;
|
|
}
|
|
|
|
void reset(Pointer aPtr = Pointer())
|
|
{
|
|
Pointer old = ptr();
|
|
ptr() = aPtr;
|
|
if (old != nullptr) {
|
|
get_deleter()(old);
|
|
}
|
|
}
|
|
|
|
void swap(UniquePtr& aOther)
|
|
{
|
|
mTuple.swap(aOther.mTuple);
|
|
}
|
|
|
|
UniquePtr(const UniquePtr& aOther) = delete; // construct using Move()!
|
|
void operator=(const UniquePtr& aOther) = delete; // assign using Move()!
|
|
};
|
|
|
|
// In case you didn't read the comment by the main definition (you should!): the
|
|
// UniquePtr<T[]> specialization exists to manage array pointers. It deletes
|
|
// such pointers using delete[], it will reject construction and modification
|
|
// attempts using U* or U[]. Otherwise it works like the normal UniquePtr.
|
|
template<typename T, class D>
|
|
class UniquePtr<T[], D>
|
|
{
|
|
public:
|
|
typedef T* Pointer;
|
|
typedef T ElementType;
|
|
typedef D DeleterType;
|
|
|
|
private:
|
|
Pair<Pointer, DeleterType> mTuple;
|
|
|
|
public:
|
|
/**
|
|
* Construct a UniquePtr containing nullptr.
|
|
*/
|
|
MOZ_CONSTEXPR UniquePtr()
|
|
: mTuple(static_cast<Pointer>(nullptr), DeleterType())
|
|
{
|
|
static_assert(!IsPointer<D>::value, "must provide a deleter instance");
|
|
static_assert(!IsReference<D>::value, "must provide a deleter instance");
|
|
}
|
|
|
|
/**
|
|
* Construct a UniquePtr containing |aPtr|.
|
|
*/
|
|
explicit UniquePtr(Pointer aPtr)
|
|
: mTuple(aPtr, DeleterType())
|
|
{
|
|
static_assert(!IsPointer<D>::value, "must provide a deleter instance");
|
|
static_assert(!IsReference<D>::value, "must provide a deleter instance");
|
|
}
|
|
|
|
// delete[] knows how to handle *only* an array of a single class type. For
|
|
// delete[] to work correctly, it must know the size of each element, the
|
|
// fields and base classes of each element requiring destruction, and so on.
|
|
// So forbid all overloads which would end up invoking delete[] on a pointer
|
|
// of the wrong type.
|
|
template<typename U>
|
|
UniquePtr(U&& aU,
|
|
typename EnableIf<IsPointer<U>::value &&
|
|
IsConvertible<U, Pointer>::value,
|
|
int>::Type aDummy = 0)
|
|
= delete;
|
|
|
|
UniquePtr(Pointer aPtr,
|
|
typename Conditional<IsReference<D>::value,
|
|
D,
|
|
const D&>::Type aD1)
|
|
: mTuple(aPtr, aD1)
|
|
{}
|
|
|
|
// If you encounter an error with MSVC10 about RemoveReference below, along
|
|
// the lines that "more than one partial specialization matches the template
|
|
// argument list": don't use UniquePtr<T[], reference to function>! See the
|
|
// comment by this constructor in the non-T[] specialization above.
|
|
UniquePtr(Pointer aPtr,
|
|
typename RemoveReference<D>::Type&& aD2)
|
|
: mTuple(aPtr, Move(aD2))
|
|
{
|
|
static_assert(!IsReference<D>::value,
|
|
"rvalue deleter can't be stored by reference");
|
|
}
|
|
|
|
// Forbidden for the same reasons as stated above.
|
|
template<typename U, typename V>
|
|
UniquePtr(U&& aU, V&& aV,
|
|
typename EnableIf<IsPointer<U>::value &&
|
|
IsConvertible<U, Pointer>::value,
|
|
int>::Type aDummy = 0)
|
|
= delete;
|
|
|
|
UniquePtr(UniquePtr&& aOther)
|
|
: mTuple(aOther.release(), Forward<DeleterType>(aOther.get_deleter()))
|
|
{}
|
|
|
|
MOZ_IMPLICIT
|
|
UniquePtr(decltype(nullptr))
|
|
: mTuple(nullptr, DeleterType())
|
|
{
|
|
static_assert(!IsPointer<D>::value, "must provide a deleter instance");
|
|
static_assert(!IsReference<D>::value, "must provide a deleter instance");
|
|
}
|
|
|
|
~UniquePtr() { reset(nullptr); }
|
|
|
|
UniquePtr& operator=(UniquePtr&& aOther)
|
|
{
|
|
reset(aOther.release());
|
|
get_deleter() = Forward<DeleterType>(aOther.get_deleter());
|
|
return *this;
|
|
}
|
|
|
|
UniquePtr& operator=(decltype(nullptr))
|
|
{
|
|
reset();
|
|
return *this;
|
|
}
|
|
|
|
explicit operator bool() const { return get() != nullptr; }
|
|
|
|
T& operator[](decltype(sizeof(int)) aIndex) const { return get()[aIndex]; }
|
|
Pointer get() const { return mTuple.first(); }
|
|
|
|
DeleterType& get_deleter() { return mTuple.second(); }
|
|
const DeleterType& get_deleter() const { return mTuple.second(); }
|
|
|
|
MOZ_MUST_USE Pointer release()
|
|
{
|
|
Pointer p = mTuple.first();
|
|
mTuple.first() = nullptr;
|
|
return p;
|
|
}
|
|
|
|
void reset(Pointer aPtr = Pointer())
|
|
{
|
|
Pointer old = mTuple.first();
|
|
mTuple.first() = aPtr;
|
|
if (old != nullptr) {
|
|
mTuple.second()(old);
|
|
}
|
|
}
|
|
|
|
void reset(decltype(nullptr))
|
|
{
|
|
Pointer old = mTuple.first();
|
|
mTuple.first() = nullptr;
|
|
if (old != nullptr) {
|
|
mTuple.second()(old);
|
|
}
|
|
}
|
|
|
|
template<typename U>
|
|
void reset(U) = delete;
|
|
|
|
void swap(UniquePtr& aOther) { mTuple.swap(aOther.mTuple); }
|
|
|
|
UniquePtr(const UniquePtr& aOther) = delete; // construct using Move()!
|
|
void operator=(const UniquePtr& aOther) = delete; // assign using Move()!
|
|
};
|
|
|
|
/**
|
|
* A default deletion policy using plain old operator delete.
|
|
*
|
|
* Note that this type can be specialized, but authors should beware of the risk
|
|
* that the specialization may at some point cease to match (either because it
|
|
* gets moved to a different compilation unit or the signature changes). If the
|
|
* non-specialized (|delete|-based) version compiles for that type but does the
|
|
* wrong thing, bad things could happen.
|
|
*
|
|
* This is a non-issue for types which are always incomplete (i.e. opaque handle
|
|
* types), since |delete|-ing such a type will always trigger a compilation
|
|
* error.
|
|
*/
|
|
template<typename T>
|
|
class DefaultDelete
|
|
{
|
|
public:
|
|
MOZ_CONSTEXPR DefaultDelete() {}
|
|
|
|
template<typename U>
|
|
MOZ_IMPLICIT DefaultDelete(const DefaultDelete<U>& aOther,
|
|
typename EnableIf<mozilla::IsConvertible<U*, T*>::value,
|
|
int>::Type aDummy = 0)
|
|
{}
|
|
|
|
void operator()(T* aPtr) const
|
|
{
|
|
static_assert(sizeof(T) > 0, "T must be complete");
|
|
delete aPtr;
|
|
}
|
|
};
|
|
|
|
/** A default deletion policy using operator delete[]. */
|
|
template<typename T>
|
|
class DefaultDelete<T[]>
|
|
{
|
|
public:
|
|
MOZ_CONSTEXPR DefaultDelete() {}
|
|
|
|
void operator()(T* aPtr) const
|
|
{
|
|
static_assert(sizeof(T) > 0, "T must be complete");
|
|
delete[] aPtr;
|
|
}
|
|
|
|
template<typename U>
|
|
void operator()(U* aPtr) const = delete;
|
|
};
|
|
|
|
template<typename T, class D>
|
|
void
|
|
Swap(UniquePtr<T, D>& aX, UniquePtr<T, D>& aY)
|
|
{
|
|
aX.swap(aY);
|
|
}
|
|
|
|
template<typename T, class D, typename U, class E>
|
|
bool
|
|
operator==(const UniquePtr<T, D>& aX, const UniquePtr<U, E>& aY)
|
|
{
|
|
return aX.get() == aY.get();
|
|
}
|
|
|
|
template<typename T, class D, typename U, class E>
|
|
bool
|
|
operator!=(const UniquePtr<T, D>& aX, const UniquePtr<U, E>& aY)
|
|
{
|
|
return aX.get() != aY.get();
|
|
}
|
|
|
|
template<typename T, class D>
|
|
bool
|
|
operator==(const UniquePtr<T, D>& aX, decltype(nullptr))
|
|
{
|
|
return !aX;
|
|
}
|
|
|
|
template<typename T, class D>
|
|
bool
|
|
operator==(decltype(nullptr), const UniquePtr<T, D>& aX)
|
|
{
|
|
return !aX;
|
|
}
|
|
|
|
template<typename T, class D>
|
|
bool
|
|
operator!=(const UniquePtr<T, D>& aX, decltype(nullptr))
|
|
{
|
|
return bool(aX);
|
|
}
|
|
|
|
template<typename T, class D>
|
|
bool
|
|
operator!=(decltype(nullptr), const UniquePtr<T, D>& aX)
|
|
{
|
|
return bool(aX);
|
|
}
|
|
|
|
// No operator<, operator>, operator<=, operator>= for now because simplicity.
|
|
|
|
namespace detail {
|
|
|
|
template<typename T>
|
|
struct UniqueSelector
|
|
{
|
|
typedef UniquePtr<T> SingleObject;
|
|
};
|
|
|
|
template<typename T>
|
|
struct UniqueSelector<T[]>
|
|
{
|
|
typedef UniquePtr<T[]> UnknownBound;
|
|
};
|
|
|
|
template<typename T, decltype(sizeof(int)) N>
|
|
struct UniqueSelector<T[N]>
|
|
{
|
|
typedef UniquePtr<T[N]> KnownBound;
|
|
};
|
|
|
|
} // namespace detail
|
|
|
|
/**
|
|
* MakeUnique is a helper function for allocating new'd objects and arrays,
|
|
* returning a UniquePtr containing the resulting pointer. The semantics of
|
|
* MakeUnique<Type>(...) are as follows.
|
|
*
|
|
* If Type is an array T[n]:
|
|
* Disallowed, deleted, no overload for you!
|
|
* If Type is an array T[]:
|
|
* MakeUnique<T[]>(size_t) is the only valid overload. The pointer returned
|
|
* is as if by |new T[n]()|, which value-initializes each element. (If T
|
|
* isn't a class type, this will zero each element. If T is a class type,
|
|
* then roughly speaking, each element will be constructed using its default
|
|
* constructor. See C++11 [dcl.init]p7 for the full gory details.)
|
|
* If Type is non-array T:
|
|
* The arguments passed to MakeUnique<T>(...) are forwarded into a
|
|
* |new T(...)| call, initializing the T as would happen if executing
|
|
* |T(...)|.
|
|
*
|
|
* There are various benefits to using MakeUnique instead of |new| expressions.
|
|
*
|
|
* First, MakeUnique eliminates use of |new| from code entirely. If objects are
|
|
* only created through UniquePtr, then (assuming all explicit release() calls
|
|
* are safe, including transitively, and no type-safety casting funniness)
|
|
* correctly maintained ownership of the UniquePtr guarantees no leaks are
|
|
* possible. (This pays off best if a class is only ever created through a
|
|
* factory method on the class, using a private constructor.)
|
|
*
|
|
* Second, initializing a UniquePtr using a |new| expression requires repeating
|
|
* the name of the new'd type, whereas MakeUnique in concert with the |auto|
|
|
* keyword names it only once:
|
|
*
|
|
* UniquePtr<char> ptr1(new char()); // repetitive
|
|
* auto ptr2 = MakeUnique<char>(); // shorter
|
|
*
|
|
* Of course this assumes the reader understands the operation MakeUnique
|
|
* performs. In the long run this is probably a reasonable assumption. In the
|
|
* short run you'll have to use your judgment about what readers can be expected
|
|
* to know, or to quickly look up.
|
|
*
|
|
* Third, a call to MakeUnique can be assigned directly to a UniquePtr. In
|
|
* contrast you can't assign a pointer into a UniquePtr without using the
|
|
* cumbersome reset().
|
|
*
|
|
* UniquePtr<char> p;
|
|
* p = new char; // ERROR
|
|
* p.reset(new char); // works, but fugly
|
|
* p = MakeUnique<char>(); // preferred
|
|
*
|
|
* (And third, although not relevant to Mozilla: MakeUnique is exception-safe.
|
|
* An exception thrown after |new T| succeeds will leak that memory, unless the
|
|
* pointer is assigned to an object that will manage its ownership. UniquePtr
|
|
* ably serves this function.)
|
|
*/
|
|
|
|
template<typename T, typename... Args>
|
|
typename detail::UniqueSelector<T>::SingleObject
|
|
MakeUnique(Args&&... aArgs)
|
|
{
|
|
return UniquePtr<T>(new T(Forward<Args>(aArgs)...));
|
|
}
|
|
|
|
template<typename T>
|
|
typename detail::UniqueSelector<T>::UnknownBound
|
|
MakeUnique(decltype(sizeof(int)) aN)
|
|
{
|
|
typedef typename RemoveExtent<T>::Type ArrayType;
|
|
return UniquePtr<T>(new ArrayType[aN]());
|
|
}
|
|
|
|
template<typename T, typename... Args>
|
|
typename detail::UniqueSelector<T>::KnownBound
|
|
MakeUnique(Args&&... aArgs) = delete;
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif /* mozilla_UniquePtr_h */
|