blob: bc139b49d6ef00580d20d02c0056fb8303c9b843 [file] [log] [blame]
/*
* Copyright 2023 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/SkJpegSourceMgr.h"
#include "include/core/SkData.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkStream.h"
#include "include/core/SkTypes.h"
#include "src/codec/SkCodecPriv.h"
#ifdef SK_CODEC_DECODES_JPEG_GAINMAPS
#include "src/codec/SkJpegConstants.h"
#include "src/codec/SkJpegSegmentScan.h"
#endif // SK_CODEC_DECODES_JPEG_GAINMAPS
////////////////////////////////////////////////////////////////////////////////////////////////////
// SkStream helpers.
/*
* Class that will will rewind an SkStream, and then restore it to its original position when it
* goes out of scope. If the SkStream is not seekable, then the stream will not be altered at all,
* and will return false from canRestore.
*/
class ScopedSkStreamRestorer {
public:
ScopedSkStreamRestorer(SkStream* stream) : fStream(stream), fPosition(stream->getPosition()) {
if (!fStream->rewind()) {
SkCodecPrintf("Failed to rewind decoder stream.\n");
}
}
~ScopedSkStreamRestorer() {
if (!fStream->seek(fPosition)) {
SkCodecPrintf("Failed to restore decoder stream.\n");
}
}
private:
SkStream* const fStream;
const size_t fPosition;
};
////////////////////////////////////////////////////////////////////////////////////////////////////
// SkJpegMemorySourceMgr
class SkJpegMemorySourceMgr : public SkJpegSourceMgr {
public:
SkJpegMemorySourceMgr(SkStream* stream) : SkJpegSourceMgr(stream) {}
~SkJpegMemorySourceMgr() override {}
void initSource(const uint8_t*& nextInputByte, size_t& bytesInBuffer) override {
nextInputByte = reinterpret_cast<const uint8_t*>(fStream->getMemoryBase());
bytesInBuffer = static_cast<size_t>(fStream->getLength());
}
bool fillInputBuffer(const uint8_t*& nextInputByte, size_t& bytesInBuffer) override {
// The whole JPEG data is expected to reside in the supplied memory buffer, so any request
// for more data beyond the given buffer size is treated as an error.
SkCodecPrintf("Asked to re-fill a memory-mapped stream.\n");
return false;
}
bool skipInputBytes(size_t bytesToSkip,
const uint8_t*& nextInputByte,
size_t& bytesInBuffer) override {
if (bytesToSkip > bytesInBuffer) {
SkCodecPrintf("Asked to read past end of a memory-mapped stream.\n");
return false;
}
nextInputByte += bytesToSkip;
bytesInBuffer -= bytesToSkip;
return true;
}
#ifdef SK_CODEC_DECODES_JPEG_GAINMAPS
const std::vector<SkJpegSegment>& getAllSegments() override {
if (fScanner) {
return fScanner->getSegments();
}
fScanner = std::make_unique<SkJpegSegmentScanner>(kJpegMarkerEndOfImage);
fScanner->onBytes(fStream->getMemoryBase(), fStream->getLength());
return fScanner->getSegments();
}
sk_sp<SkData> getSubsetData(size_t offset, size_t size, bool* wasCopied) override {
if (offset > fStream->getLength() || size > fStream->getLength() - offset) {
return nullptr;
}
if (wasCopied) {
*wasCopied = false;
}
return SkData::MakeWithoutCopy(
reinterpret_cast<const uint8_t*>(fStream->getMemoryBase()) + offset, size);
}
sk_sp<SkData> getSegmentParameters(const SkJpegSegment& segment) override {
const uint8_t* base =
reinterpret_cast<const uint8_t*>(fStream->getMemoryBase()) + segment.offset;
SkASSERT(segment.offset < fStream->getLength());
SkASSERT(kJpegMarkerCodeSize + segment.parameterLength <=
fStream->getLength() - segment.offset);
// Read the marker and verify it matches `segment`.
SkASSERT(base[0] == 0xFF);
SkASSERT(base[1] == segment.marker);
// Read the parameter length and verify it matches `segment`.
SkASSERT(256 * base[2] + base[3] == segment.parameterLength);
if (segment.parameterLength <= kJpegSegmentParameterLengthSize) {
return nullptr;
}
// Read the remainder of the segment.
return SkData::MakeWithoutCopy(base + kJpegMarkerCodeSize + kJpegSegmentParameterLengthSize,
segment.parameterLength - kJpegSegmentParameterLengthSize);
}
#endif // SK_CODEC_DECODES_JPEG_GAINMAPS
};
////////////////////////////////////////////////////////////////////////////////////////////////////
// SkJpegBufferedSourceMgr
class SkJpegBufferedSourceMgr : public SkJpegSourceMgr {
public:
SkJpegBufferedSourceMgr(SkStream* stream, size_t bufferSize) : SkJpegSourceMgr(stream) {
fBuffer = SkData::MakeUninitialized(bufferSize);
}
~SkJpegBufferedSourceMgr() override {}
void initSource(const uint8_t*& nextInputByte, size_t& bytesInBuffer) override {
nextInputByte = fBuffer->bytes();
bytesInBuffer = 0;
}
bool fillInputBuffer(const uint8_t*& nextInputByte, size_t& bytesInBuffer) override {
size_t bytesRead = fStream->read(fBuffer->writable_data(), fBuffer->size());
if (bytesRead == 0) {
// Fail if we read zero bytes (libjpeg will accept any non-zero number of bytes).
SkCodecPrintf("Hit end of file reading a buffered stream.\n");
return false;
}
nextInputByte = fBuffer->bytes();
bytesInBuffer = bytesRead;
return true;
}
bool skipInputBytes(size_t bytesToSkip,
const uint8_t*& nextInputByte,
size_t& bytesInBuffer) override {
// Skip through the already-read (or already in memory) buffer.
if (bytesToSkip <= bytesInBuffer) {
nextInputByte += bytesToSkip;
bytesInBuffer -= bytesToSkip;
return true;
}
bytesToSkip -= bytesInBuffer;
// Fail if we skip past the end of the stream.
if (fStream->skip(bytesToSkip) != bytesToSkip) {
SkCodecPrintf("Failed to skip through buffered stream.\n");
return false;
}
bytesInBuffer = 0;
nextInputByte = fBuffer->bytes();
return true;
}
#ifdef SK_CODEC_DECODES_JPEG_GAINMAPS
const std::vector<SkJpegSegment>& getAllSegments() override {
if (fScanner) {
return fScanner->getSegments();
}
ScopedSkStreamRestorer streamRestorer(fStream);
fScanner = std::make_unique<SkJpegSegmentScanner>(kJpegMarkerEndOfImage);
while (!fScanner->isDone() && !fScanner->hadError()) {
constexpr size_t kBufferSize = 1024;
uint8_t buffer[kBufferSize];
size_t bytesRead = fStream->read(buffer, kBufferSize);
if (bytesRead == 0) {
SkCodecPrintf("Unexpected EOF.\n");
break;
}
fScanner->onBytes(buffer, bytesRead);
}
return fScanner->getSegments();
}
sk_sp<SkData> getSubsetData(size_t offset, size_t size, bool* wasCopied) override {
ScopedSkStreamRestorer streamRestorer(fStream);
if (!fStream->seek(offset)) {
SkCodecPrintf("Failed to seek to subset stream position.\n");
return nullptr;
}
sk_sp<SkData> data = SkData::MakeUninitialized(size);
if (fStream->read(data->writable_data(), size) != size) {
SkCodecPrintf("Failed to read subset stream data.\n");
return nullptr;
}
if (wasCopied) {
*wasCopied = true;
}
return data;
}
sk_sp<SkData> getSegmentParameters(const SkJpegSegment& segment) override {
// If the segment's parameter length isn't longer than the two bytes for the length,
// early-out early-out.
if (segment.parameterLength <= kJpegSegmentParameterLengthSize) {
return nullptr;
}
// Seek to the start of the segment.
ScopedSkStreamRestorer streamRestorer(fStream);
if (!fStream->seek(segment.offset)) {
SkCodecPrintf("Failed to seek to segment\n");
return nullptr;
}
// Read the marker and verify it matches `segment`.
uint8_t markerCode[kJpegMarkerCodeSize] = {0};
if (fStream->read(markerCode, kJpegMarkerCodeSize) != kJpegMarkerCodeSize) {
SkCodecPrintf("Failed to read segment marker code\n");
return nullptr;
}
SkASSERT(markerCode[0] == 0xFF);
SkASSERT(markerCode[1] == segment.marker);
// Read the parameter length and verify it matches `segment`.
uint8_t parameterLength[kJpegSegmentParameterLengthSize] = {0};
if (fStream->read(parameterLength, kJpegSegmentParameterLengthSize) !=
kJpegSegmentParameterLengthSize) {
SkCodecPrintf("Failed to read parameter length\n");
return nullptr;
}
SkASSERT(256 * parameterLength[0] + parameterLength[1] == segment.parameterLength);
// Read the remainder of the segment.
size_t sizeToRead = segment.parameterLength - kJpegSegmentParameterLengthSize;
auto result = SkData::MakeUninitialized(sizeToRead);
if (fStream->read(result->writable_data(), sizeToRead) != sizeToRead) {
return nullptr;
}
return result;
}
#endif // SK_CODEC_DECODES_JPEG_GAINMAPS
private:
sk_sp<SkData> fBuffer;
};
////////////////////////////////////////////////////////////////////////////////////////////////////
// SkJpegUnseekableSourceMgr
#ifdef SK_CODEC_DECODES_JPEG_GAINMAPS
/*
* This class implements SkJpegSourceMgr for a stream that cannot seek or rewind. It scans the data
* as it is presented to the decoder. This allows it to track the position of segments, so that it
* can extract subsets at a specific offset (e.g, relative to the EndOfImage segment for JpegR or
* relative to an MPF segment for MPF).
*/
class SkJpegUnseekableSourceMgr : public SkJpegSourceMgr {
public:
SkJpegUnseekableSourceMgr(SkStream* stream, size_t bufferSize) : SkJpegSourceMgr(stream) {
fBuffer = SkData::MakeUninitialized(bufferSize);
fScanner = std::make_unique<SkJpegSegmentScanner>(kJpegMarkerEndOfImage);
}
~SkJpegUnseekableSourceMgr() override {}
void initSource(const uint8_t*& nextInputByte, size_t& bytesInBuffer) override {
nextInputByte = fBuffer->bytes();
bytesInBuffer = 0;
}
bool fillInputBuffer(const uint8_t*& nextInputByte, size_t& bytesInBuffer) override {
if (!readToBufferAndScan(fBuffer->size())) {
SkCodecPrintf("Failure filling unseekable input buffer.\n");
return false;
}
nextInputByte = fBuffer->bytes();
bytesInBuffer = fLastReadSize;
return true;
}
bool skipInputBytes(size_t bytesToSkip,
const uint8_t*& nextInputByte,
size_t& bytesInBuffer) override {
// Skip through the already-read (or already in memory) buffer.
if (bytesToSkip <= bytesInBuffer) {
nextInputByte += bytesToSkip;
bytesInBuffer -= bytesToSkip;
return true;
}
bytesToSkip -= bytesInBuffer;
// Read the remaining bytes to skip into fBuffer and feed them into fScanner.
while (bytesToSkip > 0) {
if (!readToBufferAndScan(std::min(bytesToSkip, fBuffer->size()))) {
SkCodecPrintf("Failure filling unseekable input buffer.\n");
return false;
}
bytesToSkip -= fLastReadSize;
}
// Indicate to libjpeg that it it needs to call fillInputBuffer.
bytesInBuffer = 0;
nextInputByte = fBuffer->bytes();
return true;
}
const std::vector<SkJpegSegment>& getAllSegments() override {
while (!fScanner->isDone() && !fScanner->hadError()) {
if (!readToBufferAndScan(fBuffer->size())) {
SkCodecPrintf("Failure finishing unseekable input buffer.\n");
break;
}
}
return fScanner->getSegments();
}
sk_sp<SkData> getSubsetData(size_t offset, size_t size, bool* wasCopied) override {
// If we haven't reached the EndOfImage, then we are throwing away the base image before
// decoding it. This is only reasonable for tests.
if (!fScanner->isDone()) {
SkCodecPrintf("getSubsetData is prematurely terminating scan.\n");
}
// If we have read past offset, we can never get that data back again.
if (offset < fLastReadOffset) {
SkCodecPrintf("Requested that is gone.\n");
return nullptr;
}
// Allocate the memory to return, and indicate that the result is a copy.
sk_sp<SkData> subsetData = SkData::MakeUninitialized(size);
uint8_t* subsetDataCurrent = reinterpret_cast<uint8_t*>(subsetData->writable_data());
// Determine the relationship between the offset we're reading from and |fBuffer|.
size_t offsetIntoBuffer = offset - fLastReadOffset;
if (offsetIntoBuffer >= fLastReadSize) {
// We have to skip past |fBuffer| to get to |offset|.
fLastReadOffset += fLastReadSize;
fLastReadSize = 0;
// Skip any additional bytes needed to get to |offset|.
size_t bytesToSkip = offset - fLastReadOffset;
while (bytesToSkip > 0) {
size_t bytesSkipped = fStream->skip(bytesToSkip);
if (bytesSkipped == 0) {
SkCodecPrintf("Failed to skip bytes before subset.\n");
return nullptr;
}
bytesToSkip -= bytesSkipped;
fLastReadOffset += bytesSkipped;
}
} else {
// This assert is to emphatically document the side of the branch we're on.
SkASSERT(offsetIntoBuffer < fLastReadSize);
// Some of the data we want to copy has already been read into |fBuffer|. Copy that data
// to |subsetData|
size_t bytesToReadFromBuffer = std::min(fLastReadSize - offsetIntoBuffer, size);
memcpy(subsetDataCurrent, fBuffer->bytes() + offsetIntoBuffer, bytesToReadFromBuffer);
size -= bytesToReadFromBuffer;
subsetDataCurrent += bytesToReadFromBuffer;
// If all of the data that we needed was in |fBuffer|, then return early.
if (size == 0) {
if (wasCopied) {
*wasCopied = true;
}
return subsetData;
}
// We will now have to read beyond |fBuffer|, so reset it.
fLastReadOffset += fLastReadSize;
fLastReadSize = 0;
}
// Read the remaining data from |fStream|.
while (size > 0) {
size_t bytesRead = fStream->read(subsetDataCurrent, size);
if (bytesRead == 0) {
SkCodecPrintf("Failed to read subset stream data.\n");
return nullptr;
}
size -= bytesRead;
subsetDataCurrent += bytesRead;
fLastReadOffset += bytesRead;
}
if (wasCopied) {
*wasCopied = true;
}
return subsetData;
}
sk_sp<SkData> getSegmentParameters(const SkJpegSegment& segment) override {
// The only way to implement this for an unseekable stream is to record the parameters as
// they are scanned.
return nullptr;
}
private:
// Read the specified number of bytes into fBuffer and feed them to fScanner. The number of
// bytes must not be larger than fBuffer's size.
bool readToBufferAndScan(size_t bytesToRead) {
SkASSERT(bytesToRead <= fBuffer->size());
fLastReadOffset += fLastReadSize;
fLastReadSize = fStream->read(fBuffer->writable_data(), bytesToRead);
if (fLastReadSize == 0) {
SkCodecPrintf("Hit end of file reading an unseekable stream.\n");
return false;
}
fScanner->onBytes(fBuffer->bytes(), fLastReadSize);
return true;
}
sk_sp<SkData> fBuffer;
// The number of bytes that were most recently read into fBuffer (this can be less than the size
// of fBuffer).
size_t fLastReadSize = 0;
// The offset into the stream (total number of bytes read) at the time of our most recent read
// into fBuffer.
size_t fLastReadOffset = 0;
};
#endif // SK_CODEC_DECODES_JPEG_GAINMAPS
////////////////////////////////////////////////////////////////////////////////////////////////////
// SkJpegSourceMgr
// static
std::unique_ptr<SkJpegSourceMgr> SkJpegSourceMgr::Make(SkStream* stream, size_t bufferSize) {
#ifdef SK_CODEC_DECODES_JPEG_GAINMAPS
if (!stream->hasPosition()) {
return std::make_unique<SkJpegUnseekableSourceMgr>(stream, bufferSize);
}
#endif
if (stream->hasLength() && stream->getMemoryBase()) {
return std::make_unique<SkJpegMemorySourceMgr>(stream);
}
return std::make_unique<SkJpegBufferedSourceMgr>(stream, bufferSize);
}
SkJpegSourceMgr::SkJpegSourceMgr(SkStream* stream) : fStream(stream) {}
SkJpegSourceMgr::~SkJpegSourceMgr() = default;