| /* |
| * Copyright 2011 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "src/core/SkTypefaceCache.h" |
| |
| #include "include/core/SkFontStyle.h" |
| #include "include/core/SkGraphics.h" |
| #include "include/core/SkString.h" |
| #include "include/private/base/SkDebug.h" |
| #include "include/private/base/SkMutex.h" |
| |
| #include <atomic> |
| #include <cstdint> |
| #include <utility> |
| |
| SkTypefaceCache::SkTypefaceCache() {} |
| |
| void SkTypefaceCache::add(sk_sp<SkTypeface> face) { |
| const auto limit = SkGraphics::GetTypefaceCacheCountLimit(); |
| |
| if (fTypefaces.size() >= limit) { |
| this->purge(limit >> 2); |
| } |
| if (limit > 0) { |
| fTypefaces.emplace_back(std::move(face)); |
| } |
| } |
| |
| sk_sp<SkTypeface> SkTypefaceCache::findByProcAndRef(FindProc proc, void* ctx) const { |
| for (const sk_sp<SkTypeface>& typeface : fTypefaces) { |
| if (proc(typeface.get(), ctx)) { |
| return typeface; |
| } |
| } |
| return nullptr; |
| } |
| |
| void SkTypefaceCache::purge(int numToPurge) { |
| int count = fTypefaces.size(); |
| int i = 0; |
| while (i < count) { |
| if (fTypefaces[i]->unique()) { |
| fTypefaces.removeShuffle(i); |
| --count; |
| if (--numToPurge == 0) { |
| return; |
| } |
| } else { |
| ++i; |
| } |
| } |
| } |
| |
| void SkTypefaceCache::purgeAll() { |
| this->purge(fTypefaces.size()); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| SkTypefaceCache& SkTypefaceCache::Get() { |
| static SkTypefaceCache gCache; |
| return gCache; |
| } |
| |
| SkTypefaceID SkTypefaceCache::NewTypefaceID() { |
| static std::atomic<int32_t> nextID{1}; |
| return nextID.fetch_add(1, std::memory_order_relaxed); |
| } |
| |
| static SkMutex& typeface_cache_mutex() { |
| static SkMutex& mutex = *(new SkMutex); |
| return mutex; |
| } |
| |
| void SkTypefaceCache::Add(sk_sp<SkTypeface> face) { |
| SkAutoMutexExclusive ama(typeface_cache_mutex()); |
| Get().add(std::move(face)); |
| } |
| |
| sk_sp<SkTypeface> SkTypefaceCache::FindByProcAndRef(FindProc proc, void* ctx) { |
| SkAutoMutexExclusive ama(typeface_cache_mutex()); |
| return Get().findByProcAndRef(proc, ctx); |
| } |
| |
| void SkTypefaceCache::PurgeAll() { |
| SkAutoMutexExclusive ama(typeface_cache_mutex()); |
| Get().purgeAll(); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| #ifdef SK_DEBUG |
| static bool DumpProc(SkTypeface* face, void* ctx) { |
| SkString n; |
| face->getFamilyName(&n); |
| SkFontStyle s = face->fontStyle(); |
| SkTypefaceID id = face->uniqueID(); |
| SkDebugf("SkTypefaceCache: face %p typefaceID %u weight %d width %d style %d name %s\n", |
| face, id, s.weight(), s.width(), s.slant(), n.c_str()); |
| return false; |
| } |
| #endif |
| |
| void SkTypefaceCache::Dump() { |
| #ifdef SK_DEBUG |
| (void)Get().findByProcAndRef(DumpProc, nullptr); |
| #endif |
| } |