/* -*- 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 nsTHashtable_h__ #define nsTHashtable_h__ #include "nscore.h" #include "pldhash.h" #include "nsDebug.h" #include "mozilla/MemoryChecking.h" #include "mozilla/MemoryReporting.h" #include "mozilla/Move.h" #include "mozilla/fallible.h" #include "mozilla/PodOperations.h" #include // helper function for nsTHashtable::Clear() NS_COM_GLUE PLDHashOperator PL_DHashStubEnumRemove(PLDHashTable* aTable, PLDHashEntryHdr* aEntry, uint32_t aOrdinal, void* aUserArg); /** * a base class for templated hashtables. * * Clients will rarely need to use this class directly. Check the derived * classes first, to see if they will meet your needs. * * @param EntryType the templated entry-type class that is managed by the * hashtable. EntryType must extend the following declaration, * and must not declare any virtual functions or derive from classes * with virtual functions. Any vtable pointer would break the * PLDHashTable code. *
   class EntryType : public PLDHashEntryHdr
 *   {
 *   public: or friend nsTHashtable;
 *     // KeyType is what we use when Get()ing or Put()ing this entry
 *     // this should either be a simple datatype (uint32_t, nsISupports*) or
 *     // a const reference (const nsAString&)
 *     typedef something KeyType;
 *     // KeyTypePointer is the pointer-version of KeyType, because pldhash.h
 *     // requires keys to cast to const void*
 *     typedef const something* KeyTypePointer;
 *
 *     EntryType(KeyTypePointer aKey);
 *
 *     // A copy or C++11 Move constructor must be defined, even if
 *     // AllowMemMove() == true, otherwise you will cause link errors.
 *     EntryType(const EntryType& aEnt);  // Either this...
 *     EntryType(EntryType&& aEnt);       // ...or this
 *
 *     // the destructor must be defined... or you will cause link errors!
 *     ~EntryType();
 *
 *     // KeyEquals(): does this entry match this key?
 *     bool KeyEquals(KeyTypePointer aKey) const;
 *
 *     // KeyToPointer(): Convert KeyType to KeyTypePointer
 *     static KeyTypePointer KeyToPointer(KeyType aKey);
 *
 *     // HashKey(): calculate the hash number
 *     static PLDHashNumber HashKey(KeyTypePointer aKey);
 *
 *     // ALLOW_MEMMOVE can we move this class with memmove(), or do we have
 *     // to use the copy constructor?
 *     enum { ALLOW_MEMMOVE = true/false };
 *   }
* * @see nsInterfaceHashtable * @see nsDataHashtable * @see nsClassHashtable * @author "Benjamin Smedberg " */ template class nsTHashtable { typedef mozilla::fallible_t fallible_t; public: // Separate constructors instead of default aInitLength parameter since // otherwise the default no-arg constructor isn't found. nsTHashtable() { Init(PL_DHASH_DEFAULT_INITIAL_LENGTH); } explicit nsTHashtable(uint32_t aInitLength) { Init(aInitLength); } /** * destructor, cleans up and deallocates */ ~nsTHashtable(); nsTHashtable(nsTHashtable&& aOther); /** * Return the generation number for the table. This increments whenever * the table data items are moved. */ uint32_t GetGeneration() const { return mTable.generation; } /** * KeyType is typedef'ed for ease of use. */ typedef typename EntryType::KeyType KeyType; /** * KeyTypePointer is typedef'ed for ease of use. */ typedef typename EntryType::KeyTypePointer KeyTypePointer; /** * Return the number of entries in the table. * @return number of entries */ uint32_t Count() const { return mTable.entryCount; } /** * Get the entry associated with a key. * @param aKey the key to retrieve * @return pointer to the entry class, if the key exists; nullptr if the * key doesn't exist */ EntryType* GetEntry(KeyType aKey) const { NS_ASSERTION(mTable.entrySize, "nsTHashtable was not initialized properly."); EntryType* entry = reinterpret_cast( PL_DHashTableOperate(const_cast(&mTable), EntryType::KeyToPointer(aKey), PL_DHASH_LOOKUP)); return PL_DHASH_ENTRY_IS_BUSY(entry) ? entry : nullptr; } /** * Return true if an entry for the given key exists, false otherwise. * @param aKey the key to retrieve * @return true if the key exists, false if the key doesn't exist */ bool Contains(KeyType aKey) const { return !!GetEntry(aKey); } /** * Get the entry associated with a key, or create a new entry, * @param aKey the key to retrieve * @return pointer to the entry class retreived; nullptr only if memory can't be allocated */ EntryType* PutEntry(KeyType aKey) { EntryType* e = PutEntry(aKey, fallible_t()); if (!e) { NS_ABORT_OOM(mTable.entrySize * mTable.entryCount); } return e; } EntryType* PutEntry(KeyType aKey, const fallible_t&) NS_WARN_UNUSED_RESULT { NS_ASSERTION(mTable.entrySize, "nsTHashtable was not initialized properly."); return static_cast(PL_DHashTableOperate( &mTable, EntryType::KeyToPointer(aKey), PL_DHASH_ADD)); } /** * Remove the entry associated with a key. * @param aKey of the entry to remove */ void RemoveEntry(KeyType aKey) { NS_ASSERTION(mTable.entrySize, "nsTHashtable was not initialized properly."); PL_DHashTableOperate(&mTable, EntryType::KeyToPointer(aKey), PL_DHASH_REMOVE); } /** * Remove the entry associated with a key, but don't resize the hashtable. * This is a low-level method, and is not recommended unless you know what * you're doing and you need the extra performance. This method can be used * during enumeration, while RemoveEntry() cannot. * @param aEntry the entry-pointer to remove (obtained from GetEntry or * the enumerator */ void RawRemoveEntry(EntryType* aEntry) { PL_DHashTableRawRemove(&mTable, aEntry); } /** * client must provide an Enumerator function for * EnumerateEntries * @param aEntry the entry being enumerated * @param userArg passed unchanged from EnumerateEntries * @return combination of flags * @link PLDHashOperator::PL_DHASH_NEXT PL_DHASH_NEXT @endlink , * @link PLDHashOperator::PL_DHASH_STOP PL_DHASH_STOP @endlink , * @link PLDHashOperator::PL_DHASH_REMOVE PL_DHASH_REMOVE @endlink */ typedef PLDHashOperator (*Enumerator)(EntryType* aEntry, void* userArg); /** * Enumerate all the entries of the function. * @param enumFunc the Enumerator function to call * @param userArg a pointer to pass to the * Enumerator function * @return the number of entries actually enumerated */ uint32_t EnumerateEntries(Enumerator aEnumFunc, void* aUserArg) { NS_ASSERTION(mTable.entrySize, "nsTHashtable was not initialized properly."); s_EnumArgs args = { aEnumFunc, aUserArg }; return PL_DHashTableEnumerate(&mTable, s_EnumStub, &args); } /** * remove all entries, return hashtable to "pristine" state ;) */ void Clear() { NS_ASSERTION(mTable.entrySize, "nsTHashtable was not initialized properly."); PL_DHashTableEnumerate(&mTable, PL_DHashStubEnumRemove, nullptr); } /** * client must provide a SizeOfEntryExcludingThisFun function for * SizeOfExcludingThis. * @param aEntry the entry being enumerated * @param mallocSizeOf the function used to measure heap-allocated blocks * @param arg passed unchanged from SizeOf{In,Ex}cludingThis * @return summed size of the things pointed to by the entries */ typedef size_t (*SizeOfEntryExcludingThisFun)(EntryType* aEntry, mozilla::MallocSizeOf aMallocSizeOf, void* aArg); /** * Measure the size of the table's entry storage, and if * |aSizeOfEntryExcludingThis| is non-nullptr, measure the size of things * pointed to by entries. * * @param sizeOfEntryExcludingThis the * SizeOfEntryExcludingThisFun function to call * @param mallocSizeOf the function used to measure heap-allocated blocks * @param userArg a pointer to pass to the * SizeOfEntryExcludingThisFun function * @return the summed size of all the entries */ size_t SizeOfExcludingThis(SizeOfEntryExcludingThisFun aSizeOfEntryExcludingThis, mozilla::MallocSizeOf aMallocSizeOf, void* aUserArg = nullptr) const { if (aSizeOfEntryExcludingThis) { s_SizeOfArgs args = { aSizeOfEntryExcludingThis, aUserArg }; return PL_DHashTableSizeOfExcludingThis(&mTable, s_SizeOfStub, aMallocSizeOf, &args); } return PL_DHashTableSizeOfExcludingThis(&mTable, nullptr, aMallocSizeOf); } /** * If the EntryType defines SizeOfExcludingThis, there's no need to define a new * SizeOfEntryExcludingThisFun. */ size_t SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const { return SizeOfExcludingThis(BasicSizeOfEntryExcludingThisFun, aMallocSizeOf); } /** * Like SizeOfExcludingThis, but includes sizeof(*this). */ size_t SizeOfIncludingThis(SizeOfEntryExcludingThisFun aSizeOfEntryExcludingThis, mozilla::MallocSizeOf aMallocSizeOf, void* aUserArg = nullptr) const { return aMallocSizeOf(this) + SizeOfExcludingThis(aSizeOfEntryExcludingThis, aMallocSizeOf, aUserArg); } /** * If the EntryType defines SizeOfExcludingThis, there's no need to define a new * SizeOfEntryExcludingThisFun. */ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const { return SizeOfIncludingThis(BasicSizeOfEntryExcludingThisFun, aMallocSizeOf); } #ifdef DEBUG /** * Mark the table as constant after initialization. * * This will prevent assertions when a read-only hash is accessed on multiple * threads without synchronization. */ void MarkImmutable() { NS_ASSERTION(mTable.entrySize, "nsTHashtable was not initialized properly."); PL_DHashMarkTableImmutable(&mTable); } #endif protected: PLDHashTable mTable; static const void* s_GetKey(PLDHashTable* aTable, PLDHashEntryHdr* aEntry); static PLDHashNumber s_HashKey(PLDHashTable* aTable, const void* aKey); static bool s_MatchEntry(PLDHashTable* aTable, const PLDHashEntryHdr* aEntry, const void* aKey); static void s_CopyEntry(PLDHashTable* aTable, const PLDHashEntryHdr* aFrom, PLDHashEntryHdr* aTo); static void s_ClearEntry(PLDHashTable* aTable, PLDHashEntryHdr* aEntry); static bool s_InitEntry(PLDHashTable* aTable, PLDHashEntryHdr* aEntry, const void* aKey); /** * passed internally during enumeration. Allocated on the stack. * * @param userFunc the Enumerator function passed to * EnumerateEntries by the client * @param userArg the userArg passed unaltered */ struct s_EnumArgs { Enumerator userFunc; void* userArg; }; static PLDHashOperator s_EnumStub(PLDHashTable* aTable, PLDHashEntryHdr* aEntry, uint32_t aNumber, void* aArg); /** * passed internally during sizeOf counting. Allocated on the stack. * * @param userFunc the SizeOfEntryExcludingThisFun passed to * SizeOf{In,Ex}cludingThis by the client * @param userArg the userArg passed unaltered */ struct s_SizeOfArgs { SizeOfEntryExcludingThisFun userFunc; void* userArg; }; static size_t s_SizeOfStub(PLDHashEntryHdr* aEntry, mozilla::MallocSizeOf aMallocSizeOf, void* aArg); private: // copy constructor, not implemented nsTHashtable(nsTHashtable& aToCopy) MOZ_DELETE; /** * Initialize the table. * @param aInitLength the initial number of buckets in the hashtable */ void Init(uint32_t aInitLength); /** * An implementation of SizeOfEntryExcludingThisFun that calls SizeOfExcludingThis() * on each entry. */ static size_t BasicSizeOfEntryExcludingThisFun(EntryType* aEntry, mozilla::MallocSizeOf aMallocSizeOf, void*); // assignment operator, not implemented nsTHashtable& operator=(nsTHashtable& aToEqual) MOZ_DELETE; }; // // template definitions // template nsTHashtable::nsTHashtable(nsTHashtable&& aOther) : mTable(mozilla::Move(aOther.mTable)) { // aOther shouldn't touch mTable after this, because we've stolen the table's // pointers but not overwitten them. MOZ_MAKE_MEM_UNDEFINED(&aOther.mTable, sizeof(aOther.mTable)); // Indicate that aOther is not initialized. This will make its destructor a // nop, which is what we want. aOther.mTable.entrySize = 0; } template nsTHashtable::~nsTHashtable() { if (mTable.entrySize) { PL_DHashTableFinish(&mTable); } } template void nsTHashtable::Init(uint32_t aInitLength) { static const PLDHashTableOps sOps = { ::PL_DHashAllocTable, ::PL_DHashFreeTable, s_HashKey, s_MatchEntry, EntryType::ALLOW_MEMMOVE ? ::PL_DHashMoveEntryStub : s_CopyEntry, s_ClearEntry, ::PL_DHashFinalizeStub, s_InitEntry }; PL_DHashTableInit(&mTable, &sOps, nullptr, sizeof(EntryType), aInitLength); } // static template size_t nsTHashtable::BasicSizeOfEntryExcludingThisFun(EntryType* aEntry, mozilla::MallocSizeOf aMallocSizeOf, void*) { return aEntry->SizeOfExcludingThis(aMallocSizeOf); } // static definitions template PLDHashNumber nsTHashtable::s_HashKey(PLDHashTable* aTable, const void* aKey) { return EntryType::HashKey(reinterpret_cast(aKey)); } template bool nsTHashtable::s_MatchEntry(PLDHashTable* aTable, const PLDHashEntryHdr* aEntry, const void* aKey) { return ((const EntryType*)aEntry)->KeyEquals( reinterpret_cast(aKey)); } template void nsTHashtable::s_CopyEntry(PLDHashTable* aTable, const PLDHashEntryHdr* aFrom, PLDHashEntryHdr* aTo) { EntryType* fromEntry = const_cast(reinterpret_cast(aFrom)); new (aTo) EntryType(mozilla::Move(*fromEntry)); fromEntry->~EntryType(); } template void nsTHashtable::s_ClearEntry(PLDHashTable* aTable, PLDHashEntryHdr* aEntry) { reinterpret_cast(aEntry)->~EntryType(); } template bool nsTHashtable::s_InitEntry(PLDHashTable* aTable, PLDHashEntryHdr* aEntry, const void* aKey) { new (aEntry) EntryType(reinterpret_cast(aKey)); return true; } template PLDHashOperator nsTHashtable::s_EnumStub(PLDHashTable* aTable, PLDHashEntryHdr* aEntry, uint32_t aNumber, void* aArg) { // dereferences the function-pointer to the user's enumeration function return (*reinterpret_cast(aArg)->userFunc)( reinterpret_cast(aEntry), reinterpret_cast(aArg)->userArg); } template size_t nsTHashtable::s_SizeOfStub(PLDHashEntryHdr* aEntry, mozilla::MallocSizeOf aMallocSizeOf, void* aArg) { // dereferences the function-pointer to the user's enumeration function return (*reinterpret_cast(aArg)->userFunc)( reinterpret_cast(aEntry), aMallocSizeOf, reinterpret_cast(aArg)->userArg); } class nsCycleCollectionTraversalCallback; struct MOZ_STACK_CLASS nsTHashtableCCTraversalData { nsTHashtableCCTraversalData(nsCycleCollectionTraversalCallback& aCallback, const char* aName, uint32_t aFlags) : mCallback(aCallback) , mName(aName) , mFlags(aFlags) { } nsCycleCollectionTraversalCallback& mCallback; const char* mName; uint32_t mFlags; }; template PLDHashOperator ImplCycleCollectionTraverse_EnumFunc(EntryType* aEntry, void* aUserData) { auto userData = static_cast(aUserData); ImplCycleCollectionTraverse(userData->mCallback, *aEntry, userData->mName, userData->mFlags); return PL_DHASH_NEXT; } template inline void ImplCycleCollectionUnlink(nsTHashtable& aField) { aField.Clear(); } template inline void ImplCycleCollectionTraverse(nsCycleCollectionTraversalCallback& aCallback, nsTHashtable& aField, const char* aName, uint32_t aFlags = 0) { nsTHashtableCCTraversalData userData(aCallback, aName, aFlags); aField.EnumerateEntries(ImplCycleCollectionTraverse_EnumFunc, &userData); } #endif // nsTHashtable_h__