/* -*- 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/. */ #include "prlink.h" #include "gfxTypes.h" #include "nsTArray.h" #include "gfxContext.h" #ifdef MOZ_WIDGET_GTK #include "gfxPlatformGtk.h" #endif #include "gfxFontconfigFonts.h" #include "gfxFT2FontBase.h" #include "gfxFT2Utils.h" #include "harfbuzz/hb.h" #include "harfbuzz/hb-glib.h" #include "harfbuzz/hb-ot.h" #include "nsUnicodeProperties.h" #include "nsUnicodeScriptCodes.h" #include "gfxFontconfigUtils.h" #include "gfxUserFontSet.h" #include "gfxFontConstants.h" #include "nsGkAtoms.h" #include "nsILanguageAtomService.h" #include "nsServiceManagerUtils.h" #include <cairo.h> #include <cairo-ft.h> #include "mozilla/gfx/HelpersCairo.h" #include <fontconfig/fcfreetype.h> #include <pango/pango.h> #include FT_TRUETYPE_TABLES_H #ifdef MOZ_WIDGET_GTK #include <gdk/gdk.h> #endif #include <math.h> using namespace mozilla; using namespace mozilla::unicode; #define PRINTING_FC_PROPERTY "gfx.printing" static PangoLanguage *GuessPangoLanguage(nsIAtom *aLanguage); static cairo_scaled_font_t * CreateScaledFont(FcPattern *aPattern, cairo_font_face_t *aFace); static FT_Library gFTLibrary; // FC_FAMILYLANG and FC_FULLNAME were introduced in fontconfig-2.2.97 // and so fontconfig-2.3.0 (2005). #ifndef FC_FAMILYLANG #define FC_FAMILYLANG "familylang" #endif #ifndef FC_FULLNAME #define FC_FULLNAME "fullname" #endif static PRFuncPtr FindFunctionSymbol(const char *name) { PRLibrary *lib = nullptr; PRFuncPtr result = PR_FindFunctionSymbolAndLibrary(name, &lib); if (lib) { PR_UnloadLibrary(lib); } return result; } static bool HasChar(FcPattern *aFont, FcChar32 wc) { FcCharSet *charset = nullptr; FcPatternGetCharSet(aFont, FC_CHARSET, 0, &charset); return charset && FcCharSetHasChar(charset, wc); } /** * gfxFcFontEntry: * * An abstract base class of for gfxFontEntry implementations used by * gfxFcFont and gfxUserFontSet. */ class gfxFcFontEntry : public gfxFontEntry { public: // For all FontEntrys attached to gfxFcFonts, there will be only one // pattern in this array. This is always a font pattern, not a fully // resolved pattern. gfxFcFont only uses this to construct a PangoFont. // // FontEntrys for src:local() fonts in gfxUserFontSet may return more than // one pattern. (See comment in gfxUserFcFontEntry.) const nsTArray< nsCountedRef<FcPattern> >& GetPatterns() { return mPatterns; } static gfxFcFontEntry *LookupFontEntry(cairo_font_face_t *aFace) { return static_cast<gfxFcFontEntry*> (cairo_font_face_get_user_data(aFace, &sFontEntryKey)); } // override the gfxFontEntry impl to read the name from fontconfig // instead of trying to get the 'name' table, as we don't implement // GetFontTable() here virtual nsString RealFaceName(); // This is needed to make gfxFontEntry::HasCharacter(aCh) work. virtual bool TestCharacterMap(uint32_t aCh) { for (uint32_t i = 0; i < mPatterns.Length(); ++i) { if (HasChar(mPatterns[i], aCh)) { return true; } } return false; } protected: explicit gfxFcFontEntry(const nsAString& aName) : gfxFontEntry(aName) { } // One pattern is the common case and some subclasses rely on successful // addition of the first element to the array. AutoTArray<nsCountedRef<FcPattern>,1> mPatterns; static cairo_user_data_key_t sFontEntryKey; }; cairo_user_data_key_t gfxFcFontEntry::sFontEntryKey; nsString gfxFcFontEntry::RealFaceName() { FcChar8 *name; if (!mPatterns.IsEmpty()) { if (FcPatternGetString(mPatterns[0], FC_FULLNAME, 0, &name) == FcResultMatch) { return NS_ConvertUTF8toUTF16((const char*)name); } if (FcPatternGetString(mPatterns[0], FC_FAMILY, 0, &name) == FcResultMatch) { NS_ConvertUTF8toUTF16 result((const char*)name); if (FcPatternGetString(mPatterns[0], FC_STYLE, 0, &name) == FcResultMatch) { result.Append(' '); AppendUTF8toUTF16((const char*)name, result); } return result; } } // fall back to gfxFontEntry implementation (only works for sfnt fonts) return gfxFontEntry::RealFaceName(); } /** * gfxSystemFcFontEntry: * * An implementation of gfxFcFontEntry used by gfxFcFonts for system fonts, * including those from regular family-name based font selection as well as * those from src:local(). * * All gfxFcFonts using the same cairo_font_face_t share the same FontEntry. */ class gfxSystemFcFontEntry : public gfxFcFontEntry { public: // For memory efficiency, aFontPattern should be a font pattern, // not a fully resolved pattern. gfxSystemFcFontEntry(cairo_font_face_t *aFontFace, FcPattern *aFontPattern, const nsAString& aName) : gfxFcFontEntry(aName), mFontFace(aFontFace), mFTFace(nullptr), mFTFaceInitialized(false) { cairo_font_face_reference(mFontFace); cairo_font_face_set_user_data(mFontFace, &sFontEntryKey, this, nullptr); // mPatterns is an AutoTArray with 1 space always available, so the // AppendElement always succeeds. // FIXME: Make this infallible after bug 968520 is done. MOZ_ALWAYS_TRUE(mPatterns.AppendElement(fallible)); mPatterns[0] = aFontPattern; FcChar8 *name; if (FcPatternGetString(aFontPattern, FC_FAMILY, 0, &name) == FcResultMatch) { mFamilyName = NS_ConvertUTF8toUTF16((const char*)name); } } ~gfxSystemFcFontEntry() { cairo_font_face_set_user_data(mFontFace, &sFontEntryKey, nullptr, nullptr); cairo_font_face_destroy(mFontFace); } virtual void ForgetHBFace() override; virtual void ReleaseGrFace(gr_face* aFace) override; protected: virtual nsresult CopyFontTable(uint32_t aTableTag, nsTArray<uint8_t>& aBuffer) override; void MaybeReleaseFTFace(); private: cairo_font_face_t *mFontFace; FT_Face mFTFace; bool mFTFaceInitialized; }; nsresult gfxSystemFcFontEntry::CopyFontTable(uint32_t aTableTag, nsTArray<uint8_t>& aBuffer) { if (!mFTFaceInitialized) { mFTFaceInitialized = true; FcChar8 *filename; if (FcPatternGetString(mPatterns[0], FC_FILE, 0, &filename) != FcResultMatch) { return NS_ERROR_FAILURE; } int index; if (FcPatternGetInteger(mPatterns[0], FC_INDEX, 0, &index) != FcResultMatch) { index = 0; // default to 0 if not found in pattern } if (FT_New_Face(gfxPangoFontGroup::GetFTLibrary(), (const char*)filename, index, &mFTFace) != 0) { return NS_ERROR_FAILURE; } } if (!mFTFace) { return NS_ERROR_NOT_AVAILABLE; } FT_ULong length = 0; if (FT_Load_Sfnt_Table(mFTFace, aTableTag, 0, nullptr, &length) != 0) { return NS_ERROR_NOT_AVAILABLE; } if (!aBuffer.SetLength(length, fallible)) { return NS_ERROR_OUT_OF_MEMORY; } if (FT_Load_Sfnt_Table(mFTFace, aTableTag, 0, aBuffer.Elements(), &length) != 0) { aBuffer.Clear(); return NS_ERROR_FAILURE; } return NS_OK; } void gfxSystemFcFontEntry::MaybeReleaseFTFace() { // don't release if either HB or Gr face still exists if (mHBFace || mGrFace) { return; } if (mFTFace) { FT_Done_Face(mFTFace); mFTFace = nullptr; } mFTFaceInitialized = false; } void gfxSystemFcFontEntry::ForgetHBFace() { gfxFontEntry::ForgetHBFace(); MaybeReleaseFTFace(); } void gfxSystemFcFontEntry::ReleaseGrFace(gr_face* aFace) { gfxFontEntry::ReleaseGrFace(aFace); MaybeReleaseFTFace(); } // A namespace for @font-face family names in FcPatterns so that fontconfig // aliases do not pick up families from @font-face rules and so that // fontconfig rules can distinguish between web fonts and platform fonts. // http://lists.freedesktop.org/archives/fontconfig/2008-November/003037.html #define FONT_FACE_FAMILY_PREFIX "@font-face:" /** * gfxUserFcFontEntry: * * An abstract class for objects in a gfxUserFontSet that can provide * FcPattern* handles to fonts. * * Separate implementations of this class support local fonts from src:local() * and web fonts from src:url(). */ // There is a one-to-one correspondence between gfxUserFcFontEntry objects and // @font-face rules, but sometimes a one-to-many correspondence between font // entries and font patterns. // // http://www.w3.org/TR/2002/WD-css3-webfonts-20020802#font-descriptions // provided a font-size descriptor to specify the sizes supported by the face, // but the "Editor's Draft 27 June 2008" // http://dev.w3.org/csswg/css3-fonts/#font-resources does not provide such a // descriptor, and Mozilla does not recognize such a descriptor. // // Font face names used in src:local() also do not usually specify a size. // // PCF format fonts have each size in a different file, and each of these // files is referenced by its own pattern, but really these are each // different sizes of one face with one name. // // Multiple patterns in an entry also effectively deals with a set of // PostScript Type 1 font files that all have the same face name but are in // several files because of the limit on the number of glyphs in a Type 1 font // file. (e.g. Computer Modern.) class gfxUserFcFontEntry : public gfxFcFontEntry { protected: explicit gfxUserFcFontEntry(const nsAString& aFontName, uint16_t aWeight, int16_t aStretch, uint8_t aStyle) : gfxFcFontEntry(aFontName) { mStyle = aStyle; mWeight = aWeight; mStretch = aStretch; } // Helper function to change a pattern so that it matches the CSS style // descriptors and so gets properly sorted in font selection. This also // avoids synthetic style effects being added by the renderer when the // style of the font itself does not match the descriptor provided by the // author. void AdjustPatternToCSS(FcPattern *aPattern); }; void gfxUserFcFontEntry::AdjustPatternToCSS(FcPattern *aPattern) { int fontWeight = -1; FcPatternGetInteger(aPattern, FC_WEIGHT, 0, &fontWeight); int cssWeight = gfxFontconfigUtils::FcWeightForBaseWeight(mWeight / 100); if (cssWeight != fontWeight) { FcPatternDel(aPattern, FC_WEIGHT); FcPatternAddInteger(aPattern, FC_WEIGHT, cssWeight); } int fontSlant; FcResult res = FcPatternGetInteger(aPattern, FC_SLANT, 0, &fontSlant); // gfxFontEntry doesn't understand the difference between oblique // and italic. if (res != FcResultMatch || IsItalic() != (fontSlant != FC_SLANT_ROMAN)) { FcPatternDel(aPattern, FC_SLANT); FcPatternAddInteger(aPattern, FC_SLANT, IsItalic() ? FC_SLANT_OBLIQUE : FC_SLANT_ROMAN); } int fontWidth = -1; FcPatternGetInteger(aPattern, FC_WIDTH, 0, &fontWidth); int cssWidth = gfxFontconfigUtils::FcWidthForThebesStretch(mStretch); if (cssWidth != fontWidth) { FcPatternDel(aPattern, FC_WIDTH); FcPatternAddInteger(aPattern, FC_WIDTH, cssWidth); } // Ensure that there is a fullname property (if there is a family // property) so that fontconfig rules can identify the real name of the // font, because the family property will be replaced. FcChar8 *unused; if (FcPatternGetString(aPattern, FC_FULLNAME, 0, &unused) == FcResultNoMatch) { nsAutoCString fullname; if (gfxFontconfigUtils::GetFullnameFromFamilyAndStyle(aPattern, &fullname)) { FcPatternAddString(aPattern, FC_FULLNAME, gfxFontconfigUtils::ToFcChar8(fullname)); } } nsAutoCString family; family.Append(FONT_FACE_FAMILY_PREFIX); AppendUTF16toUTF8(Name(), family); FcPatternDel(aPattern, FC_FAMILY); FcPatternDel(aPattern, FC_FAMILYLANG); FcPatternAddString(aPattern, FC_FAMILY, gfxFontconfigUtils::ToFcChar8(family)); } /** * gfxLocalFcFontEntry: * * An implementation of gfxUserFcFontEntry for local fonts from src:local(). * * This class is used only in gfxUserFontSet and for providing FcPattern* * handles to system fonts for font selection. gfxFcFonts created from these * patterns will use gfxSystemFcFontEntrys, which may be shared with * gfxFcFonts from regular family-name based font selection. */ class gfxLocalFcFontEntry : public gfxUserFcFontEntry { public: gfxLocalFcFontEntry(const nsAString& aFontName, uint16_t aWeight, int16_t aStretch, uint8_t aStyle, const nsTArray< nsCountedRef<FcPattern> >& aPatterns) : gfxUserFcFontEntry(aFontName, aWeight, aStretch, aStyle) { if (!mPatterns.SetCapacity(aPatterns.Length(), fallible)) return; // OOM for (uint32_t i = 0; i < aPatterns.Length(); ++i) { FcPattern *pattern = FcPatternDuplicate(aPatterns.ElementAt(i)); if (!pattern) return; // OOM AdjustPatternToCSS(pattern); // FIXME: Make this infallible after bug 968520 is done. MOZ_ALWAYS_TRUE(mPatterns.AppendElement(fallible)); mPatterns[i].own(pattern); } mIsLocalUserFont = true; } }; /** * gfxDownloadedFcFontEntry: * * An implementation of gfxFcFontEntry for web fonts from src:url(). * * When a cairo_font_face_t is created for these fonts, the cairo_font_face_t * keeps a reference to the FontEntry to keep the font data alive. */ class gfxDownloadedFcFontEntry : public gfxUserFcFontEntry { public: // This takes ownership of the face and its underlying data gfxDownloadedFcFontEntry(const nsAString& aFontName, uint16_t aWeight, int16_t aStretch, uint8_t aStyle, const uint8_t *aData, FT_Face aFace) : gfxUserFcFontEntry(aFontName, aWeight, aStretch, aStyle), mFontData(aData), mFace(aFace) { NS_PRECONDITION(aFace != nullptr, "aFace is NULL!"); mIsDataUserFont = true; InitPattern(); } virtual ~gfxDownloadedFcFontEntry(); // Returns true on success bool SetCairoFace(cairo_font_face_t *aFace); virtual hb_blob_t* GetFontTable(uint32_t aTableTag) override; protected: void InitPattern(); // mFontData holds the data used to instantiate the FT_Face; // this has to persist until we are finished with the face, // then be released with free(). const uint8_t* mFontData; FT_Face mFace; }; // A property for recording gfxDownloadedFcFontEntrys on FcPatterns. static const char *kFontEntryFcProp = "-moz-font-entry"; static FcBool AddDownloadedFontEntry(FcPattern *aPattern, gfxDownloadedFcFontEntry *aFontEntry) { FcValue value; value.type = FcTypeFTFace; // void* field of union value.u.f = aFontEntry; return FcPatternAdd(aPattern, kFontEntryFcProp, value, FcFalse); } static FcBool DelDownloadedFontEntry(FcPattern *aPattern) { return FcPatternDel(aPattern, kFontEntryFcProp); } static gfxDownloadedFcFontEntry *GetDownloadedFontEntry(FcPattern *aPattern) { FcValue value; if (FcPatternGet(aPattern, kFontEntryFcProp, 0, &value) != FcResultMatch) return nullptr; if (value.type != FcTypeFTFace) { NS_NOTREACHED("Wrong type for -moz-font-entry font property"); return nullptr; } return static_cast<gfxDownloadedFcFontEntry*>(value.u.f); } gfxDownloadedFcFontEntry::~gfxDownloadedFcFontEntry() { if (mPatterns.Length() != 0) { // Remove back reference to this font entry and the face in case // anyone holds a reference to the pattern. NS_ASSERTION(mPatterns.Length() == 1, "More than one pattern in gfxDownloadedFcFontEntry!"); DelDownloadedFontEntry(mPatterns[0]); FcPatternDel(mPatterns[0], FC_FT_FACE); } FT_Done_Face(mFace); free((void*)mFontData); } typedef FcPattern* (*QueryFaceFunction)(const FT_Face face, const FcChar8 *file, int id, FcBlanks *blanks); void gfxDownloadedFcFontEntry::InitPattern() { static QueryFaceFunction sQueryFacePtr = reinterpret_cast<QueryFaceFunction> (FindFunctionSymbol("FcFreeTypeQueryFace")); FcPattern *pattern; // FcFreeTypeQueryFace is the same function used to construct patterns for // system fonts and so is the preferred function to use for this purpose. // This will set up the langset property, which helps with sorting, and // the foundry, fullname, and fontversion properties, which properly // identify the font to fontconfig rules. However, FcFreeTypeQueryFace is // available only from fontconfig-2.4.2 (December 2006). (CentOS 5.0 has // fontconfig-2.4.1.) if (sQueryFacePtr) { // The "file" argument cannot be nullptr (in fontconfig-2.6.0 at // least). The dummy file passed here is removed below. // // When fontconfig scans the system fonts, FcConfigGetBlanks(nullptr) // is passed as the "blanks" argument, which provides that unexpectedly // blank glyphs are elided. Here, however, we pass nullptr for // "blanks", effectively assuming that, if the font has a blank glyph, // then the author intends any associated character to be rendered // blank. pattern = (*sQueryFacePtr)(mFace, gfxFontconfigUtils::ToFcChar8(""), 0, nullptr); if (!pattern) // Either OOM, or fontconfig chose to skip this font because it // has "no encoded characters", which I think means "BDF and PCF // fonts which are not in Unicode (or the effectively equivalent // ISO Latin-1) encoding". return; // These properties don't make sense for this face without a file. FcPatternDel(pattern, FC_FILE); FcPatternDel(pattern, FC_INDEX); } else { // Do the minimum necessary to construct a pattern for sorting. // FC_CHARSET is vital to determine which characters are supported. nsAutoRef<FcCharSet> charset(FcFreeTypeCharSet(mFace, nullptr)); // If there are no characters then assume we don't know how to read // this font. if (!charset || FcCharSetCount(charset) == 0) return; pattern = FcPatternCreate(); FcPatternAddCharSet(pattern, FC_CHARSET, charset); // FC_PIXEL_SIZE can be important for font selection of fixed-size // fonts. if (!(mFace->face_flags & FT_FACE_FLAG_SCALABLE)) { for (FT_Int i = 0; i < mFace->num_fixed_sizes; ++i) { #if HAVE_FT_BITMAP_SIZE_Y_PPEM double size = FLOAT_FROM_26_6(mFace->available_sizes[i].y_ppem); #else double size = mFace->available_sizes[i].height; #endif FcPatternAddDouble (pattern, FC_PIXEL_SIZE, size); } // Not sure whether this is important; // imitating FcFreeTypeQueryFace: FcPatternAddBool (pattern, FC_ANTIALIAS, FcFalse); } // Setting up the FC_LANGSET property is very difficult with the APIs // available prior to FcFreeTypeQueryFace. Having no FC_LANGSET // property seems better than having a property with an empty LangSet. // With no FC_LANGSET property, fontconfig sort functions will // consider this face to have the same priority as (otherwise equal) // faces that have support for the primary requested language, but // will not consider any language to have been satisfied (and so will // continue to look for a face with language support in fallback // fonts). } AdjustPatternToCSS(pattern); FcPatternAddFTFace(pattern, FC_FT_FACE, mFace); AddDownloadedFontEntry(pattern, this); // There is never more than one pattern // FIXME: Make this infallible after bug 968520 is done. MOZ_ALWAYS_TRUE(mPatterns.AppendElement(fallible)); mPatterns[0].own(pattern); } static void ReleaseDownloadedFontEntry(void *data) { gfxDownloadedFcFontEntry *downloadedFontEntry = static_cast<gfxDownloadedFcFontEntry*>(data); NS_RELEASE(downloadedFontEntry); } bool gfxDownloadedFcFontEntry::SetCairoFace(cairo_font_face_t *aFace) { if (CAIRO_STATUS_SUCCESS != cairo_font_face_set_user_data(aFace, &sFontEntryKey, this, ReleaseDownloadedFontEntry)) return false; // Hold a reference to this font entry to keep the font face data. NS_ADDREF(this); return true; } hb_blob_t * gfxDownloadedFcFontEntry::GetFontTable(uint32_t aTableTag) { // The entry already owns the (sanitized) sfnt data in mFontData, // so we can just return a blob that "wraps" the appropriate chunk of it. // The blob should not attempt to free its data, as the entire sfnt data // will be freed when the font entry is deleted. return gfxFontUtils::GetTableFromFontData(mFontData, aTableTag); } /* * gfxFcFont * * This is a gfxFont implementation using a CAIRO_FONT_TYPE_FT * cairo_scaled_font created from an FcPattern. */ class gfxFcFont : public gfxFontconfigFontBase { public: virtual ~gfxFcFont(); static already_AddRefed<gfxFcFont> GetOrMakeFont(FcPattern *aRequestedPattern, FcPattern *aFontPattern, const gfxFontStyle *aFontStyle); // return a cloned font resized and offset to simulate sub/superscript glyphs virtual already_AddRefed<gfxFont> GetSubSuperscriptFont(int32_t aAppUnitsPerDevPixel) override; protected: virtual already_AddRefed<gfxFont> MakeScaledFont(gfxFontStyle *aFontStyle, gfxFloat aFontScale); virtual already_AddRefed<gfxFont> GetSmallCapsFont() override; private: gfxFcFont(cairo_scaled_font_t *aCairoFont, FcPattern *aPattern, gfxFcFontEntry *aFontEntry, const gfxFontStyle *aFontStyle); // key for locating a gfxFcFont corresponding to a cairo_scaled_font static cairo_user_data_key_t sGfxFontKey; }; /** * gfxFcFontSet: * * Translation from a desired FcPattern to a sorted set of font references * (fontconfig cache data) and (when needed) fonts. */ class gfxFcFontSet final { public: NS_INLINE_DECL_REFCOUNTING(gfxFcFontSet) explicit gfxFcFontSet(FcPattern *aPattern, gfxUserFontSet *aUserFontSet) : mSortPattern(aPattern), mUserFontSet(aUserFontSet), mFcFontsTrimmed(0), mHaveFallbackFonts(false) { bool waitForUserFont; mFcFontSet = SortPreferredFonts(waitForUserFont); mWaitingForUserFont = waitForUserFont; } // A reference is held by the FontSet. // The caller may add a ref to keep the font alive longer than the FontSet. gfxFcFont *GetFontAt(uint32_t i, const gfxFontStyle *aFontStyle) { if (i >= mFonts.Length() || !mFonts[i].mFont) { // GetFontPatternAt sets up mFonts FcPattern *fontPattern = GetFontPatternAt(i); if (!fontPattern) return nullptr; mFonts[i].mFont = gfxFcFont::GetOrMakeFont(mSortPattern, fontPattern, aFontStyle); } return mFonts[i].mFont; } FcPattern *GetFontPatternAt(uint32_t i); bool WaitingForUserFont() const { return mWaitingForUserFont; } private: // Private destructor, to discourage deletion outside of Release(): ~gfxFcFontSet() { } nsReturnRef<FcFontSet> SortPreferredFonts(bool& aWaitForUserFont); nsReturnRef<FcFontSet> SortFallbackFonts(); struct FontEntry { explicit FontEntry(FcPattern *aPattern) : mPattern(aPattern) {} nsCountedRef<FcPattern> mPattern; RefPtr<gfxFcFont> mFont; }; struct LangSupportEntry { LangSupportEntry(FcChar8 *aLang, FcLangResult aSupport) : mLang(aLang), mBestSupport(aSupport) {} FcChar8 *mLang; FcLangResult mBestSupport; }; public: // public for nsTArray class LangComparator { public: bool Equals(const LangSupportEntry& a, const FcChar8 *b) const { return FcStrCmpIgnoreCase(a.mLang, b) == 0; } }; private: // The requested pattern nsCountedRef<FcPattern> mSortPattern; // Fonts from @font-face rules RefPtr<gfxUserFontSet> mUserFontSet; // A (trimmed) list of font patterns and fonts that is built up as // required. nsTArray<FontEntry> mFonts; // Holds a list of font patterns that will be trimmed. This is first set // to a list of preferred fonts. Then, if/when all the preferred fonts // have been trimmed and added to mFonts, this is set to a list of // fallback fonts. nsAutoRef<FcFontSet> mFcFontSet; // The set of characters supported by the fonts in mFonts. nsAutoRef<FcCharSet> mCharSet; // The index of the next font in mFcFontSet that has not yet been // considered for mFonts. int mFcFontsTrimmed; // True iff fallback fonts are either stored in mFcFontSet or have been // trimmed and added to mFonts (so that mFcFontSet is nullptr). bool mHaveFallbackFonts; // True iff there was a user font set with pending downloads, // so the set may be updated when downloads complete bool mWaitingForUserFont; }; // Find the FcPattern for an @font-face font suitable for CSS family |aFamily| // and style |aStyle| properties. static const nsTArray< nsCountedRef<FcPattern> >* FindFontPatterns(gfxUserFontSet *mUserFontSet, const nsACString &aFamily, uint8_t aStyle, uint16_t aWeight, int16_t aStretch, bool& aWaitForUserFont) { // Convert to UTF16 NS_ConvertUTF8toUTF16 utf16Family(aFamily); // needsBold is not used here. Instead synthetic bold is enabled through // FcFontRenderPrepare when the weight in the requested pattern is // compared against the weight in the font pattern. bool needsBold; gfxFontStyle style; style.style = aStyle; style.weight = aWeight; style.stretch = aStretch; gfxUserFcFontEntry *fontEntry = nullptr; gfxFontFamily *family = mUserFontSet->LookupFamily(utf16Family); if (family) { gfxUserFontEntry* userFontEntry = mUserFontSet->FindUserFontEntryAndLoad(family, style, needsBold, aWaitForUserFont); if (userFontEntry) { fontEntry = static_cast<gfxUserFcFontEntry*> (userFontEntry->GetPlatformFontEntry()); } // Accept synthetic oblique for italic and oblique. // xxx - this isn't really ideal behavior, for docs that only use a // single italic face it will also pull down the normal face // and probably never use it if (!fontEntry && aStyle != NS_FONT_STYLE_NORMAL) { style.style = NS_FONT_STYLE_NORMAL; userFontEntry = mUserFontSet->FindUserFontEntryAndLoad(family, style, needsBold, aWaitForUserFont); if (userFontEntry) { fontEntry = static_cast<gfxUserFcFontEntry*> (userFontEntry->GetPlatformFontEntry()); } } } if (!fontEntry) { return nullptr; } return &fontEntry->GetPatterns(); } typedef FcBool (*FcPatternRemoveFunction)(FcPattern *p, const char *object, int id); // FcPatternRemove is available in fontconfig-2.3.0 (2005) static FcBool moz_FcPatternRemove(FcPattern *p, const char *object, int id) { static FcPatternRemoveFunction sFcPatternRemovePtr = reinterpret_cast<FcPatternRemoveFunction> (FindFunctionSymbol("FcPatternRemove")); if (!sFcPatternRemovePtr) return FcFalse; return (*sFcPatternRemovePtr)(p, object, id); } // fontconfig prefers a matching family or lang to pixelsize of bitmap // fonts. CSS suggests a tolerance of 20% on pixelsize. static bool SizeIsAcceptable(FcPattern *aFont, double aRequestedSize) { double size; int v = 0; while (FcPatternGetDouble(aFont, FC_PIXEL_SIZE, v, &size) == FcResultMatch) { ++v; if (5.0 * fabs(size - aRequestedSize) < aRequestedSize) return true; } // No size means scalable return v == 0; } // Sorting only the preferred fonts first usually saves having to sort through // every font on the system. nsReturnRef<FcFontSet> gfxFcFontSet::SortPreferredFonts(bool &aWaitForUserFont) { aWaitForUserFont = false; gfxFontconfigUtils *utils = gfxFontconfigUtils::GetFontconfigUtils(); if (!utils) return nsReturnRef<FcFontSet>(); // The list of families in mSortPattern has values with both weak and // strong bindings. Values with strong bindings should be preferred. // Values with weak bindings are default fonts that should be considered // only when the font provides the best support for a requested language // or after other fonts have satisfied all the requested languages. // // There are no direct fontconfig APIs to get the binding type. The // binding only takes effect in the sort and match functions. // |requiredLangs| is a list of requested languages that have not yet been // satisfied. gfxFontconfigUtils only sets one FC_LANG property value, // but FcConfigSubstitute may add more values (e.g. prepending "en" to // "ja" will use western fonts to render Latin/Arabic numerals in Japanese // text.) AutoTArray<LangSupportEntry,10> requiredLangs; for (int v = 0; ; ++v) { FcChar8 *lang; FcResult result = FcPatternGetString(mSortPattern, FC_LANG, v, &lang); if (result != FcResultMatch) { // No need to check FcPatternGetLangSet() because // gfxFontconfigUtils sets only a string value for FC_LANG and // FcConfigSubstitute cannot add LangSets. NS_ASSERTION(result != FcResultTypeMismatch, "Expected a string for FC_LANG"); break; } if (!requiredLangs.Contains(lang, LangComparator())) { FcLangResult bestLangSupport = utils->GetBestLangSupport(lang); if (bestLangSupport != FcLangDifferentLang) { requiredLangs. AppendElement(LangSupportEntry(lang, bestLangSupport)); } } } nsAutoRef<FcFontSet> fontSet(FcFontSetCreate()); if (!fontSet) return fontSet.out(); // FcDefaultSubstitute() ensures a slant on mSortPattern, but, if that ever // doesn't happen, Roman will be used. int requestedSlant = FC_SLANT_ROMAN; FcPatternGetInteger(mSortPattern, FC_SLANT, 0, &requestedSlant); double requestedSize = -1.0; FcPatternGetDouble(mSortPattern, FC_PIXEL_SIZE, 0, &requestedSize); nsTHashtable<gfxFontconfigUtils::DepFcStrEntry> existingFamilies(32); FcChar8 *family; for (int v = 0; FcPatternGetString(mSortPattern, FC_FAMILY, v, &family) == FcResultMatch; ++v) { const nsTArray< nsCountedRef<FcPattern> > *familyFonts = nullptr; // Is this an @font-face family? bool isUserFont = false; if (mUserFontSet) { // Have some @font-face definitions nsDependentCString cFamily(gfxFontconfigUtils::ToCString(family)); NS_NAMED_LITERAL_CSTRING(userPrefix, FONT_FACE_FAMILY_PREFIX); if (StringBeginsWith(cFamily, userPrefix)) { isUserFont = true; // Trim off the prefix nsDependentCSubstring cssFamily(cFamily, userPrefix.Length()); uint8_t thebesStyle = gfxFontconfigUtils::FcSlantToThebesStyle(requestedSlant); uint16_t thebesWeight = gfxFontconfigUtils::GetThebesWeight(mSortPattern); int16_t thebesStretch = gfxFontconfigUtils::GetThebesStretch(mSortPattern); bool waitForUserFont; familyFonts = FindFontPatterns(mUserFontSet, cssFamily, thebesStyle, thebesWeight, thebesStretch, waitForUserFont); if (waitForUserFont) { aWaitForUserFont = true; } } } if (!isUserFont) { familyFonts = &utils->GetFontsForFamily(family); } if (!familyFonts || familyFonts->Length() == 0) { // There are no fonts matching this family, so there is no point // in searching for this family in the FontSort. // // Perhaps the original pattern should be retained for // FcFontRenderPrepare. However, the only a useful config // substitution test against missing families that i can imagine // would only be interested in the preferred family // (qual="first"), so always keep the first family and use the // same pattern for Sort and RenderPrepare. if (v != 0 && moz_FcPatternRemove(mSortPattern, FC_FAMILY, v)) { --v; } continue; } // Aliases seem to often end up occurring more than once, but // duplicate families can't be removed from the sort pattern without // knowing whether duplicates have the same binding. gfxFontconfigUtils::DepFcStrEntry *familyEntry = existingFamilies.PutEntry(family); if (familyEntry) { if (familyEntry->mKey) // old entry continue; familyEntry->mKey = family; // initialize new entry } for (uint32_t f = 0; f < familyFonts->Length(); ++f) { FcPattern *font = familyFonts->ElementAt(f); // Fix up the family name of user-font patterns, as the same // font entry may be used (via the UserFontCache) for multiple // CSS family names if (isUserFont) { font = FcPatternDuplicate(font); FcPatternDel(font, FC_FAMILY); FcPatternAddString(font, FC_FAMILY, family); } // User fonts are already filtered by slant (but not size) in // mUserFontSet->FindUserFontEntry(). if (requestedSize != -1.0 && !SizeIsAcceptable(font, requestedSize)) continue; for (uint32_t r = 0; r < requiredLangs.Length(); ++r) { const LangSupportEntry& langEntry = requiredLangs[r]; FcLangResult support = gfxFontconfigUtils::GetLangSupport(font, langEntry.mLang); if (support <= langEntry.mBestSupport) { // lower is better requiredLangs.RemoveElementAt(r); --r; } } // FcFontSetDestroy will remove a reference but FcFontSetAdd // does _not_ take a reference! if (FcFontSetAdd(fontSet, font)) { // We don't add a reference here for user fonts, because we're // using a local clone of the pattern (see above) in order to // override the family name if (!isUserFont) { FcPatternReference(font); } } } } FcPattern *truncateMarker = nullptr; for (uint32_t r = 0; r < requiredLangs.Length(); ++r) { const nsTArray< nsCountedRef<FcPattern> >& langFonts = utils->GetFontsForLang(requiredLangs[r].mLang); bool haveLangFont = false; for (uint32_t f = 0; f < langFonts.Length(); ++f) { FcPattern *font = langFonts[f]; if (requestedSize != -1.0 && !SizeIsAcceptable(font, requestedSize)) continue; haveLangFont = true; if (FcFontSetAdd(fontSet, font)) { FcPatternReference(font); } } if (!haveLangFont && langFonts.Length() > 0) { // There is a font that supports this language but it didn't pass // the slant and size criteria. Weak default font families should // not be considered until the language has been satisfied. // // Insert a font that supports the language so that it will mark // the position of fonts from weak families in the sorted set and // they can be removed. The language and weak families will be // considered in the fallback fonts, which use fontconfig's // algorithm. // // Of the fonts that don't meet slant and size criteria, strong // default font families should be considered before (other) fonts // for this language, so this marker font will be removed (as well // as the fonts from weak families), and strong families will be // reconsidered in the fallback fonts. FcPattern *font = langFonts[0]; if (FcFontSetAdd(fontSet, font)) { FcPatternReference(font); truncateMarker = font; } break; } } FcFontSet *sets[1] = { fontSet }; FcResult result; #ifdef SOLARIS // Get around a crash of FcFontSetSort when FcConfig is nullptr // Solaris's FcFontSetSort needs an FcConfig (bug 474758) fontSet.own(FcFontSetSort(FcConfigGetCurrent(), sets, 1, mSortPattern, FcFalse, nullptr, &result)); #else fontSet.own(FcFontSetSort(nullptr, sets, 1, mSortPattern, FcFalse, nullptr, &result)); #endif if (truncateMarker != nullptr && fontSet) { nsAutoRef<FcFontSet> truncatedSet(FcFontSetCreate()); for (int f = 0; f < fontSet->nfont; ++f) { FcPattern *font = fontSet->fonts[f]; if (font == truncateMarker) break; if (FcFontSetAdd(truncatedSet, font)) { FcPatternReference(font); } } fontSet.steal(truncatedSet); } return fontSet.out(); } nsReturnRef<FcFontSet> gfxFcFontSet::SortFallbackFonts() { // Setting trim to FcTrue would provide a much smaller (~ 1/10) FcFontSet, // but would take much longer due to comparing all the character sets. // // The references to fonts in this FcFontSet are almost free // as they are pointers into mmaped cache files. // // GetFontPatternAt() will trim lazily if and as needed, which will also // remove duplicates of preferred fonts. FcResult result; return nsReturnRef<FcFontSet>(FcFontSort(nullptr, mSortPattern, FcFalse, nullptr, &result)); } // GetFontAt relies on this setting up all patterns up to |i|. FcPattern * gfxFcFontSet::GetFontPatternAt(uint32_t i) { while (i >= mFonts.Length()) { while (!mFcFontSet) { if (mHaveFallbackFonts) return nullptr; mFcFontSet = SortFallbackFonts(); mHaveFallbackFonts = true; mFcFontsTrimmed = 0; // Loop to test that mFcFontSet is non-nullptr. } while (mFcFontsTrimmed < mFcFontSet->nfont) { FcPattern *font = mFcFontSet->fonts[mFcFontsTrimmed]; ++mFcFontsTrimmed; if (mFonts.Length() != 0) { // See if the next font provides support for any extra // characters. Most often the next font is not going to // support more characters so check for a SubSet first before // allocating a new CharSet with Union. FcCharSet *supportedChars = mCharSet; if (!supportedChars) { FcPatternGetCharSet(mFonts[mFonts.Length() - 1].mPattern, FC_CHARSET, 0, &supportedChars); } if (supportedChars) { FcCharSet *newChars = nullptr; FcPatternGetCharSet(font, FC_CHARSET, 0, &newChars); if (newChars) { if (FcCharSetIsSubset(newChars, supportedChars)) continue; mCharSet.own(FcCharSetUnion(supportedChars, newChars)); } else if (!mCharSet) { mCharSet.own(FcCharSetCopy(supportedChars)); } } } mFonts.AppendElement(font); if (mFonts.Length() >= i) break; } if (mFcFontsTrimmed == mFcFontSet->nfont) { // finished with this font set mFcFontSet.reset(); } } return mFonts[i].mPattern; } #ifdef MOZ_WIDGET_GTK static void ApplyGdkScreenFontOptions(FcPattern *aPattern); #endif // Apply user settings and defaults to pattern in preparation for matching. static void PrepareSortPattern(FcPattern *aPattern, double aFallbackSize, double aSizeAdjustFactor, bool aIsPrinterFont) { FcConfigSubstitute(nullptr, aPattern, FcMatchPattern); // This gets cairo_font_options_t for the Screen. We should have // different font options for printing (no hinting) but we are not told // what we are measuring for. // // If cairo adds support for lcd_filter, gdk will not provide the default // setting for that option. We could get the default setting by creating // an xlib surface once, recording its font_options, and then merging the // gdk options. // // Using an xlib surface would also be an option to get Screen font // options for non-GTK X11 toolkits, but less efficient than using GDK to // pick up dynamic changes. if(aIsPrinterFont) { cairo_font_options_t *options = cairo_font_options_create(); cairo_font_options_set_hint_style (options, CAIRO_HINT_STYLE_NONE); cairo_font_options_set_antialias (options, CAIRO_ANTIALIAS_GRAY); cairo_ft_font_options_substitute(options, aPattern); cairo_font_options_destroy(options); FcPatternAddBool(aPattern, PRINTING_FC_PROPERTY, FcTrue); } else { #ifdef MOZ_WIDGET_GTK ApplyGdkScreenFontOptions(aPattern); #endif } // Protect against any fontconfig settings that may have incorrectly // modified the pixelsize, and consider aSizeAdjustFactor. double size = aFallbackSize; if (FcPatternGetDouble(aPattern, FC_PIXEL_SIZE, 0, &size) != FcResultMatch || aSizeAdjustFactor != 1.0) { FcPatternDel(aPattern, FC_PIXEL_SIZE); FcPatternAddDouble(aPattern, FC_PIXEL_SIZE, size * aSizeAdjustFactor); } FcDefaultSubstitute(aPattern); } /** ** gfxPangoFontGroup **/ gfxPangoFontGroup::gfxPangoFontGroup(const FontFamilyList& aFontFamilyList, const gfxFontStyle *aStyle, gfxUserFontSet *aUserFontSet, gfxFloat aDevToCssSize) : gfxFontGroup(aFontFamilyList, aStyle, nullptr, aUserFontSet, aDevToCssSize), mPangoLanguage(GuessPangoLanguage(aStyle->language)) { // This language is passed to the font for shaping. // Shaping doesn't know about lang groups so make it a real language. if (mPangoLanguage) { mStyle.language = NS_Atomize(pango_language_to_string(mPangoLanguage)); } // dummy entry, will be replaced when actually needed mFonts.AppendElement(FamilyFace()); mSkipUpdateUserFonts = true; } gfxPangoFontGroup::~gfxPangoFontGroup() { } gfxFontGroup * gfxPangoFontGroup::Copy(const gfxFontStyle *aStyle) { return new gfxPangoFontGroup(mFamilyList, aStyle, mUserFontSet, mDevToCssSize); } void gfxPangoFontGroup::FindGenericFontsPFG(FontFamilyType aGenericType, nsIAtom *aLanguage, void *aClosure) { AutoTArray<nsString, 5> resolvedGenerics; ResolveGenericFontNamesPFG(aGenericType, aLanguage, resolvedGenerics); uint32_t g = 0, numGenerics = resolvedGenerics.Length(); for (g = 0; g < numGenerics; g++) { FindPlatformFontPFG(resolvedGenerics[g], false, aClosure); } } /* static */ void gfxPangoFontGroup::ResolveGenericFontNamesPFG(FontFamilyType aGenericType, nsIAtom *aLanguage, nsTArray<nsString>& aGenericFamilies) { static const char kGeneric_serif[] = "serif"; static const char kGeneric_sans_serif[] = "sans-serif"; static const char kGeneric_monospace[] = "monospace"; static const char kGeneric_cursive[] = "cursive"; static const char kGeneric_fantasy[] = "fantasy"; // treat -moz-fixed as monospace if (aGenericType == eFamily_moz_fixed) { aGenericType = eFamily_monospace; } // type should be standard generic type at this point NS_ASSERTION(aGenericType >= eFamily_serif && aGenericType <= eFamily_fantasy, "standard generic font family type required"); // create the lang string nsIAtom *langGroupAtom = nullptr; nsAutoCString langGroupString; if (aLanguage) { if (!gLangService) { CallGetService(NS_LANGUAGEATOMSERVICE_CONTRACTID, &gLangService); } if (gLangService) { nsresult rv; langGroupAtom = gLangService->GetLanguageGroup(aLanguage, &rv); } } if (!langGroupAtom) { langGroupAtom = nsGkAtoms::Unicode; } langGroupAtom->ToUTF8String(langGroupString); // map generic type to string const char *generic = nullptr; switch (aGenericType) { case eFamily_serif: generic = kGeneric_serif; break; case eFamily_sans_serif: generic = kGeneric_sans_serif; break; case eFamily_monospace: generic = kGeneric_monospace; break; case eFamily_cursive: generic = kGeneric_cursive; break; case eFamily_fantasy: generic = kGeneric_fantasy; break; default: break; } if (!generic) { return; } aGenericFamilies.Clear(); // load family for "font.name.generic.lang" nsAutoCString prefFontName("font.name."); prefFontName.Append(generic); prefFontName.Append('.'); prefFontName.Append(langGroupString); gfxFontUtils::AppendPrefsFontList(prefFontName.get(), aGenericFamilies); // if lang has pref fonts, also load fonts for "font.name-list.generic.lang" if (!aGenericFamilies.IsEmpty()) { nsAutoCString prefFontListName("font.name-list."); prefFontListName.Append(generic); prefFontListName.Append('.'); prefFontListName.Append(langGroupString); gfxFontUtils::AppendPrefsFontList(prefFontListName.get(), aGenericFamilies); } #if 0 // dump out generic mappings printf("%s ===> ", prefFontName.get()); for (uint32_t k = 0; k < aGenericFamilies.Length(); k++) { if (k > 0) printf(", "); printf("%s", NS_ConvertUTF16toUTF8(aGenericFamilies[k]).get()); } printf("\n"); #endif } void gfxPangoFontGroup::EnumerateFontListPFG(nsIAtom *aLanguage, void *aClosure) { // initialize fonts in the font family list const nsTArray<FontFamilyName>& fontlist = mFamilyList.GetFontlist(); // lookup fonts in the fontlist uint32_t i, numFonts = fontlist.Length(); for (i = 0; i < numFonts; i++) { const FontFamilyName& name = fontlist[i]; if (name.IsNamed()) { FindPlatformFontPFG(name.mName, true, aClosure); } else { FindGenericFontsPFG(name.mType, aLanguage, aClosure); } } // if necessary, append default generic onto the end if (mFamilyList.GetDefaultFontType() != eFamily_none && !mFamilyList.HasDefaultGeneric()) { FindGenericFontsPFG(mFamilyList.GetDefaultFontType(), aLanguage, aClosure); } } void gfxPangoFontGroup::FindPlatformFontPFG(const nsAString& fontName, bool aUseFontSet, void *aClosure) { nsTArray<nsString> *list = static_cast<nsTArray<nsString>*>(aClosure); if (!list->Contains(fontName)) { // names present in the user fontset are not matched against system fonts if (aUseFontSet && mUserFontSet && mUserFontSet->HasFamily(fontName)) { nsAutoString userFontName = NS_LITERAL_STRING(FONT_FACE_FAMILY_PREFIX) + fontName; list->AppendElement(userFontName); } else { list->AppendElement(fontName); } } } gfxFcFont * gfxPangoFontGroup::GetBaseFont() { if (mFonts[0].Font() == nullptr) { gfxFont* font = GetBaseFontSet()->GetFontAt(0, GetStyle()); mFonts[0] = FamilyFace(nullptr, font); } return static_cast<gfxFcFont*>(mFonts[0].Font()); } gfxFont* gfxPangoFontGroup::GetFirstValidFont(uint32_t aCh) { return GetFontAt(0); } gfxFont * gfxPangoFontGroup::GetFontAt(int32_t i, uint32_t aCh) { // If it turns out to be hard for all clients that cache font // groups to call UpdateUserFonts at appropriate times, we could // instead consider just calling UpdateUserFonts from someplace // more central (such as here). NS_ASSERTION(!mUserFontSet || mCurrGeneration == GetGeneration(), "Whoever was caching this font group should have " "called UpdateUserFonts on it"); NS_PRECONDITION(i == 0, "Only have one font"); return GetBaseFont(); } void gfxPangoFontGroup::UpdateUserFonts() { uint64_t newGeneration = GetGeneration(); if (newGeneration == mCurrGeneration) return; mFonts[0] = FamilyFace(); mFontSets.Clear(); ClearCachedData(); mCurrGeneration = newGeneration; } already_AddRefed<gfxFcFontSet> gfxPangoFontGroup::MakeFontSet(PangoLanguage *aLang, gfxFloat aSizeAdjustFactor, nsAutoRef<FcPattern> *aMatchPattern) { const char *lang = pango_language_to_string(aLang); RefPtr<nsIAtom> langGroup; if (aLang != mPangoLanguage) { // Set up langGroup for Mozilla's font prefs. langGroup = NS_Atomize(lang); } AutoTArray<nsString, 20> fcFamilyList; EnumerateFontListPFG(langGroup ? langGroup.get() : mStyle.language.get(), &fcFamilyList); // To consider: A fontset cache here could be helpful. // Get a pattern suitable for matching. nsAutoRef<FcPattern> pattern (gfxFontconfigUtils::NewPattern(fcFamilyList, mStyle, lang)); PrepareSortPattern(pattern, mStyle.size, aSizeAdjustFactor, mStyle.printerFont); RefPtr<gfxFcFontSet> fontset = new gfxFcFontSet(pattern, mUserFontSet); mSkipDrawing = fontset->WaitingForUserFont(); if (aMatchPattern) aMatchPattern->steal(pattern); return fontset.forget(); } gfxPangoFontGroup:: FontSetByLangEntry::FontSetByLangEntry(PangoLanguage *aLang, gfxFcFontSet *aFontSet) : mLang(aLang), mFontSet(aFontSet) { } gfxFcFontSet * gfxPangoFontGroup::GetFontSet(PangoLanguage *aLang) { GetBaseFontSet(); // sets mSizeAdjustFactor and mFontSets[0] if (!aLang) return mFontSets[0].mFontSet; for (uint32_t i = 0; i < mFontSets.Length(); ++i) { if (mFontSets[i].mLang == aLang) return mFontSets[i].mFontSet; } RefPtr<gfxFcFontSet> fontSet = MakeFontSet(aLang, mSizeAdjustFactor); mFontSets.AppendElement(FontSetByLangEntry(aLang, fontSet)); return fontSet; } already_AddRefed<gfxFont> gfxPangoFontGroup::FindFontForChar(uint32_t aCh, uint32_t aPrevCh, uint32_t aNextCh, Script aRunScript, gfxFont *aPrevMatchedFont, uint8_t *aMatchType) { if (aPrevMatchedFont) { // Don't switch fonts for control characters, regardless of // whether they are present in the current font, as they won't // actually be rendered (see bug 716229) uint8_t category = GetGeneralCategory(aCh); if (category == HB_UNICODE_GENERAL_CATEGORY_CONTROL) { return RefPtr<gfxFont>(aPrevMatchedFont).forget(); } // if this character is a join-control or the previous is a join-causer, // use the same font as the previous range if we can if (gfxFontUtils::IsJoinControl(aCh) || gfxFontUtils::IsJoinCauser(aPrevCh)) { if (aPrevMatchedFont->HasCharacter(aCh)) { return RefPtr<gfxFont>(aPrevMatchedFont).forget(); } } } // if this character is a variation selector, // use the previous font regardless of whether it supports VS or not. // otherwise the text run will be divided. if (gfxFontUtils::IsVarSelector(aCh)) { if (aPrevMatchedFont) { return RefPtr<gfxFont>(aPrevMatchedFont).forget(); } // VS alone. it's meaningless to search different fonts return nullptr; } // The real fonts that fontconfig provides for generic/fallback families // depend on the language used, so a different FontSet is used for each // language (except for the variation below). // // With most fontconfig configurations any real family names prior to a // fontconfig generic with corresponding fonts installed will still lead // to the same leading fonts in each FontSet. // // There is an inefficiency here therefore because the same base FontSet // could often be used if these real families support the character. // However, with fontconfig aliases, it is difficult to distinguish // where exactly alias fonts end and generic/fallback fonts begin. // // The variation from pure language-based matching used here is that the // same primary/base font is always used irrespective of the language. // This provides that SCRIPT_COMMON characters are consistently rendered // with the same font (bug 339513 and bug 416725). This is particularly // important with the word cache as script can't be reliably determined // from surrounding words. It also often avoids the unnecessary extra // FontSet efficiency mentioned above. // // However, in two situations, the base font is not checked before the // language-specific FontSet. // // 1. When we don't have a language to make a good choice for // the base font. // // 2. For system fonts, use the default Pango behavior to give // consistency with other apps. This is relevant when un-localized // builds are run in non-Latin locales. This special-case probably // wouldn't be necessary but for bug 91190. gfxFcFontSet *fontSet = GetBaseFontSet(); uint32_t nextFont = 0; FcPattern *basePattern = nullptr; if (!mStyle.systemFont && mPangoLanguage) { basePattern = fontSet->GetFontPatternAt(0); if (HasChar(basePattern, aCh)) { *aMatchType = gfxTextRange::kFontGroup; return RefPtr<gfxFont>(GetBaseFont()).forget(); } nextFont = 1; } // Our MOZ_SCRIPT_* codes may not match the PangoScript enumeration values // (if we're using ICU's codes), so convert by mapping through ISO 15924 tag. // Note that PangoScript is defined to be compatible with GUnicodeScript: // https://developer.gnome.org/pango/stable/pango-Scripts-and-Languages.html#PangoScript const hb_tag_t scriptTag = GetScriptTagForCode(aRunScript); const PangoScript script = (const PangoScript)hb_glib_script_from_script(hb_script_from_iso15924_tag(scriptTag)); // Might be nice to call pango_language_includes_script only once for the // run rather than for each character. PangoLanguage *scriptLang; if ((!basePattern || !pango_language_includes_script(mPangoLanguage, script)) && (scriptLang = pango_script_get_sample_language(script))) { fontSet = GetFontSet(scriptLang); nextFont = 0; } for (uint32_t i = nextFont; FcPattern *pattern = fontSet->GetFontPatternAt(i); ++i) { if (pattern == basePattern) { continue; // already checked basePattern } if (HasChar(pattern, aCh)) { *aMatchType = gfxTextRange::kFontGroup; return RefPtr<gfxFont>(fontSet->GetFontAt(i, GetStyle())).forget(); } } return nullptr; } /** ** gfxFcFont **/ cairo_user_data_key_t gfxFcFont::sGfxFontKey; gfxFcFont::gfxFcFont(cairo_scaled_font_t *aCairoFont, FcPattern *aPattern, gfxFcFontEntry *aFontEntry, const gfxFontStyle *aFontStyle) : gfxFontconfigFontBase(aCairoFont, aPattern, aFontEntry, aFontStyle) { cairo_scaled_font_set_user_data(mScaledFont, &sGfxFontKey, this, nullptr); } gfxFcFont::~gfxFcFont() { cairo_scaled_font_set_user_data(mScaledFont, &sGfxFontKey, nullptr, nullptr); } already_AddRefed<gfxFont> gfxFcFont::GetSubSuperscriptFont(int32_t aAppUnitsPerDevPixel) { gfxFontStyle style(*GetStyle()); style.AdjustForSubSuperscript(aAppUnitsPerDevPixel); return MakeScaledFont(&style, style.size / GetStyle()->size); } already_AddRefed<gfxFont> gfxFcFont::MakeScaledFont(gfxFontStyle *aFontStyle, gfxFloat aScaleFactor) { gfxFcFontEntry* fe = static_cast<gfxFcFontEntry*>(GetFontEntry()); RefPtr<gfxFont> font = gfxFontCache::GetCache()->Lookup(fe, aFontStyle, nullptr); if (font) { return font.forget(); } cairo_matrix_t fontMatrix; cairo_scaled_font_get_font_matrix(mScaledFont, &fontMatrix); cairo_matrix_scale(&fontMatrix, aScaleFactor, aScaleFactor); cairo_matrix_t ctm; cairo_scaled_font_get_ctm(mScaledFont, &ctm); cairo_font_options_t *options = cairo_font_options_create(); cairo_scaled_font_get_font_options(mScaledFont, options); cairo_scaled_font_t *newFont = cairo_scaled_font_create(cairo_scaled_font_get_font_face(mScaledFont), &fontMatrix, &ctm, options); cairo_font_options_destroy(options); font = new gfxFcFont(newFont, GetPattern(), fe, aFontStyle); gfxFontCache::GetCache()->AddNew(font); cairo_scaled_font_destroy(newFont); return font.forget(); } already_AddRefed<gfxFont> gfxFcFont::GetSmallCapsFont() { gfxFontStyle style(*GetStyle()); style.size *= SMALL_CAPS_SCALE_FACTOR; style.variantCaps = NS_FONT_VARIANT_CAPS_NORMAL; return MakeScaledFont(&style, SMALL_CAPS_SCALE_FACTOR); } /* static */ void gfxPangoFontGroup::Shutdown() { // Resetting gFTLibrary in case this is wanted again after a // cairo_debug_reset_static_data. gFTLibrary = nullptr; } /* static */ gfxFontEntry * gfxPangoFontGroup::NewFontEntry(const nsAString& aFontName, uint16_t aWeight, int16_t aStretch, uint8_t aStyle) { gfxFontconfigUtils *utils = gfxFontconfigUtils::GetFontconfigUtils(); if (!utils) return nullptr; // The font face name from @font-face { src: local() } is not well // defined. // // On MS Windows, this name gets compared with // ENUMLOGFONTEXW::elfFullName, which for OpenType fonts seems to be the // full font name from the name table. For CFF OpenType fonts this is the // same as the PostScript name, but for TrueType fonts it is usually // different. // // On Mac, the font face name is compared with the PostScript name, even // for TrueType fonts. // // Fontconfig only records the full font names, so the behavior here // follows that on MS Windows. However, to provide the possibility // of aliases to compensate for variations, the font face name is passed // through FcConfigSubstitute. nsAutoRef<FcPattern> pattern(FcPatternCreate()); if (!pattern) return nullptr; NS_ConvertUTF16toUTF8 fullname(aFontName); FcPatternAddString(pattern, FC_FULLNAME, gfxFontconfigUtils::ToFcChar8(fullname)); FcConfigSubstitute(nullptr, pattern, FcMatchPattern); FcChar8 *name; for (int v = 0; FcPatternGetString(pattern, FC_FULLNAME, v, &name) == FcResultMatch; ++v) { const nsTArray< nsCountedRef<FcPattern> >& fonts = utils->GetFontsForFullname(name); if (fonts.Length() != 0) return new gfxLocalFcFontEntry(aFontName, aWeight, aStretch, aStyle, fonts); } return nullptr; } /* static */ FT_Library gfxPangoFontGroup::GetFTLibrary() { if (!gFTLibrary) { // Use cairo's FT_Library so that cairo takes care of shutdown of the // FT_Library after it has destroyed its font_faces, and FT_Done_Face // has been called on each FT_Face, at least until this bug is fixed: // https://bugs.freedesktop.org/show_bug.cgi?id=18857 // // Cairo's FT_Library can be obtained from any cairo_scaled_font. The // font properties requested here are chosen to get an FT_Face that is // likely to be also used elsewhere. gfxFontStyle style; RefPtr<gfxPangoFontGroup> fontGroup = new gfxPangoFontGroup(FontFamilyList(eFamily_sans_serif), &style, nullptr, 1.0); gfxFcFont *font = fontGroup->GetBaseFont(); if (!font) return nullptr; gfxFT2LockedFace face(font); if (!face.get()) return nullptr; gFTLibrary = face.get()->glyph->library; } return gFTLibrary; } /* static */ gfxFontEntry * gfxPangoFontGroup::NewFontEntry(const nsAString& aFontName, uint16_t aWeight, int16_t aStretch, uint8_t aStyle, const uint8_t* aFontData, uint32_t aLength) { // Ownership of aFontData is passed in here, and transferred to the // new fontEntry, which will release it when no longer needed. // Using face_index = 0 for the first face in the font, as we have no // other information. FT_New_Memory_Face checks for a nullptr FT_Library. FT_Face face; FT_Error error = FT_New_Memory_Face(GetFTLibrary(), aFontData, aLength, 0, &face); if (error != 0) { free((void*)aFontData); return nullptr; } return new gfxDownloadedFcFontEntry(aFontName, aWeight, aStretch, aStyle, aFontData, face); } static double GetPixelSize(FcPattern *aPattern) { double size; if (FcPatternGetDouble(aPattern, FC_PIXEL_SIZE, 0, &size) == FcResultMatch) return size; NS_NOTREACHED("No size on pattern"); return 0.0; } /** * The following gfxFcFonts are accessed from the cairo_scaled_font or created * from the FcPattern, not from the gfxFontCache hash table. The gfxFontCache * hash table is keyed by desired family and style, whereas here we only know * actual family and style. There may be more than one of these fonts with * the same family and style, but different PangoFont and actual font face. * * The point of this is to record the exact font face for gfxTextRun glyph * indices. The style of this font does not necessarily represent the exact * gfxFontStyle used to build the text run. Notably, the language is not * recorded. */ /* static */ already_AddRefed<gfxFcFont> gfxFcFont::GetOrMakeFont(FcPattern *aRequestedPattern, FcPattern *aFontPattern, const gfxFontStyle *aFontStyle) { nsAutoRef<FcPattern> renderPattern (FcFontRenderPrepare(nullptr, aRequestedPattern, aFontPattern)); // If synthetic bold/italic is not allowed by the style, adjust the // resulting pattern to match the actual properties of the font. if (!aFontStyle->allowSyntheticWeight) { int weight; if (FcPatternGetInteger(aFontPattern, FC_WEIGHT, 0, &weight) == FcResultMatch) { FcPatternDel(renderPattern, FC_WEIGHT); FcPatternAddInteger(renderPattern, FC_WEIGHT, weight); } } if (!aFontStyle->allowSyntheticStyle) { int slant; if (FcPatternGetInteger(aFontPattern, FC_SLANT, 0, &slant) == FcResultMatch) { FcPatternDel(renderPattern, FC_SLANT); FcPatternAddInteger(renderPattern, FC_SLANT, slant); } } cairo_font_face_t *face = cairo_ft_font_face_create_for_pattern(renderPattern); // Reuse an existing font entry if available. RefPtr<gfxFcFontEntry> fe = gfxFcFontEntry::LookupFontEntry(face); if (!fe) { gfxDownloadedFcFontEntry *downloadedFontEntry = GetDownloadedFontEntry(aFontPattern); if (downloadedFontEntry) { // Web font fe = downloadedFontEntry; if (cairo_font_face_status(face) == CAIRO_STATUS_SUCCESS) { // cairo_font_face_t is using the web font data. // Hold a reference to the font entry to keep the font face // data. if (!downloadedFontEntry->SetCairoFace(face)) { // OOM. Let cairo pick a fallback font cairo_font_face_destroy(face); face = cairo_ft_font_face_create_for_pattern(aRequestedPattern); fe = gfxFcFontEntry::LookupFontEntry(face); } } } if (!fe) { // Get a unique name for the font face from the file and id. nsAutoString name; FcChar8 *fc_file; if (FcPatternGetString(renderPattern, FC_FILE, 0, &fc_file) == FcResultMatch) { int index; if (FcPatternGetInteger(renderPattern, FC_INDEX, 0, &index) != FcResultMatch) { // cairo defaults to 0. index = 0; } AppendUTF8toUTF16(gfxFontconfigUtils::ToCString(fc_file), name); if (index != 0) { name.Append('/'); name.AppendInt(index); } } fe = new gfxSystemFcFontEntry(face, aFontPattern, name); } } gfxFontStyle style(*aFontStyle); style.size = GetPixelSize(renderPattern); style.style = gfxFontconfigUtils::GetThebesStyle(renderPattern); style.weight = gfxFontconfigUtils::GetThebesWeight(renderPattern); RefPtr<gfxFont> font = gfxFontCache::GetCache()->Lookup(fe, &style, nullptr); if (!font) { // Note that a file/index pair (or FT_Face) and the gfxFontStyle are // not necessarily enough to provide a key that will describe a unique // font. cairoFont contains information from renderPattern, which is a // fully resolved pattern from FcFontRenderPrepare. // FcFontRenderPrepare takes the requested pattern and the face // pattern as input and can modify elements of the resulting pattern // that affect rendering but are not included in the gfxFontStyle. cairo_scaled_font_t *cairoFont = CreateScaledFont(renderPattern, face); font = new gfxFcFont(cairoFont, renderPattern, fe, &style); gfxFontCache::GetCache()->AddNew(font); cairo_scaled_font_destroy(cairoFont); } cairo_font_face_destroy(face); RefPtr<gfxFcFont> retval(static_cast<gfxFcFont*>(font.get())); return retval.forget(); } gfxFcFontSet * gfxPangoFontGroup::GetBaseFontSet() { if (mFontSets.Length() > 0) return mFontSets[0].mFontSet; mSizeAdjustFactor = 1.0; // will be adjusted below if necessary nsAutoRef<FcPattern> pattern; RefPtr<gfxFcFontSet> fontSet = MakeFontSet(mPangoLanguage, mSizeAdjustFactor, &pattern); double size = GetPixelSize(pattern); if (size != 0.0 && mStyle.sizeAdjust > 0.0) { gfxFcFont *font = fontSet->GetFontAt(0, GetStyle()); if (font) { const gfxFont::Metrics& metrics = font->GetMetrics(gfxFont::eHorizontal); // XXX vertical? // The factor of 0.1 ensures that xHeight is sane so fonts don't // become huge. Strictly ">" ensures that xHeight and emHeight are // not both zero. if (metrics.xHeight > 0.1 * metrics.emHeight) { mSizeAdjustFactor = mStyle.sizeAdjust * metrics.emHeight / metrics.xHeight; size *= mSizeAdjustFactor; FcPatternDel(pattern, FC_PIXEL_SIZE); FcPatternAddDouble(pattern, FC_PIXEL_SIZE, size); fontSet = new gfxFcFontSet(pattern, mUserFontSet); } } } PangoLanguage *pangoLang = mPangoLanguage; FcChar8 *fcLang; if (!pangoLang && FcPatternGetString(pattern, FC_LANG, 0, &fcLang) == FcResultMatch) { pangoLang = pango_language_from_string(gfxFontconfigUtils::ToCString(fcLang)); } mFontSets.AppendElement(FontSetByLangEntry(pangoLang, fontSet)); return fontSet; } /** ** gfxTextRun * * A serious problem: * * -- We draw with a font that's hinted for the CTM, but we measure with a font * hinted to the identity matrix, so our "bounding metrics" may not be accurate. * **/ // This will fetch an existing scaled_font if one exists. static cairo_scaled_font_t * CreateScaledFont(FcPattern *aPattern, cairo_font_face_t *aFace) { double size = GetPixelSize(aPattern); cairo_matrix_t fontMatrix; FcMatrix *fcMatrix; if (FcPatternGetMatrix(aPattern, FC_MATRIX, 0, &fcMatrix) == FcResultMatch) cairo_matrix_init(&fontMatrix, fcMatrix->xx, -fcMatrix->yx, -fcMatrix->xy, fcMatrix->yy, 0, 0); else cairo_matrix_init_identity(&fontMatrix); cairo_matrix_scale(&fontMatrix, size, size); FcBool printing; if (FcPatternGetBool(aPattern, PRINTING_FC_PROPERTY, 0, &printing) != FcResultMatch) { printing = FcFalse; } // The cairo_scaled_font is created with a unit ctm so that metrics and // positions are in user space, but this means that hinting effects will // not be estimated accurately for non-unit transformations. cairo_matrix_t identityMatrix; cairo_matrix_init_identity(&identityMatrix); // Font options are set explicitly here to improve cairo's caching // behavior and to record the relevant parts of the pattern for // SetupCairoFont (so that the pattern can be released). // // Most font_options have already been set as defaults on the FcPattern // with cairo_ft_font_options_substitute(), then user and system // fontconfig configurations were applied. The resulting font_options // have been recorded on the face during // cairo_ft_font_face_create_for_pattern(). // // None of the settings here cause this scaled_font to behave any // differently from how it would behave if it were created from the same // face with default font_options. // // We set options explicitly so that the same scaled_font will be found in // the cairo_scaled_font_map when cairo loads glyphs from a context with // the same font_face, font_matrix, ctm, and surface font_options. // // Unfortunately, _cairo_scaled_font_keys_equal doesn't know about the // font_options on the cairo_ft_font_face, and doesn't consider default // option values to not match any explicit values. // // Even after cairo_set_scaled_font is used to set font_options for the // cairo context, when cairo looks for a scaled_font for the context, it // will look for a font with some option values from the target surface if // any values are left default on the context font_options. If this // scaled_font is created with default font_options, cairo will not find // it. cairo_font_options_t *fontOptions = cairo_font_options_create(); // The one option not recorded in the pattern is hint_metrics, which will // affect glyph metrics. The default behaves as CAIRO_HINT_METRICS_ON. // We should be considering the font_options of the surface on which this // font will be used, but currently we don't have different gfxFonts for // different surface font_options, so we'll create a font suitable for the // Screen. Image and xlib surfaces default to CAIRO_HINT_METRICS_ON. if (printing) { cairo_font_options_set_hint_metrics(fontOptions, CAIRO_HINT_METRICS_OFF); } else { cairo_font_options_set_hint_metrics(fontOptions, CAIRO_HINT_METRICS_ON); } // The remaining options have been recorded on the pattern and the face. // _cairo_ft_options_merge has some logic to decide which options from the // scaled_font or from the cairo_ft_font_face take priority in the way the // font behaves. // // In the majority of cases, _cairo_ft_options_merge uses the options from // the cairo_ft_font_face, so sometimes it is not so important which // values are set here so long as they are not defaults, but we'll set // them to the exact values that we expect from the font, to be consistent // and to protect against changes in cairo. // // In some cases, _cairo_ft_options_merge uses some options from the // scaled_font's font_options rather than options on the // cairo_ft_font_face (from fontconfig). // https://bugs.freedesktop.org/show_bug.cgi?id=11838 // // Surface font options were set on the pattern in // cairo_ft_font_options_substitute. If fontconfig has changed the // hint_style then that is what the user (or distribution) wants, so we // use the setting from the FcPattern. // // Fallback values here mirror treatment of defaults in cairo-ft-font.c. FcBool hinting = FcFalse; if (FcPatternGetBool(aPattern, FC_HINTING, 0, &hinting) != FcResultMatch) { hinting = FcTrue; } cairo_hint_style_t hint_style; if (printing || !hinting) { hint_style = CAIRO_HINT_STYLE_NONE; } else { #ifdef FC_HINT_STYLE // FC_HINT_STYLE is available from fontconfig 2.2.91. int fc_hintstyle; if (FcPatternGetInteger(aPattern, FC_HINT_STYLE, 0, &fc_hintstyle ) != FcResultMatch) { fc_hintstyle = FC_HINT_FULL; } switch (fc_hintstyle) { case FC_HINT_NONE: hint_style = CAIRO_HINT_STYLE_NONE; break; case FC_HINT_SLIGHT: hint_style = CAIRO_HINT_STYLE_SLIGHT; break; case FC_HINT_MEDIUM: default: // This fallback mirrors _get_pattern_ft_options in cairo. hint_style = CAIRO_HINT_STYLE_MEDIUM; break; case FC_HINT_FULL: hint_style = CAIRO_HINT_STYLE_FULL; break; } #else // no FC_HINT_STYLE hint_style = CAIRO_HINT_STYLE_FULL; #endif } cairo_font_options_set_hint_style(fontOptions, hint_style); int rgba; if (FcPatternGetInteger(aPattern, FC_RGBA, 0, &rgba) != FcResultMatch) { rgba = FC_RGBA_UNKNOWN; } cairo_subpixel_order_t subpixel_order = CAIRO_SUBPIXEL_ORDER_DEFAULT; switch (rgba) { case FC_RGBA_UNKNOWN: case FC_RGBA_NONE: default: // There is no CAIRO_SUBPIXEL_ORDER_NONE. Subpixel antialiasing // is disabled through cairo_antialias_t. rgba = FC_RGBA_NONE; // subpixel_order won't be used by the font as we won't use // CAIRO_ANTIALIAS_SUBPIXEL, but don't leave it at default for // caching reasons described above. Fall through: MOZ_FALLTHROUGH; case FC_RGBA_RGB: subpixel_order = CAIRO_SUBPIXEL_ORDER_RGB; break; case FC_RGBA_BGR: subpixel_order = CAIRO_SUBPIXEL_ORDER_BGR; break; case FC_RGBA_VRGB: subpixel_order = CAIRO_SUBPIXEL_ORDER_VRGB; break; case FC_RGBA_VBGR: subpixel_order = CAIRO_SUBPIXEL_ORDER_VBGR; break; } cairo_font_options_set_subpixel_order(fontOptions, subpixel_order); FcBool fc_antialias; if (FcPatternGetBool(aPattern, FC_ANTIALIAS, 0, &fc_antialias) != FcResultMatch) { fc_antialias = FcTrue; } cairo_antialias_t antialias; if (!fc_antialias) { antialias = CAIRO_ANTIALIAS_NONE; } else if (rgba == FC_RGBA_NONE) { antialias = CAIRO_ANTIALIAS_GRAY; } else { antialias = CAIRO_ANTIALIAS_SUBPIXEL; } cairo_font_options_set_antialias(fontOptions, antialias); cairo_scaled_font_t *scaledFont = cairo_scaled_font_create(aFace, &fontMatrix, &identityMatrix, fontOptions); cairo_font_options_destroy(fontOptions); NS_ASSERTION(cairo_scaled_font_status(scaledFont) == CAIRO_STATUS_SUCCESS, "Failed to create scaled font"); return scaledFont; } /* static */ PangoLanguage * GuessPangoLanguage(nsIAtom *aLanguage) { if (!aLanguage) return nullptr; // Pango and fontconfig won't understand mozilla's internal langGroups, so // find a real language. nsAutoCString lang; gfxFontconfigUtils::GetSampleLangForGroup(aLanguage, &lang); if (lang.IsEmpty()) return nullptr; return pango_language_from_string(lang.get()); } #ifdef MOZ_WIDGET_GTK /*************************************************************************** * * This function must be last in the file because it uses the system cairo * library. Above this point the cairo library used is the tree cairo if * MOZ_TREE_CAIRO. */ #if MOZ_TREE_CAIRO // Tree cairo symbols have different names. Disable their activation through // preprocessor macros. #undef cairo_ft_font_options_substitute // The system cairo functions are not declared because the include paths cause // the gdk headers to pick up the tree cairo.h. extern "C" { NS_VISIBILITY_DEFAULT void cairo_ft_font_options_substitute (const cairo_font_options_t *options, FcPattern *pattern); } #endif static void ApplyGdkScreenFontOptions(FcPattern *aPattern) { const cairo_font_options_t *options = gdk_screen_get_font_options(gdk_screen_get_default()); cairo_ft_font_options_substitute(options, aPattern); } #endif // MOZ_WIDGET_GTK