/* * Copyright 2023 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef SkJpegSegmentScan_codec_DEFINED #define SkJpegSegmentScan_codec_DEFINED #include "include/core/SkRefCnt.h" #include "src/codec/SkJpegConstants.h" #include #include #include #include class SkData; class SkStream; /* * A JPEG segment. */ struct SkJpegSegment { // The offset in bytes from the initial position to where this segment starts. size_t offset = 0; // The second byte of the marker code, which determines the segment type. uint8_t marker = 0; // The length of the parameters for this segment (including the two bytes used to specify // the length). uint16_t parameterLength = 0; }; /* * Class for scanning JPEG data. The JPEG format consists of a sequence of segments that begin with * a marker, with entropy-coded data in between. */ class SkJpegSegmentScanner { public: SkJpegSegmentScanner(uint8_t stopMarker = kJpegMarkerEndOfImage); bool isDone() const { return fState == State::kDone; } bool hadError() const { return fState == State::kError; } // Provide more bytes of data to the state machine. void onBytes(const void* data, size_t size); // Return the segments that have been retrieved so far. If an error was encountered, this will // include all segments up to the error. const std::vector& getSegments() const; // Helper function to retrieve the parameters for a segment. ScannedData must be the data that // was scanned to produce segment, starting at StartOfImage. static sk_sp GetParameters(const SkData* scannedData, const SkJpegSegment& segment); private: // The scanner is a state machine. State transitions happen when a byte is read. enum class State { // The initial state, before we read the 0xFF,0xD8,0xFF JPEG signature. kStartOfImageByte0, // We have read the 0xFF of the JPEG signature. kStartOfImageByte1, // We have read the 0xFF,0xD8 of the JPEG signature. The next byte should be the 0xFF that // completes the JPEG signature and starts the second marker (the one after StartOfImage). kSecondMarkerByte0, // We have read the full JPEG signature. The next byte should be the the second byte of the // second marker. kSecondMarkerByte1, // We have read a marker that does not stand alone. The next byte is the first byte of the // length of the parameters, kSegmentParamLengthByte0, // We have read the first byte of the length of the parameters (and it is stored in // |fSegmentParamLengthByte0|). The next byte we read is the second byte of the length of // the parameters. kSegmentParamLengthByte1, // We have read the full length of the parameters. The next |fSegmentParamBytesRemaining| // bytes are the parameters. kSegmentParam, // We have read a marker and (optionally) its parameters, and we are now reading entropy- // coded data. All subsequent bytes until we reach 0xFF are entropy-coded data. kEntropyCodedData, // We reached an 0xFF in entropy-coded data. If the next byte is 0x00 then we continue // reading entropy-coded data. If the next byte is 0xFF then we are reading fill data. // If the next byte is anything else then it is the second byte of a marker. kEntropyCodedDataSentinel, // We are reading fill data. If the next byte is 0xFF then we are still reading fill data, // otherwise the next byte is the second byte of a marker. kPostEntropyCodedDataFill, // We reached |fStopMarker| and have stopped tracking our state. kDone, // We hit an error somewhere and have given up. kError, }; State fState = State::kStartOfImageByte0; // Update state transition when a single byte is read. void onByte(uint8_t byte); // Perform the appropriate state transition for when a marker is read. This will set // |fCurrentSegmentMarker| and |fCurrentSegmentOffset|, and potentially call saveCurrentSegment. void onMarkerSecondByte(uint8_t byte); // Add a new entry in |segments| for |fCurrentSegmentMarker| and offset |fCurrentSegmentOffset| // and the specified length. void saveCurrentSegment(uint16_t length); static bool MarkerStandsAlone(uint8_t marker) { // These markers are TEM (0x01), RSTm (0xD0 through 0xD7), SOI (0xD8), and // EOI (0xD9). See section B.1.1.3, Marker assignments. return marker == 0x01 || (marker >= 0xD0 && marker <= 0xD9); } // Stop tracking state when we hit this marker. If this is 0x00, then never stop. const uint8_t fStopMarker; // The number of bytes that have been processed so far. size_t fOffset = 0; // If |fState| is kSegmentParamLengthByte1, then this is the value of the the previous byte. uint8_t fSegmentParamLengthByte0 = 0; // If |fState| is kSegmentParam, then this is the number of bytes reamining in the current // segment. size_t fSegmentParamBytesRemaining = 0; // The offset and marker for the segment started by the previous call to OnMarkerSecondByte. // These are re-set when SaveCurrentSegment is called. size_t fCurrentSegmentOffset = 0; uint8_t fCurrentSegmentMarker = 0; std::vector fSegments; }; #endif