• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2019 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 "include/private/SkMacros.h"
9 #include "src/core/SkArenaAlloc.h"
10 #include "src/core/SkColorSpacePriv.h"
11 #include "src/core/SkColorSpaceXformSteps.h"
12 #include "src/core/SkCoreBlitters.h"
13 #include "src/core/SkLRUCache.h"
14 #include "src/core/SkVM.h"
15 
16 namespace {
17 
18     enum class Coverage { Full, UniformA8, MaskA8, MaskLCD16, Mask3D };
19 
20     SK_BEGIN_REQUIRE_DENSE;
21     struct Key {
22         SkColorType    colorType;
23         SkAlphaType    alphaType;
24         Coverage       coverage;
25         SkBlendMode    blendMode;
26         SkShader*      shader;
27         SkColorFilter* colorFilter;
28 
withCoverage__anon2d36eba40111::Key29         Key withCoverage(Coverage c) const {
30             Key k = *this;
31             k.coverage = c;
32             return k;
33         }
34     };
35     SK_END_REQUIRE_DENSE;
36 
operator ==(const Key & x,const Key & y)37     static bool operator==(const Key& x, const Key& y) {
38         return x.colorType   == y.colorType
39             && x.alphaType   == y.alphaType
40             && x.coverage    == y.coverage
41             && x.blendMode   == y.blendMode
42             && x.shader      == y.shader
43             && x.colorFilter == y.colorFilter;
44     }
45 
try_acquire_program_cache()46     static SkLRUCache<Key, skvm::Program>* try_acquire_program_cache() {
47     #if defined(SK_BUILD_FOR_IOS)
48         // iOS doesn't support thread_local on versions less than 9.0. pthread
49         // based fallbacks must be used there. We could also use an SkSpinlock
50         // and tryAcquire()/release(), or...
51         return nullptr;  // ... we could just not cache programs on those platforms.
52     #else
53         thread_local static auto* cache = new SkLRUCache<Key, skvm::Program>{8};
54         return cache;
55     #endif
56     }
57 
release_program_cache()58     static void release_program_cache() { }
59 
60 
61     struct Uniforms {
62         uint32_t paint_color;
63         uint8_t  coverage;   // Used when Coverage::UniformA8.
64     };
65 
66     struct Builder : public skvm::Builder {
67         //using namespace skvm;
68 
69         struct Color { skvm::I32 r,g,b,a; };
70 
71 
inv__anon2d36eba40111::Builder72         skvm::I32 inv(skvm::I32 x) {
73             return sub(splat(255), x);
74         }
75 
76         // TODO: provide this in skvm::Builder, with a custom NEON impl.
div255__anon2d36eba40111::Builder77         skvm::I32 div255(skvm::I32 v) {
78             // This should be a bit-perfect version of (v+127)/255,
79             // implemented as (v + ((v+128)>>8) + 128)>>8.
80             skvm::I32 v128 = add(v, splat(128));
81             return shr(add(v128, shr(v128, 8)), 8);
82         }
83 
mix__anon2d36eba40111::Builder84         skvm::I32 mix(skvm::I32 x, skvm::I32 y, skvm::I32 t) {
85             return div255(add(mul(x, inv(t)),
86                               mul(y,     t )));
87         }
88 
unpack_8888__anon2d36eba40111::Builder89         Color unpack_8888(skvm::I32 rgba) {
90             return {
91                 extract(rgba,  0, splat(0xff)),
92                 extract(rgba,  8, splat(0xff)),
93                 extract(rgba, 16, splat(0xff)),
94                 extract(rgba, 24, splat(0xff)),
95             };
96         }
97 
pack_8888__anon2d36eba40111::Builder98         skvm::I32 pack_8888(Color c) {
99             return pack(pack(c.r, c.g, 8),
100                         pack(c.b, c.a, 8), 16);
101         }
102 
unpack_565__anon2d36eba40111::Builder103         Color unpack_565(skvm::I32 bgr) {
104             // N.B. kRGB_565_SkColorType is named confusingly;
105             //      blue is in the low bits and red the high.
106             skvm::I32 r = extract(bgr, 11, splat(0b011'111)),
107                       g = extract(bgr,  5, splat(0b111'111)),
108                       b = extract(bgr,  0, splat(0b011'111));
109             return {
110                 // Scale 565 up to 888.
111                 bit_or(shl(r, 3), shr(r, 2)),
112                 bit_or(shl(g, 2), shr(g, 4)),
113                 bit_or(shl(b, 3), shr(b, 2)),
114                 splat(0xff),
115             };
116         }
117 
pack_565__anon2d36eba40111::Builder118         skvm::I32 pack_565(Color c) {
119             skvm::I32 r = div255(mul(c.r, splat(31))),
120                       g = div255(mul(c.g, splat(63))),
121                       b = div255(mul(c.b, splat(31)));
122             return pack(pack(b, g,5), r,11);
123         }
124 
125         // TODO: add native min/max ops to skvm::Builder
min__anon2d36eba40111::Builder126         skvm::I32 min(skvm::I32 x, skvm::I32 y) { return select(lt(x,y), x,y); }
max__anon2d36eba40111::Builder127         skvm::I32 max(skvm::I32 x, skvm::I32 y) { return select(gt(x,y), x,y); }
128 
CanBuild__anon2d36eba40111::Builder129         static bool CanBuild(const Key& key) {
130             // These checks parallel the TODOs in Builder::Builder().
131             if (key.shader)      { return false; }
132             if (key.colorFilter) { return false; }
133 
134             switch (key.colorType) {
135                 default: return false;
136                 case kRGB_565_SkColorType:   break;
137                 case kRGBA_8888_SkColorType: break;
138                 case kBGRA_8888_SkColorType: break;
139             }
140 
141             if (key.alphaType == kUnpremul_SkAlphaType) { return false; }
142 
143             switch (key.blendMode) {
144                 default: return false;
145                 case SkBlendMode::kSrc:     break;
146                 case SkBlendMode::kSrcOver: break;
147             }
148 
149             return true;
150         }
151 
Builder__anon2d36eba40111::Builder152         explicit Builder(const Key& key) {
153         #define TODO SkUNREACHABLE
154             SkASSERT(CanBuild(key));
155             skvm::Arg uniforms = uniform(),
156                       dst_ptr  = arg(SkColorTypeBytesPerPixel(key.colorType));
157             // When coverage is MaskA8 or MaskLCD16 there will be one more mask varying,
158             // and when coverage is Mask3D there will be three more mask varyings.
159 
160 
161             // When there's no shader and no color filter, the source color is the paint color.
162             if (key.shader)      { TODO; }
163             if (key.colorFilter) { TODO; }
164             Color src = unpack_8888(uniform32(uniforms, offsetof(Uniforms, paint_color)));
165 
166             // Load up the destination color.
167             Color dst;
168             switch (key.colorType) {
169                 default: TODO;
170 
171                 case kRGB_565_SkColorType:   dst = unpack_565 (load16(dst_ptr)); break;
172 
173                 case kRGBA_8888_SkColorType: dst = unpack_8888(load32(dst_ptr)); break;
174                 case kBGRA_8888_SkColorType: dst = unpack_8888(load32(dst_ptr));
175                                              std::swap(dst.r, dst.b);
176                                              break;
177             }
178 
179             // When a destination is tagged opaque, we may assume it both starts and stays fully
180             // opaque, ignoring any math that disagrees.  So anything involving force_opaque is
181             // optional, and sometimes helps cut a small amount of work in these programs.
182             const bool force_opaque = true && key.alphaType == kOpaque_SkAlphaType;
183             if (force_opaque) { dst.a = splat(0xff); }
184 
185             // We'd need to premul dst after loading and unpremul before storing.
186             if (key.alphaType == kUnpremul_SkAlphaType) { TODO; }
187 
188             // Blend src and dst.
189             switch (key.blendMode) {
190                 default: TODO;
191 
192                 case SkBlendMode::kSrc: break;
193 
194                 case SkBlendMode::kSrcOver: {
195                     src.r = add(src.r, div255(mul(dst.r, inv(src.a))));
196                     src.g = add(src.g, div255(mul(dst.g, inv(src.a))));
197                     src.b = add(src.b, div255(mul(dst.b, inv(src.a))));
198                     src.a = add(src.a, div255(mul(dst.a, inv(src.a))));
199                 } break;
200             }
201 
202             // Lerp with coverage if needed.
203             bool apply_coverage = true;
204             Color cov;
205             switch (key.coverage) {
206                 case Coverage::Full: apply_coverage = false;
207                                      break;
208 
209                 case Coverage::UniformA8: cov.r = cov.g = cov.b = cov.a =
210                                           uniform8(uniforms, offsetof(Uniforms, coverage));
211                                           break;
212 
213                 case Coverage::MaskA8: cov.r = cov.g = cov.b = cov.a =
214                                        load8(varying<uint8_t>());
215                                        break;
216 
217                 case Coverage::MaskLCD16:
218                     cov = unpack_565(load16(varying<uint16_t>()));
219                     cov.a = select(lt(src.a, dst.a), min(cov.r, min(cov.g,cov.b))
220                                                    , max(cov.r, max(cov.g,cov.b)));
221                     break;
222 
223                 case Coverage::Mask3D: TODO;
224             }
225             if (apply_coverage) {
226                 src.r = mix(dst.r, src.r, cov.r);
227                 src.g = mix(dst.g, src.g, cov.g);
228                 src.b = mix(dst.b, src.b, cov.b);
229                 src.a = mix(dst.a, src.a, cov.a);
230             }
231 
232             if (force_opaque) { src.a = splat(0xff); }
233 
234             // Store back to the destination.
235             switch (key.colorType) {
236                 default: SkUNREACHABLE;
237 
238                 case kRGB_565_SkColorType:   store16(dst_ptr, pack_565(src)); break;
239 
240                 case kBGRA_8888_SkColorType: std::swap(src.r, src.b);  // fallthrough
241                 case kRGBA_8888_SkColorType: store32(dst_ptr, pack_8888(src)); break;
242             }
243         #undef TODO
244         }
245     };
246 
247     class Blitter final : public SkBlitter {
248     public:
249         bool ok = false;
250 
Blitter(const SkPixmap & device,const SkPaint & paint)251         Blitter(const SkPixmap& device, const SkPaint& paint)
252             : fDevice(device)
253             , fKey {
254                 device.colorType(),
255                 device.alphaType(),
256                 Coverage::Full,
257                 paint.getBlendMode(),
258                 paint.getShader(),
259                 paint.getColorFilter(),
260             }
261         {
262             SkColor4f color = paint.getColor4f();
263             SkColorSpaceXformSteps{sk_srgb_singleton(), kUnpremul_SkAlphaType,
264                                    device.colorSpace(), kUnpremul_SkAlphaType}.apply(color.vec());
265 
266             if (color.fitsInBytes() && Builder::CanBuild(fKey)) {
267                 fUniforms.paint_color = color.premul().toBytes_RGBA();
268                 ok = true;
269             }
270         }
271 
~Blitter()272         ~Blitter() override {
273             if (SkLRUCache<Key, skvm::Program>* cache = try_acquire_program_cache()) {
274                 auto cache_program = [&](skvm::Program&& program, Coverage coverage) {
275                     if (!program.empty()) {
276                         Key key = fKey.withCoverage(coverage);
277                         if (skvm::Program* found = cache->find(key)) {
278                             *found = std::move(program);
279                         } else {
280                             cache->insert(key, std::move(program));
281                         }
282                     }
283                 };
284                 cache_program(std::move(fBlitH),         Coverage::Full);
285                 cache_program(std::move(fBlitAntiH),     Coverage::UniformA8);
286                 cache_program(std::move(fBlitMaskA8),    Coverage::MaskA8);
287                 cache_program(std::move(fBlitMaskLCD16), Coverage::MaskLCD16);
288 
289                 release_program_cache();
290             }
291         }
292 
293     private:
294         SkPixmap      fDevice;  // TODO: can this be const&?
295         const Key     fKey;
296         Uniforms      fUniforms;
297         skvm::Program fBlitH,
298                       fBlitAntiH,
299                       fBlitMaskA8,
300                       fBlitMaskLCD16;
301 
buildProgram(Coverage coverage)302         skvm::Program buildProgram(Coverage coverage) {
303             Key key = fKey.withCoverage(coverage);
304             {
305                 skvm::Program p;
306                 if (SkLRUCache<Key, skvm::Program>* cache = try_acquire_program_cache()) {
307                     if (skvm::Program* found = cache->find(key)) {
308                         p = std::move(*found);
309                     }
310                     release_program_cache();
311                 }
312                 if (!p.empty()) {
313                     return p;
314                 }
315             }
316         #if 0
317             static std::atomic<int> done{0};
318             if (0 == done++) {
319                 atexit([]{ SkDebugf("%d calls to done\n", done.load()); });
320             }
321         #endif
322             return Builder{key}.done();
323         }
324 
blitH(int x,int y,int w)325         void blitH(int x, int y, int w) override {
326             if (fBlitH.empty()) {
327                 fBlitH = this->buildProgram(Coverage::Full);
328             }
329             fBlitH.eval(w, &fUniforms, fDevice.addr(x,y));
330         }
331 
blitAntiH(int x,int y,const SkAlpha cov[],const int16_t runs[])332         void blitAntiH(int x, int y, const SkAlpha cov[], const int16_t runs[]) override {
333             if (fBlitAntiH.empty()) {
334                 fBlitAntiH = this->buildProgram(Coverage::UniformA8);
335             }
336             for (int16_t run = *runs; run > 0; run = *runs) {
337                 fUniforms.coverage = *cov;
338                 fBlitAntiH.eval(run, &fUniforms, fDevice.addr(x,y));
339 
340                 x    += run;
341                 runs += run;
342                 cov  += run;
343             }
344         }
345 
blitMask(const SkMask & mask,const SkIRect & clip)346         void blitMask(const SkMask& mask, const SkIRect& clip) override {
347             if (mask.fFormat == SkMask::kBW_Format) {
348                 // TODO: native BW masks?
349                 return SkBlitter::blitMask(mask, clip);
350             }
351 
352             const skvm::Program* program = nullptr;
353             switch (mask.fFormat) {
354                 default: SkUNREACHABLE;     // ARGB and SDF masks shouldn't make it here.
355 
356                 case SkMask::k3D_Format:    // TODO: the mul and add 3D mask planes too
357                 case SkMask::kA8_Format:
358                     if (fBlitMaskA8.empty()) {
359                         fBlitMaskA8 = this->buildProgram(Coverage::MaskA8);
360                     }
361                     program = &fBlitMaskA8;
362                     break;
363 
364                 case SkMask::kLCD16_Format:
365                     if (fBlitMaskLCD16.empty()) {
366                         fBlitMaskLCD16 = this->buildProgram(Coverage::MaskLCD16);
367                     }
368                     program = &fBlitMaskLCD16;
369                     break;
370             }
371 
372             SkASSERT(program);
373             if (program) {
374                 for (int y = clip.top(); y < clip.bottom(); y++) {
375                     program->eval(clip.width(),
376                                   &fUniforms,
377                                   fDevice.addr(clip.left(), y),
378                                   mask.getAddr(clip.left(), y));
379                 }
380             }
381         }
382     };
383 
384 }  // namespace
385 
386 
SkCreateSkVMBlitter(const SkPixmap & device,const SkPaint & paint,const SkMatrix & ctm,SkArenaAlloc * alloc)387 SkBlitter* SkCreateSkVMBlitter(const SkPixmap& device,
388                                const SkPaint& paint,
389                                const SkMatrix& ctm,
390                                SkArenaAlloc* alloc) {
391     auto blitter = alloc->make<Blitter>(device, paint);
392     return blitter->ok ? blitter
393                        : nullptr;
394 }
395