blob: 6f0975da7e19583a3d251a6f61bd2b71c141bc5c [file] [log] [blame]
reeda40be092016-01-14 09:11:51 -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"
Mike Klein33d20552017-03-22 13:47:51 -04009#include "sk_tool_utils.h"
reeda40be092016-01-14 09:11:51 -080010
11#include "Resources.h"
Mike Reed5dd202d2018-02-06 23:05:36 +000012#include "SkColorPriv.h"
reeda40be092016-01-14 09:11:51 -080013#include "SkGradientShader.h"
14#include "SkTypeface.h"
reeda40be092016-01-14 09:11:51 -080015#include "SkStream.h"
16#include "SkPaint.h"
17#include "SkMipMap.h"
18#include "Resources.h"
reeda40be092016-01-14 09:11:51 -080019
reed32e0b4a2016-01-15 13:17:08 -080020#define SHOW_MIP_COLOR 0xFF000000
21
reed01dc44a2016-01-15 10:51:08 -080022static SkBitmap make_bitmap(int w, int h) {
reeda40be092016-01-14 09:11:51 -080023 SkBitmap bm;
reed01dc44a2016-01-15 10:51:08 -080024 bm.allocN32Pixels(w, h);
reeda40be092016-01-14 09:11:51 -080025 SkCanvas canvas(bm);
26 canvas.clear(0xFFFFFFFF);
27 SkPaint paint;
28 paint.setStyle(SkPaint::kStroke_Style);
reed01dc44a2016-01-15 10:51:08 -080029 paint.setStrokeWidth(w / 16.0f);
reed32e0b4a2016-01-15 13:17:08 -080030 paint.setColor(SHOW_MIP_COLOR);
reed01dc44a2016-01-15 10:51:08 -080031 canvas.drawCircle(w/2.0f, h/2.0f, w/3.0f, paint);
reeda40be092016-01-14 09:11:51 -080032 return bm;
33}
34
reed01dc44a2016-01-15 10:51:08 -080035static SkBitmap make_bitmap2(int w, int h) {
reeda40be092016-01-14 09:11:51 -080036 SkBitmap bm;
reed01dc44a2016-01-15 10:51:08 -080037 bm.allocN32Pixels(w, h);
reeda40be092016-01-14 09:11:51 -080038 SkCanvas canvas(bm);
39 canvas.clear(0xFFFFFFFF);
40 SkPaint paint;
reed32e0b4a2016-01-15 13:17:08 -080041 paint.setColor(SHOW_MIP_COLOR);
reeda40be092016-01-14 09:11:51 -080042 paint.setStyle(SkPaint::kStroke_Style);
43
44 SkScalar inset = 2;
reed01dc44a2016-01-15 10:51:08 -080045 SkRect r = SkRect::MakeIWH(w, h).makeInset(0.5f, 0.5f);
reeda40be092016-01-14 09:11:51 -080046 while (r.width() > 4) {
47 canvas.drawRect(r, paint);
48 r.inset(inset, inset);
49 inset += 1;
50 }
51 return bm;
52}
53
54#include "SkNx.h"
reed01dc44a2016-01-15 10:51:08 -080055static SkBitmap make_bitmap3(int w, int h) {
reeda40be092016-01-14 09:11:51 -080056 SkBitmap bm;
reed01dc44a2016-01-15 10:51:08 -080057 bm.allocN32Pixels(w, h);
reeda40be092016-01-14 09:11:51 -080058 SkCanvas canvas(bm);
59 canvas.clear(0xFFFFFFFF);
60 SkPaint paint;
61 paint.setStyle(SkPaint::kStroke_Style);
62 paint.setStrokeWidth(2.1f);
reed32e0b4a2016-01-15 13:17:08 -080063 paint.setColor(SHOW_MIP_COLOR);
reeda40be092016-01-14 09:11:51 -080064
reed01dc44a2016-01-15 10:51:08 -080065 SkScalar s = SkIntToScalar(w);
reeda40be092016-01-14 09:11:51 -080066 Sk4f p(s, -s, -s, s);
67 Sk4f d(5);
mtklein7c249e52016-02-21 10:54:19 -080068 while (p[1] < s) {
69 canvas.drawLine(p[0],p[1], p[2], p[3], paint);
reeda40be092016-01-14 09:11:51 -080070 p = p + d;
71 }
72 return bm;
73}
74
75class ShowMipLevels : public skiagm::GM {
76 const int fN;
77 SkBitmap fBM[4];
78
79public:
80 static unsigned gamma(unsigned n) {
81 float x = n / 255.0f;
82#if 0
83 x = sqrtf(x);
84#else
85 if (x > 0.0031308f) {
86 x = 1.055f * (powf(x, (1.0f / 2.4f))) - 0.055f;
87 } else {
88 x = 12.92f * x;
89 }
90#endif
91 return (int)(x * 255);
92 }
93
94 static void apply_gamma(const SkBitmap& bm) {
95 return; // below is our experiment for sRGB correction
reeda40be092016-01-14 09:11:51 -080096 for (int y = 0; y < bm.height(); ++y) {
97 for (int x = 0; x < bm.width(); ++x) {
98 SkPMColor c = *bm.getAddr32(x, y);
99 unsigned r = gamma(SkGetPackedR32(c));
100 unsigned g = gamma(SkGetPackedG32(c));
101 unsigned b = gamma(SkGetPackedB32(c));
102 *bm.getAddr32(x, y) = SkPackARGB32(0xFF, r, g, b);
103 }
104 }
105 }
106
robertphillips55713af2016-08-26 10:04:26 -0700107 ShowMipLevels(int N) : fN(N) { }
reeda40be092016-01-14 09:11:51 -0800108
109protected:
110
111 SkString onShortName() override {
112 SkString str;
113 str.printf("showmiplevels_%d", fN);
114 return str;
115 }
116
Mike Reede32500f2017-07-19 17:20:37 -0400117 SkISize onISize() override { return { 150, 862 }; }
reeda40be092016-01-14 09:11:51 -0800118
119 static void DrawAndFrame(SkCanvas* canvas, const SkBitmap& orig, SkScalar x, SkScalar y) {
120 SkBitmap bm;
Matt Sarett68b8e3d2017-04-28 11:15:22 -0400121 sk_tool_utils::copy_to(&bm, orig.colorType(), orig);
reeda40be092016-01-14 09:11:51 -0800122 apply_gamma(bm);
123
124 canvas->drawBitmap(bm, x, y, nullptr);
125 SkPaint paint;
126 paint.setStyle(SkPaint::kStroke_Style);
127 paint.setColor(0xFFFFCCCC);
128 canvas->drawRect(SkRect::MakeIWH(bm.width(), bm.height()).makeOffset(x, y).makeOutset(0.5f, 0.5f), paint);
129 }
130
131 template <typename F> void drawLevels(SkCanvas* canvas, const SkBitmap& baseBM, F func) {
132 SkScalar x = 4;
133 SkScalar y = 4;
134
135 SkPixmap prevPM;
reeda40be092016-01-14 09:11:51 -0800136 baseBM.peekPixels(&prevPM);
137
Brian Osman7b8400d2016-11-08 17:08:54 -0500138 SkDestinationSurfaceColorMode colorMode = SkDestinationSurfaceColorMode::kLegacy;
139 sk_sp<SkMipMap> mm(SkMipMap::Build(baseBM, colorMode, nullptr));
reeda40be092016-01-14 09:11:51 -0800140
141 int index = 0;
142 SkMipMap::Level level;
143 SkScalar scale = 0.5f;
fmalita33ed3ad2016-02-09 08:20:18 -0800144 while (mm->extractLevel(SkSize::Make(scale, scale), &level)) {
reed67b09bf2016-01-16 18:50:35 -0800145 SkBitmap bm = func(prevPM, level.fPixmap);
reeda40be092016-01-14 09:11:51 -0800146 DrawAndFrame(canvas, bm, x, y);
147
reed67b09bf2016-01-16 18:50:35 -0800148 if (level.fPixmap.width() <= 2 || level.fPixmap.height() <= 2) {
reeda40be092016-01-14 09:11:51 -0800149 break;
150 }
151 if (index & 1) {
reed67b09bf2016-01-16 18:50:35 -0800152 x += level.fPixmap.width() + 4;
reeda40be092016-01-14 09:11:51 -0800153 } else {
reed67b09bf2016-01-16 18:50:35 -0800154 y += level.fPixmap.height() + 4;
reeda40be092016-01-14 09:11:51 -0800155 }
156 scale /= 2;
reed67b09bf2016-01-16 18:50:35 -0800157 prevPM = level.fPixmap;
reeda40be092016-01-14 09:11:51 -0800158 index += 1;
159 }
160 }
161
162 void drawSet(SkCanvas* canvas, const SkBitmap& orig) {
163 SkAutoCanvasRestore acr(canvas, true);
164
165 drawLevels(canvas, orig, [](const SkPixmap& prev, const SkPixmap& curr) {
166 SkBitmap bm;
167 bm.installPixels(curr);
168 return bm;
169 });
reeda40be092016-01-14 09:11:51 -0800170 }
171
robertphillips55713af2016-08-26 10:04:26 -0700172 void onOnceBeforeDraw() override {
173 fBM[0] = sk_tool_utils::create_checkerboard_bitmap(fN, fN, SK_ColorBLACK, SK_ColorWHITE, 2);
174 fBM[1] = make_bitmap(fN, fN);
175 fBM[2] = make_bitmap2(fN, fN);
176 fBM[3] = make_bitmap3(fN, fN);
177 }
178
reeda40be092016-01-14 09:11:51 -0800179 void onDraw(SkCanvas* canvas) override {
180 canvas->translate(4, 4);
181 for (const auto& bm : fBM) {
182 this->drawSet(canvas, bm);
Greg Daniel7785dd22017-06-07 10:01:48 -0400183 // round so we always produce an integral translate, so the GOLD tool won't show
184 // unimportant diffs if this is drawn on a GPU with different rounding rules
185 // since we draw the bitmaps using nearest-neighbor
186 canvas->translate(0, SkScalarRoundToScalar(bm.height() * 0.85f));
reeda40be092016-01-14 09:11:51 -0800187 }
188 }
halcanary9d524f22016-03-29 09:03:52 -0700189
reeda40be092016-01-14 09:11:51 -0800190private:
191 typedef skiagm::GM INHERITED;
192};
193DEF_GM( return new ShowMipLevels(255); )
194DEF_GM( return new ShowMipLevels(256); )
195
reed01dc44a2016-01-15 10:51:08 -0800196///////////////////////////////////////////////////////////////////////////////////////////////////
197
Matt Sarettcb6266b2017-01-17 10:48:53 -0500198void copy_to(SkBitmap* dst, SkColorType dstColorType, const SkBitmap& src) {
199 if (kGray_8_SkColorType == dstColorType) {
Matt Sarett4897fb82017-01-18 11:49:33 -0500200 return sk_tool_utils::copy_to_g8(dst, src);
Matt Sarettcb6266b2017-01-17 10:48:53 -0500201 }
202
Matt Sarett68b8e3d2017-04-28 11:15:22 -0400203 const SkBitmap* srcPtr = &src;
204 SkBitmap tmp(src);
205 if (kRGB_565_SkColorType == dstColorType) {
206 tmp.setAlphaType(kOpaque_SkAlphaType);
207 srcPtr = &tmp;
208 }
209
210 sk_tool_utils::copy_to(dst, dstColorType, *srcPtr);
Matt Sarettcb6266b2017-01-17 10:48:53 -0500211}
212
reed01dc44a2016-01-15 10:51:08 -0800213/**
214 * Show mip levels that were built, for all supported colortypes
215 */
216class ShowMipLevels2 : public skiagm::GM {
217 const int fW, fH;
218 SkBitmap fBM[4];
219
220public:
robertphillips55713af2016-08-26 10:04:26 -0700221 ShowMipLevels2(int w, int h) : fW(w), fH(h) { }
reed01dc44a2016-01-15 10:51:08 -0800222
223protected:
224
225 SkString onShortName() override {
226 SkString str;
227 str.printf("showmiplevels2_%dx%d", fW, fH);
228 return str;
229 }
230
231 SkISize onISize() override {
232 return { 824, 862 };
233 }
234
235 static void DrawAndFrame(SkCanvas* canvas, const SkBitmap& bm, SkScalar x, SkScalar y) {
236 canvas->drawBitmap(bm, x, y, nullptr);
237 SkPaint paint;
238 paint.setStyle(SkPaint::kStroke_Style);
239 paint.setColor(0xFFFFCCCC);
240 canvas->drawRect(SkRect::MakeIWH(bm.width(), bm.height()).makeOffset(x, y).makeOutset(0.5f, 0.5f), paint);
241 }
242
243 void drawLevels(SkCanvas* canvas, const SkBitmap& baseBM) {
244 SkScalar x = 4;
245 SkScalar y = 4;
246
Brian Osman7b8400d2016-11-08 17:08:54 -0500247 SkDestinationSurfaceColorMode colorMode = SkDestinationSurfaceColorMode::kLegacy;
248 sk_sp<SkMipMap> mm(SkMipMap::Build(baseBM, colorMode, nullptr));
reed01dc44a2016-01-15 10:51:08 -0800249
250 int index = 0;
251 SkMipMap::Level level;
252 SkScalar scale = 0.5f;
fmalita33ed3ad2016-02-09 08:20:18 -0800253 while (mm->extractLevel(SkSize::Make(scale, scale), &level)) {
reed01dc44a2016-01-15 10:51:08 -0800254 SkBitmap bm;
reed67b09bf2016-01-16 18:50:35 -0800255 bm.installPixels(level.fPixmap);
reed01dc44a2016-01-15 10:51:08 -0800256 DrawAndFrame(canvas, bm, x, y);
257
reed67b09bf2016-01-16 18:50:35 -0800258 if (level.fPixmap.width() <= 2 || level.fPixmap.height() <= 2) {
reed01dc44a2016-01-15 10:51:08 -0800259 break;
260 }
261 if (index & 1) {
reed67b09bf2016-01-16 18:50:35 -0800262 x += level.fPixmap.width() + 4;
reed01dc44a2016-01-15 10:51:08 -0800263 } else {
reed67b09bf2016-01-16 18:50:35 -0800264 y += level.fPixmap.height() + 4;
reed01dc44a2016-01-15 10:51:08 -0800265 }
266 scale /= 2;
267 index += 1;
268 }
269 }
270
271 void drawSet(SkCanvas* canvas, const SkBitmap& orig) {
272 const SkColorType ctypes[] = {
273 kN32_SkColorType, kRGB_565_SkColorType, kARGB_4444_SkColorType, kGray_8_SkColorType
274 };
275
276 SkAutoCanvasRestore acr(canvas, true);
277
278 for (auto ctype : ctypes) {
279 SkBitmap bm;
Matt Sarettcb6266b2017-01-17 10:48:53 -0500280 copy_to(&bm, ctype, orig);
reed01dc44a2016-01-15 10:51:08 -0800281 drawLevels(canvas, bm);
282 canvas->translate(orig.width()/2 + 8.0f, 0);
283 }
284 }
285
robertphillips55713af2016-08-26 10:04:26 -0700286 void onOnceBeforeDraw() override {
287 fBM[0] = sk_tool_utils::create_checkerboard_bitmap(fW, fH,
288 SHOW_MIP_COLOR, SK_ColorWHITE, 2);
289 fBM[1] = make_bitmap(fW, fH);
290 fBM[2] = make_bitmap2(fW, fH);
291 fBM[3] = make_bitmap3(fW, fH);
292 }
293
reed01dc44a2016-01-15 10:51:08 -0800294 void onDraw(SkCanvas* canvas) override {
295 canvas->translate(4, 4);
296 for (const auto& bm : fBM) {
297 this->drawSet(canvas, bm);
298 // round so we always produce an integral translate, so the GOLD tool won't show
299 // unimportant diffs if this is drawn on a GPU with different rounding rules
300 // since we draw the bitmaps using nearest-neighbor
301 canvas->translate(0, SkScalarRoundToScalar(bm.height() * 0.85f));
302 }
303 }
halcanary9d524f22016-03-29 09:03:52 -0700304
reed01dc44a2016-01-15 10:51:08 -0800305private:
306 typedef skiagm::GM INHERITED;
307};
308DEF_GM( return new ShowMipLevels2(255, 255); )
309DEF_GM( return new ShowMipLevels2(256, 255); )
310DEF_GM( return new ShowMipLevels2(255, 256); )
311DEF_GM( return new ShowMipLevels2(256, 256); )