/* * Copyright 2014 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkData.h" #include "SkCanvas.h" #include "SkGraphics.h" #include "SkImageGenerator.h" #include "Test.h" static bool gMyFactoryWasCalled; static std::unique_ptr my_factory(sk_sp) { gMyFactoryWasCalled = true; return nullptr; } static void test_imagegenerator_factory(skiatest::Reporter* reporter) { // just need a non-empty data to test things sk_sp data(SkData::MakeWithCString("test_imagegenerator_factory")); gMyFactoryWasCalled = false; REPORTER_ASSERT(reporter, !gMyFactoryWasCalled); std::unique_ptr gen = SkImageGenerator::MakeFromEncoded(data); REPORTER_ASSERT(reporter, nullptr == gen); REPORTER_ASSERT(reporter, !gMyFactoryWasCalled); // Test is racy, in that it hopes no other thread is changing this global... auto prev = SkGraphics::SetImageGeneratorFromEncodedDataFactory(my_factory); gen = SkImageGenerator::MakeFromEncoded(data); REPORTER_ASSERT(reporter, nullptr == gen); REPORTER_ASSERT(reporter, gMyFactoryWasCalled); SkGraphics::SetImageGeneratorFromEncodedDataFactory(prev); } class MyImageGenerator : public SkImageGenerator { public: MyImageGenerator() : SkImageGenerator(SkImageInfo::MakeN32Premul(0, 0)) {} }; DEF_TEST(ImageGenerator, reporter) { MyImageGenerator ig; SkYUVSizeInfo sizeInfo; sizeInfo.fSizes[SkYUVSizeInfo::kY] = SkISize::Make(200, 200); sizeInfo.fSizes[SkYUVSizeInfo::kU] = SkISize::Make(100, 100); sizeInfo.fSizes[SkYUVSizeInfo::kV] = SkISize::Make( 50, 50); sizeInfo.fWidthBytes[SkYUVSizeInfo::kY] = 0; sizeInfo.fWidthBytes[SkYUVSizeInfo::kU] = 0; sizeInfo.fWidthBytes[SkYUVSizeInfo::kV] = 0; void* planes[3] = { nullptr }; SkYUVColorSpace colorSpace; // Check that the YUV decoding API does not cause any crashes ig.queryYUV8(&sizeInfo, nullptr); ig.queryYUV8(&sizeInfo, &colorSpace); sizeInfo.fWidthBytes[SkYUVSizeInfo::kY] = 250; sizeInfo.fWidthBytes[SkYUVSizeInfo::kU] = 250; sizeInfo.fWidthBytes[SkYUVSizeInfo::kV] = 250; int dummy; planes[SkYUVSizeInfo::kY] = planes[SkYUVSizeInfo::kU] = planes[SkYUVSizeInfo::kV] = &dummy; ig.getYUV8Planes(sizeInfo, planes); // Suppressed due to https://code.google.com/p/skia/issues/detail?id=4339 if (false) { test_imagegenerator_factory(reporter); } } #include "SkAutoMalloc.h" #include "SkPictureRecorder.h" static sk_sp make_picture() { SkPictureRecorder recorder; recorder.beginRecording(100, 100)->drawColor(SK_ColorRED); return recorder.finishRecordingAsPicture(); } DEF_TEST(PictureImageGenerator, reporter) { const struct { SkColorType fColorType; SkAlphaType fAlphaType; bool fExpectSuccess; } recs[] = { { kRGBA_8888_SkColorType, kPremul_SkAlphaType, kRGBA_8888_SkColorType == kN32_SkColorType }, { kBGRA_8888_SkColorType, kPremul_SkAlphaType, kBGRA_8888_SkColorType == kN32_SkColorType }, { kRGBA_F16_SkColorType, kPremul_SkAlphaType, true }, { kRGBA_8888_SkColorType, kUnpremul_SkAlphaType, false }, { kBGRA_8888_SkColorType, kUnpremul_SkAlphaType, false }, { kRGBA_F16_SkColorType, kUnpremul_SkAlphaType, false }, }; auto colorspace = SkColorSpace::MakeSRGB(); auto picture = make_picture(); auto gen = SkImageGenerator::MakeFromPicture({100, 100}, picture, nullptr, nullptr, SkImage::BitDepth::kU8, colorspace); // worst case for all requests SkAutoMalloc storage(100 * 100 * SkColorTypeBytesPerPixel(kRGBA_F16_SkColorType)); for (const auto& rec : recs) { SkImageInfo info = SkImageInfo::Make(100, 100, rec.fColorType, rec.fAlphaType, colorspace); bool success = gen->getPixels(info, storage.get(), info.minRowBytes()); REPORTER_ASSERT(reporter, success == rec.fExpectSuccess); } } #include "SkImagePriv.h" DEF_TEST(ColorXformGenerator, r) { SkBitmap a, b, c, d, e; SkImageInfo info = SkImageInfo::MakeS32(1, 1, kPremul_SkAlphaType); a.allocPixels(info); b.allocPixels(info.makeColorSpace(nullptr)); c.allocPixels(info.makeColorSpace(SkColorSpace::MakeRGB(SkColorSpace::kSRGB_RenderTargetGamma, SkColorSpace::kRec2020_Gamut))); d.allocPixels(info.makeColorSpace(SkColorSpace::MakeRGB(SkColorSpace::kSRGB_RenderTargetGamma, SkColorSpace::kAdobeRGB_Gamut))); e.allocPixels(info); a.eraseColor(0); b.eraseColor(1); c.eraseColor(2); d.eraseColor(3); e.eraseColor(4); sk_sp srgb = SkColorSpace::MakeSRGB(); sk_sp ia = SkMakeImageInColorSpace(a, srgb, 0); sk_sp ib = SkMakeImageInColorSpace(b, srgb, b.getGenerationID()); sk_sp ic = SkMakeImageInColorSpace(c, srgb, c.getGenerationID()); sk_sp id = SkMakeImageInColorSpace(d, srgb, 0); sk_sp ie = SkMakeImageInColorSpace(e, srgb, e.getGenerationID(), kAlways_SkCopyPixelsMode); // Equal because sRGB->sRGB is a no-op. REPORTER_ASSERT(r, ia->uniqueID() == a.getGenerationID()); // Equal because nullptr->sRGB is a no-op (nullptr is treated as sRGB), and because // we pass the explicit id that we want. In the no-op case, the implementation // actually asserts that if we pass an id, it must match the id on the bitmap. REPORTER_ASSERT(r, ib->uniqueID() == b.getGenerationID()); // Equal because we pass in an explicit id. REPORTER_ASSERT(r, ic->uniqueID() == c.getGenerationID()); // Not equal because sRGB->Adobe is not a no-op and we do not pass an explicit id. REPORTER_ASSERT(r, id->uniqueID() != d.getGenerationID()); // Equal because we pass in an explicit id. Forcing a copy, but still want the id respected. REPORTER_ASSERT(r, ie->uniqueID() == e.getGenerationID()); }