/* * Copyright (C) 2013 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef MINIKIN_FONT_COLLECTION_H #define MINIKIN_FONT_COLLECTION_H #include #include #include #include #include #include #include #include "minikin/Buffer.h" #include "minikin/Font.h" #include "minikin/FontFamily.h" #include "minikin/Hasher.h" #include "minikin/MinikinExtent.h" #include "minikin/MinikinFont.h" #include "minikin/U16StringPiece.h" namespace minikin { // The maximum number of font families. constexpr uint32_t MAX_FAMILY_COUNT = 254; struct LocaleExtentKey { uint32_t localeId; float textSize; bool operator==(const LocaleExtentKey& o) const { return localeId == o.localeId && textSize == o.textSize; } }; inline android::hash_t hash_type(const LocaleExtentKey& key) { return Hasher().update(key.localeId).update(key.textSize).hash(); } class LocaleList; class FontCollection { public: static std::shared_ptr create( const std::vector>& typefaces); static std::shared_ptr create(std::shared_ptr&& typeface); static std::vector> readVector(BufferReader* reader); static void writeVector(BufferWriter* writer, const std::vector>& fontCollections); // Helper class for representing font family match result in packed bits. struct FamilyMatchResult { public: struct Builder { public: Builder() : mSize(0), mBits(0) {} Builder& add(uint8_t x) { if (mSize >= 7) [[unlikely]] { return *this; } mBits = mBits | (static_cast(x) << (8 * mSize)); mSize++; return *this; } Builder& reset() { mSize = 0; mBits = 0; return *this; } uint8_t size() const { return mSize; } bool empty() const { return size() == 0; } FamilyMatchResult build() { return FamilyMatchResult(mBits | (static_cast(mSize) << 56)); } private: uint8_t mSize; uint64_t mBits; }; // Helper class for iterating FamilyMatchResult class iterator { public: inline bool operator==(const iterator& o) const { return mOffset == o.mOffset && mResult == o.mResult; } inline bool operator!=(const iterator& o) const { return !(*this == o); } inline uint8_t operator*() const { return mResult[mOffset]; } inline iterator& operator++() { mOffset++; return *this; } private: friend struct FamilyMatchResult; iterator(const FamilyMatchResult& result, uint32_t offset) : mResult(result), mOffset(offset) {} const FamilyMatchResult& mResult; uint32_t mOffset; }; // Create empty FamilyMatchResult. FamilyMatchResult() : mBits(0) {} inline uint8_t size() const { return static_cast(mBits >> 56); } inline uint8_t operator[](uint32_t pos) const { return static_cast(mBits >> (pos * 8)); } inline bool empty() const { return size() == 0; } inline bool operator==(const FamilyMatchResult& o) const { return mBits == o.mBits; } // Returns the common family indices between l and r. static FamilyMatchResult intersect(FamilyMatchResult l, FamilyMatchResult r); // Iterator inline iterator begin() const { return iterator(*this, 0); } inline iterator end() const { return iterator(*this, size()); } FamilyMatchResult(const FamilyMatchResult& o) = default; FamilyMatchResult& operator=(const FamilyMatchResult& o) = default; private: explicit FamilyMatchResult(uint64_t bits) : mBits(bits) {} uint64_t mBits; }; struct Run { FamilyMatchResult familyMatch; int start; int end; }; FakedFont getBestFont(U16StringPiece textBuf, const Run& run, FontStyle style); // Perform the itemization until given max runs. std::vector itemize(U16StringPiece text, FontStyle style, uint32_t localeListId, FamilyVariant familyVariant, uint32_t runMax) const; // Perform the itemization until end of the text. std::vector itemize(U16StringPiece text, FontStyle style, uint32_t localeListId, FamilyVariant familyVariant) const { return itemize(text, style, localeListId, familyVariant, text.size()); } MinikinExtent getReferenceExtentForLocale(const MinikinPaint& paint) const; // Returns true if there is a glyph for the code point and variation selector pair. // Returns false if no fonts have a glyph for the code point and variation // selector pair, or invalid variation selector is passed. bool hasVariationSelector(uint32_t baseCodepoint, uint32_t variationSelector) const; // Get base font with fakery information (fake bold could affect metrics) FakedFont baseFontFaked(FontStyle style); // Creates new FontCollection based on this collection while applying font variations. Returns // nullptr if none of variations apply to this collection. std::shared_ptr createCollectionWithVariation( const std::vector& variations); // Creates new FontCollection that uses the specified families as top families and // families from this FontCollection as fallback. std::shared_ptr createCollectionWithFamilies( std::vector>&& families) const; size_t getSupportedAxesCount() const { return mSupportedAxesCount; } AxisTag getSupportedAxisAt(size_t index) const { return mSupportedAxes[index]; } uint32_t getId() const; size_t getFamilyCount() const { return mFamilyCount; } const std::shared_ptr& getFamilyAt(size_t index) const { if (mFamilyIndices != nullptr) { index = mFamilyIndices[index]; } return (*mMaybeSharedFamilies)[index]; } private: FRIEND_TEST(FontCollectionTest, bufferTest); explicit FontCollection(const std::vector>& typefaces); FontCollection( BufferReader* reader, const std::shared_ptr>>& allFontFamilies); // Write fields of the instance, using fontFamilyToIndexMap for finding // indices for FontFamily. void writeTo(BufferWriter* writer, const std::unordered_map, uint32_t>& fontFamilyToIndexMap) const; static void collectAllFontFamilies( const std::vector>& fontCollections, std::vector>* outAllFontFamilies, std::unordered_map, uint32_t>* outFontFamilyToIndexMap); static const int kLogCharsPerPage = 8; static const int kPageMask = (1 << kLogCharsPerPage) - 1; // mFamilyVec holds the indices of the family (as in getFamilyAt()) and // mRanges holds the range of indices of mFamilyVec. // The maximum number of pages is 0x10FF (U+10FFFF >> 8). The maximum number of // the fonts is 0xFF. Thus, technically the maximum length of mFamilyVec is 0x10EE01 // (0x10FF * 0xFF). However, in practice, 16-bit integers are enough since most fonts supports // only limited range of code points. struct Range { uint16_t start; uint16_t end; }; // Initialize the FontCollection. void init(const std::vector>& typefaces); FamilyMatchResult getFamilyForChar(uint32_t ch, uint32_t vs, uint32_t localeListId, FamilyVariant variant) const; uint32_t calcFamilyScore(uint32_t ch, uint32_t vs, FamilyVariant variant, uint32_t localeListId, const std::shared_ptr& fontFamily) const; uint32_t calcCoverageScore(uint32_t ch, uint32_t vs, uint32_t localeListId, const std::shared_ptr& fontFamily) const; bool isPrimaryFamily(const std::shared_ptr& fontFamily) const; void filterFamilyByLocale(const LocaleList& localeList, const std::function& callback) const; static uint32_t calcLocaleMatchingScore(uint32_t userLocaleListId, const FontFamily& fontFamily); static uint32_t calcVariantMatchingScore(FamilyVariant variant, const FontFamily& fontFamily); // unique id for this font collection (suitable for cache key) uint32_t mId; // Highest UTF-32 code point that can be mapped uint32_t mMaxChar; // This vector has pointers to the all font family instances in this collection. // This vector can't be empty. // This vector may be shared with other font collections. // (1) When shared, this vector is a union of all font family instances // shared by multiple font collections. // mFamilyIndices will be non-null in this case. // The i-th family in this collection will be // mMaybeSharedFamilies[mFamilyIndices[i]]. // (2) When not shared, mFamilyIndices will be null and // the i-th family in this collection will be mMaybeSharedFamilies[i]. // Use getFamilyAt(i) to access the i-th font in this family. std::shared_ptr>> mMaybeSharedFamilies; uint32_t mFamilyCount; const uint32_t* mFamilyIndices; // Following two vectors are pre-calculated tables for resolving coverage faster. // For example, to iterate over all fonts which support Unicode code point U+XXYYZZ, // iterate font families index from mFamilyVec[mRanges[0xXXYY].start] to // mFamilyVec[mRange[0xXXYY].end] instead of whole mFamilies. // This vector contains indices into mFamilies. // This vector can't be empty. uint32_t mRangesCount; const Range* mRanges; uint32_t mFamilyVecCount; const uint8_t* mFamilyVec; // This vector has pointers to the font family instances which have cmap 14 subtables. std::vector> mVSFamilyVec; // Set of supported axes in this collection. uint32_t mSupportedAxesCount; // mSupportedAxes is sorted. std::unique_ptr mSupportedAxes; // Owns allocated memory if this class is created from font families, otherwise these are // nullptr. std::unique_ptr mOwnedRanges; std::vector mOwnedFamilyVec; // Pair of locale ID and text size. mutable std::mutex mMutex; mutable android::LruCache mExtentCacheForLocale GUARDED_BY(mMutex){8}; }; } // namespace minikin #endif // MINIKIN_FONT_COLLECTION_H