Origin orientation = get_exif_orientation(decoderMgr->dinfo());
sk_sp<SkData> iccData = get_icc_profile(decoderMgr->dinfo());
sk_sp<SkColorSpace> colorSpace = nullptr;
+ bool unsupportedICC = false;
if (iccData) {
SkColorSpace_Base::InputColorFormat inputColorFormat =
SkColorSpace_Base::InputColorFormat::kRGB;
inputColorFormat);
if (!colorSpace) {
SkCodecPrintf("Could not create SkColorSpace from ICC data.\n");
+ unsupportedICC = true;
}
}
if (!colorSpace) {
const int width = decoderMgr->dinfo()->image_width;
const int height = decoderMgr->dinfo()->image_height;
- *codecOut = new SkJpegCodec(width, height, info, stream, decoderMgr.release(),
- std::move(colorSpace), orientation);
+ SkJpegCodec* codec = new SkJpegCodec(width, height, info, stream, decoderMgr.release(),
+ std::move(colorSpace), orientation);
+ codec->setUnsupportedICC(unsupportedICC);
+ *codecOut = codec;
} else {
SkASSERT(nullptr != decoderMgrOut);
*decoderMgrOut = decoderMgr.release();
#endif // LIBPNG >= 1.6
// Returns a colorSpace object that represents any color space information in
-// the encoded data. If the encoded data contains no color space, this will
-// return NULL.
+// the encoded data. If the encoded data contains an invalid/unsupported color space,
+// this will return NULL. If there is no color space information, it will guess sRGB
sk_sp<SkColorSpace> read_color_space(png_structp png_ptr, png_infop info_ptr) {
#if (PNG_LIBPNG_VER_MAJOR > 1) || (PNG_LIBPNG_VER_MAJOR == 1 && PNG_LIBPNG_VER_MINOR >= 6)
#endif // LIBPNG >= 1.6
- // Report that there is no color space information in the PNG. SkPngCodec is currently
- // implemented to guess sRGB in this case.
- return nullptr;
+ // Report that there is no color space information in the PNG.
+ // Guess sRGB in this case.
+ return SkColorSpace::MakeNamed(SkColorSpace::kSRGB_Named);
}
void SkPngCodec::allocateStorage(const SkImageInfo& dstInfo) {
if (fOutCodec) {
SkASSERT(nullptr == *fOutCodec);
sk_sp<SkColorSpace> colorSpace = read_color_space(fPng_ptr, fInfo_ptr);
+ const bool unsupportedICC = !colorSpace;
if (!colorSpace) {
- // Treat unmarked pngs as sRGB.
+ // Treat unsupported/invalid color spaces as sRGB.
colorSpace = SkColorSpace::MakeNamed(SkColorSpace::kSRGB_Named);
}
*fOutCodec = new SkPngInterlacedDecoder(encodedInfo, imageInfo, fStream,
fChunkReader, fPng_ptr, fInfo_ptr, bitDepth, numberPasses);
}
+ (*fOutCodec)->setUnsupportedICC(unsupportedICC);
}
WebPChunkIterator chunkIterator;
SkAutoTCallVProc<WebPChunkIterator, WebPDemuxReleaseChunkIterator> autoCI(&chunkIterator);
sk_sp<SkColorSpace> colorSpace = nullptr;
+ bool unsupportedICC = false;
if (WebPDemuxGetChunk(demux, "ICCP", 1, &chunkIterator)) {
colorSpace = SkColorSpace::MakeICC(chunkIterator.chunk.bytes, chunkIterator.chunk.size);
+ if (!colorSpace) {
+ unsupportedICC = true;
+ }
}
-
if (!colorSpace) {
colorSpace = SkColorSpace::MakeNamed(SkColorSpace::kSRGB_Named);
}
}
SkEncodedInfo info = SkEncodedInfo::Make(color, alpha, 8);
- return new SkWebpCodec(features.width, features.height, info, std::move(colorSpace),
- streamDeleter.release(), demux.release(), std::move(data));
+ SkWebpCodec* codecOut = new SkWebpCodec(features.width, features.height, info,
+ std::move(colorSpace), streamDeleter.release(),
+ demux.release(), std::move(data));
+ codecOut->setUnsupportedICC(unsupportedICC);
+ return codecOut;
}
SkISize SkWebpCodec::onGetScaledDimensions(float desiredScale) const {
#include "SkBitmap.h"
#include "SkCodec.h"
+#include "SkColorSpace.h"
#include "SkCommandLineFlags.h"
#include "SkData.h"
#include "SkJSONCPP.h"
#include "SkTHash.h"
+#include <iostream>
#include <map>
DEFINE_string2(skps, s, "skps", "A path to a directory of skps.");
DEFINE_string2(out, o, "img-out", "A path to an output directory.");
DEFINE_bool(testDecode, false, "Indicates if we want to test that the images decode successfully.");
-DEFINE_bool(writeImages, true, "Indicates if we want to write out images.");
+DEFINE_bool(writeImages, true,
+ "Indicates if we want to write out supported/decoded images.");
+DEFINE_bool(writeFailedImages, false,
+ "Indicates if we want to write out unsupported/failed to decode images.");
+DEFINE_bool(testICCSupport, false,
+ "Indicates if we want to test that the images with ICC profiles are supported");
DEFINE_string2(failuresJsonPath, j, "",
"Dump SKP and count of unknown images to the specified JSON file. Will not be "
"written anywhere if empty.");
static int gKnown;
static const char* gOutputDir;
static std::map<std::string, unsigned int> gSkpToUnknownCount = {};
+static std::map<std::string, unsigned int> gSkpToUnsupportedCount;
static SkTHashSet<SkMD5::Digest> gSeen;
SkASSERT(false);
}
+ auto writeImage = [&] {
+ SkString path;
+ path.appendf("%s/%d.%s", gOutputDir, gKnown, ext.c_str());
+
+ SkFILEWStream file(path.c_str());
+ file.write(ptr, len);
+
+ SkDebugf("%s\n", path.c_str());
+ };
+
+
if (FLAGS_testDecode) {
SkBitmap bitmap;
SkImageInfo info = codec->getInfo().makeColorType(kN32_SkColorType);
bitmap.allocPixels(info);
const SkCodec::Result result = codec->getPixels(
info, bitmap.getPixels(), bitmap.rowBytes());
- if (SkCodec::kIncompleteInput != result && SkCodec::kSuccess != result)
- {
+ if (SkCodec::kIncompleteInput != result && SkCodec::kSuccess != result) {
SkDebugf("Decoding failed for %s\n", skpName.c_str());
gSkpToUnknownCount[skpName]++;
+ if (FLAGS_writeFailedImages) {
+ writeImage();
+ }
return;
}
}
+#ifdef SK_DEBUG
+ if (FLAGS_testICCSupport) {
+ if (codec->fUnsupportedICC) {
+ SkDebugf("Color correction failed for %s\n", skpName.c_str());
+ gSkpToUnsupportedCount[skpName]++;
+ if (FLAGS_writeFailedImages) {
+ writeImage();
+ }
+ return;
+ }
+ }
+#endif
+
if (FLAGS_writeImages) {
- SkString path;
- path.appendf("%s/%d.%s", gOutputDir, gKnown, ext.c_str());
+ writeImage();
+ }
- SkFILEWStream file(path.c_str());
- file.write(ptr, len);
- SkDebugf("%s\n", path.c_str());
- }
gKnown++;
}
int main(int argc, char** argv) {
SkCommandLineFlags::SetUsage(
"Usage: get_images_from_skps -s <dir of skps> -o <dir for output images> --testDecode "
- "-j <output JSON path>\n");
+ "-j <output JSON path> --testICCSupport --writeImages, --writeFailedImages\n");
SkCommandLineFlags::Parse(argc, argv);
const char* inputs = FLAGS_skps[0];
SkCommandLineFlags::PrintUsage();
return 1;
}
+#ifndef SK_DEBUG
+ if (FLAGS_testICCSupport) {
+ std::cerr << "--testICCSupport unavailable outside of SK_DEBUG builds" << std::endl;
+ return 1;
+ }
+#endif
SkOSFile::Iter iter(inputs, "skp");
for (SkString file; iter.next(&file); ) {
Sniffer sniff(file.c_str());
picture->serialize(&scratch, &sniff);
}
- int totalUnknowns = 0;
/**
JSON results are written out in the following format:
{
"skp4": 2,
...
},
+ "unsupported": {
+ "skp9": 13,
+ "skp17": 3,
+ ...
+ }
"totalFailures": 32,
+ "totalUnsupported": 9,
"totalSuccesses": 21,
}
*/
Json::Value fRoot;
+ int totalFailures = 0;
for(auto it = gSkpToUnknownCount.cbegin(); it != gSkpToUnknownCount.cend(); ++it)
{
SkDebugf("%s %d\n", it->first.c_str(), it->second);
- totalUnknowns += it->second;
+ totalFailures += it->second;
fRoot["failures"][it->first.c_str()] = it->second;
}
- SkDebugf("%d known, %d unknown\n", gKnown, totalUnknowns);
- fRoot["totalFailures"] = totalUnknowns;
+ fRoot["totalFailures"] = totalFailures;
+ int totalUnsupported = 0;
+#ifdef SK_DEBUG
+ for (const auto& unsupported : gSkpToUnsupportedCount) {
+ SkDebugf("%s %d\n", unsupported.first.c_str(), unsupported.second);
+ totalUnsupported += unsupported.second;
+ fRoot["unsupported"][unsupported.first] = unsupported.second;
+ }
+ fRoot["totalUnsupported"] = totalUnsupported;
+#endif
fRoot["totalSuccesses"] = gKnown;
- if (totalUnknowns > 0) {
+ SkDebugf("%d known, %d failures, %d unsupported\n", gKnown, totalFailures, totalUnsupported);
+ if (totalFailures > 0 || totalUnsupported > 0) {
if (!FLAGS_failuresJsonPath.isEmpty()) {
SkDebugf("Writing failures to %s\n", FLAGS_failuresJsonPath[0]);
SkFILEWStream stream(FLAGS_failuresJsonPath[0]);