/*
|
* Copyright 2015 Google Inc.
|
*
|
* Use of this source code is governed by a BSD-style license that can be
|
* found in the LICENSE file.
|
*/
|
|
#include "SkAndroidCodec.h"
|
#include "SkAndroidCodecAdapter.h"
|
#include "SkCodec.h"
|
#include "SkCodecPriv.h"
|
#include "SkMakeUnique.h"
|
#include "SkPixmap.h"
|
#include "SkPixmapPriv.h"
|
#include "SkSampledCodec.h"
|
|
static bool is_valid_sample_size(int sampleSize) {
|
// FIXME: As Leon has mentioned elsewhere, surely there is also a maximum sampleSize?
|
return sampleSize > 0;
|
}
|
|
/**
|
* Loads the gamut as a set of three points (triangle).
|
*/
|
static void load_gamut(SkPoint rgb[], const skcms_Matrix3x3& xyz) {
|
// rx = rX / (rX + rY + rZ)
|
// ry = rY / (rX + rY + rZ)
|
// gx, gy, bx, and gy are calulcated similarly.
|
for (int rgbIdx = 0; rgbIdx < 3; rgbIdx++) {
|
float sum = xyz.vals[rgbIdx][0] + xyz.vals[rgbIdx][1] + xyz.vals[rgbIdx][2];
|
rgb[rgbIdx].fX = xyz.vals[rgbIdx][0] / sum;
|
rgb[rgbIdx].fY = xyz.vals[rgbIdx][1] / sum;
|
}
|
}
|
|
/**
|
* Calculates the area of the triangular gamut.
|
*/
|
static float calculate_area(SkPoint abc[]) {
|
SkPoint a = abc[0];
|
SkPoint b = abc[1];
|
SkPoint c = abc[2];
|
return 0.5f * SkTAbs(a.fX*b.fY + b.fX*c.fY - a.fX*c.fY - c.fX*b.fY - b.fX*a.fY);
|
}
|
|
static constexpr float kSRGB_D50_GamutArea = 0.084f;
|
|
static bool is_wide_gamut(const skcms_ICCProfile& profile) {
|
// Determine if the source image has a gamut that is wider than sRGB. If so, we
|
// will use P3 as the output color space to avoid clipping the gamut.
|
if (profile.has_toXYZD50) {
|
SkPoint rgb[3];
|
load_gamut(rgb, profile.toXYZD50);
|
return calculate_area(rgb) > kSRGB_D50_GamutArea;
|
}
|
|
return false;
|
}
|
|
static inline SkImageInfo adjust_info(SkCodec* codec,
|
SkAndroidCodec::ExifOrientationBehavior orientationBehavior) {
|
auto info = codec->getInfo();
|
if (orientationBehavior == SkAndroidCodec::ExifOrientationBehavior::kIgnore
|
|| !SkPixmapPriv::ShouldSwapWidthHeight(codec->getOrigin())) {
|
return info;
|
}
|
return SkPixmapPriv::SwapWidthHeight(info);
|
}
|
|
SkAndroidCodec::SkAndroidCodec(SkCodec* codec, ExifOrientationBehavior orientationBehavior)
|
: fInfo(adjust_info(codec, orientationBehavior))
|
, fOrientationBehavior(orientationBehavior)
|
, fCodec(codec)
|
{}
|
|
SkAndroidCodec::~SkAndroidCodec() {}
|
|
std::unique_ptr<SkAndroidCodec> SkAndroidCodec::MakeFromStream(std::unique_ptr<SkStream> stream,
|
SkPngChunkReader* chunkReader) {
|
auto codec = SkCodec::MakeFromStream(std::move(stream), nullptr, chunkReader);
|
return MakeFromCodec(std::move(codec));
|
}
|
|
std::unique_ptr<SkAndroidCodec> SkAndroidCodec::MakeFromCodec(std::unique_ptr<SkCodec> codec,
|
ExifOrientationBehavior orientationBehavior) {
|
if (nullptr == codec) {
|
return nullptr;
|
}
|
|
switch ((SkEncodedImageFormat)codec->getEncodedFormat()) {
|
case SkEncodedImageFormat::kPNG:
|
case SkEncodedImageFormat::kICO:
|
case SkEncodedImageFormat::kJPEG:
|
case SkEncodedImageFormat::kGIF:
|
case SkEncodedImageFormat::kBMP:
|
case SkEncodedImageFormat::kWBMP:
|
case SkEncodedImageFormat::kHEIF:
|
return skstd::make_unique<SkSampledCodec>(codec.release(), orientationBehavior);
|
|
#ifdef SK_HAS_WEBP_LIBRARY
|
case SkEncodedImageFormat::kWEBP:
|
#endif
|
#ifdef SK_CODEC_DECODES_RAW
|
case SkEncodedImageFormat::kDNG:
|
#endif
|
#if defined(SK_HAS_WEBP_LIBRARY) || defined(SK_CODEC_DECODES_RAW)
|
return skstd::make_unique<SkAndroidCodecAdapter>(codec.release(), orientationBehavior);
|
#endif
|
|
default:
|
return nullptr;
|
}
|
}
|
|
std::unique_ptr<SkAndroidCodec> SkAndroidCodec::MakeFromData(sk_sp<SkData> data,
|
SkPngChunkReader* chunkReader) {
|
if (!data) {
|
return nullptr;
|
}
|
|
return MakeFromStream(SkMemoryStream::Make(std::move(data)), chunkReader);
|
}
|
|
SkColorType SkAndroidCodec::computeOutputColorType(SkColorType requestedColorType) {
|
bool highPrecision = fCodec->getEncodedInfo().bitsPerComponent() > 8;
|
switch (requestedColorType) {
|
case kARGB_4444_SkColorType:
|
return kN32_SkColorType;
|
case kN32_SkColorType:
|
break;
|
case kAlpha_8_SkColorType:
|
// Fall through to kGray_8. Before kGray_8_SkColorType existed,
|
// we allowed clients to request kAlpha_8 when they wanted a
|
// grayscale decode.
|
case kGray_8_SkColorType:
|
if (kGray_8_SkColorType == this->getInfo().colorType()) {
|
return kGray_8_SkColorType;
|
}
|
break;
|
case kRGB_565_SkColorType:
|
if (kOpaque_SkAlphaType == this->getInfo().alphaType()) {
|
return kRGB_565_SkColorType;
|
}
|
break;
|
case kRGBA_F16_SkColorType:
|
return kRGBA_F16_SkColorType;
|
default:
|
break;
|
}
|
|
// F16 is the Android default for high precision images.
|
return highPrecision ? kRGBA_F16_SkColorType : kN32_SkColorType;
|
}
|
|
SkAlphaType SkAndroidCodec::computeOutputAlphaType(bool requestedUnpremul) {
|
if (kOpaque_SkAlphaType == this->getInfo().alphaType()) {
|
return kOpaque_SkAlphaType;
|
}
|
return requestedUnpremul ? kUnpremul_SkAlphaType : kPremul_SkAlphaType;
|
}
|
|
sk_sp<SkColorSpace> SkAndroidCodec::computeOutputColorSpace(SkColorType outputColorType,
|
sk_sp<SkColorSpace> prefColorSpace) {
|
switch (outputColorType) {
|
case kRGBA_F16_SkColorType:
|
case kRGB_565_SkColorType:
|
case kRGBA_8888_SkColorType:
|
case kBGRA_8888_SkColorType: {
|
// If |prefColorSpace| is supplied, choose it.
|
if (prefColorSpace) {
|
return prefColorSpace;
|
}
|
|
const skcms_ICCProfile* encodedProfile = fCodec->getEncodedInfo().profile();
|
if (encodedProfile) {
|
if (auto encodedSpace = SkColorSpace::Make(*encodedProfile)) {
|
// Leave the pixels in the encoded color space. Color space conversion
|
// will be handled after decode time.
|
return encodedSpace;
|
}
|
|
if (is_wide_gamut(*encodedProfile)) {
|
return SkColorSpace::MakeRGB(SkNamedTransferFn::kSRGB, SkNamedGamut::kDCIP3);
|
}
|
}
|
|
return SkColorSpace::MakeSRGB();
|
}
|
default:
|
// Color correction not supported for kGray.
|
return nullptr;
|
}
|
}
|
|
static bool supports_any_down_scale(const SkCodec* codec) {
|
return codec->getEncodedFormat() == SkEncodedImageFormat::kWEBP;
|
}
|
|
// There are a variety of ways two SkISizes could be compared. This method
|
// returns true if either dimensions of a is < that of b.
|
// computeSampleSize also uses the opposite, which means that both
|
// dimensions of a >= b.
|
static inline bool smaller_than(const SkISize& a, const SkISize& b) {
|
return a.width() < b.width() || a.height() < b.height();
|
}
|
|
// Both dimensions of a > that of b.
|
static inline bool strictly_bigger_than(const SkISize& a, const SkISize& b) {
|
return a.width() > b.width() && a.height() > b.height();
|
}
|
|
int SkAndroidCodec::computeSampleSize(SkISize* desiredSize) const {
|
SkASSERT(desiredSize);
|
|
if (!desiredSize || *desiredSize == fInfo.dimensions()) {
|
return 1;
|
}
|
|
if (smaller_than(fInfo.dimensions(), *desiredSize)) {
|
*desiredSize = fInfo.dimensions();
|
return 1;
|
}
|
|
// Handle bad input:
|
if (desiredSize->width() < 1 || desiredSize->height() < 1) {
|
*desiredSize = SkISize::Make(std::max(1, desiredSize->width()),
|
std::max(1, desiredSize->height()));
|
}
|
|
if (supports_any_down_scale(fCodec.get())) {
|
return 1;
|
}
|
|
int sampleX = fInfo.width() / desiredSize->width();
|
int sampleY = fInfo.height() / desiredSize->height();
|
int sampleSize = std::min(sampleX, sampleY);
|
auto computedSize = this->getSampledDimensions(sampleSize);
|
if (computedSize == *desiredSize) {
|
return sampleSize;
|
}
|
|
if (computedSize == fInfo.dimensions() || sampleSize == 1) {
|
// Cannot downscale
|
*desiredSize = computedSize;
|
return 1;
|
}
|
|
if (strictly_bigger_than(computedSize, *desiredSize)) {
|
// See if there is a tighter fit.
|
while (true) {
|
auto smaller = this->getSampledDimensions(sampleSize + 1);
|
if (smaller == *desiredSize) {
|
return sampleSize + 1;
|
}
|
if (smaller == computedSize || smaller_than(smaller, *desiredSize)) {
|
// Cannot get any smaller without being smaller than desired.
|
*desiredSize = computedSize;
|
return sampleSize;
|
}
|
|
sampleSize++;
|
computedSize = smaller;
|
}
|
|
SkASSERT(false);
|
}
|
|
if (!smaller_than(computedSize, *desiredSize)) {
|
// This means one of the computed dimensions is equal to desired, and
|
// the other is bigger. This is as close as we can get.
|
*desiredSize = computedSize;
|
return sampleSize;
|
}
|
|
// computedSize is too small. Make it larger.
|
while (sampleSize > 2) {
|
auto bigger = this->getSampledDimensions(sampleSize - 1);
|
if (bigger == *desiredSize || !smaller_than(bigger, *desiredSize)) {
|
*desiredSize = bigger;
|
return sampleSize - 1;
|
}
|
sampleSize--;
|
}
|
|
*desiredSize = fInfo.dimensions();
|
return 1;
|
}
|
|
SkISize SkAndroidCodec::getSampledDimensions(int sampleSize) const {
|
if (!is_valid_sample_size(sampleSize)) {
|
return {0, 0};
|
}
|
|
// Fast path for when we are not scaling.
|
if (1 == sampleSize) {
|
return fInfo.dimensions();
|
}
|
|
auto dims = this->onGetSampledDimensions(sampleSize);
|
if (fOrientationBehavior == SkAndroidCodec::ExifOrientationBehavior::kIgnore
|
|| !SkPixmapPriv::ShouldSwapWidthHeight(fCodec->getOrigin())) {
|
return dims;
|
}
|
|
return { dims.height(), dims.width() };
|
}
|
|
bool SkAndroidCodec::getSupportedSubset(SkIRect* desiredSubset) const {
|
if (!desiredSubset || !is_valid_subset(*desiredSubset, fInfo.dimensions())) {
|
return false;
|
}
|
|
return this->onGetSupportedSubset(desiredSubset);
|
}
|
|
SkISize SkAndroidCodec::getSampledSubsetDimensions(int sampleSize, const SkIRect& subset) const {
|
if (!is_valid_sample_size(sampleSize)) {
|
return {0, 0};
|
}
|
|
// We require that the input subset is a subset that is supported by SkAndroidCodec.
|
// We test this by calling getSupportedSubset() and verifying that no modifications
|
// are made to the subset.
|
SkIRect copySubset = subset;
|
if (!this->getSupportedSubset(©Subset) || copySubset != subset) {
|
return {0, 0};
|
}
|
|
// If the subset is the entire image, for consistency, use getSampledDimensions().
|
if (fInfo.dimensions() == subset.size()) {
|
return this->getSampledDimensions(sampleSize);
|
}
|
|
// This should perhaps call a virtual function, but currently both of our subclasses
|
// want the same implementation.
|
return {get_scaled_dimension(subset.width(), sampleSize),
|
get_scaled_dimension(subset.height(), sampleSize)};
|
}
|
|
static bool acceptable_result(SkCodec::Result result) {
|
switch (result) {
|
// These results mean a partial or complete image. They should be considered
|
// a success by SkPixmapPriv.
|
case SkCodec::kSuccess:
|
case SkCodec::kIncompleteInput:
|
case SkCodec::kErrorInInput:
|
return true;
|
default:
|
return false;
|
}
|
}
|
|
SkCodec::Result SkAndroidCodec::getAndroidPixels(const SkImageInfo& requestInfo,
|
void* requestPixels, size_t requestRowBytes, const AndroidOptions* options) {
|
if (!requestPixels) {
|
return SkCodec::kInvalidParameters;
|
}
|
if (requestRowBytes < requestInfo.minRowBytes()) {
|
return SkCodec::kInvalidParameters;
|
}
|
|
SkImageInfo adjustedInfo = fInfo;
|
if (ExifOrientationBehavior::kRespect == fOrientationBehavior
|
&& SkPixmapPriv::ShouldSwapWidthHeight(fCodec->getOrigin())) {
|
adjustedInfo = SkPixmapPriv::SwapWidthHeight(adjustedInfo);
|
}
|
|
AndroidOptions defaultOptions;
|
if (!options) {
|
options = &defaultOptions;
|
} else if (options->fSubset) {
|
if (!is_valid_subset(*options->fSubset, adjustedInfo.dimensions())) {
|
return SkCodec::kInvalidParameters;
|
}
|
|
if (SkIRect::MakeSize(adjustedInfo.dimensions()) == *options->fSubset) {
|
// The caller wants the whole thing, rather than a subset. Modify
|
// the AndroidOptions passed to onGetAndroidPixels to not specify
|
// a subset.
|
defaultOptions = *options;
|
defaultOptions.fSubset = nullptr;
|
options = &defaultOptions;
|
}
|
}
|
|
if (ExifOrientationBehavior::kIgnore == fOrientationBehavior) {
|
return this->onGetAndroidPixels(requestInfo, requestPixels, requestRowBytes, *options);
|
}
|
|
SkCodec::Result result;
|
auto decode = [this, options, &result](const SkPixmap& pm) {
|
result = this->onGetAndroidPixels(pm.info(), pm.writable_addr(), pm.rowBytes(), *options);
|
return acceptable_result(result);
|
};
|
|
SkPixmap dst(requestInfo, requestPixels, requestRowBytes);
|
if (SkPixmapPriv::Orient(dst, fCodec->getOrigin(), decode)) {
|
return result;
|
}
|
|
// Orient returned false. If onGetAndroidPixels succeeded, then Orient failed internally.
|
if (acceptable_result(result)) {
|
return SkCodec::kInternalError;
|
}
|
|
return result;
|
}
|
|
SkCodec::Result SkAndroidCodec::getAndroidPixels(const SkImageInfo& info, void* pixels,
|
size_t rowBytes) {
|
return this->getAndroidPixels(info, pixels, rowBytes, nullptr);
|
}
|