| /* |
| * Copyright 2013 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SkBitmap.h" |
| #include "SkMallocPixelRef.h" |
| #include "Test.h" |
| |
| static void test_peekpixels(skiatest::Reporter* reporter) { |
| const SkImageInfo info = SkImageInfo::MakeN32Premul(10, 10); |
| |
| SkPixmap pmap; |
| SkBitmap bm; |
| |
| // empty should return false |
| REPORTER_ASSERT(reporter, !bm.peekPixels(nullptr)); |
| REPORTER_ASSERT(reporter, !bm.peekPixels(&pmap)); |
| |
| // no pixels should return false |
| bm.setInfo(SkImageInfo::MakeN32Premul(10, 10)); |
| REPORTER_ASSERT(reporter, !bm.peekPixels(nullptr)); |
| REPORTER_ASSERT(reporter, !bm.peekPixels(&pmap)); |
| |
| // real pixels should return true |
| bm.allocPixels(info); |
| REPORTER_ASSERT(reporter, bm.peekPixels(nullptr)); |
| REPORTER_ASSERT(reporter, bm.peekPixels(&pmap)); |
| REPORTER_ASSERT(reporter, pmap.info() == bm.info()); |
| REPORTER_ASSERT(reporter, pmap.addr() == bm.getPixels()); |
| REPORTER_ASSERT(reporter, pmap.rowBytes() == bm.rowBytes()); |
| REPORTER_ASSERT(reporter, pmap.ctable() == bm.getColorTable()); |
| } |
| |
| // https://code.google.com/p/chromium/issues/detail?id=446164 |
| static void test_bigalloc(skiatest::Reporter* reporter) { |
| const int width = 0x40000001; |
| const int height = 0x00000096; |
| const SkImageInfo info = SkImageInfo::MakeN32Premul(width, height); |
| |
| SkBitmap bm; |
| REPORTER_ASSERT(reporter, !bm.tryAllocPixels(info)); |
| |
| SkPixelRef* pr = SkMallocPixelRef::NewAllocate(info, info.minRowBytes(), nullptr); |
| REPORTER_ASSERT(reporter, !pr); |
| } |
| |
| static void test_allocpixels(skiatest::Reporter* reporter) { |
| const int width = 10; |
| const int height = 10; |
| const SkImageInfo info = SkImageInfo::MakeN32Premul(width, height); |
| const size_t explicitRowBytes = info.minRowBytes() + 24; |
| |
| SkBitmap bm; |
| bm.setInfo(info); |
| REPORTER_ASSERT(reporter, info.minRowBytes() == bm.rowBytes()); |
| bm.allocPixels(); |
| REPORTER_ASSERT(reporter, info.minRowBytes() == bm.rowBytes()); |
| bm.reset(); |
| bm.allocPixels(info); |
| REPORTER_ASSERT(reporter, info.minRowBytes() == bm.rowBytes()); |
| |
| bm.setInfo(info, explicitRowBytes); |
| REPORTER_ASSERT(reporter, explicitRowBytes == bm.rowBytes()); |
| bm.allocPixels(); |
| REPORTER_ASSERT(reporter, explicitRowBytes == bm.rowBytes()); |
| bm.reset(); |
| bm.allocPixels(info, explicitRowBytes); |
| REPORTER_ASSERT(reporter, explicitRowBytes == bm.rowBytes()); |
| |
| bm.reset(); |
| bm.setInfo(info, 0); |
| REPORTER_ASSERT(reporter, info.minRowBytes() == bm.rowBytes()); |
| bm.reset(); |
| bm.allocPixels(info, 0); |
| REPORTER_ASSERT(reporter, info.minRowBytes() == bm.rowBytes()); |
| |
| bm.reset(); |
| bool success = bm.setInfo(info, info.minRowBytes() - 1); // invalid for 32bit |
| REPORTER_ASSERT(reporter, !success); |
| REPORTER_ASSERT(reporter, bm.isNull()); |
| } |
| |
| static void test_bigwidth(skiatest::Reporter* reporter) { |
| SkBitmap bm; |
| int width = 1 << 29; // *4 will be the high-bit of 32bit int |
| |
| SkImageInfo info = SkImageInfo::MakeA8(width, 1); |
| REPORTER_ASSERT(reporter, bm.setInfo(info)); |
| REPORTER_ASSERT(reporter, bm.setInfo(info.makeColorType(kRGB_565_SkColorType))); |
| |
| // for a 4-byte config, this width will compute a rowbytes of 0x80000000, |
| // which does not fit in a int32_t. setConfig should detect this, and fail. |
| |
| // TODO: perhaps skia can relax this, and only require that rowBytes fit |
| // in a uint32_t (or larger), but for now this is the constraint. |
| |
| REPORTER_ASSERT(reporter, !bm.setInfo(info.makeColorType(kN32_SkColorType))); |
| } |
| |
| /** |
| * This test contains basic sanity checks concerning bitmaps. |
| */ |
| DEF_TEST(Bitmap, reporter) { |
| // Zero-sized bitmaps are allowed |
| for (int width = 0; width < 2; ++width) { |
| for (int height = 0; height < 2; ++height) { |
| SkBitmap bm; |
| bool setConf = bm.setInfo(SkImageInfo::MakeN32Premul(width, height)); |
| REPORTER_ASSERT(reporter, setConf); |
| if (setConf) { |
| bm.allocPixels(); |
| } |
| REPORTER_ASSERT(reporter, SkToBool(width & height) != bm.empty()); |
| } |
| } |
| |
| test_bigwidth(reporter); |
| test_allocpixels(reporter); |
| test_bigalloc(reporter); |
| test_peekpixels(reporter); |
| } |
| |
| /** |
| * This test checks that getColor works for both swizzles. |
| */ |
| DEF_TEST(Bitmap_getColor_Swizzle, r) { |
| SkBitmap source; |
| source.allocN32Pixels(1,1); |
| source.eraseColor(SK_ColorRED); |
| SkColorType colorTypes[] = { |
| kRGBA_8888_SkColorType, |
| kBGRA_8888_SkColorType, |
| }; |
| for (SkColorType ct : colorTypes) { |
| SkBitmap copy; |
| if (!source.copyTo(©, ct)) { |
| ERRORF(r, "SkBitmap::copy failed %d", (int)ct); |
| continue; |
| } |
| SkAutoLockPixels autoLockPixels1(copy); |
| SkAutoLockPixels autoLockPixels2(source); |
| REPORTER_ASSERT(r, source.getColor(0, 0) == copy.getColor(0, 0)); |
| } |
| } |
| |
| static void test_erasecolor_premul(skiatest::Reporter* reporter, SkColorType ct, SkColor input, |
| SkColor expected) { |
| SkBitmap bm; |
| bm.allocPixels(SkImageInfo::Make(1, 1, ct, kPremul_SkAlphaType)); |
| bm.eraseColor(input); |
| if (reporter->verbose()) { |
| SkDebugf("expected: %x actual: %x\n", expected, bm.getColor(0, 0)); |
| } |
| REPORTER_ASSERT(reporter, bm.getColor(0, 0) == expected); |
| } |
| |
| /** |
| * This test checks that eraseColor premultiplies the color correctly. |
| */ |
| DEF_TEST(Bitmap_eraseColor_Premul, r) { |
| SkColor color = 0x80FF0080; |
| test_erasecolor_premul(r, kAlpha_8_SkColorType, color, 0x80000000); |
| test_erasecolor_premul(r, kRGB_565_SkColorType, color, 0xFF840042); |
| test_erasecolor_premul(r, kARGB_4444_SkColorType, color, 0x88FF0080); |
| test_erasecolor_premul(r, kRGBA_8888_SkColorType, color, color); |
| test_erasecolor_premul(r, kBGRA_8888_SkColorType, color, color); |
| } |