2017-05-04 12:53:32 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2017 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "gm/gm.h"
|
|
|
|
#include "include/codec/SkCodec.h"
|
2019-05-01 21:28:53 +00:00
|
|
|
#include "include/core/SkBitmap.h"
|
|
|
|
#include "include/core/SkCanvas.h"
|
|
|
|
#include "include/core/SkColor.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/core/SkColorSpace.h"
|
2019-05-01 21:28:53 +00:00
|
|
|
#include "include/core/SkData.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/core/SkImage.h"
|
2019-05-01 21:28:53 +00:00
|
|
|
#include "include/core/SkImageInfo.h"
|
|
|
|
#include "include/core/SkRefCnt.h"
|
|
|
|
#include "include/core/SkSize.h"
|
|
|
|
#include "include/core/SkString.h"
|
2020-07-15 14:37:50 +00:00
|
|
|
#include "include/gpu/GrDirectContext.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "src/core/SkImagePriv.h"
|
|
|
|
#include "tools/Resources.h"
|
2017-05-04 12:53:32 +00:00
|
|
|
|
2019-05-01 21:28:53 +00:00
|
|
|
#include <initializer_list>
|
|
|
|
#include <memory>
|
|
|
|
|
2018-07-12 18:44:27 +00:00
|
|
|
sk_sp<SkImage> make_raster_image(const char* path) {
|
2017-12-09 01:27:41 +00:00
|
|
|
sk_sp<SkData> resourceData = GetResourceAsData(path);
|
2017-07-23 19:30:02 +00:00
|
|
|
std::unique_ptr<SkCodec> codec = SkCodec::MakeFromData(resourceData);
|
2017-05-04 12:53:32 +00:00
|
|
|
|
|
|
|
SkBitmap bitmap;
|
|
|
|
bitmap.allocPixels(codec->getInfo());
|
|
|
|
|
2018-07-12 18:44:27 +00:00
|
|
|
codec->getPixels(codec->getInfo(), bitmap.getPixels(), bitmap.rowBytes());
|
2020-12-23 15:11:33 +00:00
|
|
|
return bitmap.asImage();
|
2017-05-04 12:53:32 +00:00
|
|
|
}
|
|
|
|
|
2020-07-15 14:37:50 +00:00
|
|
|
sk_sp<SkImage> make_color_space(
|
|
|
|
sk_sp<SkImage> orig, sk_sp<SkColorSpace> colorSpace, GrDirectContext* direct) {
|
|
|
|
sk_sp<SkImage> xform = orig->makeColorSpace(colorSpace, direct);
|
2017-05-04 12:53:32 +00:00
|
|
|
|
|
|
|
// Assign an sRGB color space on the xformed image, so we can see the effects of the xform
|
|
|
|
// when we draw.
|
|
|
|
sk_sp<SkColorSpace> srgb = SkColorSpace::MakeSRGB();
|
|
|
|
if (colorSpace->gammaIsLinear()) {
|
|
|
|
srgb = SkColorSpace::MakeSRGBLinear();
|
|
|
|
}
|
2019-08-14 20:14:51 +00:00
|
|
|
return xform->reinterpretColorSpace(std::move(srgb));
|
2017-05-04 12:53:32 +00:00
|
|
|
}
|
|
|
|
|
2019-08-14 20:14:51 +00:00
|
|
|
DEF_SIMPLE_GM_CAN_FAIL(makecolorspace, canvas, errorMsg, 128 * 3, 128 * 4) {
|
|
|
|
sk_sp<SkColorSpace> wideGamut = SkColorSpace::MakeRGB(SkNamedTransferFn::kSRGB,
|
|
|
|
SkNamedGamut::kAdobeRGB);
|
|
|
|
sk_sp<SkColorSpace> wideGamutLinear = wideGamut->makeLinearGamma();
|
|
|
|
|
|
|
|
// Lazy images
|
|
|
|
sk_sp<SkImage> opaqueImage = GetResourceAsImage("images/mandrill_128.png");
|
|
|
|
sk_sp<SkImage> premulImage = GetResourceAsImage("images/color_wheel.png");
|
|
|
|
if (!opaqueImage || !premulImage) {
|
|
|
|
*errorMsg = "Failed to load images. Did you forget to set the resourcePath?";
|
|
|
|
return skiagm::DrawResult::kFail;
|
2017-05-04 12:53:32 +00:00
|
|
|
}
|
2020-07-15 14:37:50 +00:00
|
|
|
auto direct = GrAsDirectContext(canvas->recordingContext());
|
2019-08-14 20:14:51 +00:00
|
|
|
canvas->drawImage(opaqueImage, 0.0f, 0.0f);
|
2020-07-15 14:37:50 +00:00
|
|
|
canvas->drawImage(make_color_space(opaqueImage, wideGamut, direct), 128.0f, 0.0f);
|
|
|
|
canvas->drawImage(make_color_space(opaqueImage, wideGamutLinear, direct), 256.0f, 0.0f);
|
2019-08-14 20:14:51 +00:00
|
|
|
canvas->drawImage(premulImage, 0.0f, 128.0f);
|
2020-07-15 14:37:50 +00:00
|
|
|
canvas->drawImage(make_color_space(premulImage, wideGamut, direct), 128.0f, 128.0f);
|
|
|
|
canvas->drawImage(make_color_space(premulImage, wideGamutLinear, direct), 256.0f, 128.0f);
|
2019-08-14 20:14:51 +00:00
|
|
|
canvas->translate(0.0f, 256.0f);
|
|
|
|
|
|
|
|
// Raster images
|
|
|
|
opaqueImage = make_raster_image("images/mandrill_128.png");
|
|
|
|
premulImage = make_raster_image("images/color_wheel.png");
|
|
|
|
canvas->drawImage(opaqueImage, 0.0f, 0.0f);
|
2020-07-15 14:37:50 +00:00
|
|
|
canvas->drawImage(make_color_space(opaqueImage, wideGamut, direct), 128.0f, 0.0f);
|
|
|
|
canvas->drawImage(make_color_space(opaqueImage, wideGamutLinear, direct), 256.0f, 0.0f);
|
2019-08-14 20:14:51 +00:00
|
|
|
canvas->drawImage(premulImage, 0.0f, 128.0f);
|
2020-07-15 14:37:50 +00:00
|
|
|
canvas->drawImage(make_color_space(premulImage, wideGamut, direct), 128.0f, 128.0f);
|
|
|
|
canvas->drawImage(make_color_space(premulImage, wideGamutLinear, direct), 256.0f, 128.0f);
|
2019-08-14 20:14:51 +00:00
|
|
|
return skiagm::DrawResult::kOk;
|
|
|
|
}
|
2019-01-14 16:15:50 +00:00
|
|
|
|
|
|
|
DEF_SIMPLE_GM_BG(makecolortypeandspace, canvas, 128 * 3, 128 * 4, SK_ColorWHITE) {
|
|
|
|
sk_sp<SkImage> images[] = {
|
|
|
|
GetResourceAsImage("images/mandrill_128.png"),
|
|
|
|
GetResourceAsImage("images/color_wheel.png"),
|
|
|
|
};
|
|
|
|
auto rec2020 = SkColorSpace::MakeRGB(SkNamedTransferFn::kSRGB, SkNamedGamut::kRec2020);
|
|
|
|
|
|
|
|
// Use the lazy images on the first iteration, and concrete (raster/GPU) images on the second
|
2020-07-15 14:37:50 +00:00
|
|
|
auto direct = GrAsDirectContext(canvas->recordingContext());
|
2019-01-14 16:15:50 +00:00
|
|
|
for (bool lazy : {true, false}) {
|
|
|
|
for (int j = 0; j < 2; ++j) {
|
|
|
|
const SkImage* image = images[j].get();
|
|
|
|
if (!image) {
|
|
|
|
// Can happen on bots that abandon the GPU context
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unmodified
|
|
|
|
canvas->drawImage(image, 0, 0);
|
|
|
|
|
|
|
|
// Change the color type/space of the image in a couple ways. In both cases, codec
|
|
|
|
// may fail, because we refude to decode transparent sources to opaque color types.
|
|
|
|
// Guard against that, to avoid cascading failures in DDL.
|
|
|
|
|
|
|
|
// 565 in a wide color space (should be visibly quantized). Fails with the color_wheel,
|
|
|
|
// because of the codec issues mentioned above.
|
2020-07-15 14:37:50 +00:00
|
|
|
auto image565 = image->makeColorTypeAndColorSpace(kRGB_565_SkColorType,
|
|
|
|
rec2020, direct);
|
2019-01-14 16:15:50 +00:00
|
|
|
if (!lazy || image565->makeRasterImage()) {
|
|
|
|
canvas->drawImage(image565, 128, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Grayscale in the original color space. This fails in even more cases, due to the
|
|
|
|
// above opaque issue, and because Ganesh doesn't support drawing to gray, at all.
|
|
|
|
auto imageGray = image->makeColorTypeAndColorSpace(kGray_8_SkColorType,
|
2020-07-15 14:37:50 +00:00
|
|
|
image->refColorSpace(),
|
|
|
|
direct);
|
2019-01-14 16:15:50 +00:00
|
|
|
if (!lazy || imageGray->makeRasterImage()) {
|
|
|
|
canvas->drawImage(imageGray, 256, 0);
|
|
|
|
}
|
|
|
|
|
2020-07-15 14:37:50 +00:00
|
|
|
images[j] = direct ? image->makeTextureImage(direct) : image->makeRasterImage();
|
2019-01-14 16:15:50 +00:00
|
|
|
|
|
|
|
canvas->translate(0, 128);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-08-14 20:14:51 +00:00
|
|
|
|
|
|
|
DEF_SIMPLE_GM_CAN_FAIL(reinterpretcolorspace, canvas, errorMsg, 128 * 3, 128 * 3) {
|
|
|
|
// We draw a 3x3 grid. The three rows are lazy (encoded), raster, and GPU (or another copy of
|
|
|
|
// raster so all configs look similar). In each row, we draw three variants:
|
|
|
|
// - The original image (should look normal).
|
|
|
|
// - The image, reinterpreted as being in the color-spin space. The pixel data isn't changed,
|
|
|
|
// so in untagged configs, this looks like the first column. In tagged configs, this has the
|
|
|
|
// the effect of rotating the colors (RGB -> GBR).
|
|
|
|
// - The image converted to the color-spin space, then reinterpreted as being sRGB. In all
|
|
|
|
// configs, this appears to be spun backwards (RGB -> BRG), and tests the composition of
|
|
|
|
// these two APIs.
|
|
|
|
|
|
|
|
// In all cases, every column should be identical. In tagged configs, the 'R' in the columns
|
|
|
|
// should be Red, Green, Blue.
|
|
|
|
|
|
|
|
sk_sp<SkColorSpace> srgb = SkColorSpace::MakeSRGB();
|
|
|
|
sk_sp<SkColorSpace> spin = srgb->makeColorSpin();
|
|
|
|
sk_sp<SkImage> image = GetResourceAsImage("images/color_wheel.png");
|
|
|
|
if (!image) {
|
|
|
|
*errorMsg = "Failed to load image. Did you forget to set the resourcePath?";
|
|
|
|
return skiagm::DrawResult::kFail;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lazy images
|
|
|
|
canvas->drawImage(image, 0.0f, 0.0f);
|
|
|
|
canvas->drawImage(image->reinterpretColorSpace(spin), 128.0f, 0.0f);
|
2020-07-15 14:37:50 +00:00
|
|
|
canvas->drawImage(image->makeColorSpace(spin, nullptr)->reinterpretColorSpace(srgb),
|
|
|
|
256.0f, 0.0f);
|
2019-08-14 20:14:51 +00:00
|
|
|
|
|
|
|
canvas->translate(0.0f, 128.0f);
|
|
|
|
|
|
|
|
// Raster images
|
|
|
|
image = image->makeRasterImage();
|
|
|
|
canvas->drawImage(image, 0.0f, 0.0f);
|
|
|
|
canvas->drawImage(image->reinterpretColorSpace(spin), 128.0f, 0.0f);
|
2020-07-15 14:37:50 +00:00
|
|
|
canvas->drawImage(image->makeColorSpace(spin, nullptr)->reinterpretColorSpace(srgb),
|
|
|
|
256.0f, 0.0f);
|
2019-08-14 20:14:51 +00:00
|
|
|
|
|
|
|
canvas->translate(0.0f, 128.0f);
|
|
|
|
|
|
|
|
// GPU images
|
2020-07-15 14:37:50 +00:00
|
|
|
auto direct = GrAsDirectContext(canvas->recordingContext());
|
|
|
|
if (auto gpuImage = image->makeTextureImage(direct)) {
|
2019-08-15 14:38:26 +00:00
|
|
|
image = gpuImage;
|
2019-08-14 20:14:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
canvas->drawImage(image, 0.0f, 0.0f);
|
|
|
|
canvas->drawImage(image->reinterpretColorSpace(spin), 128.0f, 0.0f);
|
2020-07-15 14:37:50 +00:00
|
|
|
canvas->drawImage(image->makeColorSpace(spin, direct)->reinterpretColorSpace(srgb),
|
|
|
|
256.0f, 0.0f);
|
2019-08-14 20:14:51 +00:00
|
|
|
|
|
|
|
return skiagm::DrawResult::kOk;
|
|
|
|
}
|