blob: d55319dab1ecb178387631a6c318902c1a9ded89 [file] [log] [blame]
Ethan Nicholasa70693b2019-03-04 13:07:36 -05001/*
2 * Copyright 2019 Google LLC
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
9#include "include/core/SkCanvas.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040010#include "include/core/SkColorFilter.h"
11#include "include/core/SkData.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkImage.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040013#include "include/core/SkPaint.h"
14#include "include/core/SkRefCnt.h"
15#include "include/core/SkSize.h"
16#include "include/core/SkString.h"
Brian Osmanee426f22020-01-02 11:55:24 -050017#include "include/effects/SkRuntimeEffect.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050018#include "tools/Resources.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040019
20#include <stddef.h>
21#include <utility>
22
Mike Kleina0f54522020-10-14 16:31:36 -050023const char* gNoop = R"(
Brian Osman577c6062021-04-12 17:17:19 -040024 half4 main(half4 color) {
25 return color;
Mike Kleina0f54522020-10-14 16:31:36 -050026 }
27)";
28
Brian Osman5240e182020-08-12 10:22:34 -040029const char* gLumaSrc = R"(
Brian Osman577c6062021-04-12 17:17:19 -040030 half4 main(half4 color) {
31 return dot(color.rgb, half3(0.3, 0.6, 0.1)).000r;
Ethan Nicholasa70693b2019-03-04 13:07:36 -050032 }
33)";
34
Mike Kleina0f54522020-10-14 16:31:36 -050035// Build up the same effect with increasingly complex control flow syntax.
36// All of these are semantically equivalent and can be reduced in principle to one basic block.
37
38// Simplest to run; hardest to write?
39const char* gTernary = R"(
Brian Osman577c6062021-04-12 17:17:19 -040040 half4 main(half4 color) {
Mike Kleina0f54522020-10-14 16:31:36 -050041 half luma = dot(color.rgb, half3(0.3, 0.6, 0.1));
Mike Klein7485ffc2020-10-14 14:05:50 -050042
Mike Kleina0f54522020-10-14 16:31:36 -050043 half scale = luma < 0.33333 ? 0.5
44 : luma < 0.66666 ? (0.166666 + 2.0 * (luma - 0.33333)) / luma
45 : /* else */ (0.833333 + 0.5 * (luma - 0.66666)) / luma;
46 return half4(color.rgb * scale, color.a);
47 }
48)";
49
50// Uses conditional if statements but no early return.
51const char* gIfs = R"(
Brian Osman577c6062021-04-12 17:17:19 -040052 half4 main(half4 color) {
Mike Klein7485ffc2020-10-14 14:05:50 -050053 half luma = dot(color.rgb, half3(0.3, 0.6, 0.1));
54
55 half scale = 0;
Mike Kleina0f54522020-10-14 16:31:36 -050056 if (luma < 0.33333) {
57 scale = 0.5;
58 } else if (luma < 0.66666) {
59 scale = (0.166666 + 2.0 * (luma - 0.33333)) / luma;
60 } else {
61 scale = (0.833333 + 0.5 * (luma - 0.66666)) / luma;
62 }
63 return half4(color.rgb * scale, color.a);
64 }
65)";
Mike Klein7485ffc2020-10-14 14:05:50 -050066
Mike Kleina0f54522020-10-14 16:31:36 -050067// Distilled from AOSP tone mapping shaders, more like what people tend to write.
68const char* gEarlyReturn = R"(
Brian Osman577c6062021-04-12 17:17:19 -040069 half4 main(half4 color) {
Mike Kleina0f54522020-10-14 16:31:36 -050070 half luma = dot(color.rgb, half3(0.3, 0.6, 0.1));
71
72 half scale = 0;
Mike Klein7485ffc2020-10-14 14:05:50 -050073 if (luma < 0.33333) {
74 return half4(color.rgb * 0.5, color.a);
75 } else if (luma < 0.66666) {
76 scale = 0.166666 + 2.0 * (luma - 0.33333);
77 } else {
78 scale = 0.833333 + 0.5 * (luma - 0.66666);
79 }
Mike Klein7485ffc2020-10-14 14:05:50 -050080 return half4(color.rgb * (scale/luma), color.a);
81 }
82)";
83
84
Mike Kleina0f54522020-10-14 16:31:36 -050085DEF_SIMPLE_GM(runtimecolorfilter, canvas, 256 * 3, 256 * 2) {
86 sk_sp<SkImage> img = GetResourceAsImage("images/mandrill_256.png");
Ethan Nicholasa70693b2019-03-04 13:07:36 -050087
Mike Kleina0f54522020-10-14 16:31:36 -050088 auto draw_filter = [&](const char* src) {
Brian Osmanf6123f12021-04-15 13:52:43 -040089 auto [effect, err] = SkRuntimeEffect::MakeForColorFilter(SkString(src));
Mike Kleina0f54522020-10-14 16:31:36 -050090 if (!effect) {
91 SkDebugf("%s\n%s\n", src, err.c_str());
92 }
Ethan Nicholas63d7ee32020-08-17 10:57:12 -040093 SkASSERT(effect);
Brian Osman5240e182020-08-12 10:22:34 -040094 SkPaint p;
Brian Osman577c6062021-04-12 17:17:19 -040095 p.setColorFilter(effect->makeColorFilter(nullptr));
Mike Reed8d29ab62021-01-23 18:10:39 -050096 canvas->drawImage(img, 0, 0, SkSamplingOptions(), &p);
Mike Kleina0f54522020-10-14 16:31:36 -050097 canvas->translate(256, 0);
98 };
99
Brian Osmanf6123f12021-04-15 13:52:43 -0400100 for (const char* src : { gNoop, gLumaSrc }) {
Mike Kleina0f54522020-10-14 16:31:36 -0500101 draw_filter(src);
102 }
Brian Osmanf6123f12021-04-15 13:52:43 -0400103 canvas->translate(-256*2, 256);
Mike Kleina0f54522020-10-14 16:31:36 -0500104 for (const char* src : { gTernary, gIfs, gEarlyReturn}) {
105 draw_filter(src);
Brian Osman5240e182020-08-12 10:22:34 -0400106 }
Ethan Nicholasa70693b2019-03-04 13:07:36 -0500107}