/* -*- 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 DOM_INDEXEDDB_FILEINFOMANAGER_H_ #define DOM_INDEXEDDB_FILEINFOMANAGER_H_ #include "mozilla/Attributes.h" #include "mozilla/Mutex.h" #include "mozilla/StaticMutex.h" #include "nsTHashMap.h" #include "nsHashKeys.h" #include "nsISupportsImpl.h" #include "FileInfo.h" #include "FlippedOnce.h" namespace mozilla { namespace dom { namespace indexedDB { class FileInfoManagerBase { public: bool Invalidated() const { return mInvalidated; } protected: bool AssertValid() const { if (NS_WARN_IF(Invalidated())) { MOZ_ASSERT(false); return false; } return true; } void Invalidate() { mInvalidated.Flip(); } private: FlippedOnce mInvalidated; }; template class FileInfoManager : public FileInfoManagerBase { public: using FileInfoType = FileInfo; using MutexType = StaticMutex; using AutoLockType = mozilla::detail::BaseAutoLock; [[nodiscard]] SafeRefPtr GetFileInfo(int64_t aId) const { return AcquireFileInfo([this, aId] { return mFileInfos.MaybeGet(aId); }); } [[nodiscard]] SafeRefPtr CreateFileInfo() { return AcquireFileInfo([this] { const int64_t id = ++mLastFileId; auto fileInfo = MakeNotNull(FileInfoManagerGuard{}, SafeRefPtr{static_cast(this), AcquireStrongRefFromRawPtr{}}, id); mFileInfos.InsertOrUpdate(id, fileInfo); return Some(fileInfo); }); } void RemoveFileInfo(const int64_t aId, const AutoLockType& aFileMutexLock) { #ifdef DEBUG aFileMutexLock.AssertOwns(FileManager::Mutex()); #endif mFileInfos.Remove(aId); } nsresult Invalidate() { AutoLockType lock(FileManager::Mutex()); FileInfoManagerBase::Invalidate(); mFileInfos.RemoveIf([](const auto& iter) { FileInfoType* info = iter.Data(); MOZ_ASSERT(info); return !info->LockedClearDBRefs(FileInfoManagerGuard{}); }); return NS_OK; } class FileInfoManagerGuard { FileInfoManagerGuard() = default; }; private: // Runs the given aFileInfoTableOp operation, which must return a FileInfo*, // under the FileManager lock, acquires a strong reference to the returned // object under the lock, and returns the strong reference. template [[nodiscard]] SafeRefPtr AcquireFileInfo( const FileInfoTableOp& aFileInfoTableOp) const { if (!AssertValid()) { // In release, the assertions are disabled. return nullptr; } // We cannot simply change this to SafeRefPtr, because // FileInfo::AddRef also acquires the FileManager::Mutex. auto fileInfo = [&aFileInfoTableOp]() -> RefPtr { AutoLockType lock(FileManager::Mutex()); const auto maybeFileInfo = aFileInfoTableOp(); if (maybeFileInfo) { const auto& fileInfo = maybeFileInfo.ref(); fileInfo->LockedAddRef(); return dont_AddRef(fileInfo.get()); } return {}; }(); return SafeRefPtr{std::move(fileInfo)}; } protected: #ifdef DEBUG ~FileInfoManager() { MOZ_ASSERT(mFileInfos.IsEmpty()); } #else ~FileInfoManager() = default; #endif // Access to the following fields must be protected by // FileManager::Mutex() int64_t mLastFileId = 0; nsTHashMap> mFileInfos; }; } // namespace indexedDB } // namespace dom } // namespace mozilla #endif // DOM_INDEXEDDB_FILEINFOMANAGER_H_