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/SkImageInfoPriv.h"
9 #include "include/private/SkMacros.h"
10 #include "src/core/SkArenaAlloc.h"
11 #include "src/core/SkBlendModePriv.h"
12 #include "src/core/SkBlenderBase.h"
13 #include "src/core/SkColorFilterBase.h"
14 #include "src/core/SkColorSpacePriv.h"
15 #include "src/core/SkColorSpaceXformSteps.h"
16 #include "src/core/SkCoreBlitters.h"
17 #include "src/core/SkLRUCache.h"
18 #include "src/core/SkMatrixProvider.h"
19 #include "src/core/SkOpts.h"
20 #include "src/core/SkPaintPriv.h"
21 #include "src/core/SkVM.h"
22 #include "src/core/SkVMBlitter.h"
23 #include "src/shaders/SkColorFilterShader.h"
24
25 #include <cinttypes>
26
27 namespace {
28
29 // Uniforms set by the Blitter itself,
30 // rather than by the Shader, which follow this struct in the skvm::Uniforms buffer.
31 struct BlitterUniforms {
32 int right; // First device x + blit run length n, used to get device x coordinate.
33 int y; // Device y coordinate.
34 };
35 static_assert(SkIsAlign4(sizeof(BlitterUniforms)), "");
36 inline static constexpr int kBlitterUniformsCount = sizeof(BlitterUniforms) / 4;
37
device_coord(skvm::Builder * p,skvm::Uniforms * uniforms)38 static skvm::Coord device_coord(skvm::Builder* p, skvm::Uniforms* uniforms) {
39 skvm::I32 dx = p->uniform32(uniforms->base, offsetof(BlitterUniforms, right))
40 - p->index(),
41 dy = p->uniform32(uniforms->base, offsetof(BlitterUniforms, y));
42 return {
43 to_F32(dx) + 0.5f,
44 to_F32(dy) + 0.5f,
45 };
46 }
47
48 struct NoopColorFilter : public SkColorFilterBase {
onProgram__anon40d8f2dd0111::NoopColorFilter49 skvm::Color onProgram(skvm::Builder*, skvm::Color c,
50 const SkColorInfo&, skvm::Uniforms*, SkArenaAlloc*) const override {
51 return c;
52 }
53
onAppendStages__anon40d8f2dd0111::NoopColorFilter54 bool onAppendStages(const SkStageRec&, bool) const override { return true; }
55
56 // Only created here, should never be flattened / unflattened.
getFactory__anon40d8f2dd0111::NoopColorFilter57 Factory getFactory() const override { return nullptr; }
getTypeName__anon40d8f2dd0111::NoopColorFilter58 const char* getTypeName() const override { return "NoopColorFilter"; }
59 };
60
61 struct SpriteShader : public SkShaderBase {
SpriteShader__anon40d8f2dd0111::SpriteShader62 explicit SpriteShader(SkPixmap sprite) : fSprite(sprite) {}
63
64 SkPixmap fSprite;
65
66 // Only created here temporarily... never serialized.
getFactory__anon40d8f2dd0111::SpriteShader67 Factory getFactory() const override { return nullptr; }
getTypeName__anon40d8f2dd0111::SpriteShader68 const char* getTypeName() const override { return "SpriteShader"; }
69
isOpaque__anon40d8f2dd0111::SpriteShader70 bool isOpaque() const override { return fSprite.isOpaque(); }
71
onProgram__anon40d8f2dd0111::SpriteShader72 skvm::Color onProgram(skvm::Builder* p,
73 skvm::Coord /*device*/, skvm::Coord /*local*/, skvm::Color /*paint*/,
74 const SkMatrixProvider&, const SkMatrix* /*localM*/,
75 const SkColorInfo& dst,
76 skvm::Uniforms* uniforms, SkArenaAlloc*) const override {
77 const SkColorType ct = fSprite.colorType();
78
79 skvm::PixelFormat fmt = skvm::SkColorType_to_PixelFormat(ct);
80
81 skvm::Color c = p->load(fmt, p->varying(SkColorTypeBytesPerPixel(ct)));
82
83 return SkColorSpaceXformSteps{fSprite, dst}.program(p, uniforms, c);
84 }
85 };
86
87 struct DitherShader : public SkShaderBase {
DitherShader__anon40d8f2dd0111::DitherShader88 explicit DitherShader(sk_sp<SkShader> shader) : fShader(std::move(shader)) {}
89
90 sk_sp<SkShader> fShader;
91
92 // Only created here temporarily... never serialized.
getFactory__anon40d8f2dd0111::DitherShader93 Factory getFactory() const override { return nullptr; }
getTypeName__anon40d8f2dd0111::DitherShader94 const char* getTypeName() const override { return "DitherShader"; }
95
isOpaque__anon40d8f2dd0111::DitherShader96 bool isOpaque() const override { return fShader->isOpaque(); }
97
onProgram__anon40d8f2dd0111::DitherShader98 skvm::Color onProgram(skvm::Builder* p,
99 skvm::Coord device, skvm::Coord local, skvm::Color paint,
100 const SkMatrixProvider& matrices, const SkMatrix* localM,
101 const SkColorInfo& dst,
102 skvm::Uniforms* uniforms, SkArenaAlloc* alloc) const override {
103 // Run our wrapped shader.
104 skvm::Color c = as_SB(fShader)->program(p, device,local, paint,
105 matrices,localM, dst, uniforms,alloc);
106 if (!c) {
107 return {};
108 }
109
110 float rate = 0.0f;
111 switch (dst.colorType()) {
112 case kARGB_4444_SkColorType: rate = 1/15.0f; break;
113 case kRGB_565_SkColorType: rate = 1/63.0f; break;
114 case kGray_8_SkColorType:
115 case kRGB_888x_SkColorType:
116 case kRGBA_8888_SkColorType:
117 case kBGRA_8888_SkColorType:
118 case kSRGBA_8888_SkColorType: rate = 1/255.0f; break;
119 case kRGB_101010x_SkColorType:
120 case kRGBA_1010102_SkColorType:
121 case kBGR_101010x_SkColorType:
122 case kBGRA_1010102_SkColorType: rate = 1/1023.0f; break;
123
124 case kUnknown_SkColorType:
125 case kAlpha_8_SkColorType:
126 case kRGBA_F16_SkColorType:
127 case kRGBA_F16Norm_SkColorType:
128 case kRGBA_F32_SkColorType:
129 case kR8G8_unorm_SkColorType:
130 case kA16_float_SkColorType:
131 case kA16_unorm_SkColorType:
132 case kR16G16_float_SkColorType:
133 case kR16G16_unorm_SkColorType:
134 case kR16G16B16A16_unorm_SkColorType: return c;
135 }
136
137 // See SkRasterPipeline dither stage.
138 // This is 8x8 ordered dithering. From here we'll only need dx and dx^dy.
139 SkASSERT(local.x.id == device.x.id);
140 SkASSERT(local.y.id == device.y.id);
141 skvm::I32 X = trunc(device.x - 0.5f),
142 Y = X ^ trunc(device.y - 0.5f);
143
144 // If X's low bits are abc and Y's def, M is fcebda,
145 // 6 bits producing all values [0,63] shuffled over an 8x8 grid.
146 skvm::I32 M = shl(Y & 1, 5)
147 | shl(X & 1, 4)
148 | shl(Y & 2, 2)
149 | shl(X & 2, 1)
150 | shr(Y & 4, 1)
151 | shr(X & 4, 2);
152
153 // Scale to [0,1) by /64, then to (-0.5,0.5) using 63/128 (~0.492) as 0.5-ε,
154 // and finally scale all that by rate. We keep dither strength strictly
155 // within ±0.5 to not change exact values like 0 or 1.
156
157 // rate could be a uniform, but since it's based on the destination SkColorType,
158 // we can bake it in without hurting the cache hit rate.
159 float scale = rate * ( 2/128.0f),
160 bias = rate * (-63/128.0f);
161 skvm::F32 dither = to_F32(M) * scale + bias;
162 c.r += dither;
163 c.g += dither;
164 c.b += dither;
165
166 c.r = clamp(c.r, 0.0f, c.a);
167 c.g = clamp(c.g, 0.0f, c.a);
168 c.b = clamp(c.b, 0.0f, c.a);
169 return c;
170 }
171 };
172
173 // This is similar to using SkShaders::Color(paint.getColor4f(), nullptr),
174 // but uses the blitter-provided paint color uniforms instead of pushing its own.
175 struct PaintColorShader : public SkShaderBase {
PaintColorShader__anon40d8f2dd0111::PaintColorShader176 explicit PaintColorShader(bool isOpaque) : fIsOpaque(isOpaque) {}
177
178 const bool fIsOpaque;
179
180 // Only created here temporarily... never serialized.
getFactory__anon40d8f2dd0111::PaintColorShader181 Factory getFactory() const override { return nullptr; }
getTypeName__anon40d8f2dd0111::PaintColorShader182 const char* getTypeName() const override { return "PaintColorShader"; }
183
isOpaque__anon40d8f2dd0111::PaintColorShader184 bool isOpaque() const override { return fIsOpaque; }
185
onProgram__anon40d8f2dd0111::PaintColorShader186 skvm::Color onProgram(skvm::Builder*,
187 skvm::Coord, skvm::Coord, skvm::Color paint,
188 const SkMatrixProvider&, const SkMatrix*, const SkColorInfo&,
189 skvm::Uniforms*, SkArenaAlloc*) const override {
190 // Incoming `paint` is unpremul in the destination color space,
191 // so we just need to premul it.
192 return premul(paint);
193 }
194 };
195 } // namespace
196
operator ==(const Key & that) const197 bool SkVMBlitter::Key::operator==(const Key& that) const {
198 return this->shader == that.shader
199 && this->clip == that.clip
200 && this->blender == that.blender
201 && this->colorSpace == that.colorSpace
202 && this->colorType == that.colorType
203 && this->alphaType == that.alphaType
204 && this->coverage == that.coverage;
205 }
206
withCoverage(Coverage c) const207 SkVMBlitter::Key SkVMBlitter::Key::withCoverage(Coverage c) const {
208 Key k = *this;
209 k.coverage = SkToU8(c);
210 return k;
211 }
212
withCoverage(Coverage c) const213 SkVMBlitter::Params SkVMBlitter::Params::withCoverage(Coverage c) const {
214 Params p = *this;
215 p.coverage = c;
216 return p;
217 }
218
EffectiveParams(const SkPixmap & device,const SkPixmap * sprite,SkPaint paint,const SkMatrixProvider & matrices,sk_sp<SkShader> clip)219 SkVMBlitter::Params SkVMBlitter::EffectiveParams(const SkPixmap& device,
220 const SkPixmap* sprite,
221 SkPaint paint,
222 const SkMatrixProvider& matrices,
223 sk_sp<SkShader> clip) {
224 // Sprites take priority over any shader. (There's rarely one set, and it's meaningless.)
225 if (sprite) {
226 paint.setShader(sk_make_sp<SpriteShader>(*sprite));
227 }
228
229 // Normal blitters will have already folded color filters into their shader,
230 // but we may still need to do that here for SpriteShaders.
231 if (paint.getColorFilter()) {
232 SkPaintPriv::RemoveColorFilter(&paint, device.colorSpace());
233 }
234 SkASSERT(!paint.getColorFilter());
235
236 // If there's no explicit shader, the paint color is the shader,
237 // but if there is a shader, it's modulated by the paint alpha.
238 sk_sp<SkShader> shader = paint.refShader();
239 if (!shader) {
240 shader = sk_make_sp<PaintColorShader>(paint.getColor4f().isOpaque());
241 } else if (paint.getAlphaf() < 1.0f) {
242 shader = sk_make_sp<SkColorFilterShader>(std::move(shader),
243 paint.getAlphaf(),
244 sk_make_sp<NoopColorFilter>());
245 }
246
247 // Add dither to the end of the shader pipeline if requested and needed.
248 if (paint.isDither() && !as_SB(shader)->isConstant()) {
249 shader = sk_make_sp<DitherShader>(std::move(shader));
250 }
251
252 // Add the blender.
253 sk_sp<SkBlender> blender = paint.refBlender();
254 if (!blender) {
255 blender = SkBlender::Mode(SkBlendMode::kSrcOver);
256 }
257
258 // The most common blend mode is SrcOver, and it can be strength-reduced
259 // _greatly_ to Src mode when the shader is opaque.
260 //
261 // In general all the information we use to make decisions here need to
262 // be reflected in Params and Key to make program caching sound, and it
263 // might appear that shader->isOpaque() is a property of the shader's
264 // uniforms than its fundamental program structure and so unsafe to use.
265 //
266 // Opacity is such a powerful property that SkShaderBase::program()
267 // forces opacity for any shader subclass that claims isOpaque(), so
268 // the opaque bit is strongly guaranteed to be part of the program and
269 // not just a property of the uniforms. The shader program hash includes
270 // this information, making it safe to use anywhere in the blitter codegen.
271 if (as_BB(blender)->asBlendMode() == SkBlendMode::kSrcOver && shader->isOpaque()) {
272 blender = SkBlender::Mode(SkBlendMode::kSrc);
273 }
274
275 SkColor4f paintColor = paint.getColor4f();
276 SkColorSpaceXformSteps{sk_srgb_singleton(), kUnpremul_SkAlphaType,
277 device.colorSpace(), kUnpremul_SkAlphaType}
278 .apply(paintColor.vec());
279
280 return {
281 std::move(shader),
282 std::move(clip),
283 std::move(blender),
284 { device.colorType(), device.alphaType(), device.refColorSpace() },
285 Coverage::Full, // Placeholder... withCoverage() will change as needed.
286 paintColor,
287 matrices,
288 };
289 }
290
DstColor(skvm::Builder * p,const Params & params)291 skvm::Color SkVMBlitter::DstColor(skvm::Builder* p, const Params& params) {
292 skvm::PixelFormat dstFormat = skvm::SkColorType_to_PixelFormat(params.dst.colorType());
293 skvm::Ptr dst_ptr = p->varying(SkColorTypeBytesPerPixel(params.dst.colorType()));
294 return p->load(dstFormat, dst_ptr);
295 }
296
BuildProgram(skvm::Builder * p,const Params & params,skvm::Uniforms * uniforms,SkArenaAlloc * alloc)297 void SkVMBlitter::BuildProgram(skvm::Builder* p, const Params& params,
298 skvm::Uniforms* uniforms, SkArenaAlloc* alloc) {
299 // First two arguments are always uniforms and the destination buffer.
300 uniforms->base = p->uniform();
301 skvm::Ptr dst_ptr = p->varying(SkColorTypeBytesPerPixel(params.dst.colorType()));
302 // A SpriteShader (in this file) may next use one argument as its varying source.
303 // Subsequent arguments depend on params.coverage:
304 // - Full: (no more arguments)
305 // - Mask3D: mul varying, add varying, 8-bit coverage varying
306 // - MaskA8: 8-bit coverage varying
307 // - MaskLCD16: 565 coverage varying
308 // - UniformF: float coverage uniform
309
310 skvm::Coord device = device_coord(p, uniforms);
311 skvm::Color paint = p->uniformColor(params.paint, uniforms);
312
313 // See note about arguments above: a SpriteShader will call p->arg() once during program().
314 skvm::Color src = as_SB(params.shader)->program(p, device, /*local=*/device, paint,
315 params.matrices, /*localM=*/nullptr,
316 params.dst, uniforms, alloc);
317 SkASSERT(src);
318 if (params.coverage == Coverage::Mask3D) {
319 skvm::F32 M = from_unorm(8, p->load8(p->varying<uint8_t>())),
320 A = from_unorm(8, p->load8(p->varying<uint8_t>()));
321
322 src.r = min(src.r * M + A, src.a);
323 src.g = min(src.g * M + A, src.a);
324 src.b = min(src.b * M + A, src.a);
325 }
326
327 // GL clamps all its color channels to limits of the format just before the blend step (~here).
328 // TODO: Below, we also clamp after the blend step. If we can prove that none of the work here
329 // (especially blending, for built-in blend modes) will produce colors outside [0, 1] we may be
330 // able to skip the second clamp. For now, we clamp twice.
331 if (SkColorTypeIsNormalized(params.dst.colorType())) {
332 src = clamp01(src);
333 }
334
335 // Load the destination color.
336 skvm::PixelFormat dstFormat = skvm::SkColorType_to_PixelFormat(params.dst.colorType());
337 skvm::Color dst = p->load(dstFormat, dst_ptr);
338 if (params.dst.isOpaque()) {
339 // When a destination is known opaque, we may assume it both starts and stays fully
340 // opaque, ignoring any math that disagrees. This sometimes trims a little work.
341 dst.a = p->splat(1.0f);
342 } else if (params.dst.alphaType() == kUnpremul_SkAlphaType) {
343 // All our blending works in terms of premul.
344 dst = premul(dst);
345 }
346
347 // Load coverage.
348 skvm::Color cov;
349 switch (params.coverage) {
350 case Coverage::Full:
351 cov.r = cov.g = cov.b = cov.a = p->splat(1.0f);
352 break;
353
354 case Coverage::UniformF:
355 cov.r = cov.g = cov.b = cov.a = p->uniformF(p->uniform(), 0);
356 break;
357
358 case Coverage::Mask3D:
359 case Coverage::MaskA8:
360 cov.r = cov.g = cov.b = cov.a = from_unorm(8, p->load8(p->varying<uint8_t>()));
361 break;
362
363 case Coverage::MaskLCD16: {
364 skvm::PixelFormat fmt = skvm::SkColorType_to_PixelFormat(kRGB_565_SkColorType);
365 cov = p->load(fmt, p->varying<uint16_t>());
366 cov.a = select(src.a < dst.a, min(cov.r, min(cov.g, cov.b)),
367 max(cov.r, max(cov.g, cov.b)));
368 } break;
369 }
370 if (params.clip) {
371 skvm::Color clip = as_SB(params.clip)->program(p, device, /*local=*/device, paint,
372 params.matrices, /*localM=*/nullptr,
373 params.dst, uniforms, alloc);
374 SkAssertResult(clip);
375 cov.r *= clip.a; // We use the alpha channel of clip for all four.
376 cov.g *= clip.a;
377 cov.b *= clip.a;
378 cov.a *= clip.a;
379 }
380
381 const SkBlenderBase* blender = as_BB(params.blender);
382 const auto as_blendmode = blender->asBlendMode();
383
384 // The math for some blend modes lets us fold coverage into src before the blend, which is
385 // simpler than the canonical post-blend lerp().
386 bool applyPostBlendCoverage = true;
387 if (as_blendmode &&
388 SkBlendMode_ShouldPreScaleCoverage(as_blendmode.value(),
389 params.coverage == Coverage::MaskLCD16)) {
390 applyPostBlendCoverage = false;
391 src.r *= cov.r;
392 src.g *= cov.g;
393 src.b *= cov.b;
394 src.a *= cov.a;
395 }
396
397 // Apply our blend function to the computed color.
398 src = blender->program(p, src, dst, params.dst, uniforms, alloc);
399
400 if (applyPostBlendCoverage) {
401 src.r = lerp(dst.r, src.r, cov.r);
402 src.g = lerp(dst.g, src.g, cov.g);
403 src.b = lerp(dst.b, src.b, cov.b);
404 src.a = lerp(dst.a, src.a, cov.a);
405 }
406
407 if (params.dst.isOpaque()) {
408 // (See the note above when loading the destination color.)
409 src.a = p->splat(1.0f);
410 } else if (params.dst.alphaType() == kUnpremul_SkAlphaType) {
411 src = unpremul(src);
412 }
413
414 // Clamp to fit destination color format if needed.
415 if (SkColorTypeIsNormalized(params.dst.colorType())) {
416 src = clamp01(src);
417 }
418
419 // Write it out!
420 store(dstFormat, dst_ptr, src);
421 }
422
423 // If BuildProgram() can't build this program, CacheKey() sets *ok to false.
CacheKey(const Params & params,skvm::Uniforms * uniforms,SkArenaAlloc * alloc,bool * ok)424 SkVMBlitter::Key SkVMBlitter::CacheKey(
425 const Params& params, skvm::Uniforms* uniforms, SkArenaAlloc* alloc, bool* ok) {
426 // Take care to match buildProgram()'s reuse of the paint color uniforms.
427 skvm::Uniform r = uniforms->pushF(params.paint.fR),
428 g = uniforms->pushF(params.paint.fG),
429 b = uniforms->pushF(params.paint.fB),
430 a = uniforms->pushF(params.paint.fA);
431
432 auto hash_shader = [&](skvm::Builder& p, const sk_sp<SkShader>& shader,
433 skvm::Color* outColor) {
434 const SkShaderBase* sb = as_SB(shader);
435
436 skvm::Coord device = device_coord(&p, uniforms);
437 skvm::Color paint = {
438 p.uniformF(r),
439 p.uniformF(g),
440 p.uniformF(b),
441 p.uniformF(a),
442 };
443
444 uint64_t hash = 0;
445 *outColor = sb->program(&p, device, /*local=*/device, paint, params.matrices,
446 /*localM=*/nullptr, params.dst, uniforms, alloc);
447 if (*outColor) {
448 hash = p.hash();
449 // p.hash() folds in all instructions to produce r,g,b,a but does not know
450 // precisely which value we'll treat as which channel. Imagine the shader
451 // called std::swap(*r,*b)... it draws differently, but p.hash() is unchanged.
452 // We'll fold the hash of their IDs in order to disambiguate.
453 const skvm::Val outputs[] = {
454 outColor->r.id,
455 outColor->g.id,
456 outColor->b.id,
457 outColor->a.id
458 };
459 hash ^= SkOpts::hash(outputs, sizeof(outputs));
460 } else {
461 *ok = false;
462 }
463 return hash;
464 };
465
466 // Use this builder for shader, clip and blender, so that color objects that pass
467 // from one to the other all 'make sense' -- i.e. have the same builder and/or have
468 // meaningful values for the hash.
469 //
470 // Question: better if we just pass in mock uniform colors, so we don't need to
471 // explicitly use the output color from one stage as input to another?
472 //
473 skvm::Builder p;
474
475 // Calculate a hash for the color shader.
476 SkASSERT(params.shader);
477 skvm::Color src;
478 uint64_t shaderHash = hash_shader(p, params.shader, &src);
479
480 // Calculate a hash for the clip shader, if one exists.
481 uint64_t clipHash = 0;
482 if (params.clip) {
483 skvm::Color cov;
484 clipHash = hash_shader(p, params.clip, &cov);
485 if (clipHash == 0) {
486 clipHash = 1;
487 }
488 }
489
490 // Calculate a hash for the blender.
491 uint64_t blendHash = 0;
492 if (auto bm = as_BB(params.blender)->asBlendMode()) {
493 blendHash = static_cast<uint8_t>(bm.value());
494 } else if (*ok) {
495 const SkBlenderBase* blender = as_BB(params.blender);
496
497 skvm::Color dst = DstColor(&p, params);
498 skvm::Color outColor = blender->program(&p, src, dst, params.dst, uniforms, alloc);
499 if (outColor) {
500 blendHash = p.hash();
501 // Like in `hash_shader` above, we must fold the color component IDs into our hash.
502 const skvm::Val outputs[] = {
503 outColor.r.id,
504 outColor.g.id,
505 outColor.b.id,
506 outColor.a.id
507 };
508 blendHash ^= SkOpts::hash(outputs, sizeof(outputs));
509 } else {
510 *ok = false;
511 }
512 if (blendHash == 0) {
513 blendHash = 1;
514 }
515 }
516
517 return {
518 shaderHash,
519 clipHash,
520 blendHash,
521 params.dst.colorSpace() ? params.dst.colorSpace()->hash() : 0,
522 SkToU8(params.dst.colorType()),
523 SkToU8(params.dst.alphaType()),
524 SkToU8(params.coverage),
525 };
526 }
527
SkVMBlitter(const SkPixmap & device,const SkPaint & paint,const SkPixmap * sprite,SkIPoint spriteOffset,const SkMatrixProvider & matrices,sk_sp<SkShader> clip,bool * ok)528 SkVMBlitter::SkVMBlitter(const SkPixmap& device,
529 const SkPaint& paint,
530 const SkPixmap* sprite,
531 SkIPoint spriteOffset,
532 const SkMatrixProvider& matrices,
533 sk_sp<SkShader> clip,
534 bool* ok)
535 : fDevice(device)
536 , fSprite(sprite ? *sprite : SkPixmap{})
537 , fSpriteOffset(spriteOffset)
538 , fUniforms(skvm::UPtr{0}, kBlitterUniformsCount)
539 , fParams(EffectiveParams(device, sprite, paint, matrices, std::move(clip)))
540 , fKey(CacheKey(fParams, &fUniforms, &fAlloc, ok)) {}
541
~SkVMBlitter()542 SkVMBlitter::~SkVMBlitter() {
543 if (SkLRUCache<Key, skvm::Program>* cache = TryAcquireProgramCache()) {
544 auto cache_program = [&](skvm::Program&& program, Coverage coverage) {
545 if (!program.empty()) {
546 cache->insert_or_update(fKey.withCoverage(coverage), std::move(program));
547 }
548 };
549 cache_program(std::move(fBlitH), Coverage::Full);
550 cache_program(std::move(fBlitAntiH), Coverage::UniformF);
551 cache_program(std::move(fBlitMaskA8), Coverage::MaskA8);
552 cache_program(std::move(fBlitMask3D), Coverage::Mask3D);
553 cache_program(std::move(fBlitMaskLCD16), Coverage::MaskLCD16);
554
555 ReleaseProgramCache();
556 }
557 }
558
TryAcquireProgramCache()559 SkLRUCache<SkVMBlitter::Key, skvm::Program>* SkVMBlitter::TryAcquireProgramCache() {
560 #if defined(SKVM_JIT)
561 thread_local static SkLRUCache<Key, skvm::Program> cache{64};
562 return &cache;
563 #else
564 // iOS now supports thread_local since iOS 9.
565 // On the other hand, we'll never be able to JIT there anyway.
566 // It's probably fine to not cache any interpreted programs, anywhere.
567 return nullptr;
568 #endif
569 }
570
DebugName(const Key & key)571 SkString SkVMBlitter::DebugName(const Key& key) {
572 return SkStringPrintf("Shader-%" PRIx64 "_Clip-%" PRIx64 "_Blender-%" PRIx64
573 "_CS-%" PRIx64 "_CT-%d_AT-%d_Cov-%d",
574 key.shader,
575 key.clip,
576 key.blender,
577 key.colorSpace,
578 key.colorType,
579 key.alphaType,
580 key.coverage);
581 }
582
ReleaseProgramCache()583 void SkVMBlitter::ReleaseProgramCache() {}
584
buildProgram(Coverage coverage)585 skvm::Program SkVMBlitter::buildProgram(Coverage coverage) {
586 Key key = fKey.withCoverage(coverage);
587 {
588 skvm::Program p;
589 if (SkLRUCache<Key, skvm::Program>* cache = TryAcquireProgramCache()) {
590 if (skvm::Program* found = cache->find(key)) {
591 p = std::move(*found);
592 }
593 ReleaseProgramCache();
594 }
595 if (!p.empty()) {
596 return p;
597 }
598 }
599 // We don't really _need_ to rebuild fUniforms here.
600 // It's just more natural to have effects unconditionally emit them,
601 // and more natural to rebuild fUniforms than to emit them into a temporary buffer.
602 // fUniforms should reuse the exact same memory, so this is very cheap.
603 SkDEBUGCODE(size_t prev = fUniforms.buf.size();)
604 fUniforms.buf.resize(kBlitterUniformsCount);
605 skvm::Builder builder;
606 BuildProgram(&builder, fParams.withCoverage(coverage), &fUniforms, &fAlloc);
607 SkASSERTF(fUniforms.buf.size() == prev,
608 "%zu, prev was %zu", fUniforms.buf.size(), prev);
609
610 skvm::Program program = builder.done(DebugName(key).c_str());
611 if (false) {
612 static std::atomic<int> missed{0},
613 total{0};
614 if (!program.hasJIT()) {
615 SkDebugf("\ncouldn't JIT %s\n", DebugName(key).c_str());
616 builder.dump();
617 program.dump();
618
619 missed++;
620 }
621 if (0 == total++) {
622 atexit([]{ SkDebugf("SkVMBlitter compiled %d programs, %d without JIT.\n",
623 total.load(), missed.load()); });
624 }
625 }
626 return program;
627 }
628
updateUniforms(int right,int y)629 void SkVMBlitter::updateUniforms(int right, int y) {
630 BlitterUniforms uniforms{right, y};
631 memcpy(fUniforms.buf.data(), &uniforms, sizeof(BlitterUniforms));
632 }
633
isSprite(int x,int y) const634 const void* SkVMBlitter::isSprite(int x, int y) const {
635 if (fSprite.colorType() != kUnknown_SkColorType) {
636 return fSprite.addr(x - fSpriteOffset.x(),
637 y - fSpriteOffset.y());
638 }
639 return nullptr;
640 }
641
blitH(int x,int y,int w)642 void SkVMBlitter::blitH(int x, int y, int w) {
643 if (fBlitH.empty()) {
644 fBlitH = this->buildProgram(Coverage::Full);
645 }
646 this->updateUniforms(x+w, y);
647 if (const void* sprite = this->isSprite(x,y)) {
648 fBlitH.eval(w, fUniforms.buf.data(), fDevice.addr(x,y), sprite);
649 } else {
650 fBlitH.eval(w, fUniforms.buf.data(), fDevice.addr(x,y));
651 }
652 }
653
blitAntiH(int x,int y,const SkAlpha cov[],const int16_t runs[])654 void SkVMBlitter::blitAntiH(int x, int y, const SkAlpha cov[], const int16_t runs[]) {
655 if (fBlitAntiH.empty()) {
656 fBlitAntiH = this->buildProgram(Coverage::UniformF);
657 }
658 if (fBlitH.empty()) {
659 fBlitH = this->buildProgram(Coverage::Full);
660 }
661 for (int16_t run = *runs; run > 0; run = *runs) {
662 const SkAlpha coverage = *cov;
663 if (coverage != 0x00) {
664 this->updateUniforms(x+run, y);
665 const void* sprite = this->isSprite(x,y);
666 if (coverage == 0xFF) {
667 if (sprite) {
668 fBlitH.eval(run, fUniforms.buf.data(), fDevice.addr(x,y), sprite);
669 } else {
670 fBlitH.eval(run, fUniforms.buf.data(), fDevice.addr(x,y));
671 }
672 } else {
673 const float covF = *cov * (1/255.0f);
674 if (sprite) {
675 fBlitAntiH.eval(run, fUniforms.buf.data(), fDevice.addr(x,y), sprite, &covF);
676 } else {
677 fBlitAntiH.eval(run, fUniforms.buf.data(), fDevice.addr(x,y), &covF);
678 }
679 }
680 }
681 x += run;
682 runs += run;
683 cov += run;
684 }
685 }
686
blitMask(const SkMask & mask,const SkIRect & clip)687 void SkVMBlitter::blitMask(const SkMask& mask, const SkIRect& clip) {
688 if (mask.fFormat == SkMask::kBW_Format) {
689 return SkBlitter::blitMask(mask, clip);
690 }
691
692 const skvm::Program* program = nullptr;
693 switch (mask.fFormat) {
694 default: SkUNREACHABLE; // ARGB and SDF masks shouldn't make it here.
695
696 case SkMask::k3D_Format:
697 if (fBlitMask3D.empty()) {
698 fBlitMask3D = this->buildProgram(Coverage::Mask3D);
699 }
700 program = &fBlitMask3D;
701 break;
702
703 case SkMask::kA8_Format:
704 if (fBlitMaskA8.empty()) {
705 fBlitMaskA8 = this->buildProgram(Coverage::MaskA8);
706 }
707 program = &fBlitMaskA8;
708 break;
709
710 case SkMask::kLCD16_Format:
711 if (fBlitMaskLCD16.empty()) {
712 fBlitMaskLCD16 = this->buildProgram(Coverage::MaskLCD16);
713 }
714 program = &fBlitMaskLCD16;
715 break;
716 }
717
718 SkASSERT(program);
719 if (program) {
720 for (int y = clip.top(); y < clip.bottom(); y++) {
721 int x = clip.left(),
722 w = clip.width();
723 void* dptr = fDevice.writable_addr(x,y);
724 auto mptr = (const uint8_t*)mask.getAddr(x,y);
725 this->updateUniforms(x+w,y);
726
727 if (program == &fBlitMask3D) {
728 size_t plane = mask.computeImageSize();
729 if (const void* sprite = this->isSprite(x,y)) {
730 program->eval(w, fUniforms.buf.data(), dptr, sprite, mptr + 1*plane
731 , mptr + 2*plane
732 , mptr + 0*plane);
733 } else {
734 program->eval(w, fUniforms.buf.data(), dptr, mptr + 1*plane
735 , mptr + 2*plane
736 , mptr + 0*plane);
737 }
738 } else {
739 if (const void* sprite = this->isSprite(x,y)) {
740 program->eval(w, fUniforms.buf.data(), dptr, sprite, mptr);
741 } else {
742 program->eval(w, fUniforms.buf.data(), dptr, mptr);
743 }
744 }
745 }
746 }
747 }
748
Make(const SkPixmap & device,const SkPaint & paint,const SkMatrixProvider & matrices,SkArenaAlloc * alloc,sk_sp<SkShader> clip)749 SkVMBlitter* SkVMBlitter::Make(const SkPixmap& device,
750 const SkPaint& paint,
751 const SkMatrixProvider& matrices,
752 SkArenaAlloc* alloc,
753 sk_sp<SkShader> clip) {
754 bool ok = true;
755 SkVMBlitter* blitter = alloc->make<SkVMBlitter>(
756 device, paint, /*sprite=*/nullptr, SkIPoint{0,0}, matrices, std::move(clip), &ok);
757 return ok ? blitter : nullptr;
758 }
759
Make(const SkPixmap & device,const SkPaint & paint,const SkPixmap & sprite,int left,int top,SkArenaAlloc * alloc,sk_sp<SkShader> clip)760 SkVMBlitter* SkVMBlitter::Make(const SkPixmap& device,
761 const SkPaint& paint,
762 const SkPixmap& sprite,
763 int left, int top,
764 SkArenaAlloc* alloc,
765 sk_sp<SkShader> clip) {
766 if (paint.getMaskFilter()) {
767 // TODO: SkVM support for mask filters? definitely possible!
768 return nullptr;
769 }
770 bool ok = true;
771 auto blitter = alloc->make<SkVMBlitter>(
772 device, paint, &sprite, SkIPoint{left,top},
773 SkSimpleMatrixProvider{SkMatrix{}}, std::move(clip), &ok);
774 return ok ? blitter : nullptr;
775 }
776