/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* 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/. */ #ifndef nsMaybeWeakPtr_h_ #define nsMaybeWeakPtr_h_ #include "mozilla/Attributes.h" #include "nsCOMPtr.h" #include "nsWeakReference.h" #include "nsTArray.h" #include "nsCycleCollectionNoteChild.h" // nsMaybeWeakPtr is a helper object to hold a strong-or-weak reference // to the template class. It's pretty minimal, but sufficient. template class nsMaybeWeakPtr { public: MOZ_IMPLICIT nsMaybeWeakPtr(nsISupports* aRef) : mPtr(aRef) {} MOZ_IMPLICIT nsMaybeWeakPtr(const nsCOMPtr& aRef) : mPtr(aRef) {} MOZ_IMPLICIT nsMaybeWeakPtr(const nsCOMPtr& aRef) : mPtr(aRef) {} bool operator==(const nsMaybeWeakPtr &other) const { return mPtr == other.mPtr; } nsISupports* GetRawValue() const { return mPtr.get(); } const nsCOMPtr GetValue() const; private: nsCOMPtr mPtr; }; // nsMaybeWeakPtrArray is an array of MaybeWeakPtr objects, that knows how to // grab a weak reference to a given object if requested. It only allows a // given object to appear in the array once. template class nsMaybeWeakPtrArray : public nsTArray> { typedef nsTArray> MaybeWeakArray; public: nsresult AppendWeakElement(T* aElement, bool aOwnsWeak) { nsCOMPtr ref; if (aOwnsWeak) { ref = do_GetWeakReference(aElement); } else { ref = aElement; } if (MaybeWeakArray::Contains(ref.get())) { return NS_ERROR_INVALID_ARG; } if (!MaybeWeakArray::AppendElement(ref)) { return NS_ERROR_OUT_OF_MEMORY; } return NS_OK; } nsresult RemoveWeakElement(T* aElement) { if (MaybeWeakArray::RemoveElement(aElement)) { return NS_OK; } // Don't use do_GetWeakReference; it should only be called if we know // the object supports weak references. nsCOMPtr supWeakRef = do_QueryInterface(aElement); NS_ENSURE_TRUE(supWeakRef, NS_ERROR_INVALID_ARG); nsCOMPtr weakRef; nsresult rv = supWeakRef->GetWeakReference(getter_AddRefs(weakRef)); NS_ENSURE_SUCCESS(rv, rv); if (MaybeWeakArray::RemoveElement(weakRef)) { return NS_OK; } return NS_ERROR_INVALID_ARG; } }; template const nsCOMPtr nsMaybeWeakPtr::GetValue() const { if (!mPtr) { return nullptr; } nsresult rv; nsCOMPtr ref = do_QueryInterface(mPtr, &rv); if (NS_SUCCEEDED(rv)) { return ref; } nsCOMPtr weakRef = do_QueryInterface(mPtr); if (weakRef) { ref = do_QueryReferent(weakRef, &rv); if (NS_SUCCEEDED(rv)) { return ref; } } return nullptr; } template inline void ImplCycleCollectionUnlink(nsMaybeWeakPtrArray& aField) { aField.Clear(); } template inline void ImplCycleCollectionTraverse(nsCycleCollectionTraversalCallback& aCallback, nsMaybeWeakPtrArray& aField, const char* aName, uint32_t aFlags = 0) { aFlags |= CycleCollectionEdgeNameArrayFlag; size_t length = aField.Length(); for (size_t i = 0; i < length; ++i) { CycleCollectionNoteChild(aCallback, aField[i].GetRawValue(), aName, aFlags); } } // Call a method on each element in the array, but only if the element is // non-null. #define ENUMERATE_WEAKARRAY(array, type, method) \ for (uint32_t array_idx = 0; array_idx < array.Length(); ++array_idx) { \ const nsCOMPtr &e = array.ElementAt(array_idx).GetValue(); \ if (e) \ e->method; \ } #endif