blob: ac53784fad4a36883c6e5767a24e234bba2fd342 [file] [log] [blame]
Matt Sarett909d3792016-12-22 10:52:25 -05001/*
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"
9#include "Resources.h"
10#include "SkCodec.h"
11#include "SkColorSpace.h"
Mike Kleine28a6b52018-07-25 13:05:17 -040012#include "SkColorSpacePriv.h"
Matt Sarett909d3792016-12-22 10:52:25 -050013#include "SkHalf.h"
14#include "SkImage.h"
Mike Reed7fcfb622018-02-09 13:26:46 -050015#include "SkImageInfoPriv.h"
Matt Sarett34855f92017-01-10 17:41:53 -050016#include "SkPictureRecorder.h"
Matt Sarett909d3792016-12-22 10:52:25 -050017
Matt Sarett909d3792016-12-22 10:52:25 -050018static const int kWidth = 64;
19static const int kHeight = 64;
20
Brian Osmanb1168a72017-03-20 14:21:18 -040021static sk_sp<SkImage> make_raster_image(SkColorType colorType) {
Hal Canaryc465d132017-12-08 10:21:31 -050022 std::unique_ptr<SkStream> stream(GetResourceAsStream("images/google_chrome.ico"));
Mike Reedede7bac2017-07-23 15:30:02 -040023 std::unique_ptr<SkCodec> codec = SkCodec::MakeFromStream(std::move(stream));
Hal Canarybaa2a282018-11-26 15:34:12 -050024 if (!codec) {
25 return nullptr;
26 }
Matt Sarett909d3792016-12-22 10:52:25 -050027
28 SkBitmap bitmap;
29 SkImageInfo info = codec->getInfo().makeWH(kWidth, kHeight)
30 .makeColorType(colorType)
Brian Osman6b622962018-08-27 19:16:02 +000031 .makeAlphaType(kPremul_SkAlphaType);
Matt Sarett909d3792016-12-22 10:52:25 -050032 bitmap.allocPixels(info);
33 codec->getPixels(info, bitmap.getPixels(), bitmap.rowBytes());
34 bitmap.setImmutable();
35 return SkImage::MakeFromBitmap(bitmap);
36}
37
Matt Sarett34855f92017-01-10 17:41:53 -050038static sk_sp<SkImage> make_codec_image() {
Hal Canaryc465d132017-12-08 10:21:31 -050039 sk_sp<SkData> encoded = GetResourceAsData("images/randPixels.png");
Matt Sarett34855f92017-01-10 17:41:53 -050040 return SkImage::MakeFromEncoded(encoded);
41}
42
43static void draw_contents(SkCanvas* canvas) {
44 SkPaint paint;
45 paint.setStyle(SkPaint::kStroke_Style);
46 paint.setStrokeWidth(20);
47 paint.setColor(0xFF800000);
48 canvas->drawCircle(40, 40, 35, paint);
49 paint.setColor(0xFF008000);
50 canvas->drawCircle(50, 50, 35, paint);
51 paint.setColor(0xFF000080);
52 canvas->drawCircle(60, 60, 35, paint);
53}
54
Matt Sarett9df70bb2017-02-14 13:50:43 -050055static sk_sp<SkImage> make_picture_image() {
Matt Sarett34855f92017-01-10 17:41:53 -050056 SkPictureRecorder recorder;
57 draw_contents(recorder.beginRecording(SkRect::MakeIWH(kWidth, kHeight)));
58 return SkImage::MakeFromPicture(recorder.finishRecordingAsPicture(),
59 SkISize::Make(kWidth, kHeight), nullptr, nullptr,
60 SkImage::BitDepth::kU8,
Matt Sarett77a7a1b2017-02-07 13:56:11 -050061 SkColorSpace::MakeSRGB());
Matt Sarett34855f92017-01-10 17:41:53 -050062}
63
Matt Sarett733340a2017-05-02 16:19:51 -040064static sk_sp<SkColorSpace> make_parametric_transfer_fn(const SkColorSpacePrimaries& primaries) {
Brian Osman82ebe042019-01-04 17:03:00 -050065 skcms_Matrix3x3 toXYZD50;
Matt Sarett909d3792016-12-22 10:52:25 -050066 SkAssertResult(primaries.toXYZD50(&toXYZD50));
Brian Osman82ebe042019-01-04 17:03:00 -050067 skcms_TransferFunction fn = { 1.8f, 1.f, 0.f, 0.f, 0.f, 0.f, 0.f };
Matt Sarett733340a2017-05-02 16:19:51 -040068 return SkColorSpace::MakeRGB(fn, toXYZD50);
Matt Sarett909d3792016-12-22 10:52:25 -050069}
70
71static sk_sp<SkColorSpace> make_wide_gamut() {
72 // ProPhoto
73 SkColorSpacePrimaries primaries;
74 primaries.fRX = 0.7347f;
75 primaries.fRY = 0.2653f;
76 primaries.fGX = 0.1596f;
77 primaries.fGY = 0.8404f;
78 primaries.fBX = 0.0366f;
79 primaries.fBY = 0.0001f;
80 primaries.fWX = 0.34567f;
81 primaries.fWY = 0.35850f;
Matt Sarett733340a2017-05-02 16:19:51 -040082 return make_parametric_transfer_fn(primaries);
Matt Sarett909d3792016-12-22 10:52:25 -050083}
84
85static sk_sp<SkColorSpace> make_small_gamut() {
86 SkColorSpacePrimaries primaries;
87 primaries.fRX = 0.50f;
88 primaries.fRY = 0.33f;
89 primaries.fGX = 0.30f;
90 primaries.fGY = 0.50f;
91 primaries.fBX = 0.25f;
92 primaries.fBY = 0.16f;
93 primaries.fWX = 0.3127f;
94 primaries.fWY = 0.3290f;
Matt Sarett733340a2017-05-02 16:19:51 -040095 return make_parametric_transfer_fn(primaries);
Matt Sarett909d3792016-12-22 10:52:25 -050096}
97
98static void draw_image(SkCanvas* canvas, SkImage* image, SkColorType dstColorType,
Matt Sarett34855f92017-01-10 17:41:53 -050099 SkAlphaType dstAlphaType, sk_sp<SkColorSpace> dstColorSpace,
100 SkImage::CachingHint hint) {
Matt Sarett909d3792016-12-22 10:52:25 -0500101 size_t rowBytes = image->width() * SkColorTypeBytesPerPixel(dstColorType);
102 sk_sp<SkData> data = SkData::MakeUninitialized(rowBytes * image->height());
Matt Sarett909d3792016-12-22 10:52:25 -0500103 SkImageInfo dstInfo = SkImageInfo::Make(image->width(), image->height(), dstColorType,
104 dstAlphaType, dstColorSpace);
Brian Osman89150582017-03-20 16:54:28 -0400105 if (!image->readPixels(dstInfo, data->writable_data(), rowBytes, 0, 0, hint)) {
106 memset(data->writable_data(), 0, rowBytes * image->height());
107 }
Matt Sarett909d3792016-12-22 10:52:25 -0500108
Matt Sarett909d3792016-12-22 10:52:25 -0500109 // Now that we have called readPixels(), dump the raw pixels into an srgb image.
Brian Osman6b622962018-08-27 19:16:02 +0000110 sk_sp<SkColorSpace> srgb = SkColorSpace::MakeSRGB();
Matt Sarett909d3792016-12-22 10:52:25 -0500111 sk_sp<SkImage> raw = SkImage::MakeRasterData(dstInfo.makeColorSpace(srgb), data, rowBytes);
112 canvas->drawImage(raw.get(), 0.0f, 0.0f, nullptr);
113}
114
115class ReadPixelsGM : public skiagm::GM {
116public:
117 ReadPixelsGM() {}
118
119protected:
120 SkString onShortName() override {
121 return SkString("readpixels");
122 }
123
124 SkISize onISize() override {
Brian Osmanb1168a72017-03-20 14:21:18 -0400125 return SkISize::Make(6 * kWidth, 9 * kHeight);
Matt Sarett909d3792016-12-22 10:52:25 -0500126 }
127
128 void onDraw(SkCanvas* canvas) override {
Matt Sarett909d3792016-12-22 10:52:25 -0500129 const SkAlphaType alphaTypes[] = {
130 kUnpremul_SkAlphaType,
131 kPremul_SkAlphaType,
132 };
133 const SkColorType colorTypes[] = {
134 kRGBA_8888_SkColorType,
135 kBGRA_8888_SkColorType,
136 kRGBA_F16_SkColorType,
137 };
138 const sk_sp<SkColorSpace> colorSpaces[] = {
139 make_wide_gamut(),
Matt Sarett77a7a1b2017-02-07 13:56:11 -0500140 SkColorSpace::MakeSRGB(),
Matt Sarett909d3792016-12-22 10:52:25 -0500141 make_small_gamut(),
142 };
143
144 for (sk_sp<SkColorSpace> dstColorSpace : colorSpaces) {
145 for (SkColorType srcColorType : colorTypes) {
Brian Osmanb1168a72017-03-20 14:21:18 -0400146 canvas->save();
147 sk_sp<SkImage> image = make_raster_image(srcColorType);
Hal Canarybaa2a282018-11-26 15:34:12 -0500148 if (!image) {
149 continue;
150 }
Brian Osman89150582017-03-20 16:54:28 -0400151 if (GrContext* context = canvas->getGrContext()) {
152 image = image->makeTextureImage(context, canvas->imageInfo().colorSpace());
153 }
154 if (image) {
155 for (SkColorType dstColorType : colorTypes) {
156 for (SkAlphaType dstAlphaType : alphaTypes) {
157 draw_image(canvas, image.get(), dstColorType, dstAlphaType,
158 dstColorSpace, SkImage::kAllow_CachingHint);
159 canvas->translate((float)kWidth, 0.0f);
160 }
Matt Sarett909d3792016-12-22 10:52:25 -0500161 }
Matt Sarett909d3792016-12-22 10:52:25 -0500162 }
Brian Osmanb1168a72017-03-20 14:21:18 -0400163 canvas->restore();
164 canvas->translate(0.0f, (float) kHeight);
Matt Sarett909d3792016-12-22 10:52:25 -0500165 }
166 }
167 }
168
169private:
170 typedef skiagm::GM INHERITED;
171};
172DEF_GM( return new ReadPixelsGM; )
Matt Sarett34855f92017-01-10 17:41:53 -0500173
174class ReadPixelsCodecGM : public skiagm::GM {
175public:
176 ReadPixelsCodecGM() {}
177
178protected:
179 SkString onShortName() override {
180 return SkString("readpixelscodec");
181 }
182
183 SkISize onISize() override {
184 return SkISize::Make(3 * (kEncodedWidth + 1), 12 * (kEncodedHeight + 1));
185 }
186
Chris Dalton50e24d72019-02-07 16:20:09 -0700187 DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg) override {
Matt Sarett34855f92017-01-10 17:41:53 -0500188 if (!canvas->imageInfo().colorSpace()) {
Chris Dalton50e24d72019-02-07 16:20:09 -0700189 *errorMsg = "This gm is only interesting in color correct modes.";
190 return DrawResult::kSkip;
Matt Sarett34855f92017-01-10 17:41:53 -0500191 }
192
193 const SkAlphaType alphaTypes[] = {
194 kUnpremul_SkAlphaType,
195 kPremul_SkAlphaType,
196 };
197 const SkColorType colorTypes[] = {
198 kRGBA_8888_SkColorType,
199 kBGRA_8888_SkColorType,
200 kRGBA_F16_SkColorType,
201 };
202 const sk_sp<SkColorSpace> colorSpaces[] = {
203 make_wide_gamut(),
Matt Sarett77a7a1b2017-02-07 13:56:11 -0500204 SkColorSpace::MakeSRGB(),
Matt Sarett34855f92017-01-10 17:41:53 -0500205 make_small_gamut(),
206 };
207 const SkImage::CachingHint hints[] = {
208 SkImage::kAllow_CachingHint,
209 SkImage::kDisallow_CachingHint,
210 };
211
212 sk_sp<SkImage> image = make_codec_image();
213 for (sk_sp<SkColorSpace> dstColorSpace : colorSpaces) {
214 canvas->save();
215 for (SkColorType dstColorType : colorTypes) {
216 for (SkAlphaType dstAlphaType : alphaTypes) {
217 for (SkImage::CachingHint hint : hints) {
218 draw_image(canvas, image.get(), dstColorType, dstAlphaType, dstColorSpace,
219 hint);
220 canvas->translate(0.0f, (float) kEncodedHeight + 1);
221 }
222 }
223 }
224 canvas->restore();
225 canvas->translate((float) kEncodedWidth + 1, 0.0f);
226 }
Chris Dalton50e24d72019-02-07 16:20:09 -0700227 return DrawResult::kOk;
Matt Sarett34855f92017-01-10 17:41:53 -0500228 }
229
230private:
231 static const int kEncodedWidth = 8;
232 static const int kEncodedHeight = 8;
233
234 typedef skiagm::GM INHERITED;
235};
236DEF_GM( return new ReadPixelsCodecGM; )
237
238class ReadPixelsPictureGM : public skiagm::GM {
239public:
240 ReadPixelsPictureGM() {}
241
242protected:
243 SkString onShortName() override {
244 return SkString("readpixelspicture");
245 }
246
247 SkISize onISize() override {
248 return SkISize::Make(3 * kWidth, 12 * kHeight);
249 }
250
Chris Dalton50e24d72019-02-07 16:20:09 -0700251 DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg) override {
Matt Sarett34855f92017-01-10 17:41:53 -0500252 if (!canvas->imageInfo().colorSpace()) {
Chris Dalton50e24d72019-02-07 16:20:09 -0700253 *errorMsg = "This gm is only interesting in color correct modes.";
254 return DrawResult::kSkip;
Matt Sarett34855f92017-01-10 17:41:53 -0500255 }
256
257 const sk_sp<SkImage> images[] = {
Matt Sarett9df70bb2017-02-14 13:50:43 -0500258 make_picture_image(),
Matt Sarett34855f92017-01-10 17:41:53 -0500259 };
260 const SkAlphaType alphaTypes[] = {
261 kUnpremul_SkAlphaType,
262 kPremul_SkAlphaType,
263 };
264 const SkColorType colorTypes[] = {
265 kRGBA_8888_SkColorType,
266 kBGRA_8888_SkColorType,
267 kRGBA_F16_SkColorType,
268 };
269 const sk_sp<SkColorSpace> colorSpaces[] = {
270 make_wide_gamut(),
Matt Sarett77a7a1b2017-02-07 13:56:11 -0500271 SkColorSpace::MakeSRGB(),
Matt Sarett34855f92017-01-10 17:41:53 -0500272 make_small_gamut(),
273 };
274 const SkImage::CachingHint hints[] = {
275 SkImage::kAllow_CachingHint,
276 SkImage::kDisallow_CachingHint,
277 };
278
279 for (sk_sp<SkImage> image : images) {
280 for (sk_sp<SkColorSpace> dstColorSpace : colorSpaces) {
281 canvas->save();
282 for (SkColorType dstColorType : colorTypes) {
283 for (SkAlphaType dstAlphaType : alphaTypes) {
284 for (SkImage::CachingHint hint : hints) {
285 draw_image(canvas, image.get(), dstColorType, dstAlphaType,
286 dstColorSpace, hint);
287 canvas->translate(0.0f, (float) kHeight);
288 }
289 }
290 }
291 canvas->restore();
292 canvas->translate((float) kWidth, 0.0f);
293 }
294 }
Chris Dalton50e24d72019-02-07 16:20:09 -0700295 return DrawResult::kOk;
Matt Sarett34855f92017-01-10 17:41:53 -0500296 }
297
298private:
299
300 typedef skiagm::GM INHERITED;
301};
302DEF_GM( return new ReadPixelsPictureGM; )