Flutter Engine
The Flutter Engine
SkImageFilterCache.cpp
Go to the documentation of this file.
1/*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
9
13#include "src/core/SkChecksum.h"
17#include "src/core/SkTHash.h"
18
19#include <vector>
20
21using namespace skia_private;
22
23#ifdef SK_BUILD_FOR_IOS
24 enum { kDefaultCacheSize = 2 * 1024 * 1024 };
25#else
26 enum { kDefaultCacheSize = 128 * 1024 * 1024 };
27#endif
28
29namespace {
30
31class CacheImpl : public SkImageFilterCache {
32public:
34 CacheImpl(size_t maxBytes) : fMaxBytes(maxBytes), fCurrentBytes(0) { }
35 ~CacheImpl() override {
36 fLookup.foreach([&](Value* v) { delete v; });
37 }
38 struct Value {
39 Value(const Key& key, const skif::FilterResult& image,
40 const SkImageFilter* filter)
41 : fKey(key), fImage(image), fFilter(filter) {}
42
43 Key fKey;
44 skif::FilterResult fImage;
45 const SkImageFilter* fFilter;
46 static const Key& GetKey(const Value& v) {
47 return v.fKey;
48 }
49 static uint32_t Hash(const Key& key) {
50 return SkChecksum::Hash32(&key, sizeof(Key));
51 }
53 };
54
55 bool get(const Key& key, skif::FilterResult* result) const override {
57
58 SkAutoMutexExclusive mutex(fMutex);
59 if (Value* v = fLookup.find(key)) {
60 if (v != fLRU.head()) {
61 fLRU.remove(v);
62 fLRU.addToHead(v);
63 }
64
65 *result = v->fImage;
66 return true;
67 }
68 return false;
69 }
70
71 void set(const Key& key, const SkImageFilter* filter,
72 const skif::FilterResult& result) override {
73 SkAutoMutexExclusive mutex(fMutex);
74 if (Value* v = fLookup.find(key)) {
75 this->removeInternal(v);
76 }
77 Value* v = new Value(key, result, filter);
78 fLookup.add(v);
79 fLRU.addToHead(v);
80 fCurrentBytes += result.image() ? result.image()->getSize() : 0;
81 if (auto* values = fImageFilterValues.find(filter)) {
82 values->push_back(v);
83 } else {
84 fImageFilterValues.set(filter, {v});
85 }
86
87 while (fCurrentBytes > fMaxBytes) {
88 Value* tail = fLRU.tail();
90 if (tail == v) {
91 break;
92 }
93 this->removeInternal(tail);
94 }
95 }
96
97 void purge() override {
98 SkAutoMutexExclusive mutex(fMutex);
99 while (fCurrentBytes > 0) {
100 Value* tail = fLRU.tail();
101 SkASSERT(tail);
102 this->removeInternal(tail);
103 }
104 }
105
106 void purgeByImageFilter(const SkImageFilter* filter) override {
107 SkAutoMutexExclusive mutex(fMutex);
108 auto* values = fImageFilterValues.find(filter);
109 if (!values) {
110 return;
111 }
112 for (Value* v : *values) {
113 // We set the filter to be null so that removeInternal() won't delete from values while
114 // we're iterating over it.
115 v->fFilter = nullptr;
116 this->removeInternal(v);
117 }
118 fImageFilterValues.remove(filter);
119 }
120
121 SkDEBUGCODE(int count() const override { return fLookup.count(); })
122private:
123 void removeInternal(Value* v) {
124 if (v->fFilter) {
125 if (auto* values = fImageFilterValues.find(v->fFilter)) {
126 if (values->size() == 1 && (*values)[0] == v) {
127 fImageFilterValues.remove(v->fFilter);
128 } else {
129 for (auto it = values->begin(); it != values->end(); ++it) {
130 if (*it == v) {
131 values->erase(it);
132 break;
133 }
134 }
135 }
136 }
137 }
138 fCurrentBytes -= v->fImage.image() ? v->fImage.image()->getSize() : 0;
139 fLRU.remove(v);
140 fLookup.remove(v->fKey);
141 delete v;
142 }
143private:
145 mutable SkTInternalLList<Value> fLRU;
146 // Value* always points to an item in fLookup.
148 size_t fMaxBytes;
149 size_t fCurrentBytes;
150 mutable SkMutex fMutex;
151};
152
153} // namespace
154
156 return sk_make_sp<CacheImpl>(maxBytes);
157}
158
160 static SkOnce once;
162
163 if (createIfNecessary == CreateIfNecessary::kNo) {
164 return cache;
165 }
166
168 return cache;
169}
int count
Definition: FontMgrTest.cpp:50
TArray< uint32_t > Key
#define SkASSERT(cond)
Definition: SkAssert.h:116
@ kDefaultCacheSize
#define SK_DECLARE_INTERNAL_LLIST_INTERFACE(ClassName)
SkDEBUGCODE(SK_SPI) SkThreadID SkGetThreadID()
virtual bool get(const SkImageFilterCacheKey &key, skif::FilterResult *result) const =0
virtual void set(const SkImageFilterCacheKey &key, const SkImageFilter *filter, const skif::FilterResult &result)=0
virtual void purge()=0
static sk_sp< SkImageFilterCache > Get(CreateIfNecessary=CreateIfNecessary::kYes)
virtual void purgeByImageFilter(const SkImageFilter *)=0
static sk_sp< SkImageFilterCache > Create(size_t maxBytes)
Definition: SkOnce.h:22
GAsyncResult * result
uint32_t Hash32(const void *data, size_t bytes, uint32_t seed)
Definition: SkChecksum.cpp:113
sk_sp< const SkImage > image
Definition: SkRecords.h:269
static uint32_t Hash(uint32_t key)
Definition: hashmap_test.cc:65
DEF_SWITCHES_START aot vmservice shared library Name of the *so containing AOT compiled Dart assets for launching the service isolate vm snapshot The VM snapshot data that will be memory mapped as read only SnapshotAssetPath must be present isolate snapshot The isolate snapshot data that will be memory mapped as read only SnapshotAssetPath must be present cache dir Path to the cache directory This is different from the persistent_cache_path in embedder which is used for Skia shader cache icu native lib Path to the library file that exports the ICU data vm service The hostname IP address on which the Dart VM Service should be served If not defaults to or::depending on whether ipv6 is specified vm service A custom Dart VM Service port The default is to pick a randomly available open port disable vm Disable the Dart VM Service The Dart VM Service is never available in release mode disable vm service Disable mDNS Dart VM Service publication Bind to the IPv6 localhost address for the Dart VM Service Ignored if vm service host is set endless trace Enable an endless trace buffer The default is a ring buffer This is useful when very old events need to viewed For during application launch Memory usage will continue to grow indefinitely however Start app with an specific route defined on the framework flutter assets Path to the Flutter assets directory enable service port Allow the VM service to fallback to automatic port selection if binding to a specified port fails trace Trace early application lifecycle Automatically switches to an endless trace buffer trace skia Filters out all Skia trace event categories except those that are specified in this comma separated list dump skp on shader Automatically dump the skp that triggers new shader compilations This is useful for writing custom ShaderWarmUp to reduce jank By this is not enabled to reduce the overhead purge persistent cache
Definition: switches.h:191