blob: 9b65aef3568629760dc306a5261e74631c06a0bb [file] [log] [blame]
/*
* Copyright 2019 Google LLC
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/core/SkAlphaType.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkColor.h"
#include "include/core/SkColorSpace.h"
#include "include/core/SkColorType.h"
#include "include/core/SkImage.h"
#include "include/core/SkImageInfo.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkSurface.h"
#include "include/core/SkTypes.h"
#include "include/gpu/GpuTypes.h"
#include "include/gpu/GrBackendSurface.h"
#include "include/gpu/GrDirectContext.h"
#include "include/gpu/GrTypes.h"
#include "include/gpu/ganesh/SkImageGanesh.h"
#include "include/gpu/ganesh/SkSurfaceGanesh.h"
#include "include/private/base/SkTo.h"
#include "src/core/SkAutoPixmapStorage.h"
#include "src/core/SkImageInfoPriv.h"
#include "src/gpu/ganesh/GrDirectContextPriv.h"
#include "tests/CtsEnforcement.h"
#include "tests/Test.h"
#include "tests/TestUtils.h"
#include "tools/ToolUtils.h"
#include <array>
#include <cstddef>
#include <cstdint>
#include <functional>
#include <initializer_list>
class SkPixmap;
struct GrContextOptions;
static constexpr int kSize = 32;
static SkColor4f get_trans_black_expected_color(SkColorChannelFlag channels) {
float a = 0;
if (!(channels & kAlpha_SkColorChannelFlag)) {
a = 1;
}
return { 0, 0, 0, a };
}
static SkColor4f get_opaque_white_expected_color(SkColorChannelFlag channels) {
if (channels & kGray_SkColorChannelFlag) {
return { 1, 1, 1, 1 };
}
float r = 1, g = 1, b = 1;
if (!(channels & kRed_SkColorChannelFlag)) {
r = 0;
}
if (!(channels & kGreen_SkColorChannelFlag)) {
g = 0;
}
if (!(channels & kBlue_SkColorChannelFlag)) {
b = 0;
}
return { r, g, b, 1.0f };
}
struct TestCase {
SkColorType fColorType;
SkAlphaType fAlphaType;
SkColorChannelFlag fChannels;
bool fGpuCanMakeSurfaces;
};
static const TestCase gTests[] = {
{ kAlpha_8_SkColorType, kPremul_SkAlphaType, kAlpha_SkColorChannelFlag, true },
{ kA16_unorm_SkColorType, kPremul_SkAlphaType, kAlpha_SkColorChannelFlag, false},
{ kA16_float_SkColorType, kPremul_SkAlphaType, kAlpha_SkColorChannelFlag, false},
{ kRGB_565_SkColorType, kOpaque_SkAlphaType, kRGB_SkColorChannelFlags, true },
{ kARGB_4444_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, true },
{ kRGBA_8888_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, true },
{ kRGB_888x_SkColorType, kOpaque_SkAlphaType, kRGB_SkColorChannelFlags, true },
{ kBGRA_8888_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, true },
{ kRGBA_1010102_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, true },
{ kRGB_101010x_SkColorType, kOpaque_SkAlphaType, kRGB_SkColorChannelFlags, true },
{ kGray_8_SkColorType, kOpaque_SkAlphaType, kGray_SkColorChannelFlag, true },
{ kRGBA_F16Norm_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, true },
{ kRGBA_F16_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, true },
{ kRGBA_F32_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, true },
{ kR8G8_unorm_SkColorType, kOpaque_SkAlphaType, kRG_SkColorChannelFlags, true },
{ kR16G16_unorm_SkColorType, kOpaque_SkAlphaType, kRG_SkColorChannelFlags, false},
{ kR16G16_float_SkColorType, kOpaque_SkAlphaType, kRG_SkColorChannelFlags, false},
{ kR16G16B16A16_unorm_SkColorType, kPremul_SkAlphaType, kRGBA_SkColorChannelFlags, false},
};
static void raster_tests(skiatest::Reporter* reporter, const TestCase& test) {
const SkImageInfo nativeII = SkImageInfo::Make(kSize, kSize, test.fColorType, test.fAlphaType);
const SkImageInfo f32Unpremul = SkImageInfo::Make(kSize, kSize, kRGBA_F32_SkColorType,
kUnpremul_SkAlphaType);
uint32_t actualChannels = SkColorTypeChannelFlags(test.fColorType);
REPORTER_ASSERT(reporter, test.fChannels == actualChannels);
// all colorTypes can be drawn to
{
auto s = SkSurfaces::Raster(nativeII);
REPORTER_ASSERT(reporter, SkToBool(s));
}
// opaque formats should make transparent black become opaque
{
SkAutoPixmapStorage pm;
pm.alloc(nativeII);
pm.erase(SkColors::kTransparent);
SkColor actual = pm.getColor(0, 0);
SkColor4f expected = get_trans_black_expected_color(test.fChannels);
REPORTER_ASSERT(reporter, expected.toSkColor() == actual);
}
// unused channels should drop out
{
SkAutoPixmapStorage pm;
pm.alloc(nativeII);
pm.erase(SkColors::kWhite);
SkColor actual = pm.getColor(0, 0);
SkColor4f expected = get_opaque_white_expected_color(test.fChannels);
REPORTER_ASSERT(reporter, expected.toSkColor() == actual);
}
// Reading back from an image to the same colorType should always work
{
SkAutoPixmapStorage srcPM;
srcPM.alloc(nativeII);
srcPM.erase(SkColors::kWhite);
auto i = SkImages::RasterFromPixmap(srcPM, nullptr, nullptr);
REPORTER_ASSERT(reporter, SkToBool(i));
SkAutoPixmapStorage readbackPM;
readbackPM.alloc(nativeII);
readbackPM.erase(SkColors::kTransparent);
REPORTER_ASSERT(reporter, i->readPixels(nullptr, readbackPM, 0, 0));
SkColor expected = srcPM.getColor(0, 0);
SkColor actual = readbackPM.getColor(0, 0);
REPORTER_ASSERT(reporter, expected == actual);
}
// Rendering to an F32 surface should always work
{
SkAutoPixmapStorage srcPM;
srcPM.alloc(nativeII);
srcPM.erase(SkColors::kWhite);
auto i = SkImages::RasterFromPixmap(srcPM, nullptr, nullptr);
REPORTER_ASSERT(reporter, SkToBool(i));
auto s = SkSurfaces::Raster(f32Unpremul);
REPORTER_ASSERT(reporter, SkToBool(s));
{
auto c = s->getCanvas();
c->drawImage(i, 0, 0);
}
SkAutoPixmapStorage readbackPM;
readbackPM.alloc(f32Unpremul);
readbackPM.erase(SkColors::kTransparent);
REPORTER_ASSERT(reporter, i->readPixels(nullptr, readbackPM, 0, 0));
SkColor expected = srcPM.getColor(0, 0);
SkColor actual = readbackPM.getColor(0, 0);
REPORTER_ASSERT(reporter, expected == actual);
}
}
static void compare_pixmaps(skiatest::Reporter* reporter,
const SkPixmap& expected, const SkPixmap& actual,
SkColorType ct, const char* label) {
const float tols[4] = {0.0f, 0.0f, 0.0f, 0};
auto error = std::function<ComparePixmapsErrorReporter>(
[reporter, ct, label](int x, int y, const float diffs[4]) {
SkASSERT(x >= 0 && y >= 0);
ERRORF(reporter, "%s %s - mismatch at %d, %d (%f, %f, %f %f)",
ToolUtils::colortype_name(ct), label, x, y,
diffs[0], diffs[1], diffs[2], diffs[3]);
});
ComparePixels(expected, actual, tols, error);
}
static void gpu_tests(GrDirectContext* dContext,
skiatest::Reporter* reporter,
const TestCase& test) {
using namespace skgpu;
const SkImageInfo nativeII = SkImageInfo::Make(kSize, kSize, test.fColorType, test.fAlphaType);
const SkImageInfo f32Unpremul = SkImageInfo::Make(kSize, kSize, kRGBA_F32_SkColorType,
kUnpremul_SkAlphaType);
// We had better not be able to render to prohibited colorTypes
if (!test.fGpuCanMakeSurfaces) {
auto s = SkSurfaces::RenderTarget(dContext, Budgeted::kNo, nativeII);
REPORTER_ASSERT(reporter, !SkToBool(s));
}
if (!dContext->colorTypeSupportedAsImage(test.fColorType)) {
return;
}
Protected isProtected = Protected(dContext->priv().caps()->supportsProtectedContent());
SkAutoPixmapStorage nativeExpected;
nativeExpected.alloc(nativeII);
nativeExpected.erase(SkColors::kWhite);
for (bool fullInit : { false, true }) {
GrBackendTexture backendTex;
bool finishedBECreate = false;
auto markFinished = [](void* context) {
*(bool*)context = true;
};
if (fullInit) {
backendTex = dContext->createBackendTexture(nativeExpected, kTopLeft_GrSurfaceOrigin,
GrRenderable::kNo, isProtected,
markFinished, &finishedBECreate);
} else {
backendTex = dContext->createBackendTexture(kSize,
kSize,
test.fColorType,
SkColors::kWhite,
Mipmapped::kNo,
GrRenderable::kNo,
isProtected,
markFinished,
&finishedBECreate);
}
REPORTER_ASSERT(reporter, backendTex.isValid());
dContext->submit();
while (backendTex.isValid() && !finishedBECreate) {
dContext->checkAsyncWorkCompletion();
}
auto img = SkImages::BorrowTextureFrom(dContext,
backendTex,
kTopLeft_GrSurfaceOrigin,
test.fColorType,
test.fAlphaType,
nullptr);
REPORTER_ASSERT(reporter, SkToBool(img));
{
SkAutoPixmapStorage nativeActual;
nativeActual.alloc(nativeII);
nativeActual.erase(SkColors::kTransparent);
if (img->readPixels(dContext, nativeActual, 0, 0)) {
compare_pixmaps(reporter, nativeExpected, nativeActual,
test.fColorType, "SkImage::readPixels to native CT");
}
// SkSurface::readPixels with the same colorType as the source pixels round trips
// (when allowed)
if (dContext->colorTypeSupportedAsSurface(test.fColorType)) {
auto s = SkSurfaces::RenderTarget(dContext, Budgeted::kNo, nativeII);
REPORTER_ASSERT(reporter, SkToBool(s));
{
SkCanvas* c = s->getCanvas();
c->drawImage(img, 0, 0);
}
nativeActual.erase(SkColors::kTransparent);
REPORTER_ASSERT(reporter, s->readPixels(nativeActual, 0, 0));
compare_pixmaps(reporter, nativeExpected, nativeActual,
test.fColorType, "SkSurface::readPixels to native CT");
}
}
{
SkAutoPixmapStorage f32Expected;
f32Expected.alloc(f32Unpremul);
f32Expected.erase(get_opaque_white_expected_color(test.fChannels));
// read back to F32 if possible
{
SkAutoPixmapStorage f32Actual;
f32Actual.alloc(f32Unpremul);
f32Actual.erase(SkColors::kTransparent);
if (img->readPixels(dContext, f32Actual, 0, 0)) {
compare_pixmaps(reporter, f32Expected, f32Actual,
test.fColorType, "SkImage::readPixels to F32");
}
}
// drawing a native SkImage works appropriately (as assessed by reading back from an
// RGBA8 surface to an F32 pixmap)
{
const SkImageInfo rgba8888Premul = SkImageInfo::Make(kSize, kSize,
kRGBA_8888_SkColorType,
kPremul_SkAlphaType);
auto s = SkSurfaces::RenderTarget(dContext, Budgeted::kNo, rgba8888Premul);
REPORTER_ASSERT(reporter, SkToBool(s));
{
SkCanvas* c = s->getCanvas();
c->drawImage(img, 0, 0);
}
SkAutoPixmapStorage f32Actual;
f32Actual.alloc(f32Unpremul);
f32Actual.erase(SkColors::kTransparent);
REPORTER_ASSERT(reporter, s->readPixels(f32Actual, 0, 0));
compare_pixmaps(reporter, f32Expected, f32Actual,
test.fColorType, "SkSurface::drawn to RGBA8888");
}
}
img.reset();
dContext->flushAndSubmit();
dContext->deleteBackendTexture(backendTex);
}
}
DEF_TEST(ExtendedSkColorTypeTests_raster, reporter) {
for (size_t i = 0; i < std::size(gTests); ++i) {
raster_tests(reporter, gTests[i]);
}
}
DEF_GANESH_TEST_FOR_RENDERING_CONTEXTS(ExtendedSkColorTypeTests_gpu,
reporter,
ctxInfo,
CtsEnforcement::kApiLevel_T) {
auto context = ctxInfo.directContext();
for (size_t i = 0; i < std::size(gTests); ++i) {
gpu_tests(context, reporter, gTests[i]);
}
}