blob: 1bf43013c969973641ccd3fad760a4ccc158edb6 [file] [log] [blame]
herbfeec8782016-02-17 10:00:07 -08001/*
2 * Copyright 2016 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 "gm.h"
Herb Derby83e939b2017-02-07 14:25:11 -05009
10#include "SkArenaAlloc.h"
fmalitad0c4e092016-02-22 17:19:04 -080011#include "SkBlitter.h"
herbfeec8782016-02-17 10:00:07 -080012#include "SkCanvas.h"
13#include "SkColor.h"
14#include "SkImage.h"
15#include "SkImageInfo.h"
16#include "SkLinearBitmapPipeline.h"
Mike Reedd4706732016-11-15 16:44:34 -050017#include "SkXfermodePriv.h"
herbfeec8782016-02-17 10:00:07 -080018#include "SkPM4fPriv.h"
19#include "SkShader.h"
20
21static void fill_in_bits(SkBitmap& bm, SkIRect ir, SkColor c, bool premul) {
22 bm.allocN32Pixels(ir.width(), ir.height());
23 SkPixmap pm;
24 bm.peekPixels(&pm);
25
26 SkPMColor b = SkColorSetARGBMacro(255, 0, 0, 0);
27 SkPMColor w;
28 if (premul) {
29 w = SkPreMultiplyColor(c);
30 } else {
31 w = SkPackARGB32NoCheck(SkColorGetA(c), SkColorGetR(c), SkColorGetG(c), SkColorGetB(c));
32 }
33
34 for (int y = 0; y < ir.height(); y++) {
35 for (int x = 0; x < ir.width(); x++) {
36 if ((x ^ y) & 16) {
37 *pm.writable_addr32(x, y) = b;
38 } else {
39 *pm.writable_addr32(x, y) = w;
40 }
41 }
42 }
43}
44
45static void draw_rect_orig(SkCanvas* canvas, const SkRect& r, SkColor c, const SkMatrix* mat, bool useBilerp) {
46 const SkIRect ir = r.round();
47
48 SkBitmap bmsrc;
49 fill_in_bits(bmsrc, ir, c, true);
50
51 SkPixmap pmsrc;
52 bmsrc.peekPixels(&pmsrc);
53
54 SkBitmap bmdst;
55 bmdst.allocN32Pixels(ir.width(), ir.height());
56 bmdst.eraseColor(0xFFFFFFFF);
57 SkPixmap pmdst;
58 bmdst.peekPixels(&pmdst);
59
brianosman52ede1d2016-06-20 08:25:02 -070060 SkImageInfo info = SkImageInfo::MakeN32Premul(ir.width(), ir.height());
herbfeec8782016-02-17 10:00:07 -080061
reed9ce9d672016-03-17 10:51:11 -070062 sk_sp<SkImage> image(SkImage::MakeRasterCopy(SkPixmap(info, pmsrc.addr32(), pmsrc.rowBytes())));
herbfeec8782016-02-17 10:00:07 -080063 SkPaint paint;
Herb Derby83e939b2017-02-07 14:25:11 -050064 SkArenaAlloc alloc{kSkBlitterContextSize * sizeof(uint32_t)};
herb6eff52a2016-03-23 09:00:33 -070065
66 sk_sp<SkShader> shader = image->makeShader(SkShader::kRepeat_TileMode,
67 SkShader::kRepeat_TileMode);
68
herbfeec8782016-02-17 10:00:07 -080069 if (useBilerp) {
70 paint.setFilterQuality(SkFilterQuality::kLow_SkFilterQuality);
71 } else {
72 paint.setFilterQuality(SkFilterQuality::kNone_SkFilterQuality);
73 }
herb6eff52a2016-03-23 09:00:33 -070074 paint.setShader(std::move(shader));
fmalitad0c4e092016-02-22 17:19:04 -080075 const SkShader::ContextRec rec(paint, *mat, nullptr,
Brian Osman11970e52016-12-05 15:26:50 -050076 SkBlitter::PreferredShaderDest(pmsrc.info()),
77 canvas->imageInfo().colorSpace());
herbfeec8782016-02-17 10:00:07 -080078
Herb Derby83e939b2017-02-07 14:25:11 -050079 SkShader::Context* ctx = paint.getShader()->makeContext(rec, &alloc);
herbfeec8782016-02-17 10:00:07 -080080
81 for (int y = 0; y < ir.height(); y++) {
82 ctx->shadeSpan(0, y, pmdst.writable_addr32(0, y), ir.width());
83 }
84
85 canvas->drawBitmap(bmdst, r.left(), r.top(), nullptr);
86
87 ctx->~Context();
herbfeec8782016-02-17 10:00:07 -080088}
89
90static void draw_rect_fp(SkCanvas* canvas, const SkRect& r, SkColor c, const SkMatrix* mat, bool useBilerp) {
91 const SkIRect ir = r.round();
92
93 SkBitmap bmsrc;
94 fill_in_bits(bmsrc, ir, c, true);
95 SkPixmap pmsrc;
96 bmsrc.peekPixels(&pmsrc);
97
98 SkBitmap bmdst;
99 bmdst.allocN32Pixels(ir.width(), ir.height());
100 bmdst.eraseColor(0xFFFFFFFF);
101 SkPixmap pmdst;
102 bmdst.peekPixels(&pmdst);
103
104 SkPM4f* dstBits = new SkPM4f[ir.width()];
herbfeec8782016-02-17 10:00:07 -0800105
106 SkMatrix inv;
107 bool trash = mat->invert(&inv);
108 sk_ignore_unused_variable(trash);
109
herbc5eddd72016-02-17 19:50:05 -0800110 SkFilterQuality filterQuality;
111 if (useBilerp) {
112 filterQuality = SkFilterQuality::kLow_SkFilterQuality;
113 } else {
114 filterQuality = SkFilterQuality::kNone_SkFilterQuality;
115 }
herbfeec8782016-02-17 10:00:07 -0800116
117 uint32_t flags = 0;
Mike Reed6a015542016-11-09 10:38:09 -0500118 auto procN = SkXfermode::GetD32Proc(SkBlendMode::kSrcOver, flags);
herbfeec8782016-02-17 10:00:07 -0800119
Herb Derbydfa492e2016-11-17 21:12:36 -0500120 char storage[512];
Herb Derby0497f082017-01-13 11:30:44 -0500121 SkArenaAlloc allocator{storage, sizeof(storage)};
herbfeec8782016-02-17 10:00:07 -0800122 SkLinearBitmapPipeline pipeline{
herbc5eddd72016-02-17 19:50:05 -0800123 inv, filterQuality,
Herb Derbydfa492e2016-11-17 21:12:36 -0500124 SkShader::kRepeat_TileMode, SkShader::kRepeat_TileMode,
125 SK_ColorBLACK, pmsrc, &allocator};
herbfeec8782016-02-17 10:00:07 -0800126
127 for (int y = 0; y < ir.height(); y++) {
128 pipeline.shadeSpan4f(0, y, dstBits, ir.width());
Mike Reed6a015542016-11-09 10:38:09 -0500129 procN(SkBlendMode::kSrcOver, pmdst.writable_addr32(0, y), dstBits, ir.width(), nullptr);
herbfeec8782016-02-17 10:00:07 -0800130 }
131
132 delete [] dstBits;
133
134 canvas->drawBitmap(bmdst, r.left(), r.top(), nullptr);
135}
136
137static void draw_rect_none(SkCanvas* canvas, const SkRect& r, SkColor c) {
138 const SkIRect ir = r.round();
139
140 SkBitmap bm;
141 fill_in_bits(bm, ir, c, true);
142
143 canvas->drawBitmap(bm, r.left(), r.top(), nullptr);
144}
145
146/*
147 * Test SkXfer4fProcs directly for src-over, comparing them to current SkColor blits.
148 */
herb94665712016-02-29 07:53:22 -0800149DEF_SIMPLE_GM(linear_pipeline, canvas, 580, 2200) {
herbfeec8782016-02-17 10:00:07 -0800150 const int IW = 50;
151 const SkScalar W = IW;
152 const SkScalar H = 100;
153
154 const SkColor colors[] = {
155 0x880000FF, 0x8800FF00, 0x88FF0000, 0x88000000,
156 SK_ColorBLUE, SK_ColorGREEN, SK_ColorRED, SK_ColorBLACK,
157 };
158
159 canvas->translate(20, 20);
160
161 SkMatrix mi = SkMatrix::I();
herb94665712016-02-29 07:53:22 -0800162 SkMatrix mlr;
163 mlr.setScale(-1.0f, 1.0f, 20, 0.0f);
herbfeec8782016-02-17 10:00:07 -0800164 SkMatrix mt;
165 mt.setTranslate(8, 8);
herb94665712016-02-29 07:53:22 -0800166 SkMatrix mt2;
167 mt2.setTranslate(-18, -18);
herbfeec8782016-02-17 10:00:07 -0800168 SkMatrix ms;
herb6eff52a2016-03-23 09:00:33 -0700169 ms.setScale(2.7f, 2.7f, -1.5f, 0);
herb94665712016-02-29 07:53:22 -0800170 SkMatrix ms2;
herb6eff52a2016-03-23 09:00:33 -0700171 ms2.setScale(-0.4f, 0.4f);
herbfeec8782016-02-17 10:00:07 -0800172 SkMatrix mr;
173 mr.setRotate(10);
174
herb94665712016-02-29 07:53:22 -0800175 const SkMatrix* mats[] = {nullptr, &mi, &mlr, &mt, &mt2, &ms, &ms2, &mr};
herbfeec8782016-02-17 10:00:07 -0800176
177 const SkRect r = SkRect::MakeWH(W, H);
178 bool useBilerp = false;
179 while (true) {
180 canvas->save();
181 for (auto mat : mats) {
182 canvas->save();
183 for (SkColor c : colors) {
184 if (mat == nullptr) {
185 SkPaint p;
186 p.setColor(c);
187 draw_rect_none(canvas, r, c);
188 canvas->translate(W + 20, 0);
189 draw_rect_none(canvas, r, c);
190
191 } else {
192 draw_rect_orig(canvas, r, c, mat, useBilerp);
193 canvas->translate(W + 20, 0);
194 draw_rect_fp(canvas, r, c, mat, useBilerp);
195 }
196 canvas->translate(W + 20, 0);
197 }
198 canvas->restore();
199 canvas->translate(0, H + 20);
200 }
201 canvas->restore();
202 canvas->translate(0, (H + 20) * SK_ARRAY_COUNT(mats));
203 if (useBilerp) break;
204 useBilerp = true;
205 }
206}