1*c8dee2aaSAndroid Build Coastguard Worker /* 2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2023 Google Inc. 3*c8dee2aaSAndroid Build Coastguard Worker * 4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be 5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file. 6*c8dee2aaSAndroid Build Coastguard Worker */ 7*c8dee2aaSAndroid Build Coastguard Worker 8*c8dee2aaSAndroid Build Coastguard Worker #ifndef SkTiffUtility_codec_DEFINED 9*c8dee2aaSAndroid Build Coastguard Worker #define SkTiffUtility_codec_DEFINED 10*c8dee2aaSAndroid Build Coastguard Worker 11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkData.h" 12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRefCnt.h" 13*c8dee2aaSAndroid Build Coastguard Worker 14*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef> 15*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint> 16*c8dee2aaSAndroid Build Coastguard Worker #include <memory> 17*c8dee2aaSAndroid Build Coastguard Worker 18*c8dee2aaSAndroid Build Coastguard Worker namespace SkTiff { 19*c8dee2aaSAndroid Build Coastguard Worker 20*c8dee2aaSAndroid Build Coastguard Worker // Constants for endian signature. 21*c8dee2aaSAndroid Build Coastguard Worker inline constexpr size_t kEndianSize = 4; 22*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint8_t kEndianBig[kEndianSize] = {'M', 'M', 0, 42}; 23*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint8_t kEndianLittle[kEndianSize] = {'I', 'I', 42, 0}; 24*c8dee2aaSAndroid Build Coastguard Worker 25*c8dee2aaSAndroid Build Coastguard Worker // Constants for types. 26*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeUnsignedByte = 1; 27*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeAsciiString = 2; 28*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeUnsignedShort = 3; 29*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeUnsignedLong = 4; 30*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeUnsignedRational = 5; 31*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeSignedByte = 6; 32*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeUndefined = 7; 33*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeSignedShort = 8; 34*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeSignedLong = 9; 35*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeSignedRational = 10; 36*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeSingleFloat = 11; 37*c8dee2aaSAndroid Build Coastguard Worker inline constexpr uint16_t kTypeDoubleFloat = 12; 38*c8dee2aaSAndroid Build Coastguard Worker 39*c8dee2aaSAndroid Build Coastguard Worker inline constexpr size_t kSizeEntry = 12; 40*c8dee2aaSAndroid Build Coastguard Worker inline constexpr size_t kSizeShort = 2; 41*c8dee2aaSAndroid Build Coastguard Worker inline constexpr size_t kSizeLong = 4; 42*c8dee2aaSAndroid Build Coastguard Worker 43*c8dee2aaSAndroid Build Coastguard Worker /* 44*c8dee2aaSAndroid Build Coastguard Worker * Helper function for parsing a Tiff Image File Directory (IFD) structure. This structure is used 45*c8dee2aaSAndroid Build Coastguard Worker * by EXIF tags, multi-picture, and maker note metadata. 46*c8dee2aaSAndroid Build Coastguard Worker */ 47*c8dee2aaSAndroid Build Coastguard Worker class ImageFileDirectory { 48*c8dee2aaSAndroid Build Coastguard Worker public: 49*c8dee2aaSAndroid Build Coastguard Worker /* 50*c8dee2aaSAndroid Build Coastguard Worker * Parse |data| to read the endian-ness into |outLittleEndian| and the IFD offset into 51*c8dee2aaSAndroid Build Coastguard Worker * |outIfdOffset|. Return true if the endian-ness was successfully parsed and there was 52*c8dee2aaSAndroid Build Coastguard Worker * the IFD offset was read. 53*c8dee2aaSAndroid Build Coastguard Worker */ 54*c8dee2aaSAndroid Build Coastguard Worker static bool ParseHeader(const SkData* data, bool* outLittleEndian, uint32_t* outIfdOffset); 55*c8dee2aaSAndroid Build Coastguard Worker 56*c8dee2aaSAndroid Build Coastguard Worker /* 57*c8dee2aaSAndroid Build Coastguard Worker * Create an object for parsing an IFD at offset |ifdOffset| inside |data| which has endianness 58*c8dee2aaSAndroid Build Coastguard Worker * indicated by |littleEndian|. If |allowTruncated| is true, then parse as much of |data| as is 59*c8dee2aaSAndroid Build Coastguard Worker * possible, otherwise reject any incomplete IFDs. 60*c8dee2aaSAndroid Build Coastguard Worker */ 61*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<ImageFileDirectory> MakeFromOffset(sk_sp<SkData> data, 62*c8dee2aaSAndroid Build Coastguard Worker bool littleEndian, 63*c8dee2aaSAndroid Build Coastguard Worker uint32_t ifdOffset, 64*c8dee2aaSAndroid Build Coastguard Worker bool allowTruncated = false); 65*c8dee2aaSAndroid Build Coastguard Worker 66*c8dee2aaSAndroid Build Coastguard Worker /* 67*c8dee2aaSAndroid Build Coastguard Worker * Return the number of entries. 68*c8dee2aaSAndroid Build Coastguard Worker */ getNumEntries()69*c8dee2aaSAndroid Build Coastguard Worker uint16_t getNumEntries() const { return fNumEntries; } 70*c8dee2aaSAndroid Build Coastguard Worker 71*c8dee2aaSAndroid Build Coastguard Worker /* 72*c8dee2aaSAndroid Build Coastguard Worker * Return the offset (within the specified SkData) of the next IFD in the list of IFDs. 73*c8dee2aaSAndroid Build Coastguard Worker */ nextIfdOffset()74*c8dee2aaSAndroid Build Coastguard Worker uint32_t nextIfdOffset() const { return fNextIfdOffset; } 75*c8dee2aaSAndroid Build Coastguard Worker 76*c8dee2aaSAndroid Build Coastguard Worker /* 77*c8dee2aaSAndroid Build Coastguard Worker * Return the tag, of a specific entry. 78*c8dee2aaSAndroid Build Coastguard Worker */ 79*c8dee2aaSAndroid Build Coastguard Worker uint16_t getEntryTag(uint16_t entryIndex) const; 80*c8dee2aaSAndroid Build Coastguard Worker 81*c8dee2aaSAndroid Build Coastguard Worker /* 82*c8dee2aaSAndroid Build Coastguard Worker * If |entryIndex| has type unsigned short (3), unsigned long (4), or signed rational (10), and 83*c8dee2aaSAndroid Build Coastguard Worker * count |count|, then populate |values| with the data for the tag and return true. Otherwise 84*c8dee2aaSAndroid Build Coastguard Worker * return false. 85*c8dee2aaSAndroid Build Coastguard Worker */ getEntryUnsignedShort(uint16_t entryIndex,uint32_t count,uint16_t * values)86*c8dee2aaSAndroid Build Coastguard Worker bool getEntryUnsignedShort(uint16_t entryIndex, uint32_t count, uint16_t* values) const { 87*c8dee2aaSAndroid Build Coastguard Worker return getEntryValuesGeneric(entryIndex, kTypeUnsignedShort, count, values); 88*c8dee2aaSAndroid Build Coastguard Worker } getEntryUnsignedLong(uint16_t entryIndex,uint32_t count,uint32_t * values)89*c8dee2aaSAndroid Build Coastguard Worker bool getEntryUnsignedLong(uint16_t entryIndex, uint32_t count, uint32_t* values) const { 90*c8dee2aaSAndroid Build Coastguard Worker return getEntryValuesGeneric(entryIndex, kTypeUnsignedLong, count, values); 91*c8dee2aaSAndroid Build Coastguard Worker } getEntrySignedRational(uint16_t entryIndex,uint32_t count,float * values)92*c8dee2aaSAndroid Build Coastguard Worker bool getEntrySignedRational(uint16_t entryIndex, uint32_t count, float* values) const { 93*c8dee2aaSAndroid Build Coastguard Worker return getEntryValuesGeneric(entryIndex, kTypeSignedRational, count, values); 94*c8dee2aaSAndroid Build Coastguard Worker } getEntryUnsignedRational(uint16_t entryIndex,uint32_t count,float * values)95*c8dee2aaSAndroid Build Coastguard Worker bool getEntryUnsignedRational(uint16_t entryIndex, uint32_t count, float* values) const { 96*c8dee2aaSAndroid Build Coastguard Worker return getEntryValuesGeneric(entryIndex, kTypeUnsignedRational, count, values); 97*c8dee2aaSAndroid Build Coastguard Worker } 98*c8dee2aaSAndroid Build Coastguard Worker 99*c8dee2aaSAndroid Build Coastguard Worker /* 100*c8dee2aaSAndroid Build Coastguard Worker * If |entryIndex| has type undefined (7), then return the bytes specified by the count field 101*c8dee2aaSAndroid Build Coastguard Worker * and the offset (read from the value field as an unsigned long). 102*c8dee2aaSAndroid Build Coastguard Worker */ 103*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkData> getEntryUndefinedData(uint16_t entryIndex) const; 104*c8dee2aaSAndroid Build Coastguard Worker 105*c8dee2aaSAndroid Build Coastguard Worker private: 106*c8dee2aaSAndroid Build Coastguard Worker static bool IsValidType(uint16_t type); 107*c8dee2aaSAndroid Build Coastguard Worker static size_t BytesForType(uint16_t type); 108*c8dee2aaSAndroid Build Coastguard Worker 109*c8dee2aaSAndroid Build Coastguard Worker ImageFileDirectory(sk_sp<SkData> data, 110*c8dee2aaSAndroid Build Coastguard Worker bool littleEndian, 111*c8dee2aaSAndroid Build Coastguard Worker uint32_t offset, 112*c8dee2aaSAndroid Build Coastguard Worker uint16_t ifdNumEntries, 113*c8dee2aaSAndroid Build Coastguard Worker uint32_t ifdNextOffset); 114*c8dee2aaSAndroid Build Coastguard Worker 115*c8dee2aaSAndroid Build Coastguard Worker /* 116*c8dee2aaSAndroid Build Coastguard Worker * Return the tag, type, count, and data for the specified entry. Return false if the type 117*c8dee2aaSAndroid Build Coastguard Worker * is invalid, or if the data in the IFD is out of bounds. 118*c8dee2aaSAndroid Build Coastguard Worker */ 119*c8dee2aaSAndroid Build Coastguard Worker bool getEntryRawData(uint16_t entryIndex, 120*c8dee2aaSAndroid Build Coastguard Worker uint16_t* outTag, 121*c8dee2aaSAndroid Build Coastguard Worker uint16_t* outType, 122*c8dee2aaSAndroid Build Coastguard Worker uint32_t* outCount, 123*c8dee2aaSAndroid Build Coastguard Worker const uint8_t** outData, 124*c8dee2aaSAndroid Build Coastguard Worker size_t* outDataSize) const; 125*c8dee2aaSAndroid Build Coastguard Worker 126*c8dee2aaSAndroid Build Coastguard Worker /* 127*c8dee2aaSAndroid Build Coastguard Worker * Helper function for assorted getTag functions. 128*c8dee2aaSAndroid Build Coastguard Worker */ 129*c8dee2aaSAndroid Build Coastguard Worker bool getEntryValuesGeneric(uint16_t entryIndex, 130*c8dee2aaSAndroid Build Coastguard Worker uint16_t type, 131*c8dee2aaSAndroid Build Coastguard Worker uint32_t count, 132*c8dee2aaSAndroid Build Coastguard Worker void* values) const; 133*c8dee2aaSAndroid Build Coastguard Worker 134*c8dee2aaSAndroid Build Coastguard Worker // The data that the IFD indexes into. 135*c8dee2aaSAndroid Build Coastguard Worker const sk_sp<SkData> fData; 136*c8dee2aaSAndroid Build Coastguard Worker 137*c8dee2aaSAndroid Build Coastguard Worker // True if the data is little endian. 138*c8dee2aaSAndroid Build Coastguard Worker const bool fLittleEndian; 139*c8dee2aaSAndroid Build Coastguard Worker 140*c8dee2aaSAndroid Build Coastguard Worker // The offset where the IFD starts. 141*c8dee2aaSAndroid Build Coastguard Worker const uint32_t fOffset; 142*c8dee2aaSAndroid Build Coastguard Worker 143*c8dee2aaSAndroid Build Coastguard Worker // The number of entries of the IFD (read from the first 2 bytes at the IFD offset). 144*c8dee2aaSAndroid Build Coastguard Worker const uint16_t fNumEntries; 145*c8dee2aaSAndroid Build Coastguard Worker 146*c8dee2aaSAndroid Build Coastguard Worker // The offset of the next IFD (read from the next 4 bytes after the IFD entries). 147*c8dee2aaSAndroid Build Coastguard Worker const uint32_t fNextIfdOffset; 148*c8dee2aaSAndroid Build Coastguard Worker }; 149*c8dee2aaSAndroid Build Coastguard Worker 150*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkTiff 151*c8dee2aaSAndroid Build Coastguard Worker 152*c8dee2aaSAndroid Build Coastguard Worker #endif 153