You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
exiv2/src/tiffcomposite.cpp

297 lines
9.4 KiB
C++

// ***************************************************************** -*- C++ -*-
/*
* Copyright (C) 2006 Andreas Huggel <ahuggel@gmx.net>
*
* 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: tiffcomposite.cpp
Version: $Rev$
Author(s): Andreas Huggel (ahu) <ahuggel@gmx.net>
History: 11-Apr-06, ahu: created
*/
// *****************************************************************************
#include "rcsid.hpp"
EXIV2_RCSID("@(#) $Id$");
// Define DEBUG to output debug information to std::cerr, e.g, by calling make
// like this: make DEFS=-DDEBUG tiffcomposite.o
//#define DEBUG
// *****************************************************************************
// included header files
#ifdef _MSC_VER
# include "exv_msvc.h"
#else
# include "exv_conf.h"
#endif
#include "tiffcomposite.hpp"
#include "tiffvisitor.hpp"
#include "makernote2.hpp"
#include "value.hpp"
// + standard includes
#include <string>
#include <iostream>
#include <iomanip>
// *****************************************************************************
// class member definitions
namespace Exiv2 {
bool TiffStructure::operator==(const TiffStructure::Key& key) const
{
return key.e_ == extendedTag_ && key.g_ == group_
|| Tag::all == extendedTag_ && key.g_ == group_;
}
void TiffHeade2::print(std::ostream& os, const std::string& prefix) const
{
os << prefix
<< "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";
} // TiffHeade2::print
TiffDirectory::~TiffDirectory()
{
Components::iterator b = components_.begin();
Components::iterator e = components_.end();
for (Components::iterator i = b; i != e; ++i) {
delete *i;
}
delete pNext_;
} // TiffDirectory::~TiffDirectory
TiffEntryBase::~TiffEntryBase()
{
if (isAllocated_) {
delete[] pData_;
}
delete pValue_;
} // TiffEntryBase::~TiffEntryBase
TiffMnEntry::~TiffMnEntry()
{
delete mn_;
} // TiffMnEntry::~TiffMnEntry
TiffArrayEntry::~TiffArrayEntry()
{
Components::iterator b = elements_.begin();
Components::iterator e = elements_.end();
for (Components::iterator i = b; i != e; ++i) {
delete *i;
}
} // TiffArrayEntry::~TiffArrayEntry
const uint16_t TiffHeade2::tag_ = 42;
bool TiffHeade2::read(const byte* pData, uint32_t size)
{
if (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;
} // TiffHeade2::read
std::string TiffComponent::groupName() const
{
// Todo: This mapping should be a table and it belongs somewhere else
// Possibly the whole function shouldn't be in this class...
std::string group;
switch (group_) {
case 1: group = "Image"; break;
case 2: group = "Thumbnail"; break;
case 3: group = "Photo"; break;
case 4: group = "GPSInfo"; break;
case 5: group = "Iop"; break;
case 257: group = "Olympus"; break;
case 258: group = "Fujifilm"; break;
case 259: group = "Canon"; break;
case 260: group = "CanonCs1"; break;
case 261: group = "CanonCs2"; break;
case 262: group = "CanonCf"; break;
default: group = "Unknown"; break;
}
return group;
}
void TiffComponent::addChild(TiffComponent::AutoPtr tiffComponent)
{
doAddChild(tiffComponent);
} // TiffComponent::addChild
void TiffDirectory::doAddChild(TiffComponent::AutoPtr tiffComponent)
{
components_.push_back(tiffComponent.release());
} // TiffDirectory::doAddChild
void TiffSubIfd::doAddChild(TiffComponent::AutoPtr tiffComponent)
{
ifd_.addChild(tiffComponent);
} // TiffSubIfd::doAddChild
void TiffMnEntry::doAddChild(TiffComponent::AutoPtr tiffComponent)
{
if (mn_) mn_->addChild(tiffComponent);
} // TiffMnEntry::doAddChild
void TiffArrayEntry::doAddChild(TiffComponent::AutoPtr tiffComponent)
{
elements_.push_back(tiffComponent.release());
} // TiffArrayEntry::doAddChild
void TiffComponent::addNext(TiffComponent::AutoPtr tiffComponent)
{
doAddNext(tiffComponent);
} // TiffComponent::addNext
void TiffDirectory::doAddNext(TiffComponent::AutoPtr tiffComponent)
{
pNext_ = tiffComponent.release();
} // TiffDirectory::doAddNext
void TiffSubIfd::doAddNext(TiffComponent::AutoPtr tiffComponent)
{
ifd_.addNext(tiffComponent);
} // TiffSubIfd::doAddNext
void TiffMnEntry::doAddNext(TiffComponent::AutoPtr tiffComponent)
{
if (mn_) mn_->addNext(tiffComponent);
} // TiffMnEntry::doAddNext
void TiffComponent::accept(TiffVisitor& visitor)
{
if (visitor.go()) doAccept(visitor); // one for NVI :)
} // TiffComponent::accept
void TiffEntry::doAccept(TiffVisitor& visitor)
{
visitor.visitEntry(this);
} // TiffEntry::doAccept
void TiffDirectory::doAccept(TiffVisitor& visitor)
{
visitor.visitDirectory(this);
Components::const_iterator b = components_.begin();
Components::const_iterator e = components_.end();
for (Components::const_iterator i = b; visitor.go() && i != e; ++i) {
(*i)->accept(visitor);
}
if (visitor.go()) visitor.visitDirectoryNext(this);
if (pNext_) {
pNext_->accept(visitor);
}
if (visitor.go()) visitor.visitDirectoryEnd(this);
} // TiffDirectory::doAccept
void TiffSubIfd::doAccept(TiffVisitor& visitor)
{
visitor.visitSubIfd(this);
ifd_.accept(visitor);
} // TiffSubIfd::doAccept
void TiffMnEntry::doAccept(TiffVisitor& visitor)
{
visitor.visitMnEntry(this);
if (mn_) mn_->accept(visitor);
} // TiffMnEntry::doAccept
void TiffArrayEntry::doAccept(TiffVisitor& visitor)
{
visitor.visitArrayEntry(this);
Components::const_iterator b = elements_.begin();
Components::const_iterator e = elements_.end();
for (Components::const_iterator i = b; visitor.go() && i != e; ++i) {
(*i)->accept(visitor);
}
} // TiffArrayEntry::doAccept
void TiffArrayElement::doAccept(TiffVisitor& visitor)
{
visitor.visitArrayElement(this);
} // TiffArrayElement::doAccept
// *************************************************************************
// free functions
TiffComponent::AutoPtr newTiffDirectory(uint16_t tag,
const TiffStructure* ts)
{
assert(ts);
return TiffComponent::AutoPtr(new TiffDirectory(tag, ts->newGroup_));
}
TiffComponent::AutoPtr newTiffSubIfd(uint16_t tag,
const TiffStructure* ts)
{
assert(ts);
return TiffComponent::AutoPtr(new TiffSubIfd(tag,
ts->group_,
ts->newGroup_));
}
TiffComponent::AutoPtr newTiffMnEntry(uint16_t tag,
const TiffStructure* ts)
{
assert(ts);
return TiffComponent::AutoPtr(new TiffMnEntry(tag,
ts->group_,
ts->newGroup_));
}
TiffComponent::AutoPtr newTiffArrayEntry(uint16_t tag,
const TiffStructure* ts)
{
assert(ts);
return TiffComponent::AutoPtr(new TiffArrayEntry(tag,
ts->group_,
ts->newGroup_));
}
TiffComponent::AutoPtr newTiffArrayElement(uint16_t tag,
const TiffStructure* ts)
{
assert(ts);
return TiffComponent::AutoPtr(new TiffArrayElement(tag,
ts->group_));
}
} // namespace Exiv2