2015-09-17 05:08:19 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
|
|
* vim: set ts=2 sw=2 et tw=78:
|
|
|
|
* 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 for strong references to nsPresArena-allocated objects
|
|
|
|
that might be held onto until the arena's destruction */
|
|
|
|
|
|
|
|
#include "mozilla/Assertions.h"
|
2015-10-18 08:24:48 +03:00
|
|
|
#include "mozilla/RefPtr.h"
|
2015-09-17 05:08:19 +03:00
|
|
|
|
|
|
|
#ifndef mozilla_ArenaRefPtr_h
|
|
|
|
#define mozilla_ArenaRefPtr_h
|
|
|
|
|
|
|
|
class nsPresArena;
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A class for holding strong references to nsPresArena-allocated
|
|
|
|
* objects.
|
|
|
|
*
|
|
|
|
* Since the arena's lifetime is not related to the refcounts
|
|
|
|
* of the objects allocated within it, it is possible to have a strong
|
|
|
|
* reference to an arena-allocated object that lives until the
|
|
|
|
* destruction of the arena. An ArenaRefPtr acts like a weak reference
|
|
|
|
* in that it will clear its referent if the arena is about to go away.
|
|
|
|
*
|
|
|
|
* T must be a class that has these two methods:
|
|
|
|
*
|
|
|
|
* static mozilla::ArenaObjectID ArenaObjectID();
|
|
|
|
* U* Arena();
|
|
|
|
*
|
|
|
|
* where U is a class that has these two methods:
|
|
|
|
*
|
|
|
|
* void RegisterArenaRefPtr(ArenaRefPtr<T>*);
|
|
|
|
* void DeregisterArenaRefPtr(ArenaRefPtr<T>*);
|
|
|
|
*
|
|
|
|
* Currently, both nsPresArena and nsIPresShell can be used as U.
|
|
|
|
*
|
|
|
|
* The ArenaObjectID method must return the mozilla::ArenaObjectID that
|
|
|
|
* uniquely identifies T, and the Arena method must return the nsPresArena
|
|
|
|
* (or a proxy for it) in which the object was allocated.
|
|
|
|
*/
|
|
|
|
template<typename T>
|
|
|
|
class ArenaRefPtr
|
|
|
|
{
|
|
|
|
friend class ::nsPresArena;
|
|
|
|
|
|
|
|
public:
|
|
|
|
ArenaRefPtr()
|
|
|
|
{
|
|
|
|
AssertValidType();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename I>
|
|
|
|
MOZ_IMPLICIT ArenaRefPtr(already_AddRefed<I>& aRhs)
|
|
|
|
{
|
|
|
|
AssertValidType();
|
|
|
|
assign(aRhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename I>
|
|
|
|
MOZ_IMPLICIT ArenaRefPtr(already_AddRefed<I>&& aRhs)
|
|
|
|
{
|
|
|
|
AssertValidType();
|
|
|
|
assign(aRhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
MOZ_IMPLICIT ArenaRefPtr(T* aRhs)
|
|
|
|
{
|
|
|
|
AssertValidType();
|
|
|
|
assign(aRhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename I>
|
|
|
|
ArenaRefPtr<T>& operator=(already_AddRefed<I>& aRhs)
|
|
|
|
{
|
|
|
|
assign(aRhs);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename I>
|
|
|
|
ArenaRefPtr<T>& operator=(already_AddRefed<I>&& aRhs)
|
|
|
|
{
|
|
|
|
assign(aRhs);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
ArenaRefPtr<T>& operator=(T* aRhs)
|
|
|
|
{
|
|
|
|
assign(aRhs);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
~ArenaRefPtr() { assign(nullptr); }
|
|
|
|
|
|
|
|
operator T*() const & { return get(); }
|
|
|
|
operator T*() const && = delete;
|
|
|
|
explicit operator bool() const { return !!mPtr; }
|
|
|
|
bool operator!() const { return !mPtr; }
|
|
|
|
T* operator->() const { return mPtr.operator->(); }
|
|
|
|
T& operator*() const { return *get(); }
|
|
|
|
|
|
|
|
T* get() const { return mPtr; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
void AssertValidType();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clears the pointer to the arena-allocated object but skips the usual
|
|
|
|
* step of deregistering the ArenaRefPtr from the nsPresArena. This
|
|
|
|
* method is called by nsPresArena when clearing all registered ArenaRefPtrs
|
|
|
|
* so that it can deregister them all at once, avoiding hash table churn.
|
|
|
|
*/
|
|
|
|
void ClearWithoutDeregistering()
|
|
|
|
{
|
|
|
|
mPtr = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename I>
|
|
|
|
void assign(already_AddRefed<I>& aSmartPtr)
|
|
|
|
{
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<T> newPtr(aSmartPtr);
|
2015-09-17 05:08:19 +03:00
|
|
|
assignFrom(newPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename I>
|
|
|
|
void assign(already_AddRefed<I>&& aSmartPtr)
|
|
|
|
{
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<T> newPtr(aSmartPtr);
|
2015-09-17 05:08:19 +03:00
|
|
|
assignFrom(newPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void assign(T* aPtr) { assignFrom(aPtr); }
|
|
|
|
|
|
|
|
template<typename I>
|
|
|
|
void assignFrom(I& aPtr)
|
|
|
|
{
|
|
|
|
if (aPtr == mPtr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
bool sameArena = mPtr && aPtr && mPtr->Arena() == aPtr->Arena();
|
|
|
|
if (mPtr && !sameArena) {
|
|
|
|
MOZ_ASSERT(mPtr->Arena());
|
|
|
|
mPtr->Arena()->DeregisterArenaRefPtr(this);
|
|
|
|
}
|
|
|
|
mPtr = Move(aPtr);
|
|
|
|
if (mPtr && !sameArena) {
|
|
|
|
MOZ_ASSERT(mPtr->Arena());
|
|
|
|
mPtr->Arena()->RegisterArenaRefPtr(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-18 08:24:48 +03:00
|
|
|
RefPtr<T> mPtr;
|
2015-09-17 05:08:19 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace mozilla
|
|
|
|
|
|
|
|
#endif // mozilla_ArenaRefPtr_h
|