2014-06-06 13:16:28 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2014 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2014-06-18 21:32:48 +00:00
|
|
|
#include "Resources.h"
|
2014-06-06 13:16:28 +00:00
|
|
|
#include "SkBitmap.h"
|
|
|
|
#include "SkData.h"
|
2015-01-08 02:04:45 +00:00
|
|
|
#include "SkImageGenerator.h"
|
2014-06-06 13:16:28 +00:00
|
|
|
#include "SkForceLinking.h"
|
|
|
|
#include "SkImageDecoder.h"
|
|
|
|
#include "SkOSFile.h"
|
|
|
|
#include "SkRandom.h"
|
|
|
|
#include "SkStream.h"
|
|
|
|
#include "Test.h"
|
|
|
|
|
|
|
|
__SK_FORCE_IMAGE_DECODER_LINKING;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* First, make sure that writing an 8-bit RGBA KTX file and then
|
|
|
|
* reading it produces the same bitmap.
|
|
|
|
*/
|
|
|
|
DEF_TEST(KtxReadWrite, reporter) {
|
|
|
|
|
|
|
|
// Random number generator with explicit seed for reproducibility
|
|
|
|
SkRandom rand(0x1005cbad);
|
|
|
|
|
|
|
|
SkBitmap bm8888;
|
2014-09-02 19:50:45 +00:00
|
|
|
bm8888.allocN32Pixels(128, 128);
|
2014-06-06 13:16:28 +00:00
|
|
|
|
|
|
|
uint8_t *pixels = reinterpret_cast<uint8_t*>(bm8888.getPixels());
|
2014-09-05 20:34:00 +00:00
|
|
|
REPORTER_ASSERT(reporter, pixels);
|
2014-06-06 13:16:28 +00:00
|
|
|
|
2015-08-27 14:41:13 +00:00
|
|
|
if (nullptr == pixels) {
|
2014-06-06 13:16:28 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t *row = pixels;
|
|
|
|
for (int y = 0; y < bm8888.height(); ++y) {
|
|
|
|
for (int x = 0; x < bm8888.width(); ++x) {
|
|
|
|
uint8_t a = rand.nextRangeU(0, 255);
|
|
|
|
uint8_t r = rand.nextRangeU(0, 255);
|
|
|
|
uint8_t g = rand.nextRangeU(0, 255);
|
|
|
|
uint8_t b = rand.nextRangeU(0, 255);
|
|
|
|
|
|
|
|
SkPMColor &pixel = *(reinterpret_cast<SkPMColor*>(row + x*sizeof(SkPMColor)));
|
|
|
|
pixel = SkPreMultiplyARGB(a, r, g, b);
|
|
|
|
}
|
|
|
|
row += bm8888.rowBytes();
|
|
|
|
}
|
|
|
|
REPORTER_ASSERT(reporter, !(bm8888.empty()));
|
|
|
|
|
|
|
|
SkAutoDataUnref encodedData(SkImageEncoder::EncodeData(bm8888, SkImageEncoder::kKTX_Type, 0));
|
2014-09-05 20:34:00 +00:00
|
|
|
REPORTER_ASSERT(reporter, encodedData);
|
2014-06-06 13:16:28 +00:00
|
|
|
|
2015-08-26 20:07:48 +00:00
|
|
|
SkAutoTDelete<SkMemoryStream> stream(new SkMemoryStream(encodedData));
|
2014-09-05 20:34:00 +00:00
|
|
|
REPORTER_ASSERT(reporter, stream);
|
2014-06-06 13:16:28 +00:00
|
|
|
|
|
|
|
SkBitmap decodedBitmap;
|
|
|
|
bool imageDecodeSuccess = SkImageDecoder::DecodeStream(stream, &decodedBitmap);
|
|
|
|
REPORTER_ASSERT(reporter, imageDecodeSuccess);
|
|
|
|
|
2014-06-17 15:38:31 +00:00
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.colorType() == bm8888.colorType());
|
2014-06-06 13:16:28 +00:00
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.alphaType() == bm8888.alphaType());
|
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.width() == bm8888.width());
|
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.height() == bm8888.height());
|
|
|
|
REPORTER_ASSERT(reporter, !(decodedBitmap.empty()));
|
|
|
|
|
|
|
|
uint8_t *decodedPixels = reinterpret_cast<uint8_t*>(decodedBitmap.getPixels());
|
2014-09-05 20:34:00 +00:00
|
|
|
REPORTER_ASSERT(reporter, decodedPixels);
|
2014-06-06 13:16:28 +00:00
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.getSize() == bm8888.getSize());
|
|
|
|
|
2015-08-27 14:41:13 +00:00
|
|
|
if (nullptr == decodedPixels) {
|
2014-06-06 13:16:28 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, memcmp(decodedPixels, pixels, decodedBitmap.getSize()) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Next test is to see whether or not reading an unpremultiplied KTX file accurately
|
|
|
|
* creates a premultiplied buffer...
|
|
|
|
*/
|
|
|
|
DEF_TEST(KtxReadUnpremul, reporter) {
|
|
|
|
|
|
|
|
static const uint8_t kHalfWhiteKTX[] = {
|
|
|
|
0xAB, 0x4B, 0x54, 0x58, 0x20, 0x31, // First twelve bytes is magic
|
|
|
|
0x31, 0xBB, 0x0D, 0x0A, 0x1A, 0x0A, // KTX identifier string
|
|
|
|
0x01, 0x02, 0x03, 0x04, // Then magic endian specifier
|
|
|
|
0x01, 0x14, 0x00, 0x00, // uint32_t fGLType;
|
|
|
|
0x01, 0x00, 0x00, 0x00, // uint32_t fGLTypeSize;
|
|
|
|
0x08, 0x19, 0x00, 0x00, // uint32_t fGLFormat;
|
|
|
|
0x58, 0x80, 0x00, 0x00, // uint32_t fGLInternalFormat;
|
|
|
|
0x08, 0x19, 0x00, 0x00, // uint32_t fGLBaseInternalFormat;
|
|
|
|
0x02, 0x00, 0x00, 0x00, // uint32_t fPixelWidth;
|
|
|
|
0x02, 0x00, 0x00, 0x00, // uint32_t fPixelHeight;
|
|
|
|
0x00, 0x00, 0x00, 0x00, // uint32_t fPixelDepth;
|
|
|
|
0x00, 0x00, 0x00, 0x00, // uint32_t fNumberOfArrayElements;
|
|
|
|
0x01, 0x00, 0x00, 0x00, // uint32_t fNumberOfFaces;
|
|
|
|
0x01, 0x00, 0x00, 0x00, // uint32_t fNumberOfMipmapLevels;
|
|
|
|
0x00, 0x00, 0x00, 0x00, // uint32_t fBytesOfKeyValueData;
|
|
|
|
0x10, 0x00, 0x00, 0x00, // image size: 2x2 image of RGBA = 4 * 4 = 16 bytes
|
|
|
|
0xFF, 0xFF, 0xFF, 0x80, // Pixel 1
|
|
|
|
0xFF, 0xFF, 0xFF, 0x80, // Pixel 2
|
|
|
|
0xFF, 0xFF, 0xFF, 0x80, // Pixel 3
|
|
|
|
0xFF, 0xFF, 0xFF, 0x80};// Pixel 4
|
|
|
|
|
2015-08-26 20:07:48 +00:00
|
|
|
SkAutoTDelete<SkMemoryStream> stream(new SkMemoryStream(kHalfWhiteKTX, sizeof(kHalfWhiteKTX)));
|
2014-09-05 20:34:00 +00:00
|
|
|
REPORTER_ASSERT(reporter, stream);
|
2014-06-06 13:16:28 +00:00
|
|
|
|
|
|
|
SkBitmap decodedBitmap;
|
|
|
|
bool imageDecodeSuccess = SkImageDecoder::DecodeStream(stream, &decodedBitmap);
|
|
|
|
REPORTER_ASSERT(reporter, imageDecodeSuccess);
|
|
|
|
|
2014-06-14 12:30:20 +00:00
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.colorType() == kN32_SkColorType);
|
2014-06-06 13:16:28 +00:00
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.alphaType() == kPremul_SkAlphaType);
|
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.width() == 2);
|
|
|
|
REPORTER_ASSERT(reporter, decodedBitmap.height() == 2);
|
|
|
|
REPORTER_ASSERT(reporter, !(decodedBitmap.empty()));
|
|
|
|
|
|
|
|
uint8_t *decodedPixels = reinterpret_cast<uint8_t*>(decodedBitmap.getPixels());
|
2014-09-05 20:34:00 +00:00
|
|
|
REPORTER_ASSERT(reporter, decodedPixels);
|
2014-06-06 13:16:28 +00:00
|
|
|
|
|
|
|
uint8_t *row = decodedPixels;
|
|
|
|
for (int j = 0; j < decodedBitmap.height(); ++j) {
|
|
|
|
for (int i = 0; i < decodedBitmap.width(); ++i) {
|
|
|
|
SkPMColor pixel = *(reinterpret_cast<SkPMColor*>(row + i*sizeof(SkPMColor)));
|
|
|
|
REPORTER_ASSERT(reporter, SkPreMultiplyARGB(0x80, 0xFF, 0xFF, 0xFF) == pixel);
|
|
|
|
}
|
|
|
|
row += decodedBitmap.rowBytes();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Finally, make sure that if we get ETC1 data from a PKM file that we can then
|
|
|
|
* accurately write it out into a KTX file (i.e. transferring the ETC1 data from
|
|
|
|
* the PKM to the KTX should produce an identical KTX to the one we have on file)
|
|
|
|
*/
|
|
|
|
DEF_TEST(KtxReexportPKM, reporter) {
|
2014-07-01 19:35:49 +00:00
|
|
|
SkString pkmFilename = GetResourcePath("mandrill_128.pkm");
|
2014-06-06 13:16:28 +00:00
|
|
|
|
|
|
|
// Load PKM file into a bitmap
|
|
|
|
SkBitmap etcBitmap;
|
2014-07-01 19:35:49 +00:00
|
|
|
SkAutoTUnref<SkData> fileData(SkData::NewFromFileName(pkmFilename.c_str()));
|
2015-08-27 14:41:13 +00:00
|
|
|
if (nullptr == fileData) {
|
2014-10-28 14:12:37 +00:00
|
|
|
SkDebugf("KtxReexportPKM: can't load test file %s\n", pkmFilename.c_str());
|
2014-07-02 12:57:50 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-06-06 13:16:28 +00:00
|
|
|
|
|
|
|
bool installDiscardablePixelRefSuccess =
|
2015-01-08 02:04:45 +00:00
|
|
|
SkInstallDiscardablePixelRef(fileData, &etcBitmap);
|
2014-06-06 13:16:28 +00:00
|
|
|
REPORTER_ASSERT(reporter, installDiscardablePixelRefSuccess);
|
|
|
|
|
|
|
|
// Write the bitmap out to a KTX file.
|
|
|
|
SkData *ktxDataPtr = SkImageEncoder::EncodeData(etcBitmap, SkImageEncoder::kKTX_Type, 0);
|
|
|
|
SkAutoDataUnref newKtxData(ktxDataPtr);
|
2014-09-05 20:34:00 +00:00
|
|
|
REPORTER_ASSERT(reporter, ktxDataPtr);
|
2014-06-06 13:16:28 +00:00
|
|
|
|
|
|
|
// See is this data is identical to data in existing ktx file.
|
2014-07-01 19:35:49 +00:00
|
|
|
SkString ktxFilename = GetResourcePath("mandrill_128.ktx");
|
2014-06-06 13:16:28 +00:00
|
|
|
SkAutoDataUnref oldKtxData(SkData::NewFromFileName(ktxFilename.c_str()));
|
|
|
|
REPORTER_ASSERT(reporter, oldKtxData->equals(newKtxData));
|
|
|
|
}
|