mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-24 05:11:16 +00:00
680815cd6e
This patch was generated automatically by the "modeline.py" script, available here: https://github.com/amccreight/moz-source-tools/blob/master/modeline.py For every file that is modified in this patch, the changes are as follows: (1) The patch changes the file to use the exact C++ mode lines from the Mozilla coding style guide, available here: https://developer.mozilla.org/en-US/docs/Mozilla/Developer_guide/Coding_Style#Mode_Line (2) The patch deletes any blank lines between the mode line & the MPL boilerplate comment. (3) If the file previously had the mode lines and MPL boilerplate in a single contiguous C++ comment, then the patch splits them into separate C++ comments, to match the boilerplate in the coding style. MozReview-Commit-ID: EuRsDue63tK --HG-- extra : rebase_source : 3356d4b80ff6213935192e87cdbc9103fec6084c
163 lines
3.9 KiB
C++
163 lines
3.9 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 for strong references to nsPresArena-allocated objects
|
|
that might be held onto until the arena's destruction */
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/RefPtr.h"
|
|
|
|
#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)
|
|
{
|
|
RefPtr<T> newPtr(aSmartPtr);
|
|
assignFrom(newPtr);
|
|
}
|
|
|
|
template<typename I>
|
|
void assign(already_AddRefed<I>&& aSmartPtr)
|
|
{
|
|
RefPtr<T> newPtr(aSmartPtr);
|
|
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);
|
|
}
|
|
}
|
|
|
|
RefPtr<T> mPtr;
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif // mozilla_ArenaRefPtr_h
|