| /* |
| * 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]); |
| } |
| } |