| /* |
| * 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; |