// ***************************************************************** -*- C++ -*- /* * Copyright (C) 2006 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: tiffvisitor.cpp Version: $Rev$ Author(s): Andreas Huggel (ahu) History: 11-Apr-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 "tiffvisitor.hpp" #include "tiffcomposite.hpp" #include "makernote2.hpp" #include "exif.hpp" #include "value.hpp" #include "image.hpp" // + standard includes #include #include #include #include // ***************************************************************************** // class member definitions namespace Exiv2 { void TiffFinder::init(uint16_t tag, uint16_t group) { tag_ = tag; group_ = group; tiffComponent_ = 0; } void TiffFinder::findObject(TiffComponent* object) { if (object->tag() == tag_ && object->group() == group_) { tiffComponent_ = object; setGo(false); } } void TiffFinder::visitEntry(TiffEntry* object) { findObject(object); } void TiffFinder::visitDataEntry(TiffDataEntry* object) { findObject(object); } void TiffFinder::visitSizeEntry(TiffSizeEntry* object) { findObject(object); } void TiffFinder::visitDirectory(TiffDirectory* object) { findObject(object); } void TiffFinder::visitSubIfd(TiffSubIfd* object) { findObject(object); } void TiffFinder::visitMnEntry(TiffMnEntry* object) { findObject(object); } void TiffFinder::visitIfdMakernote(TiffIfdMakernote* object) { findObject(object); } void TiffFinder::visitArrayEntry(TiffArrayEntry* object) { findObject(object); } void TiffFinder::visitArrayElement(TiffArrayElement* object) { findObject(object); } void TiffMetadataDecoder::visitEntry(TiffEntry* object) { decodeTiffEntry(object); } void TiffMetadataDecoder::visitDataEntry(TiffDataEntry* object) { decodeTiffEntry(object); } void TiffMetadataDecoder::visitSizeEntry(TiffSizeEntry* object) { decodeTiffEntry(object); } void TiffMetadataDecoder::visitDirectory(TiffDirectory* object) { // Nothing to do } void TiffMetadataDecoder::visitSubIfd(TiffSubIfd* object) { decodeTiffEntry(object); } void TiffMetadataDecoder::visitMnEntry(TiffMnEntry* object) { if (!object->mn_) decodeTiffEntry(object); } void TiffMetadataDecoder::visitIfdMakernote(TiffIfdMakernote* object) { // Nothing to do } void TiffMetadataDecoder::decodeTiffEntry(const TiffEntryBase* object) { assert(object != 0); if (object->group() == Group::ignr) return; // Experimental code to decode Olympus Thumbnail from the TIFF makernote // into IFD1. // Todo: this should go elsewhere. Ideally, such code to decode specific // tag/group combinations could be added from outside if (object->tag() == 0x0100 && object->group() == 257) { const DataValue* v = dynamic_cast(object->pValue()); if (v != 0) { ExifData& exifData = pImage_->exifData(); exifData["Exif.Thumbnail.Compression"] = uint16_t(6); DataBuf buf(v->size()); v->copy(buf.pData_); Exifdatum& ed = exifData["Exif.Thumbnail.JPEGInterchangeFormat"]; ed = uint32_t(0); ed.setDataArea(buf.pData_, buf.size_); exifData["Exif.Thumbnail.JPEGInterchangeFormatLength"] = uint32_t(buf.size_); return; } } // Todo: ExifKey should have an appropriate c'tor, it should not be // necessary to use groupName here ExifKey k(object->tag(), object->groupName()); const Value* v = object->pValue(); if ( threshold_ > 0 && v != 0 && static_cast(v->size()) > threshold_ && k.tagName().substr(0, 2) == "0x") { #ifndef SUPPRESS_WARNINGS std::cerr << "Warning: " << "Size " << v->size() << " of " << k.key() << " exceeds " << threshold_ << " bytes limit. Not decoded.\n"; #endif return; } assert(pImage_ != 0); pImage_->exifData().add(k, object->pValue()); } // TiffMetadataDecoder::decodeTiffEntry void TiffMetadataDecoder::visitArrayEntry(TiffArrayEntry* object) { assert(object != 0); // Array entry degenerates to a normal entry if type is not unsignedShort if (object->typeId() != unsignedShort) { decodeTiffEntry(object); } } void TiffMetadataDecoder::visitArrayElement(TiffArrayElement* object) { decodeTiffEntry(object); } const std::string TiffPrinter::indent_(" "); void TiffPrinter::incIndent() { prefix_ += indent_; } // TiffPrinter::incIndent void TiffPrinter::decIndent() { if (prefix_.length() >= indent_.length()) { prefix_.erase(prefix_.length() - indent_.length(), indent_.length()); } } // TiffPrinter::decIndent void TiffPrinter::visitEntry(TiffEntry* object) { printTiffEntry(object, prefix()); } // TiffPrinter::visitEntry void TiffPrinter::visitDataEntry(TiffDataEntry* object) { printTiffEntry(object, prefix()); if (object->pValue()) { os_ << prefix() << "Data area " << object->pValue()->sizeDataArea() << " bytes.\n"; } } // TiffPrinter::visitEntry void TiffPrinter::visitSizeEntry(TiffSizeEntry* object) { printTiffEntry(object, prefix()); } void TiffPrinter::visitDirectory(TiffDirectory* object) { assert(object != 0); os_ << prefix() << object->groupName() << " directory with " // cast to make MSVC happy << std::dec << static_cast(object->components_.size()); if (object->components_.size() == 1) os_ << " entry:\n"; else os_ << " entries:\n"; incIndent(); } // TiffPrinter::visitDirectory void TiffPrinter::visitDirectoryNext(TiffDirectory* object) { decIndent(); if (object->hasNext()) { if (object->pNext_) os_ << prefix() << "Next directory:\n"; else os_ << prefix() << "No next directory\n"; } } // TiffPrinter::visitDirectoryNext void TiffPrinter::visitDirectoryEnd(TiffDirectory* object) { // Nothing to do } // TiffPrinter::visitDirectoryEnd void TiffPrinter::visitSubIfd(TiffSubIfd* object) { os_ << prefix() << "Sub-IFD "; printTiffEntry(object); } // TiffPrinter::visitSubIfd void TiffPrinter::visitMnEntry(TiffMnEntry* object) { if (!object->mn_) printTiffEntry(object, prefix()); else os_ << prefix() << "Makernote "; } // TiffPrinter::visitMnEntry void TiffPrinter::visitIfdMakernote(TiffIfdMakernote* object) { os_ << prefix() << "Todo: Print IFD makernote header\n"; } // TiffPrinter::visitIfdMakernote void TiffPrinter::printTiffEntry(TiffEntryBase* object, const std::string& px) const { assert(object != 0); os_ << px << object->groupName() << " tag 0x" << std::setw(4) << std::setfill('0') << std::hex << std::right << object->tag() << ", type " << TypeInfo::typeName(object->typeId()) << ", " << std::dec << object->count() << " component"; if (object->count() > 1) os_ << "s"; os_ << " in " << object->size() << " bytes"; if (object->size() > 4) os_ << ", offset " << object->offset(); os_ << "\n"; const Value* vp = object->pValue(); if (vp && vp->count() < 100) os_ << prefix() << *vp; else os_ << prefix() << "..."; os_ << "\n"; } // TiffPrinter::printTiffEntry void TiffPrinter::visitArrayEntry(TiffArrayEntry* object) { // Array entry degenerates to a normal entry if type is not unsignedShort if (object->typeId() != unsignedShort) { printTiffEntry(object, prefix()); } else { os_ << prefix() << "Array Entry " << object->groupName() << " tag 0x" << std::setw(4) << std::setfill('0') << std::hex << std::right << object->tag() << "\n"; } } // TiffPrinter::visitArrayEntry void TiffPrinter::visitArrayElement(TiffArrayElement* object) { printTiffEntry(object, prefix()); } // TiffPrinter::visitArrayElement TiffReader::TiffReader(const byte* pData, uint32_t size, TiffComponent* pRoot, TiffRwState::AutoPtr state) : pData_(pData), size_(size), pLast_(pData + size - 1), pRoot_(pRoot), pState_(state.release()), pOrigState_(pState_) { assert(pData_); assert(size_ > 0); } // TiffReader::TiffReader TiffReader::~TiffReader() { if (pOrigState_ != pState_) delete pOrigState_; delete pState_; } void TiffReader::resetState() { if (pOrigState_ != pState_) delete pState_; pState_ = pOrigState_; } void TiffReader::changeState(TiffRwState::AutoPtr state) { if (state.get() != 0) { if (pOrigState_ != pState_) delete pState_; // 0 for create function indicates 'no change' if (state->createFct_ == 0) state->createFct_ = pState_->createFct_; pState_ = state.release(); } } ByteOrder TiffReader::byteOrder() const { assert(pState_); return pState_->byteOrder_; } uint32_t TiffReader::baseOffset() const { assert(pState_); return pState_->baseOffset_; } TiffComponent::AutoPtr TiffReader::create(uint32_t extendedTag, uint16_t group) const { assert(pState_); assert(pState_->createFct_); return pState_->createFct_(extendedTag, group); } void TiffReader::visitEntry(TiffEntry* object) { readTiffEntry(object); } void TiffReader::visitDataEntry(TiffDataEntry* object) { assert(object != 0); readTiffEntry(object); TiffFinder finder(object->szTag(), object->szGroup()); pRoot_->accept(finder); TiffEntryBase* te = dynamic_cast(finder.result()); if (te && te->pValue()) { long size = te->pValue()->toLong(); long offset = object->pValue()->toLong(); if (baseOffset() + offset + size <= size_) { object->pValue_->setDataArea(pData_ + baseOffset() + offset, size); } #ifndef SUPPRESS_WARNINGS else { std::cerr << "Warning: " << "Directory " << object->groupName() << ", entry 0x" << std::setw(4) << std::setfill('0') << std::hex << object->tag() << " Data area exceeds data buffer, ignoring it.\n"; } #endif } } void TiffReader::visitSizeEntry(TiffSizeEntry* object) { assert(object != 0); readTiffEntry(object); TiffFinder finder(object->dtTag(), object->dtGroup()); pRoot_->accept(finder); TiffEntryBase* te = dynamic_cast(finder.result()); if (te && te->pValue()) { long offset = te->pValue()->toLong(); long size = object->pValue()->toLong(); if (baseOffset() + offset + size <= size_) { te->pValue_->setDataArea(pData_ + baseOffset() + offset, size); } #ifndef SUPPRESS_WARNINGS else { std::cerr << "Warning: " << "Directory " << object->groupName() << ", entry 0x" << std::setw(4) << std::setfill('0') << std::hex << object->tag() << " Data area exceeds data buffer, ignoring it.\n"; } #endif } } void TiffReader::visitDirectory(TiffDirectory* object) { assert(object != 0); const byte* p = object->start(); assert(p >= pData_); if (p + 2 > pLast_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: " << "Directory " << object->groupName() << ": " << " IFD exceeds data buffer, cannot read entry count.\n"; #endif return; } const uint16_t n = getUShort(p, byteOrder()); p += 2; for (uint16_t i = 0; i < n; ++i) { if (p + 12 > pLast_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: " << "Directory " << object->groupName() << ": " << " IFD entry " << i << " lies outside of the data buffer.\n"; #endif return; } uint16_t tag = getUShort(p, byteOrder()); TiffComponent::AutoPtr tc = create(tag, object->group()); assert(tc.get()); tc->setStart(p); object->addChild(tc); p += 12; } if (p + 4 > pLast_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: " << "Directory " << object->groupName() << ": " << " IFD exceeds data buffer, cannot read next pointer.\n"; #endif return; } if (object->hasNext()) { TiffComponent::AutoPtr tc(0); uint32_t next = getLong(p, byteOrder()); if (next) { tc = create(Tag::next, object->group()); #ifndef SUPPRESS_WARNINGS if (tc.get() == 0) { std::cerr << "Warning: " << "Directory " << object->groupName() << " has an unhandled next pointer.\n"; } #endif } if (tc.get()) { if (baseOffset() + next > size_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: " << "Directory " << object->groupName() << ": " << " Next pointer is out of bounds.\n"; #endif return; } tc->setStart(pData_ + baseOffset() + next); object->addNext(tc); } } // object->hasNext() } // TiffReader::visitDirectory void TiffReader::visitSubIfd(TiffSubIfd* object) { assert(object != 0); readTiffEntry(object); if (object->typeId() == unsignedLong && object->count() >= 1) { uint32_t offset = getULong(object->pData(), byteOrder()); if (baseOffset() + offset > size_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: " << "Directory " << object->groupName() << ", entry 0x" << std::setw(4) << std::setfill('0') << std::hex << object->tag() << " Sub-IFD pointer is out of bounds; ignoring it.\n"; #endif return; } object->ifd_.setStart(pData_ + baseOffset() + offset); } #ifndef SUPPRESS_WARNINGS else { std::cerr << "Warning: " << "Directory " << object->groupName() << ", entry 0x" << std::setw(4) << std::setfill('0') << std::hex << object->tag() << " doesn't look like a sub-IFD."; } #endif } // TiffReader::visitSubIfd void TiffReader::visitMnEntry(TiffMnEntry* object) { assert(object != 0); readTiffEntry(object); // Find camera make TiffFinder finder(0x010f, Group::ifd0); pRoot_->accept(finder); TiffEntryBase* te = dynamic_cast(finder.result()); std::string make; if (te && te->pValue()) { make = te->pValue()->toString(); // create concrete makernote, based on make and makernote contents object->mn_ = TiffMnCreator::create(object->tag(), object->mnGroup_, make, object->pData(), object->size(), byteOrder()); } if (object->mn_) object->mn_->setStart(object->pData()); } // TiffReader::visitMnEntry void TiffReader::visitIfdMakernote(TiffIfdMakernote* object) { assert(object != 0); if (!object->readHeader(object->start(), static_cast(pLast_ - object->start()), byteOrder())) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: Failed to read " << object->ifd_.groupName() << " (" << object->ifd_.group() << ") IFD Makernote header.\n"; #endif return; // todo: signal error to parent, delete object } // Modify reader for Makernote peculiarities, byte order, offset, // component factory changeState(object->getState(static_cast(object->start() - pData_), byteOrder())); object->ifd_.setStart(object->start() + object->ifdOffset()); } // TiffReader::visitIfdMakernote void TiffReader::visitIfdMakernoteEnd(TiffIfdMakernote* object) { // Reset state (byte order, create function, offset) back to that // for the image resetState(); } // TiffReader::visitIfdMakernoteEnd void TiffReader::readTiffEntry(TiffEntryBase* object) { assert(object != 0); const byte* p = object->start(); assert(p >= pData_); if (p + 12 > pLast_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: Entry in directory " << object->groupName() << "requests access to memory beyond the data buffer. " << "Skipping entry.\n"; #endif return; } // Component already has tag p += 2; object->type_ = getUShort(p, byteOrder()); // todo: check type p += 2; object->count_ = getULong(p, byteOrder()); p += 4; object->size_ = TypeInfo::typeSize(object->typeId()) * object->count(); object->offset_ = getULong(p, byteOrder()); object->pData_ = p; if (object->size() > 4) { if (baseOffset() + object->offset() >= size_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: Offset of " << "directory " << object->groupName() << ", " << " entry 0x" << std::setw(4) << std::setfill('0') << std::hex << object->tag() << " is out of bounds:\n" << "Offset = 0x" << std::setw(8) << std::setfill('0') << std::hex << object->offset() << "; truncating the entry\n"; #endif object->size_ = 0; object->count_ = 0; object->offset_ = 0; return; } object->pData_ = pData_ + baseOffset() + object->offset(); if (object->pData() + object->size() > pLast_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Warning: Upper boundary of data for " << "directory " << object->groupName() << ", " << " entry 0x" << std::setw(4) << std::setfill('0') << std::hex << object->tag() << " is out of bounds:\n" << "Offset = 0x" << std::setw(8) << std::setfill('0') << std::hex << object->offset() << ", size = " << std::dec << object->size() << ", exceeds buffer size by " // cast to make MSVC happy << static_cast(object->pData() + object->size() - pLast_) << " Bytes; adjusting the size\n"; #endif object->size_ = static_cast(pLast_ - object->pData() + 1); // todo: adjust count_, make size_ a multiple of typeSize } } Value::AutoPtr v = Value::create(object->typeId()); if (v.get()) { v->read(object->pData(), object->size(), byteOrder()); object->pValue_ = v.release(); } } // TiffReader::readTiffEntry void TiffReader::visitArrayEntry(TiffArrayEntry* object) { assert(object != 0); readTiffEntry(object); if (object->typeId() == unsignedShort) { for (uint16_t i = 0; i < static_cast(object->count()); ++i) { uint16_t tag = i; TiffComponent::AutoPtr tc = create(tag, object->elGroup()); assert(tc.get()); tc->setStart(object->pData() + i * 2); object->addChild(tc); } } } // TiffReader::visitArrayEntry void TiffReader::visitArrayElement(TiffArrayElement* object) { assert(object != 0); const byte* p = object->start(); assert(p >= pData_); if (p + 2 > pLast_) { #ifndef SUPPRESS_WARNINGS std::cerr << "Error: Array element in group " << object->groupName() << "requests access to memory beyond the data buffer. " << "Skipping element.\n"; #endif return; } object->type_ = unsignedShort; object->count_ = 1; object->size_ = TypeInfo::typeSize(object->typeId()) * object->count(); object->offset_ = 0; object->pData_ = p; Value::AutoPtr v = Value::create(object->typeId()); if (v.get()) { v->read(object->pData(), object->size(), byteOrder()); object->pValue_ = v.release(); } } // TiffReader::visitArrayElement } // namespace Exiv2