Coverage Report

Created: 2026-01-10 06:22

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/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