blob: 5532f5b5e8e0dad6caa275a0465490e7e93277e9 [file] [log] [blame]
/*
* Copyright 2022 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/codec/SkJpegRCodec.h"
#ifdef SK_CODEC_DECODES_JPEGR
#include <array>
#include <csetjmp>
#include <cstdlib>
#include <cstring>
#include <utility>
#include "include/android/SkAndroidFrameworkUtils.h"
#include "include/codec/SkCodec.h"
#include "include/core/SkAlphaType.h"
#include "include/core/SkColorType.h"
#include "include/core/SkData.h"
#include "include/core/SkImageInfo.h"
#include "include/core/SkPixmap.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkStream.h"
#include "include/core/SkTypes.h"
#include "include/core/SkYUVAInfo.h"
#include "include/private/base/SkTemplates.h"
#include "include/private/base/SkMalloc.h"
#include "include/private/base/SkTo.h"
#include "modules/skcms/skcms.h"
#include "src/codec/SkCodecPriv.h"
#include "src/codec/SkParseEncodedOrigin.h"
#include "src/core/SkStreamPriv.h"
static SkEncodedOrigin get_orientation(const std::vector<uint8_t> &exifData) {
SkEncodedOrigin orientation = kDefault_SkEncodedOrigin;
if (exifData.size() > 6) {
constexpr size_t kOffset = 6;
const size_t exifSize = exifData.size();
const uint8_t *data = exifData.data();
SkParseEncodedOrigin(data + kOffset, exifSize - kOffset,
&orientation);
}
return orientation;
}
bool SkJpegRCodec::IsJpegR(const void* buffer, size_t bytesRead) {
// We expect JPEGR file will have EXIF packet frist and custom
// 'JR' EXIF tag first in this EXIF packet. 32 bytes is not enough to
// parse the whole EXIF (or even the first TAG), but enough to detect
// this TAG.
constexpr uint8_t jpegSig[] = {0xFF, 0xD8, 0xFF, 0xE1};
constexpr uint8_t exifSig[]{'E', 'x', 'i', 'f', '\0', '\0'};
if (bytesRead < 32 || memcmp(buffer, jpegSig, sizeof(jpegSig))) {
return false;
}
const uint8_t* dataPtr = static_cast<const uint8_t*>(buffer) + 4;
const uint16_t exifSize = dataPtr[1] | (dataPtr[0] << 8);
if (exifSize < 26) {
return false;
}
dataPtr += 2;
if (memcmp(dataPtr, exifSig, sizeof(exifSig))) {
return false;
}
dataPtr += sizeof(exifSig);
bool isBigEndian = false;
if (dataPtr[0] == 0x49 && dataPtr[1] == 0x49) {
isBigEndian = false;
} else if (dataPtr[0] == 0x4d && dataPtr[1] == 0x4d) {
isBigEndian = true;
} else {
// Wrong EXIF format
return false;
}
dataPtr += 8; // points to the TAG number
uint16_t tagNum;
if (isBigEndian) {
tagNum = (dataPtr[0] << 8) | dataPtr[1];
} else {
tagNum = (dataPtr[1] << 8) | dataPtr[0];
}
if (tagNum < 1) {
return false;
}
dataPtr += 2;
if (dataPtr[0] != 0x4a || dataPtr[1] != 0x52) {
// Wrong TAG. JPEGR shall start with 0x4a 0x52 TAG
return false;
}
return true;
}
SkCodec::Result SkJpegRCodec::ReadHeader(SkStream* stream,
SkCodec** codecOut,
RecoveryMap** recoveryMapOut) {
// Create a RecoveryMap object to own all of the decompress information
std::unique_ptr<RecoveryMap> recoveryMap = std::make_unique<RecoveryMap>();
jpegr_info_struct jpegRInfo;
if (codecOut) {
// Get the encoded color type
std::unique_ptr<SkEncodedInfo::ICCProfile> profile = nullptr;
sk_sp<SkData> data = nullptr;
if (stream->getMemoryBase()) {
// It is safe to make without copy because we'll hold onto the stream.
data = SkData::MakeWithoutCopy(stream->getMemoryBase(), stream->getLength());
} else {
data = SkCopyStreamToData(stream);
// We don't need to hold the stream anymore
stream = nullptr;
}
jpegr_compressed_struct compressedImage;
if (data->data() == nullptr || data->size() == 0) {
return kIncompleteInput;
}
compressedImage.data = (void*)data->data();
compressedImage.length = data->size();
std::vector<uint8_t> exifData;
std::vector<uint8_t> iccData;
jpegRInfo.exifData = &exifData;
jpegRInfo.iccData = &iccData;
if (recoveryMap->getJPEGRInfo(&compressedImage, &jpegRInfo) != 0) {
return kInvalidInput;
}
// JPEGR always report 10-bit color depth
const uint8_t colorDepth = 10;
// iccSkData will outlive iccData as it is passed to profile, hence we need a copy
sk_sp<SkData> iccSkData = SkData::MakeWithCopy(iccData.data(), iccData.size());
profile = SkEncodedInfo::ICCProfile::Make(std::move(iccSkData));
// TODO: Figure out if we need to expose default profile for JPEGR and what it should be
SkEncodedInfo info = SkEncodedInfo::Make(jpegRInfo.width,
jpegRInfo.height,
SkEncodedInfo::kRGBA_Color,
SkEncodedInfo::kOpaque_Alpha,
colorDepth,
std::move(profile));
SkEncodedOrigin orientation = get_orientation(exifData);
SkJpegRCodec* codec = new SkJpegRCodec(std::move(info),
std::unique_ptr<SkStream>(stream),
recoveryMap.release(),
orientation,
std::move(data));
*codecOut = codec;
} else {
SkASSERT(nullptr != recoveryMap);
*recoveryMapOut = recoveryMap.release();
}
return kSuccess;
}
std::unique_ptr<SkCodec> SkJpegRCodec::MakeFromStream(std::unique_ptr<SkStream> stream,
Result* result) {
SkCodec* codec = nullptr;
*result = ReadHeader(stream.get(), &codec, nullptr);
if (kSuccess == *result) {
// Codec has taken ownership of the stream, we do not need to delete it
SkASSERT(codec);
stream.release();
return std::unique_ptr<SkCodec>(codec);
}
return nullptr;
}
SkJpegRCodec::SkJpegRCodec(SkEncodedInfo&& info,
std::unique_ptr<SkStream> stream,
RecoveryMap* recoveryMap,
SkEncodedOrigin origin,
sk_sp<SkData> data)
: SkCodec(std::move(info), skcms_PixelFormat_RGBA_1010102, std::move(stream), origin)
, fRecoveryMap(recoveryMap)
, fData(std::move(data)) {}
SkJpegRCodec::~SkJpegRCodec() = default;
bool SkJpegRCodec::conversionSupported(const SkImageInfo& dstInfo,
bool srcIsOpaque,
bool needsColorXform) {
SkASSERT(srcIsOpaque);
// TODO: Implement color XForm
SkASSERT(needsColorXform == false);
if (kUnknown_SkAlphaType == dstInfo.alphaType()) {
return false;
}
if (kOpaque_SkAlphaType != dstInfo.alphaType()) {
SkCodecPrintf(
"Warning: an opaque image should be decoded as opaque "
"- it is being decoded as non-opaque, which will draw slower\n");
}
switch (dstInfo.colorType()) {
case kRGBA_1010102_SkColorType:
this->setSrcXformFormat(skcms_PixelFormat_RGBA_1010102);
return true;
case kRGBA_8888_SkColorType:
this->setSrcXformFormat(skcms_PixelFormat_RGBA_8888);
return true;
default:
return false;
}
}
/*
* Performs the JpegR decode
*/
SkCodec::Result SkJpegRCodec::onGetPixels(const SkImageInfo& dstInfo,
void* dst,
size_t dstRowBytes,
const Options& options,
int* rowsDecoded) {
if (options.fSubset) {
return kUnimplemented;
}
const SkColorType dstColorType = dstInfo.colorType();
if (dstColorType != kRGBA_1010102_SkColorType && dstColorType != kRGBA_8888_SkColorType) {
// We only support RGBA1010102 and RGBA8888 colors
return kUnimplemented;
}
if (this->dimensions() != dstInfo.dimensions()) {
// No Scaling
return kUnimplemented;
}
if (dstRowBytes != static_cast<size_t>(this->dimensions().width()) * 4) {
// TODO: Add stride handling
return kUnimplemented;
}
const bool decodeSDR = (dstColorType == kRGBA_8888_SkColorType);
jpegr_compressed_struct compressedImage;
jpegr_uncompressed_struct decompressedImage;
compressedImage.data = (void*)fData->data();
compressedImage.length = fData->size();
decompressedImage.data = dst;
if (fRecoveryMap->decodeJPEGR(&compressedImage, &decompressedImage, nullptr, decodeSDR) != 0) {
return kInternalError;
}
*rowsDecoded = dstInfo.height();
return kSuccess;
}
#endif // SK_CODEC_DECODES_JPEGR