/src/exiv2/src/photoshop.cpp
Line | Count | Source |
1 | | #include "config.h" |
2 | | |
3 | | #include "photoshop.hpp" |
4 | | |
5 | | #include "enforce.hpp" |
6 | | #include "image.hpp" |
7 | | #include "safe_op.hpp" |
8 | | |
9 | | #include <cstring> |
10 | | |
11 | | #ifdef EXIV2_DEBUG_MESSAGES |
12 | | #include <iostream> |
13 | | #endif |
14 | | |
15 | | namespace Exiv2 { |
16 | | |
17 | 130M | bool Photoshop::isIrb(const byte* pPsData) { |
18 | 130M | if (pPsData == nullptr) { |
19 | 0 | return false; |
20 | 0 | } |
21 | 383M | return std::any_of(irbId_.begin(), irbId_.end(), [pPsData](auto id) { return memcmp(pPsData, id, 4) == 0; }); |
22 | 130M | } |
23 | | |
24 | 111k | bool Photoshop::valid(const byte* pPsData, size_t sizePsData) { |
25 | 111k | const byte* record = nullptr; |
26 | 111k | uint32_t sizeIptc = 0; |
27 | 111k | uint32_t sizeHdr = 0; |
28 | 111k | const byte* pCur = pPsData; |
29 | 111k | const byte* pEnd = pPsData + sizePsData; |
30 | 111k | int ret = 0; |
31 | 86.7M | while (pCur < pEnd && 0 == (ret = Photoshop::locateIptcIrb(pCur, (pEnd - pCur), &record, sizeHdr, sizeIptc))) { |
32 | 86.6M | pCur = record + sizeHdr + sizeIptc + (sizeIptc & 1); |
33 | 86.6M | } |
34 | 111k | return ret >= 0; |
35 | 111k | } |
36 | | |
37 | | // Todo: Generalised from JpegBase::locateIptcData without really understanding |
38 | | // the format (in particular the header). So it remains to be confirmed |
39 | | // if this also makes sense for psTag != Photoshop::iptc |
40 | | int Photoshop::locateIrb(const byte* pPsData, size_t sizePsData, uint16_t psTag, const byte** record, uint32_t& sizeHdr, |
41 | 86.8M | uint32_t& sizeData) { |
42 | 86.8M | if (sizePsData < 12) { |
43 | 3.07k | return 3; |
44 | 3.07k | } |
45 | | |
46 | | // Used for error checking |
47 | 86.8M | size_t position = 0; |
48 | | #ifdef EXIV2_DEBUG_MESSAGES |
49 | | std::cerr << "Photoshop::locateIrb: "; |
50 | | #endif |
51 | | // Data should follow Photoshop format, if not exit |
52 | 130M | while (position <= (sizePsData - 12) && isIrb(pPsData + position)) { |
53 | 130M | const byte* hrd = pPsData + position; |
54 | 130M | position += 4; |
55 | 130M | uint16_t type = getUShort(pPsData + position, bigEndian); |
56 | 130M | position += 2; |
57 | | #ifdef EXIV2_DEBUG_MESSAGES |
58 | | std::cerr << "0x" << std::hex << type << std::dec << " "; |
59 | | #endif |
60 | | // Pascal string is padded to have an even size (including size byte) |
61 | 130M | byte psSize = pPsData[position] + 1; |
62 | 130M | psSize += (psSize & 1); |
63 | 130M | position += psSize; |
64 | 130M | if (position + 4 > sizePsData) { |
65 | | #ifdef EXIV2_DEBUG_MESSAGES |
66 | | std::cerr << "Warning: " |
67 | | << "Invalid or extended Photoshop IRB\n"; |
68 | | #endif |
69 | 38.0k | return -2; |
70 | 38.0k | } |
71 | 130M | uint32_t dataSize = getULong(pPsData + position, bigEndian); |
72 | 130M | position += 4; |
73 | 130M | if (dataSize > (sizePsData - position)) { |
74 | | #ifdef EXIV2_DEBUG_MESSAGES |
75 | | std::cerr << "Warning: " |
76 | | << "Invalid Photoshop IRB data size " << dataSize << " or extended Photoshop IRB\n"; |
77 | | #endif |
78 | 58.3k | return -2; |
79 | 58.3k | } |
80 | | #ifdef EXIV2_DEBUG_MESSAGES |
81 | | if ((dataSize & 1) && position + dataSize == sizePsData) { |
82 | | std::cerr << "Warning: " |
83 | | << "Photoshop IRB data is not padded to even size\n"; |
84 | | } |
85 | | #endif |
86 | 130M | if (type == psTag) { |
87 | | #ifdef EXIV2_DEBUG_MESSAGES |
88 | | std::cerr << "ok\n"; |
89 | | #endif |
90 | 86.7M | sizeData = dataSize; |
91 | 86.7M | sizeHdr = psSize + 10; |
92 | 86.7M | *record = hrd; |
93 | 86.7M | return 0; |
94 | 86.7M | } |
95 | | // Data size is also padded to be even |
96 | 43.4M | position += dataSize + (dataSize & 1); |
97 | 43.4M | } |
98 | | #ifdef EXIV2_DEBUG_MESSAGES |
99 | | std::cerr << "pPsData doesn't start with '8BIM'\n"; |
100 | | #endif |
101 | 15.7k | if (position < sizePsData) { |
102 | | #ifdef EXIV2_DEBUG_MESSAGES |
103 | | std::cerr << "Warning: " |
104 | | << "Invalid or extended Photoshop IRB\n"; |
105 | | #endif |
106 | 15.3k | return -2; |
107 | 15.3k | } |
108 | 429 | return 3; |
109 | 15.7k | } |
110 | | |
111 | | int Photoshop::locateIptcIrb(const byte* pPsData, size_t sizePsData, const byte** record, uint32_t& sizeHdr, |
112 | 86.8M | uint32_t& sizeData) { |
113 | 86.8M | return locateIrb(pPsData, sizePsData, iptc_, record, sizeHdr, sizeData); |
114 | 86.8M | } |
115 | | |
116 | | int Photoshop::locatePreviewIrb(const byte* pPsData, size_t sizePsData, const byte** record, uint32_t& sizeHdr, |
117 | 0 | uint32_t& sizeData) { |
118 | 0 | return locateIrb(pPsData, sizePsData, preview_, record, sizeHdr, sizeData); |
119 | 0 | } |
120 | | |
121 | 1.55k | DataBuf Photoshop::setIptcIrb(const byte* pPsData, size_t sizePsData, const IptcData& iptcData) { |
122 | | #ifdef EXIV2_DEBUG_MESSAGES |
123 | | std::cerr << "IRB block at the beginning of Photoshop::setIptcIrb\n"; |
124 | | if (sizePsData == 0) |
125 | | std::cerr << " None.\n"; |
126 | | else |
127 | | hexdump(std::cerr, pPsData, sizePsData); |
128 | | #endif |
129 | 1.55k | const byte* record = pPsData; |
130 | 1.55k | uint32_t sizeIptc = 0; |
131 | 1.55k | uint32_t sizeHdr = 0; |
132 | 1.55k | DataBuf rc; |
133 | 1.55k | if (0 > Photoshop::locateIptcIrb(pPsData, sizePsData, &record, sizeHdr, sizeIptc)) { |
134 | 166 | return rc; |
135 | 166 | } |
136 | | |
137 | 1.38k | Blob psBlob; |
138 | 1.38k | const auto sizeFront = static_cast<size_t>(record - pPsData); |
139 | | // Write data before old record. |
140 | 1.38k | if (sizePsData > 0 && sizeFront > 0) { |
141 | 102 | append(psBlob, pPsData, sizeFront); |
142 | 102 | } |
143 | | |
144 | | // Write new iptc record if we have it |
145 | 1.38k | if (DataBuf rawIptc = IptcParser::encode(iptcData); !rawIptc.empty()) { |
146 | 412 | std::array<byte, 12> tmpBuf; |
147 | 412 | std::copy_n(Photoshop::irbId_.front(), 4, tmpBuf.begin()); |
148 | 412 | us2Data(tmpBuf.data() + 4, iptc_, bigEndian); |
149 | 412 | tmpBuf[6] = 0; |
150 | 412 | tmpBuf[7] = 0; |
151 | 412 | ul2Data(tmpBuf.data() + 8, static_cast<uint32_t>(rawIptc.size()), bigEndian); |
152 | 412 | append(psBlob, tmpBuf.data(), 12); |
153 | 412 | append(psBlob, rawIptc.c_data(), rawIptc.size()); |
154 | | // Data is padded to be even (but not included in size) |
155 | 412 | if (rawIptc.size() & 1) |
156 | 227 | psBlob.push_back(0x00); |
157 | 412 | } |
158 | | |
159 | | // Write existing stuff after record, skip the current and all remaining IPTC blocks |
160 | 1.38k | size_t pos = sizeFront; |
161 | 1.38k | auto nextSizeData = Safe::add<long>(static_cast<long>(sizePsData), -static_cast<long>(pos)); |
162 | 1.38k | Internal::enforce(nextSizeData >= 0, ErrorCode::kerCorruptedMetadata); |
163 | 44.6k | while (0 == Photoshop::locateIptcIrb(pPsData + pos, nextSizeData, &record, sizeHdr, sizeIptc)) { |
164 | 43.2k | const auto newPos = static_cast<size_t>(record - pPsData); |
165 | 43.2k | if (newPos > pos) { // Copy data up to the IPTC IRB |
166 | 19.3k | append(psBlob, pPsData + pos, newPos - pos); |
167 | 19.3k | } |
168 | 43.2k | pos = newPos + sizeHdr + sizeIptc + (sizeIptc & 1); // Skip the IPTC IRB |
169 | 43.2k | nextSizeData = Safe::add<long>(static_cast<long>(sizePsData), -static_cast<long>(pos)); |
170 | 43.2k | Internal::enforce(nextSizeData >= 0, ErrorCode::kerCorruptedMetadata); |
171 | 43.2k | } |
172 | 1.38k | if (pos < sizePsData) { |
173 | 530 | append(psBlob, pPsData + pos, sizePsData - pos); |
174 | 530 | } |
175 | | |
176 | | // Data is rounded to be even |
177 | 1.38k | if (!psBlob.empty()) |
178 | 846 | rc = DataBuf(psBlob.data(), psBlob.size()); |
179 | | #ifdef EXIV2_DEBUG_MESSAGES |
180 | | std::cerr << "IRB block at the end of Photoshop::setIptcIrb\n"; |
181 | | if (rc.empty()) |
182 | | std::cerr << " None.\n"; |
183 | | else |
184 | | hexdump(std::cerr, rc.c_data(), rc.size()); |
185 | | #endif |
186 | 1.38k | return rc; |
187 | 1.55k | } |
188 | | |
189 | | } // namespace Exiv2 |