aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--src/core/SkColorSpaceXformer.cpp96
-rw-r--r--src/core/SkColorSpaceXformer.h16
2 files changed, 96 insertions, 16 deletions
diff --git a/src/core/SkColorSpaceXformer.cpp b/src/core/SkColorSpaceXformer.cpp
index 71715add87..58b3f4958a 100644
--- a/src/core/SkColorSpaceXformer.cpp
+++ b/src/core/SkColorSpaceXformer.cpp
@@ -19,7 +19,8 @@
SkColorSpaceXformer::SkColorSpaceXformer(sk_sp<SkColorSpace> dst,
std::unique_ptr<SkColorSpaceXform> fromSRGB)
: fDst(std::move(dst))
- , fFromSRGB(std::move(fromSRGB)) {}
+ , fFromSRGB(std::move(fromSRGB))
+ , fReentryCount(0) {}
SkColorSpaceXformer::~SkColorSpaceXformer() {}
@@ -33,11 +34,76 @@ std::unique_ptr<SkColorSpaceXformer> SkColorSpaceXformer::Make(sk_sp<SkColorSpac
: nullptr;
}
+// So what's up with these caches?
+//
+// We want to cache transformed objects for a couple of reasons:
+//
+// 1) to avoid redundant work - the inputs are a DAG, not a tree (e.g. same SkImage drawn multiple
+// times in a SkPicture), so if we blindly recurse we could end up transforming the same objects
+// repeatedly.
+//
+// 2) to avoid topology changes - we want the output to remain isomorphic with the input -- this is
+// particularly important for image filters (to maintain their original DAG structure in order
+// to not defeat their own/internal caching), but also for avoiding unnecessary cloning
+// (e.g. duplicated SkImages allocated for the example in #1 above).
+//
+// The caching scope is naturaly bound by the lifetime of the SkColorSpaceXformer object, but
+// clients may choose to not discard xformers immediately - in which case, caching indefinitely
+// is problematic. The solution is to limit the cache scope to the top level apply() call
+// (i.e. we only keep cached objects alive while transforming).
+
+class SkColorSpaceXformer::AutoCachePurge {
+public:
+ AutoCachePurge(SkColorSpaceXformer* xformer)
+ : fXformer(xformer) {
+ fXformer->fReentryCount++;
+ }
+
+ ~AutoCachePurge() {
+ SkASSERT(fXformer->fReentryCount > 0);
+ if (--fXformer->fReentryCount == 0) {
+ fXformer->purgeCaches();
+ }
+ }
+
+private:
+ SkColorSpaceXformer* fXformer;
+};
+
+template <typename T>
+sk_sp<T> SkColorSpaceXformer::cachedApply(const T* src, Cache<T>* cache,
+ sk_sp<T> (*applyFunc)(const T*, SkColorSpaceXformer*)) {
+ if (!src) {
+ return nullptr;
+ }
+
+ auto key = sk_ref_sp(const_cast<T*>(src));
+ if (auto* xformed = cache->find(key)) {
+ return sk_ref_sp(xformed->get());
+ }
+
+ auto xformed = applyFunc(src, this);
+ cache->set(std::move(key), xformed);
+
+ return xformed;
+}
+
+void SkColorSpaceXformer::purgeCaches() {
+ fImageCache.reset();
+ fColorFilterCache.reset();
+ fImageFilterCache.reset();
+}
+
sk_sp<SkImage> SkColorSpaceXformer::apply(const SkImage* src) {
- return src->makeColorSpace(fDst, SkTransferFunctionBehavior::kIgnore);
+ const AutoCachePurge autoPurge(this);
+ return this->cachedApply<SkImage>(src, &fImageCache,
+ [](const SkImage* img, SkColorSpaceXformer* xformer) {
+ return img->makeColorSpace(xformer->fDst, SkTransferFunctionBehavior::kIgnore);
+ });
}
sk_sp<SkImage> SkColorSpaceXformer::apply(const SkBitmap& src) {
+ const AutoCachePurge autoPurge(this);
sk_sp<SkImage> image = SkMakeImageFromRasterBitmap(src, kNever_SkCopyPixelsMode);
if (!image) {
return nullptr;
@@ -50,25 +116,23 @@ sk_sp<SkImage> SkColorSpaceXformer::apply(const SkBitmap& src) {
}
sk_sp<SkColorFilter> SkColorSpaceXformer::apply(const SkColorFilter* colorFilter) {
- return colorFilter->makeColorSpace(this);
+ const AutoCachePurge autoPurge(this);
+ return this->cachedApply<SkColorFilter>(colorFilter, &fColorFilterCache,
+ [](const SkColorFilter* f, SkColorSpaceXformer* xformer) {
+ return f->makeColorSpace(xformer);
+ });
}
sk_sp<SkImageFilter> SkColorSpaceXformer::apply(const SkImageFilter* imageFilter) {
- if (!imageFilter) {
- return nullptr;
- }
-
- if (auto* xformedFilter = fFilterCache.find(imageFilter->fUniqueID)) {
- return sk_ref_sp(xformedFilter->get());
- }
-
- auto xformedFilter = imageFilter->makeColorSpace(this);
- fFilterCache.set(imageFilter->fUniqueID, xformedFilter);
-
- return xformedFilter;
+ const AutoCachePurge autoPurge(this);
+ return this->cachedApply<SkImageFilter>(imageFilter, &fImageFilterCache,
+ [](const SkImageFilter* f, SkColorSpaceXformer* xformer) {
+ return f->makeColorSpace(xformer);
+ });
}
sk_sp<SkShader> SkColorSpaceXformer::apply(const SkShader* shader) {
+ const AutoCachePurge autoPurge(this);
return as_SB(shader)->makeColorSpace(this);
}
@@ -85,6 +149,8 @@ SkColor SkColorSpaceXformer::apply(SkColor srgb) {
}
SkPaint SkColorSpaceXformer::apply(const SkPaint& src) {
+ const AutoCachePurge autoPurge(this);
+
SkPaint dst = src;
// All SkColorSpaces have the same black point.
diff --git a/src/core/SkColorSpaceXformer.h b/src/core/SkColorSpaceXformer.h
index fce8d03e43..5fa4d836b7 100644
--- a/src/core/SkColorSpaceXformer.h
+++ b/src/core/SkColorSpaceXformer.h
@@ -41,10 +41,24 @@ public:
private:
SkColorSpaceXformer(sk_sp<SkColorSpace> dst, std::unique_ptr<SkColorSpaceXform> fromSRGB);
+ template <typename T>
+ using Cache = SkTHashMap<sk_sp<T>, sk_sp<T>>;
+
+ template <typename T>
+ sk_sp<T> cachedApply(const T*, Cache<T>*, sk_sp<T> (*)(const T*, SkColorSpaceXformer*));
+
+ void purgeCaches();
+
+ class AutoCachePurge;
+
sk_sp<SkColorSpace> fDst;
std::unique_ptr<SkColorSpaceXform> fFromSRGB;
- SkTHashMap<uint32_t, sk_sp<SkImageFilter>> fFilterCache;
+ size_t fReentryCount; // tracks the number of nested apply() calls for cache purging.
+
+ Cache<SkImage > fImageCache;
+ Cache<SkColorFilter> fColorFilterCache;
+ Cache<SkImageFilter> fImageFilterCache;
};
#endif