blob: 63d9e5193f5bad1d2c062b87c046823660fc6e8b [file] [log] [blame]
Matt Sarettcdc651d2017-03-30 12:41:48 -04001/*
2 * Copyright 2017 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
8#include "SkColorFilter.h"
Mike Kleine28a6b52018-07-25 13:05:17 -04009#include "SkColorSpacePriv.h"
Matt Sarettcdc651d2017-03-30 12:41:48 -040010#include "SkColorSpaceXformer.h"
Matt Sarettcdc651d2017-03-30 12:41:48 -040011#include "SkDrawLooper.h"
12#include "SkGradientShader.h"
Florin Malita39e08552017-07-06 14:16:18 -040013#include "SkImage.h"
Matt Sarettcdc651d2017-03-30 12:41:48 -040014#include "SkImage_Base.h"
Matt Sarett31abf1f2017-04-07 16:54:04 -040015#include "SkImageFilter.h"
Matt Sarette6844832017-04-03 10:35:42 -040016#include "SkImagePriv.h"
Florin Malita4aed1382017-05-25 10:38:07 -040017#include "SkShaderBase.h"
Matt Sarettcdc651d2017-03-30 12:41:48 -040018
Mike Klein9faf9fd2018-08-09 10:04:38 -040019SkColorSpaceXformer::SkColorSpaceXformer(sk_sp<SkColorSpace> dst)
20 : fDst(std::move(dst))
Mike Klein8f3d36c2018-08-14 10:28:05 -040021 , fFromSRGBSteps(sk_srgb_singleton(), kUnpremul_SkAlphaType,
22 fDst.get() , kUnpremul_SkAlphaType)
Mike Klein9faf9fd2018-08-09 10:04:38 -040023 , fReentryCount(0) {
Mike Kleinc8579562018-08-07 13:21:21 -040024
Mike Klein9faf9fd2018-08-09 10:04:38 -040025 SkRasterPipeline p(&fAlloc);
Mike Klein1a3eb522018-10-18 10:11:00 -040026 p.append(SkRasterPipeline::load_8888, &fFromSRGBSrc);
27 p.append(SkRasterPipeline::swap_rb);
Mike Klein9faf9fd2018-08-09 10:04:38 -040028 fFromSRGBSteps.apply(&p);
Mike Klein1a3eb522018-10-18 10:11:00 -040029 p.append(SkRasterPipeline::swap_rb);
30 p.append(SkRasterPipeline::store_8888, &fFromSRGBDst);
Mike Klein9faf9fd2018-08-09 10:04:38 -040031 fFromSRGB = p.compile();
32}
Florin Malita24a2ecf2017-07-06 15:12:12 -040033
Florin Malita39e08552017-07-06 14:16:18 -040034SkColorSpaceXformer::~SkColorSpaceXformer() {}
35
Matt Sarettcdc651d2017-03-30 12:41:48 -040036std::unique_ptr<SkColorSpaceXformer> SkColorSpaceXformer::Make(sk_sp<SkColorSpace> dst) {
Mike Kleinc8579562018-08-07 13:21:21 -040037 return std::unique_ptr<SkColorSpaceXformer>(new SkColorSpaceXformer{std::move(dst)});
Matt Sarettcdc651d2017-03-30 12:41:48 -040038}
39
Florin Malita5449aad2017-07-10 11:14:40 -040040// So what's up with these caches?
41//
42// We want to cache transformed objects for a couple of reasons:
43//
44// 1) to avoid redundant work - the inputs are a DAG, not a tree (e.g. same SkImage drawn multiple
45// times in a SkPicture), so if we blindly recurse we could end up transforming the same objects
46// repeatedly.
47//
48// 2) to avoid topology changes - we want the output to remain isomorphic with the input -- this is
49// particularly important for image filters (to maintain their original DAG structure in order
50// to not defeat their own/internal caching), but also for avoiding unnecessary cloning
51// (e.g. duplicated SkImages allocated for the example in #1 above).
52//
53// The caching scope is naturaly bound by the lifetime of the SkColorSpaceXformer object, but
54// clients may choose to not discard xformers immediately - in which case, caching indefinitely
55// is problematic. The solution is to limit the cache scope to the top level apply() call
56// (i.e. we only keep cached objects alive while transforming).
57
58class SkColorSpaceXformer::AutoCachePurge {
59public:
60 AutoCachePurge(SkColorSpaceXformer* xformer)
61 : fXformer(xformer) {
62 fXformer->fReentryCount++;
63 }
64
65 ~AutoCachePurge() {
66 SkASSERT(fXformer->fReentryCount > 0);
67 if (--fXformer->fReentryCount == 0) {
68 fXformer->purgeCaches();
69 }
70 }
71
72private:
73 SkColorSpaceXformer* fXformer;
74};
75
76template <typename T>
77sk_sp<T> SkColorSpaceXformer::cachedApply(const T* src, Cache<T>* cache,
78 sk_sp<T> (*applyFunc)(const T*, SkColorSpaceXformer*)) {
79 if (!src) {
80 return nullptr;
81 }
82
83 auto key = sk_ref_sp(const_cast<T*>(src));
84 if (auto* xformed = cache->find(key)) {
85 return sk_ref_sp(xformed->get());
86 }
87
88 auto xformed = applyFunc(src, this);
89 cache->set(std::move(key), xformed);
90
91 return xformed;
92}
93
94void SkColorSpaceXformer::purgeCaches() {
95 fImageCache.reset();
96 fColorFilterCache.reset();
97 fImageFilterCache.reset();
98}
99
Matt Sarettcdc651d2017-03-30 12:41:48 -0400100sk_sp<SkImage> SkColorSpaceXformer::apply(const SkImage* src) {
Florin Malita5449aad2017-07-10 11:14:40 -0400101 const AutoCachePurge autoPurge(this);
102 return this->cachedApply<SkImage>(src, &fImageCache,
103 [](const SkImage* img, SkColorSpaceXformer* xformer) {
Brian Osmanb62f50c2018-07-12 14:44:27 -0400104 return img->makeColorSpace(xformer->fDst);
Florin Malita5449aad2017-07-10 11:14:40 -0400105 });
Matt Sarettcdc651d2017-03-30 12:41:48 -0400106}
107
Matt Sarette6844832017-04-03 10:35:42 -0400108sk_sp<SkImage> SkColorSpaceXformer::apply(const SkBitmap& src) {
Florin Malita5449aad2017-07-10 11:14:40 -0400109 const AutoCachePurge autoPurge(this);
Matt Sarette6844832017-04-03 10:35:42 -0400110 sk_sp<SkImage> image = SkMakeImageFromRasterBitmap(src, kNever_SkCopyPixelsMode);
111 if (!image) {
112 return nullptr;
113 }
114
Brian Osmanb62f50c2018-07-12 14:44:27 -0400115 sk_sp<SkImage> xformed = image->makeColorSpace(fDst);
Matt Sarette6844832017-04-03 10:35:42 -0400116 // We want to be sure we don't let the kNever_SkCopyPixelsMode image escape this stack frame.
117 SkASSERT(xformed != image);
118 return xformed;
119}
120
Matt Sarett6d72ed92017-04-10 16:35:33 -0400121sk_sp<SkColorFilter> SkColorSpaceXformer::apply(const SkColorFilter* colorFilter) {
Florin Malita5449aad2017-07-10 11:14:40 -0400122 const AutoCachePurge autoPurge(this);
123 return this->cachedApply<SkColorFilter>(colorFilter, &fColorFilterCache,
124 [](const SkColorFilter* f, SkColorSpaceXformer* xformer) {
125 return f->makeColorSpace(xformer);
126 });
Matt Sarett6d72ed92017-04-10 16:35:33 -0400127}
128
Mike Kleine908b942017-04-25 13:09:40 -0400129sk_sp<SkImageFilter> SkColorSpaceXformer::apply(const SkImageFilter* imageFilter) {
Florin Malita5449aad2017-07-10 11:14:40 -0400130 const AutoCachePurge autoPurge(this);
131 return this->cachedApply<SkImageFilter>(imageFilter, &fImageFilterCache,
132 [](const SkImageFilter* f, SkColorSpaceXformer* xformer) {
133 return f->makeColorSpace(xformer);
134 });
Mike Kleine908b942017-04-25 13:09:40 -0400135}
136
Mike Klein2814d912017-05-10 12:35:51 -0400137sk_sp<SkShader> SkColorSpaceXformer::apply(const SkShader* shader) {
Florin Malita5449aad2017-07-10 11:14:40 -0400138 const AutoCachePurge autoPurge(this);
Florin Malita4aed1382017-05-25 10:38:07 -0400139 return as_SB(shader)->makeColorSpace(this);
Mike Klein2814d912017-05-10 12:35:51 -0400140}
141
Matt Sarettcdc651d2017-03-30 12:41:48 -0400142void SkColorSpaceXformer::apply(SkColor* xformed, const SkColor* srgb, int n) {
Mike Kleinc8579562018-08-07 13:21:21 -0400143 fFromSRGBSrc.pixels = const_cast<SkColor*>(srgb);
144 fFromSRGBDst.pixels = xformed;
145 fFromSRGB(0,0,n,1);
Matt Sarettcdc651d2017-03-30 12:41:48 -0400146}
147
148SkColor SkColorSpaceXformer::apply(SkColor srgb) {
149 SkColor xformed;
150 this->apply(&xformed, &srgb, 1);
151 return xformed;
152}
153
Matt Sarettc15bb7b2017-04-25 13:53:11 -0400154SkPaint SkColorSpaceXformer::apply(const SkPaint& src) {
Florin Malita5449aad2017-07-10 11:14:40 -0400155 const AutoCachePurge autoPurge(this);
156
Matt Sarettc15bb7b2017-04-25 13:53:11 -0400157 SkPaint dst = src;
Matt Sarettcdc651d2017-03-30 12:41:48 -0400158
159 // All SkColorSpaces have the same black point.
160 if (src.getColor() & 0xffffff) {
Matt Sarettc15bb7b2017-04-25 13:53:11 -0400161 dst.setColor(this->apply(src.getColor()));
Matt Sarettcdc651d2017-03-30 12:41:48 -0400162 }
163
164 if (auto shader = src.getShader()) {
Mike Klein2814d912017-05-10 12:35:51 -0400165 dst.setShader(this->apply(shader));
Matt Sarettcdc651d2017-03-30 12:41:48 -0400166 }
167
Matt Sarettcdc651d2017-03-30 12:41:48 -0400168 if (auto cf = src.getColorFilter()) {
Matt Sarettc15bb7b2017-04-25 13:53:11 -0400169 dst.setColorFilter(this->apply(cf));
Matt Sarettcdc651d2017-03-30 12:41:48 -0400170 }
171
172 if (auto looper = src.getDrawLooper()) {
Matt Sarettc15bb7b2017-04-25 13:53:11 -0400173 dst.setDrawLooper(looper->makeColorSpace(this));
Matt Sarettcdc651d2017-03-30 12:41:48 -0400174 }
175
Matt Sarett31abf1f2017-04-07 16:54:04 -0400176 if (auto imageFilter = src.getImageFilter()) {
Matt Sarettc15bb7b2017-04-25 13:53:11 -0400177 dst.setImageFilter(this->apply(imageFilter));
Matt Sarett31abf1f2017-04-07 16:54:04 -0400178 }
179
Matt Sarettc15bb7b2017-04-25 13:53:11 -0400180 return dst;
Matt Sarettcdc651d2017-03-30 12:41:48 -0400181}
Stan Iliev45fd9952017-12-18 14:46:30 -0500182
183SkCanvas::Lattice SkColorSpaceXformer::apply(const SkCanvas::Lattice& lattice,
184 SkColor* colorBuffer, int count) {
185 if (count) {
186 this->apply(colorBuffer, lattice.fColors, count);
187 return {lattice.fXDivs, lattice.fYDivs, lattice.fRectTypes,
188 lattice.fXCount, lattice.fYCount, lattice.fBounds, colorBuffer};
189 }
190
191 return lattice;
192}