| // Copyright 2019 Google LLC. |
| #include <memory> |
| |
| #include "modules/skparagraph/include/FontArguments.h" |
| #include "modules/skparagraph/include/ParagraphCache.h" |
| #include "modules/skparagraph/src/ParagraphImpl.h" |
| |
| using namespace skia_private; |
| |
| namespace skia { |
| namespace textlayout { |
| |
| namespace { |
| int32_t relax(SkScalar a) { |
| // This rounding is done to match Flutter tests. Must be removed.. |
| if (SkScalarIsFinite(a)) { |
| auto threshold = SkIntToScalar(1 << 12); |
| return SkFloat2Bits(SkScalarRoundToScalar(a * threshold)/threshold); |
| } else { |
| return SkFloat2Bits(a); |
| } |
| } |
| |
| bool exactlyEqual(SkScalar x, SkScalar y) { |
| return x == y || (x != x && y != y); |
| } |
| |
| } // namespace |
| |
| class ParagraphCacheKey { |
| public: |
| ParagraphCacheKey(const ParagraphImpl* paragraph) |
| : fText(paragraph->fText.c_str(), paragraph->fText.size()) |
| , fPlaceholders(paragraph->fPlaceholders) |
| , fTextStyles(paragraph->fTextStyles) |
| , fParagraphStyle(paragraph->paragraphStyle()) { |
| fHash = computeHash(); |
| } |
| |
| ParagraphCacheKey(const ParagraphCacheKey& other) = default; |
| |
| ParagraphCacheKey(ParagraphCacheKey&& other) |
| : fText(std::move(other.fText)) |
| , fPlaceholders(std::move(other.fPlaceholders)) |
| , fTextStyles(std::move(other.fTextStyles)) |
| , fParagraphStyle(std::move(other.fParagraphStyle)) |
| , fHash(other.fHash) { |
| other.fHash = 0; |
| } |
| |
| bool operator==(const ParagraphCacheKey& other) const; |
| |
| uint32_t hash() const { return fHash; } |
| |
| const SkString& text() const { return fText; } |
| |
| private: |
| static uint32_t mix(uint32_t hash, uint32_t data); |
| uint32_t computeHash() const; |
| |
| SkString fText; |
| TArray<Placeholder, true> fPlaceholders; |
| TArray<Block, true> fTextStyles; |
| ParagraphStyle fParagraphStyle; |
| uint32_t fHash; |
| }; |
| |
| class ParagraphCacheValue { |
| public: |
| ParagraphCacheValue(ParagraphCacheKey&& key, const ParagraphImpl* paragraph) |
| : fKey(std::move(key)) |
| , fRuns(paragraph->fRuns) |
| , fClusters(paragraph->fClusters) |
| , fClustersIndexFromCodeUnit(paragraph->fClustersIndexFromCodeUnit) |
| , fCodeUnitProperties(paragraph->fCodeUnitProperties) |
| , fWords(paragraph->fWords) |
| , fBidiRegions(paragraph->fBidiRegions) |
| , fHasLineBreaks(paragraph->fHasLineBreaks) |
| , fHasWhitespacesInside(paragraph->fHasWhitespacesInside) |
| , fTrailingSpaces(paragraph->fTrailingSpaces) { } |
| |
| // Input == key |
| ParagraphCacheKey fKey; |
| |
| // Shaped results |
| TArray<Run, false> fRuns; |
| TArray<Cluster, true> fClusters; |
| TArray<size_t, true> fClustersIndexFromCodeUnit; |
| // ICU results |
| TArray<SkUnicode::CodeUnitFlags, true> fCodeUnitProperties; |
| std::vector<size_t> fWords; |
| std::vector<SkUnicode::BidiRegion> fBidiRegions; |
| bool fHasLineBreaks; |
| bool fHasWhitespacesInside; |
| TextIndex fTrailingSpaces; |
| }; |
| |
| uint32_t ParagraphCacheKey::mix(uint32_t hash, uint32_t data) { |
| hash += data; |
| hash += (hash << 10); |
| hash ^= (hash >> 6); |
| return hash; |
| } |
| |
| uint32_t ParagraphCacheKey::computeHash() const { |
| uint32_t hash = 0; |
| for (auto& ph : fPlaceholders) { |
| if (ph.fRange.width() == 0) { |
| continue; |
| } |
| hash = mix(hash, SkGoodHash()(ph.fRange)); |
| hash = mix(hash, SkGoodHash()(relax(ph.fStyle.fHeight))); |
| hash = mix(hash, SkGoodHash()(relax(ph.fStyle.fWidth))); |
| hash = mix(hash, SkGoodHash()(ph.fStyle.fAlignment)); |
| hash = mix(hash, SkGoodHash()(ph.fStyle.fBaseline)); |
| if (ph.fStyle.fAlignment == PlaceholderAlignment::kBaseline) { |
| hash = mix(hash, SkGoodHash()(relax(ph.fStyle.fBaselineOffset))); |
| } |
| } |
| |
| for (auto& ts : fTextStyles) { |
| if (ts.fStyle.isPlaceholder()) { |
| continue; |
| } |
| hash = mix(hash, SkGoodHash()(relax(ts.fStyle.getLetterSpacing()))); |
| hash = mix(hash, SkGoodHash()(relax(ts.fStyle.getWordSpacing()))); |
| hash = mix(hash, SkGoodHash()(ts.fStyle.getLocale())); |
| hash = mix(hash, SkGoodHash()(relax(ts.fStyle.getHeight()))); |
| hash = mix(hash, SkGoodHash()(relax(ts.fStyle.getBaselineShift()))); |
| for (auto& ff : ts.fStyle.getFontFamilies()) { |
| hash = mix(hash, SkGoodHash()(ff)); |
| } |
| for (auto& ff : ts.fStyle.getFontFeatures()) { |
| hash = mix(hash, SkGoodHash()(ff.fValue)); |
| hash = mix(hash, SkGoodHash()(ff.fName)); |
| } |
| hash = mix(hash, std::hash<std::optional<FontArguments>>()(ts.fStyle.getFontArguments())); |
| hash = mix(hash, SkGoodHash()(ts.fStyle.getFontStyle())); |
| hash = mix(hash, SkGoodHash()(relax(ts.fStyle.getFontSize()))); |
| hash = mix(hash, SkGoodHash()(ts.fRange)); |
| } |
| |
| hash = mix(hash, SkGoodHash()(relax(fParagraphStyle.getHeight()))); |
| hash = mix(hash, SkGoodHash()(fParagraphStyle.getTextDirection())); |
| hash = mix(hash, SkGoodHash()(fParagraphStyle.getReplaceTabCharacters() ? 1 : 0)); |
| |
| auto& strutStyle = fParagraphStyle.getStrutStyle(); |
| if (strutStyle.getStrutEnabled()) { |
| hash = mix(hash, SkGoodHash()(relax(strutStyle.getHeight()))); |
| hash = mix(hash, SkGoodHash()(relax(strutStyle.getLeading()))); |
| hash = mix(hash, SkGoodHash()(relax(strutStyle.getFontSize()))); |
| hash = mix(hash, SkGoodHash()(strutStyle.getHeightOverride())); |
| hash = mix(hash, SkGoodHash()(strutStyle.getFontStyle())); |
| hash = mix(hash, SkGoodHash()(strutStyle.getForceStrutHeight())); |
| for (auto& ff : strutStyle.getFontFamilies()) { |
| hash = mix(hash, SkGoodHash()(ff)); |
| } |
| } |
| |
| hash = mix(hash, SkGoodHash()(fText)); |
| return hash; |
| } |
| |
| uint32_t ParagraphCache::KeyHash::operator()(const ParagraphCacheKey& key) const { |
| return key.hash(); |
| } |
| |
| bool ParagraphCacheKey::operator==(const ParagraphCacheKey& other) const { |
| if (fText.size() != other.fText.size()) { |
| return false; |
| } |
| if (fPlaceholders.size() != other.fPlaceholders.size()) { |
| return false; |
| } |
| if (fText != other.fText) { |
| return false; |
| } |
| if (fTextStyles.size() != other.fTextStyles.size()) { |
| return false; |
| } |
| |
| // There is no need to compare default paragraph styles - they are included into fTextStyles |
| if (!exactlyEqual(fParagraphStyle.getHeight(), other.fParagraphStyle.getHeight())) { |
| return false; |
| } |
| if (fParagraphStyle.getTextDirection() != other.fParagraphStyle.getTextDirection()) { |
| return false; |
| } |
| |
| if (!(fParagraphStyle.getStrutStyle() == other.fParagraphStyle.getStrutStyle())) { |
| return false; |
| } |
| |
| if (!(fParagraphStyle.getReplaceTabCharacters() == other.fParagraphStyle.getReplaceTabCharacters())) { |
| return false; |
| } |
| |
| for (int i = 0; i < fTextStyles.size(); ++i) { |
| auto& tsa = fTextStyles[i]; |
| auto& tsb = other.fTextStyles[i]; |
| if (tsa.fStyle.isPlaceholder()) { |
| continue; |
| } |
| if (!(tsa.fStyle.equalsByFonts(tsb.fStyle))) { |
| return false; |
| } |
| if (tsa.fRange.width() != tsb.fRange.width()) { |
| return false; |
| } |
| if (tsa.fRange.start != tsb.fRange.start) { |
| return false; |
| } |
| } |
| for (int i = 0; i < fPlaceholders.size(); ++i) { |
| auto& tsa = fPlaceholders[i]; |
| auto& tsb = other.fPlaceholders[i]; |
| if (tsa.fRange.width() == 0 && tsb.fRange.width() == 0) { |
| continue; |
| } |
| if (!(tsa.fStyle.equals(tsb.fStyle))) { |
| return false; |
| } |
| if (tsa.fRange.width() != tsb.fRange.width()) { |
| return false; |
| } |
| if (tsa.fRange.start != tsb.fRange.start) { |
| return false; |
| } |
| } |
| |
| return true; |
| } |
| |
| struct ParagraphCache::Entry { |
| |
| Entry(ParagraphCacheValue* value) : fValue(value) {} |
| std::unique_ptr<ParagraphCacheValue> fValue; |
| }; |
| |
| ParagraphCache::ParagraphCache() |
| : fChecker([](ParagraphImpl* impl, const char*, bool){ }) |
| , fLRUCacheMap(kMaxEntries) |
| , fCacheIsOn(true) |
| , fLastCachedValue(nullptr) |
| #ifdef PARAGRAPH_CACHE_STATS |
| , fTotalRequests(0) |
| , fCacheMisses(0) |
| , fHashMisses(0) |
| #endif |
| { } |
| |
| ParagraphCache::~ParagraphCache() { } |
| |
| void ParagraphCache::updateTo(ParagraphImpl* paragraph, const Entry* entry) { |
| |
| paragraph->fRuns.clear(); |
| paragraph->fRuns = entry->fValue->fRuns; |
| paragraph->fClusters = entry->fValue->fClusters; |
| paragraph->fClustersIndexFromCodeUnit = entry->fValue->fClustersIndexFromCodeUnit; |
| paragraph->fCodeUnitProperties = entry->fValue->fCodeUnitProperties; |
| paragraph->fWords = entry->fValue->fWords; |
| paragraph->fBidiRegions = entry->fValue->fBidiRegions; |
| paragraph->fHasLineBreaks = entry->fValue->fHasLineBreaks; |
| paragraph->fHasWhitespacesInside = entry->fValue->fHasWhitespacesInside; |
| paragraph->fTrailingSpaces = entry->fValue->fTrailingSpaces; |
| for (auto& run : paragraph->fRuns) { |
| run.setOwner(paragraph); |
| } |
| for (auto& cluster : paragraph->fClusters) { |
| cluster.setOwner(paragraph); |
| } |
| } |
| |
| void ParagraphCache::printStatistics() { |
| SkDebugf("--- Paragraph Cache ---\n"); |
| SkDebugf("Total requests: %d\n", fTotalRequests); |
| SkDebugf("Cache misses: %d\n", fCacheMisses); |
| SkDebugf("Cache miss %%: %f\n", (fTotalRequests > 0) ? 100.f * fCacheMisses / fTotalRequests : 0.f); |
| int cacheHits = fTotalRequests - fCacheMisses; |
| SkDebugf("Hash miss %%: %f\n", (cacheHits > 0) ? 100.f * fHashMisses / cacheHits : 0.f); |
| SkDebugf("---------------------\n"); |
| } |
| |
| void ParagraphCache::abandon() { |
| this->reset(); |
| } |
| |
| void ParagraphCache::reset() { |
| SkAutoMutexExclusive lock(fParagraphMutex); |
| #ifdef PARAGRAPH_CACHE_STATS |
| fTotalRequests = 0; |
| fCacheMisses = 0; |
| fHashMisses = 0; |
| #endif |
| fLRUCacheMap.reset(); |
| fLastCachedValue = nullptr; |
| } |
| |
| bool ParagraphCache::findParagraph(ParagraphImpl* paragraph) { |
| if (!fCacheIsOn) { |
| return false; |
| } |
| #ifdef PARAGRAPH_CACHE_STATS |
| ++fTotalRequests; |
| #endif |
| SkAutoMutexExclusive lock(fParagraphMutex); |
| ParagraphCacheKey key(paragraph); |
| std::unique_ptr<Entry>* entry = fLRUCacheMap.find(key); |
| |
| if (!entry) { |
| // We have a cache miss |
| #ifdef PARAGRAPH_CACHE_STATS |
| ++fCacheMisses; |
| #endif |
| fChecker(paragraph, "missingParagraph", true); |
| return false; |
| } |
| updateTo(paragraph, entry->get()); |
| fChecker(paragraph, "foundParagraph", true); |
| return true; |
| } |
| |
| bool ParagraphCache::updateParagraph(ParagraphImpl* paragraph) { |
| if (!fCacheIsOn) { |
| return false; |
| } |
| #ifdef PARAGRAPH_CACHE_STATS |
| ++fTotalRequests; |
| #endif |
| SkAutoMutexExclusive lock(fParagraphMutex); |
| |
| ParagraphCacheKey key(paragraph); |
| std::unique_ptr<Entry>* entry = fLRUCacheMap.find(key); |
| if (!entry) { |
| // isTooMuchMemoryWasted(paragraph) not needed for now |
| if (isPossiblyTextEditing(paragraph)) { |
| // Skip this paragraph |
| return false; |
| } |
| ParagraphCacheValue* value = new ParagraphCacheValue(std::move(key), paragraph); |
| fLRUCacheMap.insert(value->fKey, std::make_unique<Entry>(value)); |
| fChecker(paragraph, "addedParagraph", true); |
| fLastCachedValue = value; |
| return true; |
| } else { |
| // We do not have to update the paragraph |
| return false; |
| } |
| } |
| |
| // Special situation: (very) long paragraph that is close to the last formatted paragraph |
| #define NOCACHE_PREFIX_LENGTH 40 |
| bool ParagraphCache::isPossiblyTextEditing(ParagraphImpl* paragraph) { |
| if (fLastCachedValue == nullptr) { |
| return false; |
| } |
| |
| auto& lastText = fLastCachedValue->fKey.text(); |
| auto& text = paragraph->fText; |
| |
| if ((lastText.size() < NOCACHE_PREFIX_LENGTH) || (text.size() < NOCACHE_PREFIX_LENGTH)) { |
| // Either last text or the current are too short |
| return false; |
| } |
| |
| if (std::strncmp(lastText.c_str(), text.c_str(), NOCACHE_PREFIX_LENGTH) == 0) { |
| // Texts have the same starts |
| return true; |
| } |
| |
| if (std::strncmp(lastText.c_str() + lastText.size() - NOCACHE_PREFIX_LENGTH, &text[text.size() - NOCACHE_PREFIX_LENGTH], NOCACHE_PREFIX_LENGTH) == 0) { |
| // Texts have the same ends |
| return true; |
| } |
| |
| // It does not look like editing the text |
| return false; |
| } |
| } // namespace textlayout |
| } // namespace skia |