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/makernote.hpp

471 lines
20 KiB
C++

// ***************************************************************** -*- C++ -*-
/*
* Copyright (C) 2004 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
/*!
@file makernote.hpp
@brief Contains the Exif %MakerNote interface, IFD %MakerNote and a
MakerNote factory
@version $Name: $ $Revision: 1.18 $
@author Andreas Huggel (ahu)
<a href="mailto:ahuggel@gmx.net">ahuggel@gmx.net</a>
@date 18-Feb-04, ahu: created
*/
#ifndef MAKERNOTE_HPP_
#define MAKERNOTE_HPP_
// *****************************************************************************
// included header files
#include "types.hpp"
#include "ifd.hpp"
// + standard includes
#include <string>
#include <iosfwd>
#include <utility>
#include <vector>
// *****************************************************************************
// namespace extensions
namespace Exiv2 {
// *****************************************************************************
// class declarations
class Value;
// *****************************************************************************
// type definitions
//! Type for a pointer to a function creating a makernote
typedef MakerNote* (*CreateFct)(bool);
// *****************************************************************************
// class definitions
/*!
@brief Exif makernote interface
%MakerNote is a low-level container for makernote entries. The ExifData
container uses makernote entries just like the other Exif metadata. Thus,
clients can access Exif and makernote tags and their values uniformly
through the ExifData interface. The role of %MakerNote is very similar to
that of class Ifd (but makernotes do not need to be in IFD format, see
below). In addition, it provides %MakerNote specific tag descriptions and
print functions to interpret the makernote values.
MakerNote holds methods and functionality to
- read the makernote from a character buffer
- copy the makernote to a character buffer
- maintain a list of makernote entries (similar to IFD entries)
- provide makernote specific tag names and keys
- interpret (print) the values of makernote tags
Makernotes can be added to the system by subclassing %MakerNote and
registering a create function for the new subclass together with the
camera make and model (which may contain wildcards) in the
MakerNoteFactory. Since the majority of makernotes are in IFD format,
subclass IfdMakerNote is provided. It contains an IFD container and
implements all interface methods related to the makernote entries. <BR>
To implement a new IFD makernote, all that you need to do is
- subclass %IfdMakerNote,
- implement methods to read and check the header (if any) as well as
clone and create functions,
- add a list of tag descriptions and appropriate print functions and
- register the camera make/model and create function in the makernote factory.
.
See existing makernote implementations for examples, e.g., CanonMakerNote
or FujiMakerNote.
Finally, the implementation files should be named *mn.[ch]pp, so that the
magic, which ensures that the makernote is automatically registered
in the factory, will pick it up (see mn.sh for details).
*/
class MakerNote {
public:
//! MakerNote Tag information
struct MnTagInfo {
//! Constructor
MnTagInfo(uint16 tag, const char* name, const char* desc)
: tag_(tag), name_(name), desc_(desc) {}
uint16 tag_; //!< Tag
const char* name_; //!< One word tag label
const char* desc_; //!< Short tag description
}; // struct MnTagInfo
//! @name Creators
//@{
/*!
@brief Constructor. Takes an optional makernote info tag array and
allows to choose whether or not memory management is required
for the Entries.
*/
explicit MakerNote(const MnTagInfo* pMnTagInfo =0, bool alloc =true);
//! Virtual destructor.
virtual ~MakerNote() {}
//@}
//! @name Manipulators
//@{
/*!
@brief Read the makernote, including the makernote header, from
character buffer buf of length len at position offset (from the
start of the TIFF header) and encoded in byte order byteOrder.
Return 0 if successful.
*/
virtual int read(const char* buf,
long len,
ByteOrder byteOrder,
long offset) =0;
/*!
@brief Copy (write) the makerNote to the character buffer buf at
position offset (from the start of the TIFF header), encoded
in byte order byteOrder. Update internal offsets if necessary.
Return the number of bytes written.
*/
virtual long copy(char* buf, ByteOrder byteOrder, long offset) =0;
/*!
@brief Add the entry to the makernote. No duplicate-check is performed,
i.e., it is possible to add multiple entries with the same tag.
The memory allocation mode of the entry to be added must be the
same as that of the makernote and the IFD id of the entry must
be set to 'makerIfd'.
*/
virtual void add(const Entry& entry) =0;
//! The first makernote entry
virtual Entries::iterator begin() =0;
//! End of the makernote entries
virtual Entries::iterator end() =0;
//@}
//! @name Accessors
//@{
//! Return the key for the tag.
std::string makeKey(uint16 tag) const;
//! Return the associated tag for a makernote key.
uint16 decomposeKey(const std::string& key) const;
//! Return the byte order (little or big endian).
ByteOrder byteOrder() const { return byteOrder_; }
//! Return the offset of the makernote from the start of the TIFF header
long offset() const { return offset_; }
/*!
@brief Return the name of a makernote tag. The default implementation
looks up the makernote info tag array if one is set, else
it translates the tag to a string with the hexadecimal value of
the tag.
*/
virtual std::string tagName(uint16 tag) const;
/*!
@brief Return the description of a makernote tag. The default
implementation looks up the makernote info tag array if one is
set, else it returns an empty string.
*/
virtual uint16 tag(const std::string& tagName) const;
/*!
@brief Print a list of all tags known by this MakerNote to the output
stream os. The default implementation prints all tags in the
makernote info tag array if one is set.
*/
virtual std::string tagDesc(uint16 tag) const;
/*!
@brief Return the tag associated with a makernote tag name. The
default implementation looks up the makernote info tag array
if one is set, else it expects tag names in the form \"0x01ff\"
and converts them to unsigned integer.
*/
virtual void taglist(std::ostream& os) const;
/*!
@brief Write the makernote tag info of tag to the output stream os.
*/
virtual std::ostream& writeMnTagInfo(std::ostream& os, uint16 tag) const;
/*!
@brief Return a pointer to an newly created, empty instance of the
same type as this. The makernote entries are <B>not</B> copied.
The caller owns the new object and is responsible to delete it!
@param alloc Memory management model for the clone. Indicates if
memory required to store data should be allocated and deallocated
(true) or not (false). If false, only pointers to the buffer
provided to read() will be kept. See Ifd for more background on
this concept.
*/
virtual MakerNote* clone(bool alloc =true) const =0;
//! The first makernote entry
virtual Entries::const_iterator begin() const =0;
//! End of the makernote entries
virtual Entries::const_iterator end() const =0;
//! Find an entry by idx, return a const iterator to the record
virtual Entries::const_iterator findIdx(int idx) const =0;
//! Return the size of the makernote in bytes
virtual long size() const =0;
//! Return the name of the makernote section
virtual std::string sectionName(uint16 tag) const =0;
//! Interpret and print the value of a makernote tag
virtual std::ostream& printTag(std::ostream& os,
uint16 tag,
const Value& value) const =0;
//@}
protected:
// DATA
//! Pointer to an array of makernote tag infos
const MnTagInfo* pMnTagInfo_;
/*!
@brief Flag to control the memory management: <BR>
True: requires memory allocation and deallocation, <BR>
False: no memory management needed.
*/
const bool alloc_;
/*!
@brief Offset of the makernote from the start of the TIFF header
(for offset()).
*/
long offset_;
/*!
@brief Alternative byte order to use, invalid if the byte order of the
Exif block can be used
*/
ByteOrder byteOrder_;
}; // class MakerNote
/*!
@brief Interface for MakerNotes in IFD format. See MakerNote.
*/
class IfdMakerNote : public MakerNote {
public:
//! @name Creators
//@{
/*!
@brief Constructor. Takes an optional makernote info tag array and
allows to choose whether or not memory management is required
for the Entries.
*/
explicit IfdMakerNote(const MakerNote::MnTagInfo* pMnTagInfo =0,
bool alloc =true);
//! Virtual destructor
virtual ~IfdMakerNote() {}
//@}
//! @name Manipulators
//@{
virtual int read(const char* buf,
long len,
ByteOrder byteOrder,
long offset);
/*!
@brief Read the makernote header from the makernote databuffer. This
method must set the offset adjustment (adjOffset_), if needed
(assuming that the required information is in the header).
Return 0 if successful.
@note The default implementation does nothing, assuming there is no
header
*/
virtual int readHeader(const char* buf,
long len,
ByteOrder byteOrder);
virtual long copy(char* buf, ByteOrder byteOrder, long offset);
void add(const Entry& entry) { ifd_.add(entry); }
Entries::iterator begin() { return ifd_.begin(); }
Entries::iterator end() { return ifd_.end(); }
//@}
//! @name Accessors
//@{
Entries::const_iterator begin() const { return ifd_.begin(); }
Entries::const_iterator end() const { return ifd_.end(); }
Entries::const_iterator findIdx(int idx) const;
long size() const;
/*!
@brief Check the makernote header. This will typically check if a
required prefix string is present in the header. Return 0 if
successful.
@note The default implementation does nothing, assuming there is no
header
*/
virtual int checkHeader() const;
/*!
@brief Write the makernote header to a character buffer, return the
number of characters written.
@note The default implementation copies the header_ buffer.
*/
virtual long copyHeader(char* buf) const;
/*!
@brief Return the size of the makernote header in bytes.
@note The default implementation returns the size of the header_
buffer.
*/
virtual long headerSize() const;
virtual IfdMakerNote* clone(bool alloc =true) const =0;
virtual std::string sectionName(uint16 tag) const =0;
virtual std::ostream& printTag(std::ostream& os,
uint16 tag,
const Value& value) const =0;
//@}
protected:
// DATA
/*!
@brief True: Adjustment of the IFD offsets is to be added to the
offset from the start of the TIFF header,
False: Adjustment of the IFD offsets is a suitable absolute
value. Ignore the offset from the start of the TIFF
header.
*/
bool absOffset_;
/*!
@brief Adjustment of the IFD offsets relative to the start of the
TIFF header or to the start of the makernote, depending on
the setting of absOffset_.
*/
long adjOffset_;
//! Data buffer for the makernote header
DataBuf header_;
//! The makernote IFD
Ifd ifd_;
}; // class IfdMakerNote
/*!
@brief Factory for MakerNote objects.
Maintains an associative list (tree) of camera makes/models and
corresponding %MakerNote create functions. Creates an instance of the
%MakerNote for one camera make/model. The factory is implemented as a
singleton, which can be accessed only through the static member function
instance().
*/
class MakerNoteFactory {
public:
/*!
@brief Access the MakerNote factory. Clients access the task factory
exclusively through this method.
*/
static MakerNoteFactory& instance();
//! @name Manipulators
//@{
/*!
@brief Register a %MakerNote create function for a camera make and model.
Registers a create function for a %MakerNote for a given make and
model combination with the factory. Both the make and model strings
may contain wildcards ('*', e.g., "Canon*"). If the make already
exists in the registry, then a new branch for the model is added. If
the model also already exists, then the new create function replaces
the old one.
@param make Camera manufacturer. (Typically the string from the Exif
make tag.)
@param model Camera model. (Typically the string from the Exif
model tag.)
@param createMakerNote Pointer to a function to create a new
%MakerNote of a particular type.
*/
void registerMakerNote(const std::string& make,
const std::string& model,
CreateFct createMakerNote);
//@}
//! @name Accessors
//@{
/*!
@brief Create the appropriate %MakerNote based on camera make and
model, return a pointer to the newly created MakerNote
instance. Return 0 if no %MakerNote is defined for the camera
model.
The method searches the make-model tree for a make and model
combination in the registry that matches the search key. The search is
case insensitive (Todo: implement case-insensitive comparisons) and
wildcards in the registry entries are supported. First the best
matching make is searched, then the best matching model for this make
is searched. If there is no matching make or no matching model within
the models registered for the best matching make, then no maker note
is created and the function returns 0. If a match is found, the
function invokes the registered create function and returns a pointer
to the newly created MakerNote. The makernote pointed to is owned by
the caller of the function, i.e., the caller is responsible to delete
the returned makernote when it is no longer needed. The best match is
the match with the most matching characters.
@param make Camera manufacturer. (Typically the string from the Exif
make tag.)
@param model Camera model. (Typically the string from the Exif
model tag.)
@param alloc Memory management model for the new MakerNote. Determines
if memory required to store data should be allocated and
deallocated (true) or not (false). If false, only pointers to
the buffer provided to read() will be kept. See Ifd for more
background on this concept.
@return A pointer that owns a %MakerNote for the camera model. If the
camera is not supported, the pointer is 0.
*/
MakerNote* create(const std::string& make,
const std::string& model,
bool alloc =true) const;
//@}
/*!
@brief Match a registry entry with a key (used for make and model).
The matching algorithm is case insensitive and wildcards ('*') in the
registry entry are supported. The best match is an exact match, then
a match is rated according to the number of matching characters.
@return A score value indicating how good the key and registry entry
match. 0 means no match, values greater than 0 indicate a
match, larger values are better matches:<BR>
0: key and registry entry do not match<BR>
1: a pure wildcard match, i.e., the registry entry is just
a wildcard.<BR>
Score values greater than 1 are computed by adding 1 to the
number of matching characters, except for an exact match,
which scores 2 plus the number of matching characters.
*/
static int match(const std::string& regEntry, const std::string& key);
private:
//! @name Creators
//@{
//! Prevent construction other than through instance().
MakerNoteFactory() {}
//! Prevent copy construction: not implemented.
MakerNoteFactory(const MakerNoteFactory& rhs);
//@}
//! Type used to store model labels and %MakerNote create functions
typedef std::vector<std::pair<std::string, CreateFct> > ModelRegistry;
//! Type used to store a list of make labels and model registries
typedef std::vector<std::pair<std::string, ModelRegistry*> > Registry;
// DATA
//! Pointer to the one and only instance of this class.
static MakerNoteFactory* pInstance_;
//! List of makernote types and corresponding makernote create functions.
Registry registry_;
}; // class MakerNoteFactory
} // namespace Exiv2
#endif // #ifndef MAKERNOTE_HPP_