2014-01-10 18:28:23 +00:00
|
|
|
|
2011-07-28 14:26:00 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2011 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
2009-05-01 04:00:01 +00:00
|
|
|
#include "Test.h"
|
|
|
|
#include "SkBitmap.h"
|
2009-05-05 23:13:23 +00:00
|
|
|
#include "SkRect.h"
|
2009-05-01 04:00:01 +00:00
|
|
|
|
|
|
|
static const char* boolStr(bool value) {
|
|
|
|
return value ? "true" : "false";
|
|
|
|
}
|
|
|
|
|
|
|
|
// these are in the same order as the SkBitmap::Config enum
|
|
|
|
static const char* gConfigName[] = {
|
2013-12-02 13:50:38 +00:00
|
|
|
"None", "A8", "Index8", "565", "4444", "8888"
|
2009-05-01 04:00:01 +00:00
|
|
|
};
|
|
|
|
|
2009-08-22 03:44:57 +00:00
|
|
|
static void report_opaqueness(skiatest::Reporter* reporter, const SkBitmap& src,
|
|
|
|
const SkBitmap& dst) {
|
2014-01-10 14:58:10 +00:00
|
|
|
ERRORF(reporter, "src %s opaque:%d, dst %s opaque:%d",
|
|
|
|
gConfigName[src.config()], src.isOpaque(),
|
|
|
|
gConfigName[dst.config()], dst.isOpaque());
|
2009-08-22 03:44:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool canHaveAlpha(SkBitmap::Config config) {
|
|
|
|
return config != SkBitmap::kRGB_565_Config;
|
|
|
|
}
|
|
|
|
|
|
|
|
// copyTo() should preserve isOpaque when it makes sense
|
2013-10-10 14:44:56 +00:00
|
|
|
static void test_isOpaque(skiatest::Reporter* reporter,
|
|
|
|
const SkBitmap& srcOpaque, const SkBitmap& srcPremul,
|
2009-08-22 03:44:57 +00:00
|
|
|
SkBitmap::Config dstConfig) {
|
|
|
|
SkBitmap dst;
|
|
|
|
|
2013-10-10 14:44:56 +00:00
|
|
|
if (canHaveAlpha(srcPremul.config()) && canHaveAlpha(dstConfig)) {
|
|
|
|
REPORTER_ASSERT(reporter, srcPremul.copyTo(&dst, dstConfig));
|
2009-08-22 03:44:57 +00:00
|
|
|
REPORTER_ASSERT(reporter, dst.config() == dstConfig);
|
2013-10-10 14:44:56 +00:00
|
|
|
if (srcPremul.isOpaque() != dst.isOpaque()) {
|
|
|
|
report_opaqueness(reporter, srcPremul, dst);
|
2009-08-22 03:44:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-10 14:44:56 +00:00
|
|
|
REPORTER_ASSERT(reporter, srcOpaque.copyTo(&dst, dstConfig));
|
2009-08-22 03:44:57 +00:00
|
|
|
REPORTER_ASSERT(reporter, dst.config() == dstConfig);
|
2013-10-10 14:44:56 +00:00
|
|
|
if (srcOpaque.isOpaque() != dst.isOpaque()) {
|
|
|
|
report_opaqueness(reporter, srcOpaque, dst);
|
2009-08-22 03:44:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-10 14:44:56 +00:00
|
|
|
static void init_src(const SkBitmap& bitmap) {
|
2009-05-03 18:23:30 +00:00
|
|
|
SkAutoLockPixels lock(bitmap);
|
2009-05-01 04:00:01 +00:00
|
|
|
if (bitmap.getPixels()) {
|
2013-10-10 14:44:56 +00:00
|
|
|
if (bitmap.getColorTable()) {
|
2011-01-10 14:04:07 +00:00
|
|
|
sk_bzero(bitmap.getPixels(), bitmap.getSize());
|
|
|
|
} else {
|
|
|
|
bitmap.eraseColor(SK_ColorWHITE);
|
|
|
|
}
|
2009-05-01 04:00:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-10 14:44:56 +00:00
|
|
|
static SkColorTable* init_ctable(SkAlphaType alphaType) {
|
2009-05-01 04:00:01 +00:00
|
|
|
static const SkColor colors[] = {
|
|
|
|
SK_ColorBLACK, SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE
|
|
|
|
};
|
2013-10-10 14:44:56 +00:00
|
|
|
return new SkColorTable(colors, SK_ARRAY_COUNT(colors), alphaType);
|
2009-05-01 04:00:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct Pair {
|
|
|
|
SkBitmap::Config fConfig;
|
|
|
|
const char* fValid;
|
|
|
|
};
|
|
|
|
|
2010-11-16 20:22:41 +00:00
|
|
|
// Utility functions for copyPixelsTo()/copyPixelsFrom() tests.
|
|
|
|
// getPixel()
|
|
|
|
// setPixel()
|
|
|
|
// getSkConfigName()
|
|
|
|
// struct Coordinates
|
|
|
|
// reportCopyVerification()
|
|
|
|
// writeCoordPixels()
|
|
|
|
|
|
|
|
// Utility function to read the value of a given pixel in bm. All
|
|
|
|
// values converted to uint32_t for simplification of comparisons.
|
2012-06-06 12:03:39 +00:00
|
|
|
static uint32_t getPixel(int x, int y, const SkBitmap& bm) {
|
2010-11-16 20:22:41 +00:00
|
|
|
uint32_t val = 0;
|
|
|
|
uint16_t val16;
|
2013-12-02 13:50:38 +00:00
|
|
|
uint8_t val8;
|
2010-11-16 20:22:41 +00:00
|
|
|
SkAutoLockPixels lock(bm);
|
|
|
|
const void* rawAddr = bm.getAddr(x,y);
|
|
|
|
|
2013-10-31 17:28:30 +00:00
|
|
|
switch (bm.config()) {
|
2010-11-16 20:22:41 +00:00
|
|
|
case SkBitmap::kARGB_8888_Config:
|
|
|
|
memcpy(&val, rawAddr, sizeof(uint32_t));
|
|
|
|
break;
|
|
|
|
case SkBitmap::kARGB_4444_Config:
|
|
|
|
case SkBitmap::kRGB_565_Config:
|
|
|
|
memcpy(&val16, rawAddr, sizeof(uint16_t));
|
|
|
|
val = val16;
|
|
|
|
break;
|
|
|
|
case SkBitmap::kA8_Config:
|
|
|
|
case SkBitmap::kIndex8_Config:
|
|
|
|
memcpy(&val8, rawAddr, sizeof(uint8_t));
|
|
|
|
val = val8;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Utility function to set value of any pixel in bm.
|
|
|
|
// bm.getConfig() specifies what format 'val' must be
|
|
|
|
// converted to, but at present uint32_t can handle all formats.
|
2012-06-06 12:03:39 +00:00
|
|
|
static void setPixel(int x, int y, uint32_t val, SkBitmap& bm) {
|
2010-11-16 20:22:41 +00:00
|
|
|
uint16_t val16;
|
2013-12-02 13:50:38 +00:00
|
|
|
uint8_t val8;
|
2010-11-16 20:22:41 +00:00
|
|
|
SkAutoLockPixels lock(bm);
|
|
|
|
void* rawAddr = bm.getAddr(x,y);
|
|
|
|
|
2013-10-31 17:28:30 +00:00
|
|
|
switch (bm.config()) {
|
2010-11-16 20:22:41 +00:00
|
|
|
case SkBitmap::kARGB_8888_Config:
|
|
|
|
memcpy(rawAddr, &val, sizeof(uint32_t));
|
|
|
|
break;
|
|
|
|
case SkBitmap::kARGB_4444_Config:
|
|
|
|
case SkBitmap::kRGB_565_Config:
|
|
|
|
val16 = val & 0xFFFF;
|
|
|
|
memcpy(rawAddr, &val16, sizeof(uint16_t));
|
|
|
|
break;
|
|
|
|
case SkBitmap::kA8_Config:
|
|
|
|
case SkBitmap::kIndex8_Config:
|
|
|
|
val8 = val & 0xFF;
|
|
|
|
memcpy(rawAddr, &val8, sizeof(uint8_t));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// Ignore.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Utility to return string containing name of each format, to
|
|
|
|
// simplify diagnostic output.
|
2012-06-06 12:03:39 +00:00
|
|
|
static const char* getSkConfigName(const SkBitmap& bm) {
|
2013-10-31 17:28:30 +00:00
|
|
|
switch (bm.config()) {
|
2010-11-16 20:22:41 +00:00
|
|
|
case SkBitmap::kNo_Config: return "SkBitmap::kNo_Config";
|
|
|
|
case SkBitmap::kA8_Config: return "SkBitmap::kA8_Config";
|
|
|
|
case SkBitmap::kIndex8_Config: return "SkBitmap::kIndex8_Config";
|
|
|
|
case SkBitmap::kRGB_565_Config: return "SkBitmap::kRGB_565_Config";
|
|
|
|
case SkBitmap::kARGB_4444_Config: return "SkBitmap::kARGB_4444_Config";
|
|
|
|
case SkBitmap::kARGB_8888_Config: return "SkBitmap::kARGB_8888_Config";
|
|
|
|
default: return "Unknown SkBitmap configuration.";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper struct to contain pixel locations, while avoiding need for STL.
|
|
|
|
struct Coordinates {
|
|
|
|
|
|
|
|
const int length;
|
|
|
|
SkIPoint* const data;
|
|
|
|
|
|
|
|
explicit Coordinates(int _length): length(_length)
|
|
|
|
, data(new SkIPoint[length]) { }
|
|
|
|
|
|
|
|
~Coordinates(){
|
|
|
|
delete [] data;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkIPoint* operator[](int i) const {
|
|
|
|
// Use with care, no bounds checking.
|
|
|
|
return data + i;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// A function to verify that two bitmaps contain the same pixel values
|
|
|
|
// at all coordinates indicated by coords. Simplifies verification of
|
|
|
|
// copied bitmaps.
|
2012-06-06 12:03:39 +00:00
|
|
|
static void reportCopyVerification(const SkBitmap& bm1, const SkBitmap& bm2,
|
2010-11-16 20:22:41 +00:00
|
|
|
Coordinates& coords,
|
|
|
|
const char* msg,
|
|
|
|
skiatest::Reporter* reporter){
|
|
|
|
bool success = true;
|
|
|
|
|
|
|
|
// Confirm all pixels in the list match.
|
2012-08-22 15:00:05 +00:00
|
|
|
for (int i = 0; i < coords.length; ++i) {
|
2010-11-16 20:22:41 +00:00
|
|
|
success = success &&
|
|
|
|
(getPixel(coords[i]->fX, coords[i]->fY, bm1) ==
|
|
|
|
getPixel(coords[i]->fX, coords[i]->fY, bm2));
|
2012-08-22 15:00:05 +00:00
|
|
|
}
|
2010-11-16 20:22:41 +00:00
|
|
|
|
|
|
|
if (!success) {
|
2014-01-10 14:58:10 +00:00
|
|
|
ERRORF(reporter, "%s [config = %s]", msg, getSkConfigName(bm1));
|
2010-11-16 20:22:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Writes unique pixel values at locations specified by coords.
|
2012-06-06 12:03:39 +00:00
|
|
|
static void writeCoordPixels(SkBitmap& bm, const Coordinates& coords) {
|
2010-11-16 20:22:41 +00:00
|
|
|
for (int i = 0; i < coords.length; ++i)
|
|
|
|
setPixel(coords[i]->fX, coords[i]->fY, i, bm);
|
|
|
|
}
|
|
|
|
|
2014-01-10 18:28:23 +00:00
|
|
|
DEF_TEST(BitmapCopy, reporter) {
|
|
|
|
static const Pair gPairs[] = {
|
|
|
|
{ SkBitmap::kNo_Config, "0000000" },
|
|
|
|
{ SkBitmap::kA8_Config, "0101010" },
|
|
|
|
{ SkBitmap::kIndex8_Config, "0111010" },
|
|
|
|
{ SkBitmap::kRGB_565_Config, "0101010" },
|
|
|
|
{ SkBitmap::kARGB_4444_Config, "0101110" },
|
|
|
|
{ SkBitmap::kARGB_8888_Config, "0101110" },
|
|
|
|
};
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2014-01-10 18:28:23 +00:00
|
|
|
static const bool isExtracted[] = {
|
|
|
|
false, true
|
|
|
|
};
|
2009-05-03 18:23:30 +00:00
|
|
|
|
2014-01-10 18:28:23 +00:00
|
|
|
const int W = 20;
|
|
|
|
const int H = 33;
|
Fix genID cloning bugs.
SkBitmap.cpp:
When copyTo calls readPixels, only clone the genID if the resulting
SkPixelRef has the same dimensions as the original. This catches a
bug where copying an SkBitmap representing the subset of an SkPixelRef
(which implements onReadPixels) would result in the copy sharing the
genID. (Thanks to r6710, this case can only happen using setPixelRef,
so the updated GpuBitmapCopyTest checks for that.)
Move some unnecessary NULL checks to asserts.
When copyTo performs a memcpy, only clone the genID if the resulting
SkPixelRef has the same dimensions as the original. This catches a bug
where copying an extracted SkBitmap with the same width as its original
SkPixelRef would incorrectly have the same genID.
Add a comment and assert in deepCopyTo, when cloning the genID, since
that case correctly clones it.
BitmapCopyTest.cpp:
Pull redundant work out of the inner loop (setting up the source bitmaps
and testing extractSubset). Create a new inner loop for extractSubset, to
test copying the result to each different config.
Extract a subset that has the same width as the original, to catch the
bug mentioned above.
Remove the reporter assert which checks for the resulting rowbytes.
Add checks to ensure that copying the extracted subset changes the genID.
GpuBitmapCopyTest:
Create an SkBitmap that shares an existing SkPixelRef, but only represents
a subset. This is to test the first call to cloneGenID in SkBitmap::copyTo.
In this case, the genID should NOT be copied, since only a portion of the
SkPixelRef was copied.
Also test deepCopy on this subset.
TestIndividualCopy now takes a parameter stating whether the genID should
change in the copy. It also does a read back using the appropriate subset.
It no longer differentiates between copyTo and deepCopyTo, since that
distinction was only necessary for copying from/to configs other than 8888
(which are no longer being tested), where copyTo did a read back in 8888 and
then drew the result to the desired config (resulting in an imperfect copy).
BUG=skia:1742
R=mtklein@google.com, bsalomon@google.com, reed@google.com
Author: scroggo@google.com
Review URL: https://codereview.chromium.org/112113005
git-svn-id: http://skia.googlecode.com/svn/trunk@13021 2bbb7eff-a529-9590-31e7-b0007b416f81
2014-01-10 17:22:01 +00:00
|
|
|
|
2009-05-01 04:00:01 +00:00
|
|
|
for (size_t i = 0; i < SK_ARRAY_COUNT(gPairs); i++) {
|
2014-01-10 18:28:23 +00:00
|
|
|
for (size_t j = 0; j < SK_ARRAY_COUNT(gPairs); j++) {
|
|
|
|
SkBitmap srcOpaque, srcPremul, dst;
|
2013-10-11 07:01:39 +00:00
|
|
|
|
2014-01-10 18:28:23 +00:00
|
|
|
{
|
|
|
|
SkColorTable* ctOpaque = NULL;
|
|
|
|
SkColorTable* ctPremul = NULL;
|
2013-10-10 14:44:56 +00:00
|
|
|
|
2014-01-10 18:28:23 +00:00
|
|
|
srcOpaque.setConfig(gPairs[i].fConfig, W, H, 0, kOpaque_SkAlphaType);
|
|
|
|
srcPremul.setConfig(gPairs[i].fConfig, W, H, 0, kPremul_SkAlphaType);
|
|
|
|
if (SkBitmap::kIndex8_Config == gPairs[i].fConfig) {
|
|
|
|
ctOpaque = init_ctable(kOpaque_SkAlphaType);
|
|
|
|
ctPremul = init_ctable(kPremul_SkAlphaType);
|
2013-10-10 14:44:56 +00:00
|
|
|
}
|
2014-01-10 18:28:23 +00:00
|
|
|
srcOpaque.allocPixels(ctOpaque);
|
|
|
|
srcPremul.allocPixels(ctPremul);
|
|
|
|
SkSafeUnref(ctOpaque);
|
|
|
|
SkSafeUnref(ctPremul);
|
2009-05-01 04:00:01 +00:00
|
|
|
}
|
2014-01-10 18:28:23 +00:00
|
|
|
init_src(srcOpaque);
|
|
|
|
init_src(srcPremul);
|
2009-05-01 04:00:01 +00:00
|
|
|
|
2013-10-10 14:44:56 +00:00
|
|
|
bool success = srcPremul.copyTo(&dst, gPairs[j].fConfig);
|
2009-05-01 04:00:01 +00:00
|
|
|
bool expected = gPairs[i].fValid[j] != '0';
|
|
|
|
if (success != expected) {
|
2014-01-10 14:58:10 +00:00
|
|
|
ERRORF(reporter, "SkBitmap::copyTo from %s to %s. expected %s "
|
|
|
|
"returned %s", gConfigName[i], gConfigName[j],
|
|
|
|
boolStr(expected), boolStr(success));
|
2009-05-01 04:00:01 +00:00
|
|
|
}
|
2011-01-05 15:50:27 +00:00
|
|
|
|
2013-10-10 14:44:56 +00:00
|
|
|
bool canSucceed = srcPremul.canCopyTo(gPairs[j].fConfig);
|
2009-05-06 17:44:34 +00:00
|
|
|
if (success != canSucceed) {
|
2014-01-10 14:58:10 +00:00
|
|
|
ERRORF(reporter, "SkBitmap::copyTo from %s to %s. returned %s "
|
|
|
|
"canCopyTo %s", gConfigName[i], gConfigName[j],
|
|
|
|
boolStr(success), boolStr(canSucceed));
|
2009-05-06 17:44:34 +00:00
|
|
|
}
|
2009-05-03 18:23:30 +00:00
|
|
|
|
2009-05-01 04:00:01 +00:00
|
|
|
if (success) {
|
2013-10-10 14:44:56 +00:00
|
|
|
REPORTER_ASSERT(reporter, srcPremul.width() == dst.width());
|
|
|
|
REPORTER_ASSERT(reporter, srcPremul.height() == dst.height());
|
2009-05-03 18:23:30 +00:00
|
|
|
REPORTER_ASSERT(reporter, dst.config() == gPairs[j].fConfig);
|
2013-10-10 14:44:56 +00:00
|
|
|
test_isOpaque(reporter, srcOpaque, srcPremul, dst.config());
|
|
|
|
if (srcPremul.config() == dst.config()) {
|
|
|
|
SkAutoLockPixels srcLock(srcPremul);
|
2009-05-01 04:00:01 +00:00
|
|
|
SkAutoLockPixels dstLock(dst);
|
2013-10-10 14:44:56 +00:00
|
|
|
REPORTER_ASSERT(reporter, srcPremul.readyToDraw());
|
2009-05-01 04:00:01 +00:00
|
|
|
REPORTER_ASSERT(reporter, dst.readyToDraw());
|
2013-10-10 14:44:56 +00:00
|
|
|
const char* srcP = (const char*)srcPremul.getAddr(0, 0);
|
2009-05-01 04:00:01 +00:00
|
|
|
const char* dstP = (const char*)dst.getAddr(0, 0);
|
|
|
|
REPORTER_ASSERT(reporter, srcP != dstP);
|
|
|
|
REPORTER_ASSERT(reporter, !memcmp(srcP, dstP,
|
2013-10-10 14:44:56 +00:00
|
|
|
srcPremul.getSize()));
|
|
|
|
REPORTER_ASSERT(reporter, srcPremul.getGenerationID() == dst.getGenerationID());
|
2012-08-22 15:00:05 +00:00
|
|
|
} else {
|
2013-10-10 14:44:56 +00:00
|
|
|
REPORTER_ASSERT(reporter, srcPremul.getGenerationID() != dst.getGenerationID());
|
2009-05-01 04:00:01 +00:00
|
|
|
}
|
2014-01-10 18:28:23 +00:00
|
|
|
// test extractSubset
|
|
|
|
{
|
|
|
|
SkBitmap bitmap(srcOpaque);
|
|
|
|
SkBitmap subset;
|
|
|
|
SkIRect r;
|
|
|
|
r.set(1, 1, 2, 2);
|
|
|
|
bitmap.setIsVolatile(true);
|
|
|
|
if (bitmap.extractSubset(&subset, r)) {
|
|
|
|
REPORTER_ASSERT(reporter, subset.width() == 1);
|
|
|
|
REPORTER_ASSERT(reporter, subset.height() == 1);
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.alphaType() == bitmap.alphaType());
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.isVolatile() == true);
|
|
|
|
|
|
|
|
SkBitmap copy;
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyTo(©, subset.config()));
|
|
|
|
REPORTER_ASSERT(reporter, copy.width() == 1);
|
|
|
|
REPORTER_ASSERT(reporter, copy.height() == 1);
|
|
|
|
REPORTER_ASSERT(reporter, copy.rowBytes() <= 4);
|
|
|
|
|
|
|
|
SkAutoLockPixels alp0(subset);
|
|
|
|
SkAutoLockPixels alp1(copy);
|
|
|
|
// they should both have, or both not-have, a colortable
|
|
|
|
bool hasCT = subset.getColorTable() != NULL;
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
(copy.getColorTable() != NULL) == hasCT);
|
|
|
|
}
|
|
|
|
bitmap = srcPremul;
|
|
|
|
bitmap.setIsVolatile(false);
|
|
|
|
if (bitmap.extractSubset(&subset, r)) {
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.alphaType() == bitmap.alphaType());
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.isVolatile() == false);
|
|
|
|
}
|
|
|
|
}
|
2009-05-01 04:00:01 +00:00
|
|
|
} else {
|
|
|
|
// dst should be unchanged from its initial state
|
|
|
|
REPORTER_ASSERT(reporter, dst.config() == SkBitmap::kNo_Config);
|
|
|
|
REPORTER_ASSERT(reporter, dst.width() == 0);
|
|
|
|
REPORTER_ASSERT(reporter, dst.height() == 0);
|
|
|
|
}
|
2010-11-16 20:22:41 +00:00
|
|
|
} // for (size_t j = ...
|
|
|
|
|
|
|
|
// Tests for getSafeSize(), getSafeSize64(), copyPixelsTo(),
|
|
|
|
// copyPixelsFrom().
|
|
|
|
//
|
|
|
|
for (size_t copyCase = 0; copyCase < SK_ARRAY_COUNT(isExtracted);
|
|
|
|
++copyCase) {
|
|
|
|
// Test copying to/from external buffer.
|
|
|
|
// Note: the tests below have hard-coded values ---
|
|
|
|
// Please take care if modifying.
|
|
|
|
|
2013-06-26 14:35:02 +00:00
|
|
|
// Tests for getSafeSize64().
|
|
|
|
// Test with a very large configuration without pixel buffer
|
|
|
|
// attached.
|
|
|
|
SkBitmap tstSafeSize;
|
|
|
|
tstSafeSize.setConfig(gPairs[i].fConfig, 100000000U,
|
|
|
|
100000000U);
|
2013-12-30 14:40:38 +00:00
|
|
|
int64_t safeSize = tstSafeSize.computeSafeSize64();
|
|
|
|
if (safeSize < 0) {
|
2014-01-10 14:58:10 +00:00
|
|
|
ERRORF(reporter, "getSafeSize64() negative: %s",
|
|
|
|
getSkConfigName(tstSafeSize));
|
2013-06-26 14:35:02 +00:00
|
|
|
}
|
|
|
|
bool sizeFail = false;
|
|
|
|
// Compare against hand-computed values.
|
|
|
|
switch (gPairs[i].fConfig) {
|
|
|
|
case SkBitmap::kNo_Config:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SkBitmap::kA8_Config:
|
|
|
|
case SkBitmap::kIndex8_Config:
|
2013-12-30 14:40:38 +00:00
|
|
|
if (safeSize != 0x2386F26FC10000LL) {
|
2013-06-26 14:35:02 +00:00
|
|
|
sizeFail = true;
|
2013-12-20 14:24:21 +00:00
|
|
|
}
|
2013-06-26 14:35:02 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SkBitmap::kRGB_565_Config:
|
|
|
|
case SkBitmap::kARGB_4444_Config:
|
2013-12-30 14:40:38 +00:00
|
|
|
if (safeSize != 0x470DE4DF820000LL) {
|
2013-06-26 14:35:02 +00:00
|
|
|
sizeFail = true;
|
2013-12-20 14:24:21 +00:00
|
|
|
}
|
2013-06-26 14:35:02 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SkBitmap::kARGB_8888_Config:
|
2013-12-30 14:40:38 +00:00
|
|
|
if (safeSize != 0x8E1BC9BF040000LL) {
|
2013-06-26 14:35:02 +00:00
|
|
|
sizeFail = true;
|
2013-12-20 14:24:21 +00:00
|
|
|
}
|
2013-06-26 14:35:02 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (sizeFail) {
|
2014-01-10 14:58:10 +00:00
|
|
|
ERRORF(reporter, "computeSafeSize64() wrong size: %s",
|
|
|
|
getSkConfigName(tstSafeSize));
|
2013-06-26 14:35:02 +00:00
|
|
|
}
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2013-12-02 13:50:38 +00:00
|
|
|
int subW = 2;
|
|
|
|
int subH = 2;
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2013-06-26 14:35:02 +00:00
|
|
|
// Create bitmap to act as source for copies and subsets.
|
|
|
|
SkBitmap src, subset;
|
|
|
|
SkColorTable* ct = NULL;
|
|
|
|
if (isExtracted[copyCase]) { // A larger image to extract from.
|
|
|
|
src.setConfig(gPairs[i].fConfig, 2 * subW + 1, subH);
|
|
|
|
} else { // Tests expect a 2x2 bitmap, so make smaller.
|
|
|
|
src.setConfig(gPairs[i].fConfig, subW, subH);
|
|
|
|
}
|
|
|
|
if (SkBitmap::kIndex8_Config == src.config()) {
|
2013-10-10 14:44:56 +00:00
|
|
|
ct = init_ctable(kPremul_SkAlphaType);
|
2013-06-26 14:35:02 +00:00
|
|
|
}
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2013-06-26 14:35:02 +00:00
|
|
|
src.allocPixels(ct);
|
|
|
|
SkSafeUnref(ct);
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2013-06-26 14:35:02 +00:00
|
|
|
// Either copy src or extract into 'subset', which is used
|
|
|
|
// for subsequent calls to copyPixelsTo/From.
|
|
|
|
bool srcReady = false;
|
|
|
|
if (isExtracted[copyCase]) {
|
|
|
|
// The extractedSubset() test case allows us to test copy-
|
|
|
|
// ing when src and dst mave possibly different strides.
|
|
|
|
SkIRect r;
|
2013-12-02 13:50:38 +00:00
|
|
|
r.set(1, 0, 1 + subW, subH); // 2x2 extracted bitmap
|
2013-06-26 14:35:02 +00:00
|
|
|
|
|
|
|
srcReady = src.extractSubset(&subset, r);
|
|
|
|
} else {
|
2013-10-31 17:28:30 +00:00
|
|
|
srcReady = src.copyTo(&subset, src.config());
|
2013-06-26 14:35:02 +00:00
|
|
|
}
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2013-06-26 14:35:02 +00:00
|
|
|
// Not all configurations will generate a valid 'subset'.
|
|
|
|
if (srcReady) {
|
|
|
|
|
|
|
|
// Allocate our target buffer 'buf' for all copies.
|
|
|
|
// To simplify verifying correctness of copies attach
|
|
|
|
// buf to a SkBitmap, but copies are done using the
|
|
|
|
// raw buffer pointer.
|
2013-10-16 17:48:11 +00:00
|
|
|
const size_t bufSize = subH *
|
2013-10-31 17:28:30 +00:00
|
|
|
SkBitmap::ComputeRowBytes(src.config(), subW) * 2;
|
2013-06-26 14:35:02 +00:00
|
|
|
SkAutoMalloc autoBuf (bufSize);
|
|
|
|
uint8_t* buf = static_cast<uint8_t*>(autoBuf.get());
|
|
|
|
|
|
|
|
SkBitmap bufBm; // Attach buf to this bitmap.
|
|
|
|
bool successExpected;
|
|
|
|
|
|
|
|
// Set up values for each pixel being copied.
|
|
|
|
Coordinates coords(subW * subH);
|
2013-10-16 17:48:11 +00:00
|
|
|
for (int x = 0; x < subW; ++x)
|
|
|
|
for (int y = 0; y < subH; ++y)
|
2013-06-26 14:35:02 +00:00
|
|
|
{
|
|
|
|
int index = y * subW + x;
|
|
|
|
SkASSERT(index < coords.length);
|
|
|
|
coords[index]->fX = x;
|
|
|
|
coords[index]->fY = y;
|
|
|
|
}
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2013-06-26 14:35:02 +00:00
|
|
|
writeCoordPixels(subset, coords);
|
|
|
|
|
|
|
|
// Test #1 ////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Before/after comparisons easier if we attach buf
|
|
|
|
// to an appropriately configured SkBitmap.
|
|
|
|
memset(buf, 0xFF, bufSize);
|
|
|
|
// Config with stride greater than src but that fits in buf.
|
|
|
|
bufBm.setConfig(gPairs[i].fConfig, subW, subH,
|
2013-10-31 17:28:30 +00:00
|
|
|
SkBitmap::ComputeRowBytes(subset.config(), subW) * 2);
|
2013-06-26 14:35:02 +00:00
|
|
|
bufBm.setPixels(buf);
|
|
|
|
successExpected = false;
|
|
|
|
// Then attempt to copy with a stride that is too large
|
|
|
|
// to fit in the buffer.
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyPixelsTo(buf, bufSize, bufBm.rowBytes() * 3)
|
|
|
|
== successExpected);
|
|
|
|
|
|
|
|
if (successExpected)
|
|
|
|
reportCopyVerification(subset, bufBm, coords,
|
|
|
|
"copyPixelsTo(buf, bufSize, 1.5*maxRowBytes)",
|
2010-11-16 20:22:41 +00:00
|
|
|
reporter);
|
|
|
|
|
2013-06-26 14:35:02 +00:00
|
|
|
// Test #2 ////////////////////////////////////////////
|
|
|
|
// This test should always succeed, but in the case
|
|
|
|
// of extracted bitmaps only because we handle the
|
|
|
|
// issue of getSafeSize(). Without getSafeSize()
|
|
|
|
// buffer overrun/read would occur.
|
|
|
|
memset(buf, 0xFF, bufSize);
|
|
|
|
bufBm.setConfig(gPairs[i].fConfig, subW, subH,
|
|
|
|
subset.rowBytes());
|
|
|
|
bufBm.setPixels(buf);
|
|
|
|
successExpected = subset.getSafeSize() <= bufSize;
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyPixelsTo(buf, bufSize) ==
|
|
|
|
successExpected);
|
|
|
|
if (successExpected)
|
|
|
|
reportCopyVerification(subset, bufBm, coords,
|
|
|
|
"copyPixelsTo(buf, bufSize)", reporter);
|
|
|
|
|
|
|
|
// Test #3 ////////////////////////////////////////////
|
|
|
|
// Copy with different stride between src and dst.
|
|
|
|
memset(buf, 0xFF, bufSize);
|
|
|
|
bufBm.setConfig(gPairs[i].fConfig, subW, subH,
|
|
|
|
subset.rowBytes()+1);
|
|
|
|
bufBm.setPixels(buf);
|
|
|
|
successExpected = true; // Should always work.
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyPixelsTo(buf, bufSize,
|
|
|
|
subset.rowBytes()+1) == successExpected);
|
|
|
|
if (successExpected)
|
|
|
|
reportCopyVerification(subset, bufBm, coords,
|
|
|
|
"copyPixelsTo(buf, bufSize, rowBytes+1)", reporter);
|
|
|
|
|
|
|
|
// Test #4 ////////////////////////////////////////////
|
|
|
|
// Test copy with stride too small.
|
|
|
|
memset(buf, 0xFF, bufSize);
|
|
|
|
bufBm.setConfig(gPairs[i].fConfig, subW, subH);
|
|
|
|
bufBm.setPixels(buf);
|
|
|
|
successExpected = false;
|
|
|
|
// Request copy with stride too small.
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyPixelsTo(buf, bufSize, bufBm.rowBytes()-1)
|
|
|
|
== successExpected);
|
|
|
|
if (successExpected)
|
|
|
|
reportCopyVerification(subset, bufBm, coords,
|
|
|
|
"copyPixelsTo(buf, bufSize, rowBytes()-1)", reporter);
|
|
|
|
|
|
|
|
#if 0 // copyPixelsFrom is gone
|
|
|
|
// Test #5 ////////////////////////////////////////////
|
|
|
|
// Tests the case where the source stride is too small
|
|
|
|
// for the source configuration.
|
|
|
|
memset(buf, 0xFF, bufSize);
|
|
|
|
bufBm.setConfig(gPairs[i].fConfig, subW, subH);
|
|
|
|
bufBm.setPixels(buf);
|
|
|
|
writeCoordPixels(bufBm, coords);
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyPixelsFrom(buf, bufSize, 1) == false);
|
|
|
|
|
|
|
|
// Test #6 ///////////////////////////////////////////
|
|
|
|
// Tests basic copy from an external buffer to the bitmap.
|
|
|
|
// If the bitmap is "extracted", this also tests the case
|
|
|
|
// where the source stride is different from the dest.
|
|
|
|
// stride.
|
|
|
|
// We've made the buffer large enough to always succeed.
|
|
|
|
bufBm.setConfig(gPairs[i].fConfig, subW, subH);
|
|
|
|
bufBm.setPixels(buf);
|
|
|
|
writeCoordPixels(bufBm, coords);
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyPixelsFrom(buf, bufSize, bufBm.rowBytes()) ==
|
|
|
|
true);
|
|
|
|
reportCopyVerification(bufBm, subset, coords,
|
|
|
|
"copyPixelsFrom(buf, bufSize)",
|
|
|
|
reporter);
|
|
|
|
|
|
|
|
// Test #7 ////////////////////////////////////////////
|
|
|
|
// Tests the case where the source buffer is too small
|
|
|
|
// for the transfer.
|
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
subset.copyPixelsFrom(buf, 1, subset.rowBytes()) ==
|
|
|
|
false);
|
2010-11-16 20:22:41 +00:00
|
|
|
|
2011-11-01 16:03:35 +00:00
|
|
|
#endif
|
2010-11-16 20:22:41 +00:00
|
|
|
}
|
|
|
|
} // for (size_t copyCase ...
|
2009-05-01 04:00:01 +00:00
|
|
|
}
|
|
|
|
}
|