mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-23 12:51:06 +00:00
141fb31205
Differential Revision: https://phabricator.services.mozilla.com/D209460
578 lines
21 KiB
C++
578 lines
21 KiB
C++
/* 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 CacheEntry__h__
|
|
#define CacheEntry__h__
|
|
|
|
#include "mozilla/LinkedList.h"
|
|
#include "nsICacheEntry.h"
|
|
#include "CacheFile.h"
|
|
|
|
#include "nsIRunnable.h"
|
|
#include "nsIOutputStream.h"
|
|
#include "nsICacheEntryOpenCallback.h"
|
|
#include "nsICacheEntryDoomCallback.h"
|
|
#include "nsITransportSecurityInfo.h"
|
|
|
|
#include "nsCOMPtr.h"
|
|
#include "nsRefPtrHashtable.h"
|
|
#include "nsHashKeys.h"
|
|
#include "nsString.h"
|
|
#include "nsCOMArray.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "mozilla/Attributes.h"
|
|
#include "mozilla/Mutex.h"
|
|
#include "mozilla/TimeStamp.h"
|
|
|
|
static inline uint32_t PRTimeToSeconds(PRTime t_usec) {
|
|
return uint32_t(t_usec / PR_USEC_PER_SEC);
|
|
}
|
|
|
|
#define NowInSeconds() PRTimeToSeconds(PR_Now())
|
|
|
|
class nsIOutputStream;
|
|
class nsIURI;
|
|
class nsIThread;
|
|
|
|
namespace mozilla {
|
|
namespace net {
|
|
|
|
class CacheStorageService;
|
|
class CacheStorage;
|
|
class CacheOutputCloseListener;
|
|
class CacheEntryHandle;
|
|
|
|
class CacheEntry final : public nsIRunnable,
|
|
public CacheFileListener,
|
|
// Used by CacheStorageService::MemoryPool
|
|
public LinkedListElement<RefPtr<CacheEntry>> {
|
|
public:
|
|
NS_DECL_THREADSAFE_ISUPPORTS
|
|
NS_DECL_NSIRUNNABLE
|
|
|
|
static uint64_t GetNextId();
|
|
|
|
CacheEntry(const nsACString& aStorageID, const nsACString& aURI,
|
|
const nsACString& aEnhanceID, bool aUseDisk, bool aSkipSizeCheck,
|
|
bool aPin);
|
|
|
|
void AsyncOpen(nsICacheEntryOpenCallback* aCallback, uint32_t aFlags);
|
|
|
|
CacheEntryHandle* NewHandle();
|
|
// For a new and recreated entry w/o a callback, we need to wrap it
|
|
// with a handle to detect writing consumer is gone.
|
|
CacheEntryHandle* NewWriteHandle();
|
|
|
|
// Forwarded to from CacheEntryHandle : nsICacheEntry
|
|
nsresult GetKey(nsACString& aKey);
|
|
nsresult GetCacheEntryId(uint64_t* aCacheEntryId);
|
|
nsresult GetPersistent(bool* aPersistToDisk);
|
|
nsresult GetFetchCount(uint32_t* aFetchCount);
|
|
nsresult GetLastFetched(uint32_t* aLastFetched);
|
|
nsresult GetLastModified(uint32_t* aLastModified);
|
|
nsresult GetExpirationTime(uint32_t* aExpirationTime);
|
|
nsresult SetExpirationTime(uint32_t expirationTime);
|
|
nsresult GetOnStartTime(uint64_t* aTime);
|
|
nsresult GetOnStopTime(uint64_t* aTime);
|
|
nsresult SetNetworkTimes(uint64_t onStartTime, uint64_t onStopTime);
|
|
nsresult SetContentType(uint8_t aContentType);
|
|
nsresult ForceValidFor(uint32_t aSecondsToTheFuture);
|
|
nsresult GetIsForcedValid(bool* aIsForcedValid);
|
|
nsresult MarkForcedValidUse();
|
|
nsresult OpenInputStream(int64_t offset, nsIInputStream** _retval);
|
|
nsresult OpenOutputStream(int64_t offset, int64_t predictedSize,
|
|
nsIOutputStream** _retval);
|
|
nsresult GetSecurityInfo(nsITransportSecurityInfo** aSecurityInfo);
|
|
nsresult SetSecurityInfo(nsITransportSecurityInfo* aSecurityInfo);
|
|
nsresult GetStorageDataSize(uint32_t* aStorageDataSize);
|
|
nsresult AsyncDoom(nsICacheEntryDoomCallback* aCallback);
|
|
nsresult GetMetaDataElement(const char* key, char** aRetval);
|
|
nsresult SetMetaDataElement(const char* key, const char* value);
|
|
nsresult VisitMetaData(nsICacheEntryMetaDataVisitor* visitor);
|
|
nsresult MetaDataReady(void);
|
|
nsresult SetValid(void);
|
|
nsresult GetDiskStorageSizeInKB(uint32_t* aDiskStorageSizeInKB);
|
|
nsresult Recreate(bool aMemoryOnly, nsICacheEntry** _retval);
|
|
nsresult GetDataSize(int64_t* aDataSize);
|
|
nsresult GetAltDataSize(int64_t* aDataSize);
|
|
nsresult GetAltDataType(nsACString& aAltDataType);
|
|
nsresult OpenAlternativeOutputStream(const nsACString& type,
|
|
int64_t predictedSize,
|
|
nsIAsyncOutputStream** _retval);
|
|
nsresult OpenAlternativeInputStream(const nsACString& type,
|
|
nsIInputStream** _retval);
|
|
nsresult GetLoadContextInfo(nsILoadContextInfo** aInfo);
|
|
|
|
public:
|
|
uint32_t GetMetadataMemoryConsumption();
|
|
nsCString const& GetStorageID() const { return mStorageID; }
|
|
nsCString const& GetEnhanceID() const { return mEnhanceID; }
|
|
nsCString const& GetURI() const { return mURI; }
|
|
// Accessible at any time
|
|
bool IsUsingDisk() const { return mUseDisk; }
|
|
bool IsReferenced() const MOZ_NO_THREAD_SAFETY_ANALYSIS;
|
|
bool IsFileDoomed();
|
|
bool IsDoomed() const { return mIsDoomed; }
|
|
bool IsPinned() const { return mPinned; }
|
|
|
|
// Methods for entry management (eviction from memory),
|
|
// called only on the management thread.
|
|
|
|
// TODO make these inline
|
|
double GetFrecency() const;
|
|
uint32_t GetExpirationTime() const;
|
|
uint32_t UseCount() const { return mUseCount; }
|
|
|
|
bool IsRegistered() const;
|
|
bool CanRegister() const;
|
|
void SetRegistered(bool aRegistered);
|
|
|
|
TimeStamp const& LoadStart() const { return mLoadStart; }
|
|
|
|
enum EPurge {
|
|
PURGE_DATA_ONLY_DISK_BACKED,
|
|
PURGE_WHOLE_ONLY_DISK_BACKED,
|
|
PURGE_WHOLE,
|
|
};
|
|
|
|
bool DeferOrBypassRemovalOnPinStatus(bool aPinned);
|
|
bool Purge(uint32_t aWhat);
|
|
void PurgeAndDoom();
|
|
void DoomAlreadyRemoved();
|
|
|
|
nsresult HashingKeyWithStorage(nsACString& aResult) const;
|
|
nsresult HashingKey(nsACString& aResult) const;
|
|
|
|
static nsresult HashingKey(const nsACString& aStorageID,
|
|
const nsACString& aEnhanceID, nsIURI* aURI,
|
|
nsACString& aResult);
|
|
|
|
static nsresult HashingKey(const nsACString& aStorageID,
|
|
const nsACString& aEnhanceID,
|
|
const nsACString& aURISpec, nsACString& aResult);
|
|
|
|
// Accessed only on the service management thread
|
|
double mFrecency{0};
|
|
::mozilla::Atomic<uint32_t, ::mozilla::Relaxed> mSortingExpirationTime{
|
|
uint32_t(-1)};
|
|
|
|
// Memory reporting
|
|
size_t SizeOfExcludingThis(mozilla::MallocSizeOf mallocSizeOf);
|
|
size_t SizeOfIncludingThis(mozilla::MallocSizeOf mallocSizeOf);
|
|
|
|
private:
|
|
virtual ~CacheEntry();
|
|
|
|
// CacheFileListener
|
|
NS_IMETHOD OnFileReady(nsresult aResult, bool aIsNew) override;
|
|
NS_IMETHOD OnFileDoomed(nsresult aResult) override;
|
|
|
|
// Keep the service alive during life-time of an entry
|
|
RefPtr<CacheStorageService> mService;
|
|
|
|
// We must monitor when a cache entry whose consumer is responsible
|
|
// for writing it the first time gets released. We must then invoke
|
|
// waiting callbacks to not break the chain.
|
|
class Callback {
|
|
public:
|
|
Callback(CacheEntry* aEntry, nsICacheEntryOpenCallback* aCallback,
|
|
bool aReadOnly, bool aCheckOnAnyThread, bool aSecret);
|
|
// Special constructor for Callback objects added to the chain
|
|
// just to ensure proper defer dooming (recreation) of this entry.
|
|
Callback(CacheEntry* aEntry, bool aDoomWhenFoundInPinStatus);
|
|
Callback(Callback const& aThat);
|
|
~Callback();
|
|
|
|
// Called when this callback record changes it's owning entry,
|
|
// mainly during recreation.
|
|
void ExchangeEntry(CacheEntry* aEntry) MOZ_REQUIRES(aEntry->mLock);
|
|
|
|
// Returns true when an entry is about to be "defer" doomed and this is
|
|
// a "defer" callback. The caller must hold a lock (this entry is in the
|
|
// caller's mCallback array)
|
|
bool DeferDoom(bool* aDoom) const;
|
|
|
|
// We are raising reference count here to take into account the pending
|
|
// callback (that virtually holds a ref to this entry before it gets
|
|
// it's pointer).
|
|
RefPtr<CacheEntry> mEntry;
|
|
nsCOMPtr<nsICacheEntryOpenCallback> mCallback;
|
|
nsCOMPtr<nsIEventTarget> mTarget;
|
|
bool mReadOnly : 1;
|
|
bool mRevalidating : 1;
|
|
bool mCheckOnAnyThread : 1;
|
|
bool mRecheckAfterWrite : 1;
|
|
bool mNotWanted : 1;
|
|
bool mSecret : 1;
|
|
|
|
// These are set only for the defer-doomer Callback instance inserted
|
|
// to the callback chain. When any of these is set and also any of
|
|
// the corressponding flags on the entry is set, this callback will
|
|
// indicate (via DeferDoom()) the entry have to be recreated/doomed.
|
|
bool mDoomWhenFoundPinned : 1;
|
|
bool mDoomWhenFoundNonPinned : 1;
|
|
|
|
nsresult OnCheckThread(bool* aOnCheckThread) const;
|
|
nsresult OnAvailThread(bool* aOnAvailThread) const;
|
|
};
|
|
|
|
// Since OnCacheEntryAvailable must be invoked on the main thread
|
|
// we need a runnable for it...
|
|
class AvailableCallbackRunnable : public Runnable {
|
|
public:
|
|
AvailableCallbackRunnable(CacheEntry* aEntry, Callback const& aCallback)
|
|
: Runnable("CacheEntry::AvailableCallbackRunnable"),
|
|
mEntry(aEntry),
|
|
mCallback(aCallback) {}
|
|
|
|
private:
|
|
NS_IMETHOD Run() override {
|
|
mEntry->InvokeAvailableCallback(mCallback);
|
|
return NS_OK;
|
|
}
|
|
|
|
RefPtr<CacheEntry> mEntry;
|
|
Callback mCallback;
|
|
};
|
|
|
|
// Since OnCacheEntryDoomed must be invoked on the main thread
|
|
// we need a runnable for it...
|
|
class DoomCallbackRunnable : public Runnable {
|
|
public:
|
|
DoomCallbackRunnable(CacheEntry* aEntry, nsresult aRv)
|
|
: Runnable("net::CacheEntry::DoomCallbackRunnable"),
|
|
mEntry(aEntry),
|
|
mRv(aRv) {}
|
|
|
|
private:
|
|
NS_IMETHOD Run() override {
|
|
nsCOMPtr<nsICacheEntryDoomCallback> callback;
|
|
{
|
|
mozilla::MutexAutoLock lock(mEntry->mLock);
|
|
mEntry->mDoomCallback.swap(callback);
|
|
}
|
|
|
|
if (callback) callback->OnCacheEntryDoomed(mRv);
|
|
return NS_OK;
|
|
}
|
|
|
|
RefPtr<CacheEntry> mEntry;
|
|
nsresult mRv;
|
|
};
|
|
|
|
// Starts the load or just invokes the callback, bypasses (when required)
|
|
// if busy. Returns true on job done, false on bypass.
|
|
bool Open(Callback& aCallback, bool aTruncate, bool aPriority,
|
|
bool aBypassIfBusy);
|
|
// Loads from disk asynchronously
|
|
bool Load(bool aTruncate, bool aPriority);
|
|
|
|
void RememberCallback(Callback& aCallback) MOZ_REQUIRES(mLock);
|
|
void InvokeCallbacksLock();
|
|
void InvokeCallbacks();
|
|
bool InvokeCallbacks(bool aReadOnly);
|
|
bool InvokeCallback(Callback& aCallback);
|
|
void InvokeAvailableCallback(Callback const& aCallback);
|
|
void OnFetched(Callback const& aCallback);
|
|
|
|
nsresult OpenOutputStreamInternal(int64_t offset, nsIOutputStream** _retval);
|
|
nsresult OpenInputStreamInternal(int64_t offset, const char* aAltDataType,
|
|
nsIInputStream** _retval);
|
|
|
|
void OnHandleClosed(CacheEntryHandle const* aHandle);
|
|
|
|
private:
|
|
friend class CacheEntryHandle;
|
|
// Increment/decrements the number of handles keeping this entry.
|
|
void AddHandleRef() MOZ_REQUIRES(mLock) { ++mHandlesCount; }
|
|
void ReleaseHandleRef() MOZ_REQUIRES(mLock) { --mHandlesCount; }
|
|
// Current number of handles keeping this entry.
|
|
uint32_t HandlesCount() const MOZ_REQUIRES(mLock) { return mHandlesCount; }
|
|
|
|
private:
|
|
friend class CacheOutputCloseListener;
|
|
void OnOutputClosed();
|
|
|
|
private:
|
|
// Schedules a background operation on the management thread.
|
|
// When executed on the management thread directly, the operation(s)
|
|
// is (are) executed immediately.
|
|
void BackgroundOp(uint32_t aOperation, bool aForceAsync = false);
|
|
void StoreFrecency(double aFrecency);
|
|
|
|
// Called only from DoomAlreadyRemoved()
|
|
void DoomFile() MOZ_REQUIRES(mLock);
|
|
// When this entry is doomed the first time, this method removes
|
|
// any force-valid timing info for this entry.
|
|
void RemoveForcedValidity();
|
|
|
|
already_AddRefed<CacheEntryHandle> ReopenTruncated(
|
|
bool aMemoryOnly, nsICacheEntryOpenCallback* aCallback);
|
|
void TransferCallbacks(CacheEntry& aFromEntry);
|
|
|
|
mozilla::Mutex mLock{"CacheEntry"};
|
|
|
|
// Reflects the number of existing handles for this entry
|
|
::mozilla::ThreadSafeAutoRefCnt mHandlesCount MOZ_GUARDED_BY(mLock);
|
|
|
|
nsTArray<Callback> mCallbacks MOZ_GUARDED_BY(mLock);
|
|
nsCOMPtr<nsICacheEntryDoomCallback> mDoomCallback;
|
|
|
|
// Set in CacheEntry::Load(), only - shouldn't need to be under lock
|
|
// XXX FIX? is this correct?
|
|
RefPtr<CacheFile> mFile;
|
|
|
|
// Using ReleaseAcquire since we only control access to mFile with this.
|
|
// When mFileStatus is read and found success it is ensured there is mFile and
|
|
// that it is after a successful call to Init().
|
|
Atomic<nsresult, ReleaseAcquire> mFileStatus{NS_ERROR_NOT_INITIALIZED};
|
|
// Set in constructor
|
|
nsCString const mURI;
|
|
nsCString const mEnhanceID;
|
|
nsCString const mStorageID;
|
|
|
|
// mUseDisk, mSkipSizeCheck, mIsDoomed are plain "bool", not "bool:1",
|
|
// so as to avoid bitfield races with the byte containing
|
|
// mSecurityInfoLoaded et al. See bug 1278524.
|
|
//
|
|
// Whether it's allowed to persist the data to disk
|
|
bool const mUseDisk;
|
|
// Whether it should skip max size check.
|
|
bool const mSkipSizeCheck;
|
|
// Set when entry is doomed with AsyncDoom() or DoomAlreadyRemoved().
|
|
Atomic<bool, Relaxed> mIsDoomed{false};
|
|
// The indication of pinning this entry was open with
|
|
Atomic<bool, Relaxed> mPinned;
|
|
|
|
// Following flags are all synchronized with the cache entry lock.
|
|
|
|
// Whether security info has already been looked up in metadata.
|
|
bool mSecurityInfoLoaded : 1 MOZ_GUARDED_BY(mLock);
|
|
// Prevents any callback invocation
|
|
bool mPreventCallbacks : 1 MOZ_GUARDED_BY(mLock);
|
|
// true: after load and an existing file, or after output stream has been
|
|
// opened.
|
|
// note - when opening an input stream, and this flag is false, output
|
|
// stream is open along ; this makes input streams on new entries
|
|
// behave correctly when EOF is reached (WOULD_BLOCK is returned).
|
|
// false: after load and a new file, or dropped to back to false when a
|
|
// writer fails to open an output stream.
|
|
bool mHasData : 1 MOZ_GUARDED_BY(mLock);
|
|
// Whether the pinning state of the entry is known (equals to the actual state
|
|
// of the cache file)
|
|
bool mPinningKnown : 1 MOZ_GUARDED_BY(mLock);
|
|
|
|
static char const* StateString(uint32_t aState);
|
|
|
|
enum EState { // transiting to:
|
|
NOTLOADED = 0, // -> LOADING | EMPTY
|
|
LOADING = 1, // -> EMPTY | READY
|
|
EMPTY = 2, // -> WRITING
|
|
WRITING = 3, // -> EMPTY | READY
|
|
READY = 4, // -> REVALIDATING
|
|
REVALIDATING = 5 // -> READY
|
|
};
|
|
|
|
// State of this entry.
|
|
EState mState MOZ_GUARDED_BY(mLock){NOTLOADED};
|
|
|
|
enum ERegistration {
|
|
NEVERREGISTERED = 0, // The entry has never been registered
|
|
REGISTERED = 1, // The entry is stored in the memory pool index
|
|
DEREGISTERED = 2 // The entry has been removed from the pool
|
|
};
|
|
|
|
// Accessed only on the management thread. Records the state of registration
|
|
// this entry in the memory pool intermediate cache.
|
|
ERegistration mRegistration{NEVERREGISTERED};
|
|
|
|
// If a new (empty) entry is requested to open an input stream before
|
|
// output stream has been opened, we must open output stream internally
|
|
// on CacheFile and hold until writer releases the entry or opens the output
|
|
// stream for read (then we trade him mOutputStream).
|
|
nsCOMPtr<nsIOutputStream> mOutputStream MOZ_GUARDED_BY(mLock);
|
|
|
|
// Weak reference to the current writter. There can be more then one
|
|
// writer at a time and OnHandleClosed() must be processed only for the
|
|
// current one.
|
|
CacheEntryHandle* mWriter MOZ_GUARDED_BY(mLock){nullptr};
|
|
|
|
// Background thread scheduled operation. Set (under the lock) one
|
|
// of this flags to tell the background thread what to do.
|
|
class Ops {
|
|
public:
|
|
static uint32_t const REGISTER = 1 << 0;
|
|
static uint32_t const FRECENCYUPDATE = 1 << 1;
|
|
static uint32_t const CALLBACKS = 1 << 2;
|
|
static uint32_t const UNREGISTER = 1 << 3;
|
|
|
|
Ops() = default;
|
|
uint32_t Grab() {
|
|
uint32_t flags = mFlags;
|
|
mFlags = 0;
|
|
return flags;
|
|
}
|
|
bool Set(uint32_t aFlags) {
|
|
if (mFlags & aFlags) return false;
|
|
mFlags |= aFlags;
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
uint32_t mFlags{0};
|
|
} mBackgroundOperations;
|
|
|
|
nsCOMPtr<nsITransportSecurityInfo> mSecurityInfo;
|
|
mozilla::TimeStamp mLoadStart;
|
|
uint32_t mUseCount{0};
|
|
|
|
const uint64_t mCacheEntryId;
|
|
};
|
|
|
|
class CacheEntryHandle final : public nsICacheEntry {
|
|
public:
|
|
explicit CacheEntryHandle(CacheEntry* aEntry);
|
|
CacheEntry* Entry() const { return mEntry; }
|
|
|
|
NS_DECL_THREADSAFE_ISUPPORTS
|
|
|
|
// Default implementation is simply safely forwarded.
|
|
NS_IMETHOD GetKey(nsACString& aKey) override { return mEntry->GetKey(aKey); }
|
|
NS_IMETHOD GetCacheEntryId(uint64_t* aCacheEntryId) override {
|
|
return mEntry->GetCacheEntryId(aCacheEntryId);
|
|
}
|
|
NS_IMETHOD GetPersistent(bool* aPersistent) override {
|
|
return mEntry->GetPersistent(aPersistent);
|
|
}
|
|
NS_IMETHOD GetFetchCount(uint32_t* aFetchCount) override {
|
|
return mEntry->GetFetchCount(aFetchCount);
|
|
}
|
|
NS_IMETHOD GetLastFetched(uint32_t* aLastFetched) override {
|
|
return mEntry->GetLastFetched(aLastFetched);
|
|
}
|
|
NS_IMETHOD GetLastModified(uint32_t* aLastModified) override {
|
|
return mEntry->GetLastModified(aLastModified);
|
|
}
|
|
NS_IMETHOD GetExpirationTime(uint32_t* aExpirationTime) override {
|
|
return mEntry->GetExpirationTime(aExpirationTime);
|
|
}
|
|
NS_IMETHOD SetExpirationTime(uint32_t expirationTime) override {
|
|
return mEntry->SetExpirationTime(expirationTime);
|
|
}
|
|
NS_IMETHOD GetOnStartTime(uint64_t* aOnStartTime) override {
|
|
return mEntry->GetOnStartTime(aOnStartTime);
|
|
}
|
|
NS_IMETHOD GetOnStopTime(uint64_t* aOnStopTime) override {
|
|
return mEntry->GetOnStopTime(aOnStopTime);
|
|
}
|
|
NS_IMETHOD SetNetworkTimes(uint64_t onStartTime,
|
|
uint64_t onStopTime) override {
|
|
return mEntry->SetNetworkTimes(onStartTime, onStopTime);
|
|
}
|
|
NS_IMETHOD SetContentType(uint8_t contentType) override {
|
|
return mEntry->SetContentType(contentType);
|
|
}
|
|
NS_IMETHOD ForceValidFor(uint32_t aSecondsToTheFuture) override {
|
|
return mEntry->ForceValidFor(aSecondsToTheFuture);
|
|
}
|
|
NS_IMETHOD GetIsForcedValid(bool* aIsForcedValid) override {
|
|
return mEntry->GetIsForcedValid(aIsForcedValid);
|
|
}
|
|
NS_IMETHOD MarkForcedValidUse() override {
|
|
return mEntry->MarkForcedValidUse();
|
|
}
|
|
NS_IMETHOD OpenInputStream(int64_t offset,
|
|
nsIInputStream** _retval) override {
|
|
return mEntry->OpenInputStream(offset, _retval);
|
|
}
|
|
NS_IMETHOD OpenOutputStream(int64_t offset, int64_t predictedSize,
|
|
nsIOutputStream** _retval) override {
|
|
return mEntry->OpenOutputStream(offset, predictedSize, _retval);
|
|
}
|
|
NS_IMETHOD GetSecurityInfo(
|
|
nsITransportSecurityInfo** aSecurityInfo) override {
|
|
return mEntry->GetSecurityInfo(aSecurityInfo);
|
|
}
|
|
NS_IMETHOD SetSecurityInfo(nsITransportSecurityInfo* aSecurityInfo) override {
|
|
return mEntry->SetSecurityInfo(aSecurityInfo);
|
|
}
|
|
NS_IMETHOD GetStorageDataSize(uint32_t* aStorageDataSize) override {
|
|
return mEntry->GetStorageDataSize(aStorageDataSize);
|
|
}
|
|
NS_IMETHOD AsyncDoom(nsICacheEntryDoomCallback* listener) override {
|
|
return mEntry->AsyncDoom(listener);
|
|
}
|
|
NS_IMETHOD GetMetaDataElement(const char* key, char** _retval) override {
|
|
return mEntry->GetMetaDataElement(key, _retval);
|
|
}
|
|
NS_IMETHOD SetMetaDataElement(const char* key, const char* value) override {
|
|
return mEntry->SetMetaDataElement(key, value);
|
|
}
|
|
NS_IMETHOD VisitMetaData(nsICacheEntryMetaDataVisitor* visitor) override {
|
|
return mEntry->VisitMetaData(visitor);
|
|
}
|
|
NS_IMETHOD MetaDataReady(void) override { return mEntry->MetaDataReady(); }
|
|
NS_IMETHOD SetValid(void) override { return mEntry->SetValid(); }
|
|
NS_IMETHOD GetDiskStorageSizeInKB(uint32_t* aDiskStorageSizeInKB) override {
|
|
return mEntry->GetDiskStorageSizeInKB(aDiskStorageSizeInKB);
|
|
}
|
|
NS_IMETHOD Recreate(bool aMemoryOnly, nsICacheEntry** _retval) override {
|
|
return mEntry->Recreate(aMemoryOnly, _retval);
|
|
}
|
|
NS_IMETHOD GetDataSize(int64_t* aDataSize) override {
|
|
return mEntry->GetDataSize(aDataSize);
|
|
}
|
|
NS_IMETHOD GetAltDataSize(int64_t* aAltDataSize) override {
|
|
return mEntry->GetAltDataSize(aAltDataSize);
|
|
}
|
|
NS_IMETHOD GetAltDataType(nsACString& aType) override {
|
|
return mEntry->GetAltDataType(aType);
|
|
}
|
|
NS_IMETHOD OpenAlternativeOutputStream(
|
|
const nsACString& type, int64_t predictedSize,
|
|
nsIAsyncOutputStream** _retval) override {
|
|
return mEntry->OpenAlternativeOutputStream(type, predictedSize, _retval);
|
|
}
|
|
NS_IMETHOD OpenAlternativeInputStream(const nsACString& type,
|
|
nsIInputStream** _retval) override {
|
|
return mEntry->OpenAlternativeInputStream(type, _retval);
|
|
}
|
|
NS_IMETHOD GetLoadContextInfo(
|
|
nsILoadContextInfo** aLoadContextInfo) override {
|
|
return mEntry->GetLoadContextInfo(aLoadContextInfo);
|
|
}
|
|
|
|
// Specific implementation:
|
|
NS_IMETHOD Dismiss() override;
|
|
|
|
private:
|
|
virtual ~CacheEntryHandle();
|
|
RefPtr<CacheEntry> mEntry;
|
|
|
|
// This is |false| until Dismiss() was called and prevents OnHandleClosed
|
|
// being called more than once.
|
|
Atomic<bool, ReleaseAcquire> mClosed{false};
|
|
};
|
|
|
|
class CacheOutputCloseListener final : public Runnable {
|
|
public:
|
|
void OnOutputClosed();
|
|
|
|
private:
|
|
friend class CacheEntry;
|
|
|
|
virtual ~CacheOutputCloseListener() = default;
|
|
|
|
NS_DECL_NSIRUNNABLE
|
|
explicit CacheOutputCloseListener(CacheEntry* aEntry);
|
|
|
|
private:
|
|
RefPtr<CacheEntry> mEntry;
|
|
};
|
|
|
|
} // namespace net
|
|
} // namespace mozilla
|
|
|
|
#endif
|