/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- * 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 GFXPLATFORMFONTLIST_H_ #define GFXPLATFORMFONTLIST_H_ #include "nsDataHashtable.h" #include "nsRefPtrHashtable.h" #include "nsTHashtable.h" #include "gfxFontUtils.h" #include "gfxFontInfoLoader.h" #include "gfxFont.h" #include "gfxPlatform.h" #include "nsIMemoryReporter.h" #include "mozilla/Attributes.h" #include "mozilla/MemoryReporting.h" class CharMapHashKey : public PLDHashEntryHdr { public: typedef gfxCharacterMap* KeyType; typedef const gfxCharacterMap* KeyTypePointer; CharMapHashKey(const gfxCharacterMap *aCharMap) : mCharMap(const_cast(aCharMap)) { MOZ_COUNT_CTOR(CharMapHashKey); } CharMapHashKey(const CharMapHashKey& toCopy) : mCharMap(toCopy.mCharMap) { MOZ_COUNT_CTOR(CharMapHashKey); } ~CharMapHashKey() { MOZ_COUNT_DTOR(CharMapHashKey); } gfxCharacterMap* GetKey() const { return mCharMap; } bool KeyEquals(const gfxCharacterMap *aCharMap) const { NS_ASSERTION(!aCharMap->mBuildOnTheFly && !mCharMap->mBuildOnTheFly, "custom cmap used in shared cmap hashtable"); // cmaps built on the fly never match if (aCharMap->mHash != mCharMap->mHash) { return false; } return mCharMap->Equals(aCharMap); } static const gfxCharacterMap* KeyToPointer(gfxCharacterMap *aCharMap) { return aCharMap; } static PLDHashNumber HashKey(const gfxCharacterMap *aCharMap) { return aCharMap->mHash; } enum { ALLOW_MEMMOVE = true }; protected: gfxCharacterMap *mCharMap; }; // gfxPlatformFontList is an abstract class for the global font list on the system; // concrete subclasses for each platform implement the actual interface to the system fonts. // This class exists because we cannot rely on the platform font-finding APIs to behave // in sensible/similar ways, particularly with rich, complex OpenType families, // so we do our own font family/style management here instead. // Much of this is based on the old gfxQuartzFontCache, but adapted for use on all platforms. struct FontListSizes { uint32_t mFontListSize; // size of the font list and dependent objects // (font family and face names, etc), but NOT // including the font table cache and the cmaps uint32_t mFontTableCacheSize; // memory used for the gfxFontEntry table caches uint32_t mCharMapsSize; // memory used for cmap coverage info }; class gfxPlatformFontList : public gfxFontInfoLoader { public: static gfxPlatformFontList* PlatformFontList() { return sPlatformFontList; } static nsresult Init() { NS_ASSERTION(!sPlatformFontList, "What's this doing here?"); gfxPlatform::GetPlatform()->CreatePlatformFontList(); if (!sPlatformFontList) { return NS_ERROR_OUT_OF_MEMORY; } return NS_OK; } static void Shutdown() { delete sPlatformFontList; sPlatformFontList = nullptr; } virtual ~gfxPlatformFontList(); // initialize font lists virtual nsresult InitFontList(); void GetFontList (nsIAtom *aLangGroup, const nsACString& aGenericFamily, nsTArray& aListOfFonts); virtual bool ResolveFontName(const nsAString& aFontName, nsAString& aResolvedFontName); void UpdateFontList() { InitFontList(); } void ClearPrefFonts() { mPrefFonts.Clear(); } virtual void GetFontFamilyList(nsTArray >& aFamilyArray); virtual gfxFontEntry* SystemFindFontForChar(const uint32_t aCh, int32_t aRunScript, const gfxFontStyle* aStyle); // TODO: make this virtual, for lazily adding to the font list virtual gfxFontFamily* FindFamily(const nsAString& aFamily); gfxFontEntry* FindFontForFamily(const nsAString& aFamily, const gfxFontStyle* aStyle, bool& aNeedsBold); bool GetPrefFontFamilyEntries(eFontPrefLang aLangGroup, nsTArray > *array); void SetPrefFontFamilyEntries(eFontPrefLang aLangGroup, nsTArray >& array); // name lookup table methods void AddOtherFamilyName(gfxFontFamily *aFamilyEntry, nsAString& aOtherFamilyName); void AddFullname(gfxFontEntry *aFontEntry, nsAString& aFullname); void AddPostscriptName(gfxFontEntry *aFontEntry, nsAString& aPostscriptName); bool NeedFullnamePostscriptNames() { return mExtraNames != nullptr; } // pure virtual functions, to be provided by concrete subclasses // get the system default font family virtual gfxFontFamily* GetDefaultFont(const gfxFontStyle* aStyle) = 0; // look up a font by name on the host platform virtual gfxFontEntry* LookupLocalFont(const gfxProxyFontEntry *aProxyEntry, const nsAString& aFontName) = 0; // create a new platform font from downloaded data (@font-face) // this method is responsible to ensure aFontData is NS_Free()'d virtual gfxFontEntry* MakePlatformFont(const gfxProxyFontEntry *aProxyEntry, const uint8_t *aFontData, uint32_t aLength) = 0; // get the standard family name on the platform for a given font name // (platforms may override, eg Mac) virtual bool GetStandardFamilyName(const nsAString& aFontName, nsAString& aFamilyName); virtual void AddSizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf, FontListSizes* aSizes) const; virtual void AddSizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf, FontListSizes* aSizes) const; // search for existing cmap that matches the input // return the input if no match is found gfxCharacterMap* FindCharMap(gfxCharacterMap *aCmap); // add a cmap to the shared cmap set gfxCharacterMap* AddCmap(const gfxCharacterMap *aCharMap); // remove the cmap from the shared cmap set void RemoveCmap(const gfxCharacterMap *aCharMap); protected: class MemoryReporter MOZ_FINAL : public nsIMemoryReporter { public: NS_DECL_ISUPPORTS NS_DECL_NSIMEMORYREPORTER }; gfxPlatformFontList(bool aNeedFullnamePostscriptNames = true); static gfxPlatformFontList *sPlatformFontList; static PLDHashOperator FindFontForCharProc(nsStringHashKey::KeyType aKey, nsRefPtr& aFamilyEntry, void* userArg); // returns default font for a given character, null otherwise gfxFontEntry* CommonFontFallback(const uint32_t aCh, int32_t aRunScript, const gfxFontStyle* aMatchStyle, gfxFontFamily** aMatchedFamily); // search fonts system-wide for a given character, null otherwise virtual gfxFontEntry* GlobalFontFallback(const uint32_t aCh, int32_t aRunScript, const gfxFontStyle* aMatchStyle, uint32_t& aCmapCount, gfxFontFamily** aMatchedFamily); // whether system-based font fallback is used or not // if system fallback is used, no need to load all cmaps virtual bool UsesSystemFallback() { return false; } // verifies that a family contains a non-zero font count gfxFontFamily* CheckFamily(gfxFontFamily *aFamily); // separate initialization for reading in name tables, since this is expensive void InitOtherFamilyNames(); static PLDHashOperator InitOtherFamilyNamesProc(nsStringHashKey::KeyType aKey, nsRefPtr& aFamilyEntry, void* userArg); // read in all fullname/Postscript names for all font faces void InitFaceNameLists(); static PLDHashOperator InitFaceNameListsProc(nsStringHashKey::KeyType aKey, nsRefPtr& aFamilyEntry, void* userArg); // commonly used fonts for which the name table should be loaded at startup virtual void PreloadNamesList(); // load the bad underline blacklist from pref. void LoadBadUnderlineList(); // explicitly set fixed-pitch flag for all faces void SetFixedPitch(const nsAString& aFamilyName); void GenerateFontListKey(const nsAString& aKeyName, nsAString& aResult); static PLDHashOperator HashEnumFuncForFamilies(nsStringHashKey::KeyType aKey, nsRefPtr& aFamilyEntry, void* aUserArg); virtual void GetFontFamilyNames(nsTArray& aFontFamilyNames); // gfxFontInfoLoader overrides, used to load in font cmaps virtual void InitLoader(); virtual bool LoadFontInfo(); virtual void CleanupLoader(); // read the loader initialization prefs, and start it void GetPrefsAndStartLoader(); // used by memory reporter to accumulate sizes of family names in the hash static size_t SizeOfFamilyNameEntryExcludingThis(const nsAString& aKey, const nsRefPtr& aFamily, mozilla::MallocSizeOf aMallocSizeOf, void* aUserArg); // canonical family name ==> family entry (unique, one name per family entry) nsRefPtrHashtable mFontFamilies; // other family name ==> family entry (not unique, can have multiple names per // family entry, only names *other* than the canonical names are stored here) nsRefPtrHashtable mOtherFamilyNames; // flag set after InitOtherFamilyNames is called upon first name lookup miss bool mOtherFamilyNamesInitialized; // flag set after fullname and Postcript name lists are populated bool mFaceNamesInitialized; struct ExtraNames { ExtraNames() : mFullnames(100), mPostscriptNames(100) {} // fullname ==> font entry (unique, one name per font entry) nsRefPtrHashtable mFullnames; // Postscript name ==> font entry (unique, one name per font entry) nsRefPtrHashtable mPostscriptNames; }; nsAutoPtr mExtraNames; // cached pref font lists // maps list of family names ==> array of family entries, one per lang group nsDataHashtable > > mPrefFonts; // when system-wide font lookup fails for a character, cache it to skip future searches gfxSparseBitSet mCodepointsWithNoFonts; // the family to use for U+FFFD fallback, to avoid expensive search every time // on pages with lots of problems nsRefPtr mReplacementCharFallbackFamily; nsTHashtable mBadUnderlineFamilyNames; // character map data shared across families // contains weak ptrs to cmaps shared by font entry objects nsTHashtable mSharedCmaps; // data used as part of the font cmap loading process nsTArray > mFontFamiliesToLoad; uint32_t mStartIndex; uint32_t mIncrement; uint32_t mNumFamilies; }; #endif /* GFXPLATFORMFONTLIST_H_ */