#include "photoshop.hpp" #include "enforce.hpp" #include "image.hpp" #include "safe_op.hpp" #include namespace Exiv2 { bool Photoshop::isIrb(const byte* data) { if (data == nullptr) { return false; } return std::any_of(irbId_.begin(), irbId_.end(), [data](auto id) { return memcmp(data, id, 4) == 0; }); } bool Photoshop::valid(const byte* pPsData, size_t sizePsData) { const byte* record = nullptr; uint32_t sizeIptc = 0; uint32_t sizeHdr = 0; const byte* pCur = pPsData; const byte* pEnd = pPsData + sizePsData; int ret = 0; while (pCur < pEnd && 0 == (ret = Photoshop::locateIptcIrb(pCur, (pEnd - pCur), &record, sizeHdr, sizeIptc))) { pCur = record + sizeHdr + sizeIptc + (sizeIptc & 1); } return ret >= 0; } // Todo: Generalised from JpegBase::locateIptcData without really understanding // the format (in particular the header). So it remains to be confirmed // if this also makes sense for psTag != Photoshop::iptc int Photoshop::locateIrb(const byte* pPsData, size_t sizePsData, uint16_t psTag, const byte** record, uint32_t& sizeHdr, uint32_t& sizeData) { if (sizePsData < 12) { return 3; } // Used for error checking size_t position = 0; #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "Photoshop::locateIrb: "; #endif // Data should follow Photoshop format, if not exit while (position <= (sizePsData - 12) && isIrb(pPsData + position)) { const byte* hrd = pPsData + position; position += 4; uint16_t type = getUShort(pPsData + position, bigEndian); position += 2; #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "0x" << std::hex << type << std::dec << " "; #endif // Pascal string is padded to have an even size (including size byte) byte psSize = pPsData[position] + 1; psSize += (psSize & 1); position += psSize; if (position + 4 > sizePsData) { #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "Warning: " << "Invalid or extended Photoshop IRB\n"; #endif return -2; } uint32_t dataSize = getULong(pPsData + position, bigEndian); position += 4; if (dataSize > (sizePsData - position)) { #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "Warning: " << "Invalid Photoshop IRB data size " << dataSize << " or extended Photoshop IRB\n"; #endif return -2; } #ifdef EXIV2_DEBUG_MESSAGES if ((dataSize & 1) && position + dataSize == sizePsData) { std::cerr << "Warning: " << "Photoshop IRB data is not padded to even size\n"; } #endif if (type == psTag) { #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "ok\n"; #endif sizeData = dataSize; sizeHdr = psSize + 10; *record = hrd; return 0; } // Data size is also padded to be even position += dataSize + (dataSize & 1); } #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "pPsData doesn't start with '8BIM'\n"; #endif if (position < sizePsData) { #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "Warning: " << "Invalid or extended Photoshop IRB\n"; #endif return -2; } return 3; } int Photoshop::locateIptcIrb(const byte* pPsData, size_t sizePsData, const byte** record, uint32_t& sizeHdr, uint32_t& sizeData) { return locateIrb(pPsData, sizePsData, iptc_, record, sizeHdr, sizeData); } int Photoshop::locatePreviewIrb(const byte* pPsData, size_t sizePsData, const byte** record, uint32_t& sizeHdr, uint32_t& sizeData) { return locateIrb(pPsData, sizePsData, preview_, record, sizeHdr, sizeData); } DataBuf Photoshop::setIptcIrb(const byte* pPsData, size_t sizePsData, const IptcData& iptcData) { #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "IRB block at the beginning of Photoshop::setIptcIrb\n"; if (sizePsData == 0) std::cerr << " None.\n"; else hexdump(std::cerr, pPsData, sizePsData); #endif const byte* record = pPsData; uint32_t sizeIptc = 0; uint32_t sizeHdr = 0; DataBuf rc; if (0 > Photoshop::locateIptcIrb(pPsData, sizePsData, &record, sizeHdr, sizeIptc)) { return rc; } Blob psBlob; const auto sizeFront = static_cast(record - pPsData); // Write data before old record. if (sizePsData > 0 && sizeFront > 0) { append(psBlob, pPsData, sizeFront); } // Write new iptc record if we have it if (DataBuf rawIptc = IptcParser::encode(iptcData); !rawIptc.empty()) { std::array tmpBuf; std::copy_n(Photoshop::irbId_.front(), 4, tmpBuf.data()); us2Data(tmpBuf.data() + 4, iptc_, bigEndian); tmpBuf[6] = 0; tmpBuf[7] = 0; ul2Data(tmpBuf.data() + 8, static_cast(rawIptc.size()), bigEndian); append(psBlob, tmpBuf.data(), 12); append(psBlob, rawIptc.c_data(), rawIptc.size()); // Data is padded to be even (but not included in size) if (rawIptc.size() & 1) psBlob.push_back(0x00); } // Write existing stuff after record, skip the current and all remaining IPTC blocks size_t pos = sizeFront; auto nextSizeData = Safe::add(static_cast(sizePsData), -static_cast(pos)); Internal::enforce(nextSizeData >= 0, ErrorCode::kerCorruptedMetadata); while (0 == Photoshop::locateIptcIrb(pPsData + pos, nextSizeData, &record, sizeHdr, sizeIptc)) { const auto newPos = static_cast(record - pPsData); if (newPos > pos) { // Copy data up to the IPTC IRB append(psBlob, pPsData + pos, newPos - pos); } pos = newPos + sizeHdr + sizeIptc + (sizeIptc & 1); // Skip the IPTC IRB nextSizeData = Safe::add(static_cast(sizePsData), -static_cast(pos)); Internal::enforce(nextSizeData >= 0, ErrorCode::kerCorruptedMetadata); } if (pos < sizePsData) { append(psBlob, pPsData + pos, sizePsData - pos); } // Data is rounded to be even if (!psBlob.empty()) rc = DataBuf(psBlob.data(), psBlob.size()); #ifdef EXIV2_DEBUG_MESSAGES std::cerr << "IRB block at the end of Photoshop::setIptcIrb\n"; if (rc.empty()) std::cerr << " None.\n"; else hexdump(std::cerr, rc.c_data(), rc.size()); #endif return rc; } } // namespace Exiv2