/* -*- 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/. */ #ifndef mozilla_HoldDropJSObjects_h #define mozilla_HoldDropJSObjects_h #include #include "nsCycleCollectionNoteChild.h" class nsISupports; class nsScriptObjectTracer; class nsCycleCollectionParticipant; namespace JS { class Zone; } // Only HoldJSObjects and DropJSObjects should be called directly. namespace mozilla { namespace cyclecollector { void HoldJSObjectsImpl(void* aHolder, nsScriptObjectTracer* aTracer, JS::Zone* aZone = nullptr); void HoldJSObjectsImpl(nsISupports* aHolder); void DropJSObjectsImpl(void* aHolder); void DropJSObjectsImpl(nsISupports* aHolder); } // namespace cyclecollector template ::value, typename P = typename T::NS_CYCLE_COLLECTION_INNERCLASS> struct HoldDropJSObjectsHelper { static void Hold(T* aHolder) { cyclecollector::HoldJSObjectsImpl(aHolder, NS_CYCLE_COLLECTION_PARTICIPANT(T)); } static void Drop(T* aHolder) { cyclecollector::DropJSObjectsImpl(aHolder); } }; template struct HoldDropJSObjectsHelper { static void Hold(T* aHolder) { cyclecollector::HoldJSObjectsImpl(ToSupports(aHolder)); } static void Drop(T* aHolder) { cyclecollector::DropJSObjectsImpl(ToSupports(aHolder)); } }; /** Classes that hold strong references to JS GC things such as `JSObjects` and `JS::Values` (e.g. `JS::Heap mFoo;`) must use these, generally by calling `HoldJSObjects(this)` and `DropJSObjects(this)` in the ctor and dtor respectively. For classes that are wrapper cached and hold no other strong references to JS GC things, there's no need to call these; it will be taken care of automatically by nsWrapperCache. **/ template void HoldJSObjects(T* aHolder) { static_assert(!std::is_base_of::value, "Don't call this on the CC participant but on the object that " "it's for (in an Unlink implementation it's usually stored in " "a variable named 'tmp')."); HoldDropJSObjectsHelper::Hold(aHolder); } template void DropJSObjects(T* aHolder) { static_assert(!std::is_base_of::value, "Don't call this on the CC participant but on the object that " "it's for (in an Unlink implementation it's usually stored in " "a variable named 'tmp')."); HoldDropJSObjectsHelper::Drop(aHolder); } } // namespace mozilla #endif // mozilla_HoldDropJSObjects_h