// ***************************************************************** -*- C++ -*- /* * Copyright (C) 2004-2008 Andreas Huggel * * This program is part of the Exiv2 distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, 5th Floor, Boston, MA 02110-1301 USA. */ /* File: tiffimage.cpp Version: $Rev$ Author(s): Andreas Huggel (ahu) History: 15-Mar-06, ahu: created */ // ***************************************************************************** #include "rcsid.hpp" EXIV2_RCSID("@(#) $Id$") // ***************************************************************************** // included header files #ifdef _MSC_VER # include "exv_msvc.h" #else # include "exv_conf.h" #endif #include "tiffimage.hpp" #include "tiffimage_int.hpp" #include "tiffcomposite_int.hpp" #include "tiffvisitor_int.hpp" #include "makernote_int.hpp" #include "image.hpp" #include "error.hpp" #include "futils.hpp" #include "types.hpp" #include "i18n.h" // NLS support. // + standard includes #include #include #include #include #include /* -------------------------------------------------------------------------- Todo: + Review boundary checking, is it better to check the offsets? + Define and implement consistent error handling for recursive hierarchy + Make TiffImage a template StandardImage, which can be parametrized with a parser and the necessary checking functions to cover all types of images which need to be loaded completely. + TiffComponent: should it have end() and setEnd() or pData and size? + Can NewTiffCompFct and TiffCompFactoryFct be combined? + Create function is repeated when actually only the table changes. Fix it. + Is it easier (for writing) to combine all creation tables into one? + CR2 Makernotes don't seem to have a next pointer but Canon Jpeg Makernotes do. What a mess. (That'll become an issue when it comes to writing to CR2) + Sony makernotes in RAW files do not seem to have header like those in Jpegs. And maybe no next pointer either. + Filtering of large unknown tags: Should be moved to writing/encoding code and done only if really needed (i.e., if writing to a Jpeg segment) + Make Tiff parser completely standalone, depending only on very low level stuff from exiv2 in crwimage.* : + Fix CiffHeader according to TiffHeader + Combine Error(15) and Error(33), add format argument %1 + Search crwimage for todos, fix writeMetadata comment + rename loadStack to getPath for consistency -------------------------------------------------------------------------- */ // ***************************************************************************** // class member definitions namespace Exiv2 { using namespace Internal; TiffImage::TiffImage(BasicIo::AutoPtr io, bool /*create*/) : Image(ImageType::tiff, mdExif | mdIptc, io) { } // TiffImage::TiffImage int TiffImage::pixelWidth() const { ExifData::const_iterator imageWidth; if ((imageWidth = exifData_.findKey(Exiv2::ExifKey("Exif.Image.ImageWidth"))) != exifData_.end()) { return imageWidth->toLong(); } return 0; } int TiffImage::pixelHeight() const { ExifData::const_iterator imageHeight; if ((imageHeight = exifData_.findKey(Exiv2::ExifKey("Exif.Image.ImageLength"))) != exifData_.end()) { return imageHeight->toLong(); } return 0; } void TiffImage::setComment(const std::string& /*comment*/) { // not supported throw(Error(32, "Image comment", "TIFF")); } void TiffImage::readMetadata() { #ifdef DEBUG std::cerr << "Reading TIFF file " << io_->path() << "\n"; #endif if (io_->open() != 0) throw Error(9, io_->path(), strError()); IoCloser closer(*io_); // Ensure that this is the correct image type if (!isTiffType(*io_, false)) { if (io_->error() || io_->eof()) throw Error(14); throw Error(3, "TIFF"); } clearMetadata(); ByteOrder bo = TiffParser::decode(exifData_, iptcData_, xmpData_, io_->mmap(), io_->size()); setByteOrder(bo); } // TiffImage::readMetadata void TiffImage::writeMetadata() { #ifdef DEBUG std::cerr << "Writing TIFF file " << io_->path() << "\n"; #endif // Read existing image ByteOrder bo = byteOrder(); DataBuf buf; if (io_->open() == 0) { IoCloser closer(*io_); // Ensure that this is the correct image type if (isTiffType(*io_, false)) { // Read the image into a memory buffer buf.alloc(io_->size()); io_->read(buf.pData_, buf.size_); if (io_->error() || io_->eof()) { buf.reset(); } TiffHeader tiffHeader; if (0 == tiffHeader.read(buf.pData_, 8)) { bo = tiffHeader.byteOrder(); } } } if (bo == invalidByteOrder) { bo = littleEndian; } setByteOrder(bo); Blob blob; WriteMethod wm = TiffParser::encode(blob, buf.pData_, buf.size_, bo, exifData_, iptcData_, xmpData_); // Write updated or new buffer to file BasicIo::AutoPtr tempIo(io_->temporary()); // may throw assert(tempIo.get() != 0); if (wm == wmNonIntrusive) { // Buffer may be modified but size is unchanged, write buffer back tempIo->write(buf.pData_, buf.size_); } else { // Size of the buffer changed, write from blob tempIo->write(&blob[0], static_cast(blob.size())); } io_->close(); io_->transfer(*tempIo); // may throw } // TiffImage::writeMetadata ByteOrder TiffParser::decode( ExifData& exifData, IptcData& iptcData, XmpData& xmpData, const byte* pData, uint32_t size ) { return TiffParserWorker::decode(exifData, iptcData, xmpData, pData, size, TiffCreator::create, TiffMapping::findDecoder); } // TiffParser::decode WriteMethod TiffParser::encode( Blob& blob, const byte* pData, uint32_t size, ByteOrder byteOrder, const ExifData& exifData, const IptcData& iptcData, const XmpData& xmpData ) { std::auto_ptr header(new TiffHeader(byteOrder)); return TiffParserWorker::encode(blob, pData, size, exifData, iptcData, xmpData, TiffCreator::create, TiffMapping::findEncoder, header.get()); } // TiffParser::encode // ************************************************************************* // free functions Image::AutoPtr newTiffInstance(BasicIo::AutoPtr io, bool create) { Image::AutoPtr image(new TiffImage(io, create)); if (!image->good()) { image.reset(); } return image; } bool isTiffType(BasicIo& iIo, bool advance) { const int32_t len = 8; byte buf[len]; iIo.read(buf, len); if (iIo.error() || iIo.eof()) { return false; } TiffHeader tiffHeader; bool rc = tiffHeader.read(buf, len); if (!advance || !rc) { iIo.seek(-len, BasicIo::cur); } return rc; } } // namespace Exiv2 namespace Exiv2 { namespace Internal { /* This table describes the standard TIFF layout (including non-standard Makernote structures) and determines the corresponding Exiv2 TIFF components. The key of the table consists of the first two attributes, (extended) tag and group. Tag is the TIFF tag or one of a few extended tags, group identifies the IFD or any other composite TIFF component. Each entry of the table defines for a particular tag and group combination, which create function is used, what the group and parent tag and group of the new component are. */ const TiffStructure TiffCreator::tiffStructure_[] = { // ext. tag group child group parent tag parent group create function //--------- ----------------- ----------------- ---------- ----------------- ------------------- // Root directory { Tag::root, Group::none, Group::ifd0, Tag::root, Group::none, newTiffDirectory }, // IFD0 { 0x8769, Group::ifd0, Group::exif, Tag::root, Group::none, newTiffSubIfd }, { 0x8825, Group::ifd0, Group::gps, Tag::root, Group::none, newTiffSubIfd }, { 0x0111, Group::ifd0, Group::ifd0, Tag::root, Group::none, newTiffImageData<0x0117, Group::ifd0> }, { 0x0117, Group::ifd0, Group::ifd0, Tag::root, Group::none, newTiffImageSize<0x0111, Group::ifd0> }, { 0x0201, Group::ifd0, Group::ifd0, Tag::root, Group::ifd0, newTiffImageData<0x0202, Group::ifd0> }, { 0x0202, Group::ifd0, Group::ifd0, Tag::root, Group::ifd0, newTiffImageSize<0x0201, Group::ifd0> }, { 0x014a, Group::ifd0, Group::subimg1, Tag::root, Group::none, newTiffSubIfd }, { Tag::next, Group::ifd0, Group::ifd1, Tag::root, Group::none, newTiffDirectory }, { Tag::all, Group::ifd0, Group::ifd0, Tag::root, Group::none, newTiffEntry }, // Subdir subimg1 { 0x0111, Group::subimg1, Group::subimg1, 0x014a, Group::ifd0, newTiffImageData<0x0117, Group::subimg1> }, { 0x0117, Group::subimg1, Group::subimg1, 0x014a, Group::ifd0, newTiffImageSize<0x0111, Group::subimg1> }, { 0x0201, Group::subimg1, Group::subimg1, 0x014a, Group::ifd0, newTiffImageData<0x0202, Group::subimg1> }, { 0x0202, Group::subimg1, Group::subimg1, 0x014a, Group::ifd0, newTiffImageSize<0x0201, Group::subimg1> }, { Tag::next, Group::subimg1, Group::ignr, 0x014a, Group::ifd0, newTiffDirectory }, { Tag::all, Group::subimg1, Group::subimg1, 0x014a, Group::ifd0, newTiffEntry }, // Subdir subimg2 { 0x0111, Group::subimg2, Group::subimg2, 0x014a, Group::ifd0, newTiffImageData<0x0117, Group::subimg2> }, { 0x0117, Group::subimg2, Group::subimg2, 0x014a, Group::ifd0, newTiffImageSize<0x0111, Group::subimg2> }, { 0x0201, Group::subimg2, Group::subimg2, 0x014a, Group::ifd0, newTiffImageData<0x0202, Group::subimg2> }, { 0x0202, Group::subimg2, Group::subimg2, 0x014a, Group::ifd0, newTiffImageSize<0x0201, Group::subimg2> }, { Tag::next, Group::subimg2, Group::ignr, 0x014a, Group::ifd0, newTiffDirectory }, { Tag::all, Group::subimg2, Group::subimg2, 0x014a, Group::ifd0, newTiffEntry }, // Subdir subimg3 { 0x0111, Group::subimg3, Group::subimg3, 0x014a, Group::ifd0, newTiffImageData<0x0117, Group::subimg3> }, { 0x0117, Group::subimg3, Group::subimg3, 0x014a, Group::ifd0, newTiffImageSize<0x0111, Group::subimg3> }, { 0x0201, Group::subimg3, Group::subimg3, 0x014a, Group::ifd0, newTiffImageData<0x0202, Group::subimg3> }, { 0x0202, Group::subimg3, Group::subimg3, 0x014a, Group::ifd0, newTiffImageSize<0x0201, Group::subimg3> }, { Tag::next, Group::subimg3, Group::ignr, 0x014a, Group::ifd0, newTiffDirectory }, { Tag::all, Group::subimg3, Group::subimg3, 0x014a, Group::ifd0, newTiffEntry }, // Subdir subimg4 { 0x0111, Group::subimg4, Group::subimg4, 0x014a, Group::ifd0, newTiffImageData<0x0117, Group::subimg4> }, { 0x0117, Group::subimg4, Group::subimg4, 0x014a, Group::ifd0, newTiffImageSize<0x0111, Group::subimg4> }, { 0x0201, Group::subimg4, Group::subimg4, 0x014a, Group::ifd0, newTiffImageData<0x0202, Group::subimg4> }, { 0x0202, Group::subimg4, Group::subimg4, 0x014a, Group::ifd0, newTiffImageSize<0x0201, Group::subimg4> }, { Tag::next, Group::subimg4, Group::ignr, 0x014a, Group::ifd0, newTiffDirectory }, { Tag::all, Group::subimg4, Group::subimg4, 0x014a, Group::ifd0, newTiffEntry }, // Exif subdir { 0xa005, Group::exif, Group::iop, 0x8769, Group::ifd0, newTiffSubIfd }, { 0x927c, Group::exif, Group::mn, 0x8769, Group::ifd0, newTiffMnEntry }, { Tag::next, Group::exif, Group::ignr, 0x8769, Group::ifd0, newTiffDirectory }, { Tag::all, Group::exif, Group::exif, 0x8769, Group::ifd0, newTiffEntry }, // GPS subdir { Tag::next, Group::gps, Group::ignr, 0x8825, Group::ifd0, newTiffDirectory }, { Tag::all, Group::gps, Group::gps, 0x8825, Group::ifd0, newTiffEntry }, // IOP subdir { Tag::next, Group::iop, Group::ignr, 0xa005, Group::exif, newTiffDirectory }, { Tag::all, Group::iop, Group::iop, 0xa005, Group::exif, newTiffEntry }, // IFD1 { 0x0111, Group::ifd1, Group::ifd1, Tag::next, Group::ifd0, newTiffThumbData<0x0117, Group::ifd1> }, { 0x0117, Group::ifd1, Group::ifd1, Tag::next, Group::ifd0, newTiffThumbSize<0x0111, Group::ifd1> }, { 0x0201, Group::ifd1, Group::ifd1, Tag::next, Group::ifd0, newTiffThumbData<0x0202, Group::ifd1> }, { 0x0202, Group::ifd1, Group::ifd1, Tag::next, Group::ifd0, newTiffThumbSize<0x0201, Group::ifd1> }, { Tag::next, Group::ifd1, Group::ignr, Tag::next, Group::ifd0, newTiffDirectory }, { Tag::all, Group::ifd1, Group::ifd1, Tag::next, Group::ifd0, newTiffEntry }, // Olympus makernote - some Olympus cameras use Minolta structures // Todo: Adding such tags will not work (maybe result in a Minolta makernote), need separate groups { 0x0001, Group::olympmn, Group::minocso, 0x927c, Group::exif, newTiffArrayEntry }, { 0x0003, Group::olympmn, Group::minocsn, 0x927c, Group::exif, newTiffArrayEntry }, { Tag::next, Group::olympmn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::olympmn, Group::olympmn, 0x927c, Group::exif, newTiffEntry }, // Fujifilm makernote { Tag::next, Group::fujimn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::fujimn, Group::fujimn, 0x927c, Group::exif, newTiffEntry }, // Canon makernote { 0x0001, Group::canonmn, Group::canoncs, 0x927c, Group::exif, newTiffArrayEntry }, { 0x0004, Group::canonmn, Group::canonsi, 0x927c, Group::exif, newTiffArrayEntry }, { 0x0005, Group::canonmn, Group::canonpa, 0x927c, Group::exif, newTiffArrayEntry }, { 0x000f, Group::canonmn, Group::canoncf, 0x927c, Group::exif, newTiffArrayEntry }, { 0x0012, Group::canonmn, Group::canonpi, 0x927c, Group::exif, newTiffArrayEntry }, { Tag::next, Group::canonmn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::canonmn, Group::canonmn, 0x927c, Group::exif, newTiffEntry }, // Canon makernote composite tags { Tag::all, Group::canoncs, Group::canoncs, 0x0001, Group::canonmn, newTiffArrayElement }, { Tag::all, Group::canonsi, Group::canonsi, 0x0004, Group::canonmn, newTiffArrayElement }, { Tag::all, Group::canonpa, Group::canonpa, 0x0005, Group::canonmn, newTiffArrayElement }, { Tag::all, Group::canoncf, Group::canoncf, 0x000f, Group::canonmn, newTiffArrayElement }, { Tag::all, Group::canonpi, Group::canonpi, 0x0012, Group::canonmn, newTiffArrayElement }, // Nikon1 makernote { Tag::next, Group::nikon1mn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::nikon1mn, Group::nikon1mn, 0x927c, Group::exif, newTiffEntry }, // Nikon2 makernote { Tag::next, Group::nikon2mn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::nikon2mn, Group::nikon2mn, 0x927c, Group::exif, newTiffEntry }, // Nikon3 makernote { Tag::next, Group::nikon3mn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { 0x0011, Group::nikon3mn, Group::nikonpv, 0x927c, Group::exif, newTiffSubIfd }, { Tag::all, Group::nikon3mn, Group::nikon3mn, 0x927c, Group::exif, newTiffEntry }, // Nikon3 makernote preview sub-IFD { 0x0201, Group::nikonpv, Group::nikonpv, 0x0011, Group::nikon3mn, newTiffThumbData<0x0202, Group::nikonpv> }, { 0x0202, Group::nikonpv, Group::nikonpv, 0x0011, Group::nikon3mn, newTiffThumbSize<0x0201, Group::nikonpv> }, { Tag::next, Group::nikonpv, Group::ignr, 0x0011, Group::nikon3mn, newTiffDirectory }, { Tag::all, Group::nikonpv, Group::nikonpv, 0x0011, Group::nikon3mn, newTiffEntry }, // Panasonic makernote { Tag::next, Group::panamn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::panamn, Group::panamn, 0x927c, Group::exif, newTiffEntry }, // Pentax makernote { Tag::next, Group::pentaxmn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::pentaxmn, Group::pentaxmn, 0x927c, Group::exif, newTiffEntry }, // Sigma/Foveon makernote { Tag::next, Group::sigmamn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::sigmamn, Group::sigmamn, 0x927c, Group::exif, newTiffEntry }, // Sony1 makernote { Tag::next, Group::sony1mn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::sony1mn, Group::sony1mn, 0x927c, Group::exif, newTiffEntry }, // Sony2 makernote { Tag::next, Group::sony2mn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::sony2mn, Group::sony2mn, 0x927c, Group::exif, newTiffEntry }, // Minolta makernote { 0x0001, Group::minoltamn, Group::minocso, 0x927c, Group::exif, newTiffArrayEntry }, { 0x0003, Group::minoltamn, Group::minocsn, 0x927c, Group::exif, newTiffArrayEntry }, { 0x0004, Group::minoltamn, Group::minocs7, 0x927c, Group::exif, newTiffArrayEntry }, { 0x0088, Group::minoltamn, Group::minoltamn, 0x927c, Group::exif, newTiffThumbData<0x0089, Group::minoltamn> }, { 0x0089, Group::minoltamn, Group::minoltamn, 0x927c, Group::exif, newTiffThumbSize<0x0088, Group::minoltamn> }, { 0x0114, Group::minoltamn, Group::minocs5, 0x927c, Group::exif, newTiffArrayEntry }, { Tag::next, Group::minoltamn, Group::ignr, 0x927c, Group::exif, newTiffDirectory }, { Tag::all, Group::minoltamn, Group::minoltamn, 0x927c, Group::exif, newTiffEntry }, // Minolta makernote composite tags { Tag::all, Group::minocso, Group::minocso, 0x0001, Group::minoltamn, newTiffArrayElement }, { Tag::all, Group::minocsn, Group::minocsn, 0x0003, Group::minoltamn, newTiffArrayElement }, { Tag::all, Group::minocs7, Group::minocs7, 0x0004, Group::minoltamn, newTiffArrayElement }, { Tag::all, Group::minocs5, Group::minocs5, 0x0114, Group::minoltamn, newTiffArrayElement }, // Tags which are not de/encoded { Tag::next, Group::ignr, Group::ignr, Tag::none, Group::none, newTiffDirectory }, { Tag::all, Group::ignr, Group::ignr, Tag::none, Group::none, newTiffEntry } }; // TIFF mapping table for special decoding and encoding requirements const TiffMappingInfo TiffMapping::tiffMappingInfo_[] = { { "*", Tag::all, Group::ignr, 0, 0 }, // Do not decode tags with group == Group::ignr { "OLYMPUS", 0x0100, Group::olympmn, &TiffDecoder::decodeOlympThumb, &TiffEncoder::encodeOlympThumb }, { "*", 0x02bc, Group::ifd0, &TiffDecoder::decodeXmp, 0 /*done before the tree is traversed*/ }, { "*", 0x83bb, Group::ifd0, &TiffDecoder::decodeIptc, 0 /*done before the tree is traversed*/ }, { "*", 0x8649, Group::ifd0, &TiffDecoder::decodeIptc, 0 /*done before the tree is traversed*/ }, // Minolta makernote entries which need to be encoded in big endian byte order { "*", Tag::all, Group::minocso, &TiffDecoder::decodeStdTiffEntry, &TiffEncoder::encodeBigEndianEntry }, { "*", Tag::all, Group::minocso, &TiffDecoder::decodeStdTiffEntry, &TiffEncoder::encodeBigEndianEntry }, { "*", Tag::all, Group::minocso, &TiffDecoder::decodeStdTiffEntry, &TiffEncoder::encodeBigEndianEntry }, { "*", Tag::all, Group::minocso, &TiffDecoder::decodeStdTiffEntry, &TiffEncoder::encodeBigEndianEntry } }; DecoderFct TiffMapping::findDecoder(const std::string& make, uint32_t extendedTag, uint16_t group) { DecoderFct decoderFct = &TiffDecoder::decodeStdTiffEntry; const TiffMappingInfo* td = find(tiffMappingInfo_, TiffMappingInfo::Key(make, extendedTag, group)); if (td) { // This may set decoderFct to 0, meaning that the tag should not be decoded decoderFct = td->decoderFct_; } return decoderFct; } EncoderFct TiffMapping::findEncoder( const std::string& make, uint32_t extendedTag, uint16_t group ) { EncoderFct encoderFct = 0; const TiffMappingInfo* td = find(tiffMappingInfo_, TiffMappingInfo::Key(make, extendedTag, group)); if (td) { // Returns 0 if no special encoder function is found encoderFct = td->encoderFct_; } return encoderFct; } TiffComponent::AutoPtr TiffCreator::create(uint32_t extendedTag, uint16_t group) { TiffComponent::AutoPtr tc(0); uint16_t tag = static_cast(extendedTag & 0xffff); const TiffStructure* ts = find(tiffStructure_, TiffStructure::Key(extendedTag, group)); if (ts && ts->newTiffCompFct_) { tc = ts->newTiffCompFct_(tag, ts); } #ifdef DEBUG else { if (!ts) { std::cerr << "Warning: No TIFF structure entry found for "; } else { std::cerr << "Warning: No TIFF component creator found for "; } std::cerr << "extended tag 0x" << std::setw(4) << std::setfill('0') << std::hex << std::right << extendedTag << ", group " << tiffGroupName(group) << "\n"; } #endif return tc; } // TiffCreator::create void TiffCreator::getPath(TiffPath& tiffPath, uint32_t extendedTag, uint16_t group) { const TiffStructure* ts = 0; do { ts = find(tiffStructure_, TiffStructure::Key(extendedTag, group)); assert(ts != 0); tiffPath.push(ts); extendedTag = ts->parentExtTag_; group = ts->parentGroup_; } while (!(ts->extendedTag_ == Tag::root && ts->group_ == Group::none)); } // TiffCreator::getPath ByteOrder TiffParserWorker::decode( ExifData& exifData, IptcData& iptcData, XmpData& xmpData, const byte* pData, uint32_t size, TiffCompFactoryFct createFct, FindDecoderFct findDecoderFct, TiffHeaderBase* pHeader ) { // Create standard TIFF header if necessary std::auto_ptr ph; if (!pHeader) { ph = std::auto_ptr(new TiffHeader); pHeader = ph.get(); } TiffComponent::AutoPtr rootDir = parse(pData, size, createFct, pHeader); if (0 != rootDir.get()) { TiffDecoder decoder(exifData, iptcData, xmpData, rootDir.get(), findDecoderFct); rootDir->accept(decoder); } return pHeader->byteOrder(); } // TiffParserWorker::decode WriteMethod TiffParserWorker::encode( Blob& blob, const byte* pData, uint32_t size, const ExifData& exifData, const IptcData& iptcData, const XmpData& xmpData, TiffCompFactoryFct createFct, FindEncoderFct findEncoderFct, TiffHeaderBase* pHeader ) { /* 1) parse the binary image, if one is provided, and 2) attempt updating the parsed tree in-place ("non-intrusive writing") 3) else, create a new tree and write a new TIFF structure ("intrusive writing"). If there is a parsed tree, it is only used to access the image data in this case. */ assert(pHeader); assert(pHeader->byteOrder() != invalidByteOrder); blob.clear(); WriteMethod writeMethod = wmIntrusive; TiffComponent::AutoPtr createdTree; TiffComponent::AutoPtr parsedTree = parse(pData, size, createFct, pHeader); if (0 != parsedTree.get()) { // Attempt to update existing TIFF components based on metadata entries TiffEncoder encoder(exifData, iptcData, xmpData, parsedTree.get(), pHeader->byteOrder(), findEncoderFct); parsedTree->accept(encoder); if (!encoder.dirty()) writeMethod = wmNonIntrusive; } if (writeMethod == wmIntrusive) { createdTree = createFct(Tag::root, Group::none); TiffEncoder encoder(exifData, iptcData, xmpData, createdTree.get(), pHeader->byteOrder(), findEncoderFct); // Add entries from metadata to composite encoder.add(createdTree.get(), parsedTree.get(), createFct); // Write binary representation from the composite tree uint32_t offset = pHeader->write(blob); uint32_t imageIdx(-1); uint32_t len = createdTree->write(blob, pHeader->byteOrder(), offset, uint32_t(-1), uint32_t(-1), imageIdx); // Avoid writing just the header if there is no IFD data if (len == 0) blob.clear(); #ifdef DEBUG std::cerr << "Intrusive writing\n"; #endif } #ifdef DEBUG else { std::cerr << "Non-intrusive writing\n"; } #endif return writeMethod; } // TiffParserWorker::encode TiffComponent::AutoPtr TiffParserWorker::parse( const byte* pData, uint32_t size, TiffCompFactoryFct createFct, TiffHeaderBase* pHeader ) { if (pData == 0 || size == 0) return TiffComponent::AutoPtr(0); if (!pHeader->read(pData, size) || pHeader->offset() >= size) { throw Error(3, "TIFF"); } TiffComponent::AutoPtr rootDir = createFct(Tag::root, Group::none); if (0 != rootDir.get()) { rootDir->setStart(pData + pHeader->offset()); TiffRwState::AutoPtr state( new TiffRwState(pHeader->byteOrder(), 0, createFct)); TiffReader reader(pData, size, rootDir.get(), state); rootDir->accept(reader); } return rootDir; } // TiffParserWorker::parse TiffHeaderBase::TiffHeaderBase(uint16_t tag, uint32_t size, ByteOrder byteOrder, uint32_t offset) : tag_(tag), size_(size), byteOrder_(byteOrder), offset_(offset) { } TiffHeaderBase::~TiffHeaderBase() { } bool TiffHeaderBase::read(const byte* pData, uint32_t size) { if (!pData || size < 8) return false; if (pData[0] == 0x49 && pData[1] == 0x49) { byteOrder_ = littleEndian; } else if (pData[0] == 0x4d && pData[1] == 0x4d) { byteOrder_ = bigEndian; } else { return false; } if (tag_ != getUShort(pData + 2, byteOrder_)) return false; offset_ = getULong(pData + 4, byteOrder_); return true; } // TiffHeaderBase::read uint32_t TiffHeaderBase::write(Blob& blob) const { byte buf[8]; switch (byteOrder_) { case littleEndian: buf[0] = 0x49; buf[1] = 0x49; break; case bigEndian: buf[0] = 0x4d; buf[1] = 0x4d; break; case invalidByteOrder: assert(false); break; } us2Data(buf + 2, tag_, byteOrder_); ul2Data(buf + 4, 0x00000008, byteOrder_); append(blob, buf, 8); return 8; } void TiffHeaderBase::print(std::ostream& os, const std::string& prefix) const { os << prefix << _("TIFF header, offset") << " = 0x" << std::setw(8) << std::setfill('0') << std::hex << std::right << offset_; switch (byteOrder_) { case littleEndian: os << ", " << _("little endian encoded"); break; case bigEndian: os << ", " << _("big endian encoded"); break; case invalidByteOrder: break; } os << "\n"; } // TiffHeaderBase::print ByteOrder TiffHeaderBase::byteOrder() const { return byteOrder_; } void TiffHeaderBase::setByteOrder(ByteOrder byteOrder) { byteOrder_ = byteOrder; } uint32_t TiffHeaderBase::offset() const { return offset_; } void TiffHeaderBase::setOffset(uint32_t offset) { offset_ = offset; } uint32_t TiffHeaderBase::size() const { return size_; } uint16_t TiffHeaderBase::tag() const { return tag_; } TiffHeader::TiffHeader(ByteOrder byteOrder) : TiffHeaderBase(42, 8, byteOrder, 0x00000008) { } TiffHeader::~TiffHeader() { } }} // namespace Internal, Exiv2