/* 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 CacheFileIOManager__h__ #define CacheFileIOManager__h__ #include "CacheIOThread.h" #include "CacheEntriesEnumerator.h" #include "nsIEventTarget.h" #include "nsITimer.h" #include "nsCOMPtr.h" #include "mozilla/SHA1.h" #include "nsTArray.h" #include "nsString.h" #include "nsTHashtable.h" #include "prio.h" class nsIFile; namespace mozilla { namespace net { class CacheFileHandle : public nsISupports { public: NS_DECL_THREADSAFE_ISUPPORTS bool DispatchRelease(); CacheFileHandle(const SHA1Sum::Hash *aHash, bool aPriority); CacheFileHandle(const CacheFileHandle &aOther); bool IsDoomed() { return mIsDoomed; } const SHA1Sum::Hash *Hash() { return mHash; } int64_t FileSize() { return mFileSize; } bool IsPriority() { return mPriority; } bool FileExists() { return mFileExists; } bool IsClosed() { return mClosed; } nsCString & Key() { return mKey; } private: friend class CacheFileIOManager; friend class CacheFileHandles; friend class ReleaseNSPRHandleEvent; virtual ~CacheFileHandle(); const SHA1Sum::Hash *mHash; bool mIsDoomed; bool mPriority; bool mClosed; bool mInvalid; bool mFileExists; // This means that the file should exists, // but it can be still deleted by OS/user // and then a subsequent OpenNSPRFileDesc() // will fail. nsCOMPtr mFile; int64_t mFileSize; PRFileDesc *mFD; // if null then the file doesn't exists on the disk nsCString mKey; }; class CacheFileHandles { public: CacheFileHandles(); ~CacheFileHandles(); nsresult GetHandle(const SHA1Sum::Hash *aHash, bool aReturnDoomed, CacheFileHandle **_retval); nsresult NewHandle(const SHA1Sum::Hash *aHash, bool aPriority, CacheFileHandle **_retval); void RemoveHandle(CacheFileHandle *aHandlle); void GetAllHandles(nsTArray > *_retval); void ClearAll(); uint32_t HandleCount(); class HandleHashKey : public PLDHashEntryHdr { public: typedef const SHA1Sum::Hash& KeyType; typedef const SHA1Sum::Hash* KeyTypePointer; HandleHashKey(KeyTypePointer aKey) { MOZ_COUNT_CTOR(HandleHashKey); mHash = (SHA1Sum::Hash*)new uint8_t[SHA1Sum::HashSize]; memcpy(mHash, aKey, sizeof(SHA1Sum::Hash)); } HandleHashKey(const HandleHashKey& aOther) { NS_NOTREACHED("HandleHashKey copy constructor is forbidden!"); } ~HandleHashKey() { MOZ_COUNT_DTOR(HandleHashKey); } bool KeyEquals(KeyTypePointer aKey) const { return memcmp(mHash, aKey, sizeof(SHA1Sum::Hash)) == 0; } static KeyTypePointer KeyToPointer(KeyType aKey) { return &aKey; } static PLDHashNumber HashKey(KeyTypePointer aKey) { return (reinterpret_cast(aKey))[0]; } void AddHandle(CacheFileHandle* aHandle); void RemoveHandle(CacheFileHandle* aHandle); already_AddRefed GetNewestHandle(); void GetHandles(nsTArray > &aResult); SHA1Sum::Hash *Hash() { return mHash; } bool IsEmpty() { return mHandles.Length() == 0; } enum { ALLOW_MEMMOVE = true }; #ifdef DEBUG void AssertHandlesState(); #endif private: nsAutoArrayPtr mHash; // Use weak pointers since the hash table access is on a single thread // only and CacheFileHandle removes itself from this table in its dtor // that may only be called on the same thread as we work with the hashtable // since we dispatch its Release() to this thread. nsTArray mHandles; }; private: nsTHashtable mTable; }; //////////////////////////////////////////////////////////////////////////////// class OpenFileEvent; class CloseFileEvent; class ReadEvent; class WriteEvent; class MetadataWriteScheduleEvent; #define CACHEFILEIOLISTENER_IID \ { /* dcaf2ddc-17cf-4242-bca1-8c86936375a5 */ \ 0xdcaf2ddc, \ 0x17cf, \ 0x4242, \ {0xbc, 0xa1, 0x8c, 0x86, 0x93, 0x63, 0x75, 0xa5} \ } class CacheFileIOListener : public nsISupports { public: NS_DECLARE_STATIC_IID_ACCESSOR(CACHEFILEIOLISTENER_IID) NS_IMETHOD OnFileOpened(CacheFileHandle *aHandle, nsresult aResult) = 0; NS_IMETHOD OnDataWritten(CacheFileHandle *aHandle, const char *aBuf, nsresult aResult) = 0; NS_IMETHOD OnDataRead(CacheFileHandle *aHandle, char *aBuf, nsresult aResult) = 0; NS_IMETHOD OnFileDoomed(CacheFileHandle *aHandle, nsresult aResult) = 0; NS_IMETHOD OnEOFSet(CacheFileHandle *aHandle, nsresult aResult) = 0; }; NS_DEFINE_STATIC_IID_ACCESSOR(CacheFileIOListener, CACHEFILEIOLISTENER_IID) class CacheFileIOManager : public nsITimerCallback { public: NS_DECL_THREADSAFE_ISUPPORTS NS_DECL_NSITIMERCALLBACK enum { OPEN = 0U, CREATE = 1U, CREATE_NEW = 2U, PRIORITY = 4U, NOHASH = 8U }; CacheFileIOManager(); static nsresult Init(); static nsresult Shutdown(); static nsresult OnProfile(); static already_AddRefed IOTarget(); static already_AddRefed IOThread(); static bool IsOnIOThreadOrCeased(); static bool IsShutdown(); // Make aFile's WriteMetadataIfNeeded be called automatically after // a short interval. static nsresult ScheduleMetadataWrite(CacheFile * aFile); // Remove aFile from the scheduling registry array. // WriteMetadataIfNeeded will not be automatically called. static nsresult UnscheduleMetadataWrite(CacheFile * aFile); // Shuts the scheduling off and flushes all pending metadata writes. static nsresult ShutdownMetadataWriteScheduling(); static nsresult OpenFile(const nsACString &aKey, uint32_t aFlags, CacheFileIOListener *aCallback); static nsresult Read(CacheFileHandle *aHandle, int64_t aOffset, char *aBuf, int32_t aCount, CacheFileIOListener *aCallback); static nsresult Write(CacheFileHandle *aHandle, int64_t aOffset, const char *aBuf, int32_t aCount, bool aValidate, CacheFileIOListener *aCallback); static nsresult DoomFile(CacheFileHandle *aHandle, CacheFileIOListener *aCallback); static nsresult DoomFileByKey(const nsACString &aKey, CacheFileIOListener *aCallback); static nsresult ReleaseNSPRHandle(CacheFileHandle *aHandle); static nsresult TruncateSeekSetEOF(CacheFileHandle *aHandle, int64_t aTruncatePos, int64_t aEOFPos, CacheFileIOListener *aCallback); enum EEnumerateMode { ENTRIES, DOOMED }; static nsresult EnumerateEntryFiles(EEnumerateMode aMode, CacheEntriesEnumerator** aEnumerator); static void GetCacheDirectory(nsIFile** result); private: friend class CacheFileHandle; friend class CacheFileChunk; friend class CacheFile; friend class ShutdownEvent; friend class OpenFileEvent; friend class CloseHandleEvent; friend class ReadEvent; friend class WriteEvent; friend class DoomFileEvent; friend class DoomFileByKeyEvent; friend class ReleaseNSPRHandleEvent; friend class TruncateSeekSetEOFEvent; friend class MetadataWriteScheduleEvent; virtual ~CacheFileIOManager(); nsresult InitInternal(); nsresult ShutdownInternal(); nsresult OpenFileInternal(const SHA1Sum::Hash *aHash, uint32_t aFlags, CacheFileHandle **_retval); nsresult CloseHandleInternal(CacheFileHandle *aHandle); nsresult ReadInternal(CacheFileHandle *aHandle, int64_t aOffset, char *aBuf, int32_t aCount); nsresult WriteInternal(CacheFileHandle *aHandle, int64_t aOffset, const char *aBuf, int32_t aCount, bool aValidate); nsresult DoomFileInternal(CacheFileHandle *aHandle); nsresult DoomFileByKeyInternal(const SHA1Sum::Hash *aHash); nsresult ReleaseNSPRHandleInternal(CacheFileHandle *aHandle); nsresult TruncateSeekSetEOFInternal(CacheFileHandle *aHandle, int64_t aTruncatePos, int64_t aEOFPos); nsresult CreateFile(CacheFileHandle *aHandle); static void GetHashStr(const SHA1Sum::Hash *aHash, nsACString &_retval); nsresult GetFile(const SHA1Sum::Hash *aHash, nsIFile **_retval); nsresult GetDoomedFile(nsIFile **_retval); nsresult CheckAndCreateDir(nsIFile *aFile, const char *aDir); nsresult CreateCacheTree(); nsresult OpenNSPRHandle(CacheFileHandle *aHandle, bool aCreate = false); void NSPRHandleUsed(CacheFileHandle *aHandle); nsresult ScheduleMetadataWriteInternal(CacheFile * aFile); nsresult UnscheduleMetadataWriteInternal(CacheFile * aFile); nsresult ShutdownMetadataWriteSchedulingInternal(); static CacheFileIOManager *gInstance; bool mShuttingDown; nsRefPtr mIOThread; nsCOMPtr mCacheDirectory; bool mTreeCreated; CacheFileHandles mHandles; nsTArray mHandlesByLastUsed; nsTArray > mScheduledMetadataWrites; nsCOMPtr mMetadataWritesTimer; }; } // net } // mozilla #endif