test_pie/external/exiftool/lib/Image/ExifTool/MakerNotes.pm

1791 lines
68 KiB
Perl
Raw Normal View History

2023-09-14 11:12:02 +02:00
#------------------------------------------------------------------------------
# File: MakerNotes.pm
#
# Description: Read and write EXIF maker notes
#
# Revisions: 11/11/2004 - P. Harvey Created
#------------------------------------------------------------------------------
package Image::ExifTool::MakerNotes;
use strict;
use vars qw($VERSION);
use Image::ExifTool qw(:DataAccess);
use Image::ExifTool::Exif;
sub ProcessUnknown($$$);
sub ProcessUnknownOrPreview($$$);
sub ProcessCanon($$$);
sub ProcessGE2($$$);
sub ProcessKodakPatch($$$);
sub WriteUnknownOrPreview($$$);
sub FixLeicaBase($$;$);
$VERSION = '2.06';
my $debug; # set to 1 to enable debugging code
# conditional list of maker notes
# Notes:
# - This is NOT a normal tag table!
# - All byte orders are now specified because we can now
# write maker notes into a file with different byte ordering!
# - Put these in alphabetical order to make TagNames documentation nicer.
@Image::ExifTool::MakerNotes::Main = (
# decide which MakerNotes to use (based on camera make/model)
{
Name => 'MakerNoteApple',
Condition => '$$valPt =~ /^Apple iOS\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Apple::Main',
Start => '$valuePtr + 14',
Base => '$start - 14',
ByteOrder => 'Unknown',
},
},
{
# this maker notes starts with a standard TIFF header at offset 0x0a
# (must check Nikon signature first because Nikon Capture NX can generate
# NEF images containing Nikon maker notes from JPEG images of any camera model)
Name => 'MakerNoteNikon',
Condition => '$$valPt=~/^Nikon\x00\x02/',
SubDirectory => {
TagTable => 'Image::ExifTool::Nikon::Main',
Start => '$valuePtr + 18',
Base => '$start - 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteCanon',
# (starts with an IFD)
Condition => '$$self{Make} =~ /^Canon/',
SubDirectory => {
TagTable => 'Image::ExifTool::Canon::Main',
ProcessProc => \&ProcessCanon,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteCasio',
# do negative lookahead assertion just to get tags
# in a nice order for documentation
# (starts with an IFD)
Condition => '$$self{Make}=~/^CASIO/ and $$valPt!~/^(QVC|DCI)\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Casio::Main',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteCasio2',
# (starts with "QVC\0" [Casio] or "DCI\0" [Concord])
# (also found in AVI and MOV videos)
Condition => '$$valPt =~ /^(QVC|DCI)\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Casio::Type2',
Start => '$valuePtr + 6',
ByteOrder => 'Unknown',
FixBase => 1, # necessary for AVI and MOV videos
},
},
{
Name => 'MakerNoteDJI',
Condition => '$$self{Make} eq "DJI" and $$valPt !~ /^...\@AMBA/s',
SubDirectory => {
TagTable => 'Image::ExifTool::DJI::Main',
Start => '$valuePtr',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteFLIR',
# (starts with IFD, Make is 'FLIR Systems AB' or 'FLIR Systems')
Condition => '$$self{Make} =~ /^FLIR Systems/',
SubDirectory => {
TagTable => 'Image::ExifTool::FLIR::Main',
Start => '$valuePtr',
ByteOrder => 'Unknown',
},
},
{
# The Fuji maker notes use a structure similar to a self-contained
# TIFF file, but with "FUJIFILM" instead of the standard TIFF header
Name => 'MakerNoteFujiFilm',
# (starts with "FUJIFILM" -- also used by some Leica, Minolta and Sharp models)
# (GE FujiFilm models start with "GENERALE")
Condition => '$$valPt =~ /^(FUJIFILM|GENERALE)/',
SubDirectory => {
TagTable => 'Image::ExifTool::FujiFilm::Main',
# there is an 8-byte maker tag (FUJIFILM) we must skip over
OffsetPt => '$valuePtr+8',
# the pointers are relative to the subdirectory start
# (before adding the offsetPt) - PH
Base => '$start',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNoteGE',
Condition => '$$valPt =~ /^GE(\0\0|NIC\0)/',
SubDirectory => {
TagTable => 'Image::ExifTool::GE::Main',
Start => '$valuePtr + 18',
FixBase => 1,
AutoFix => 1,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteGE2',
Condition => '$$valPt =~ /^GE\x0c\0\0\0\x16\0\0\0/',
# Note: we will get a "Maker notes could not be parsed" warning when writing
# these maker notes because they aren't currently supported for writing
SubDirectory => {
TagTable => 'Image::ExifTool::FujiFilm::Main',
ProcessProc => \&ProcessGE2,
Start => '$valuePtr + 12',
Base => '$start - 6',
ByteOrder => 'LittleEndian',
# hard patch for crazy offsets
FixOffsets => '$valuePtr -= 210 if $tagID >= 0x1303',
},
},
{
Name => 'MakerNoteHasselblad',
Condition => '$$self{Make} eq "Hasselblad"',
SubDirectory => {
TagTable => 'Image::ExifTool::Unknown::Main',
ByteOrder => 'Unknown',
Start => '$valuePtr',
Base => 0, # (avoids warnings since maker notes are not self-contained)
},
},
# (the GE X5 has really messed up EXIF-like maker notes starting with
# "GENIC\x0c\0" --> currently not decoded)
{
Name => 'MakerNoteHP', # PhotoSmart 720 (also Vivitar 3705, 3705B and 3715)
Condition => '$$valPt =~ /^(Hewlett-Packard|Vivitar)/',
SubDirectory => {
TagTable => 'Image::ExifTool::HP::Main',
ProcessProc => \&ProcessUnknown,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteHP2', # PhotoSmart E427
# (this type of maker note also used by BenQ, Mustek, Sanyo, Traveler and Vivitar)
Condition => '$$valPt =~ /^610[\0-\4]/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::HP::Type2',
Start => '$valuePtr',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNoteHP4', # PhotoSmart M627
Condition => '$$valPt =~ /^IIII\x04\0/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::HP::Type4',
Start => '$valuePtr',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNoteHP6', # PhotoSmart M425, M525 and M527
Condition => '$$valPt =~ /^IIII\x06\0/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::HP::Type6',
Start => '$valuePtr',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNoteISL', # (used in Samsung GX20 samples)
Condition => '$$valPt =~ /^ISLMAKERNOTE000\0/',
# this maker notes starts with a TIFF-like header at offset 0x10
SubDirectory => {
TagTable => 'Image::ExifTool::Unknown::Main',
Start => '$valuePtr + 24',
Base => '$start - 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteJVC',
Condition => '$$valPt=~/^JVC /',
SubDirectory => {
TagTable => 'Image::ExifTool::JVC::Main',
Start => '$valuePtr + 4',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteJVCText',
Condition => '$$self{Make}=~/^(JVC|Victor)/ and $$valPt=~/^VER:/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::JVC::Text',
},
},
{
Name => 'MakerNoteKodak1a',
Condition => '$$self{Make}=~/^EASTMAN KODAK/ and $$valPt=~/^KDK INFO/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Main',
Start => '$valuePtr + 8',
ByteOrder => 'BigEndian',
},
},
{
Name => 'MakerNoteKodak1b',
Condition => '$$self{Make}=~/^EASTMAN KODAK/ and $$valPt=~/^KDK/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Main',
Start => '$valuePtr + 8',
ByteOrder => 'LittleEndian',
},
},
{
# used by various Kodak, HP, Pentax and Minolta models
Name => 'MakerNoteKodak2',
Condition => q{
$$valPt =~ /^.{8}Eastman Kodak/s or
$$valPt =~ /^\x01\0[\0\x01]\0\0\0\x04\0[a-zA-Z]{4}/
},
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type2',
ByteOrder => 'BigEndian',
},
},
{
# not much to key on here, but we know the
# upper byte of the year should be 0x07:
Name => 'MakerNoteKodak3',
Condition => q{
$$self{Make} =~ /^EASTMAN KODAK/ and
$$valPt =~ /^(?!MM|II).{12}\x07/s and
$$valPt !~ /^(MM|II|AOC)/
},
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type3',
ByteOrder => 'BigEndian',
},
},
{
Name => 'MakerNoteKodak4',
Condition => q{
$$self{Make} =~ /^Eastman Kodak/ and
$$valPt =~ /^.{41}JPG/s and
$$valPt !~ /^(MM|II|AOC)/
},
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type4',
ByteOrder => 'BigEndian',
},
},
{
Name => 'MakerNoteKodak5',
Condition => q{
$$self{Make}=~/^EASTMAN KODAK/ and
($$self{Model}=~/CX(4200|4230|4300|4310|6200|6230)/ or
# try to pick up similar models we haven't tested yet
$$valPt=~/^\0(\x1a\x18|\x3a\x08|\x59\xf8|\x14\x80)\0/)
},
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type5',
ByteOrder => 'BigEndian',
},
},
{
Name => 'MakerNoteKodak6a',
Condition => q{
$$self{Make}=~/^EASTMAN KODAK/ and
$$self{Model}=~/DX3215/
},
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type6',
ByteOrder => 'BigEndian',
},
},
{
Name => 'MakerNoteKodak6b',
Condition => q{
$$self{Make}=~/^EASTMAN KODAK/ and
$$self{Model}=~/DX3700/
},
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type6',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNoteKodak7',
# look for something that looks like a serial number
# (confirmed serial numbers have the format KXXXX########, but we also
# accept other strings from sample images that may be serial numbers)
Condition => q{
$$self{Make}=~/Kodak/i and
$$valPt =~ /^[CK][A-Z\d]{3} ?[A-Z\d]{1,2}\d{2}[A-Z\d]\d{4}[ \0]/
},
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type7',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNoteKodak8a',
# IFD-format maker notes: look for reasonable number of
# entries and check format and count of first IFD entry
Condition => q{
$$self{Make}=~/Kodak/i and
($$valPt =~ /^\0[\x02-\x7f]..\0[\x01-\x0c]\0\0/s or
$$valPt =~ /^[\x02-\x7f]\0..[\x01-\x0c]\0..\0\0/s)
},
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type8',
ProcessProc => \&ProcessUnknown,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteKodak8b',
# these maker notes have an extra 2 bytes after the entry count
# (this is handled by the patch). Also, the IFD uses a Format 13,
# which is some 2-byte format (not Float, as decoded by ExifTool)
# - written by the PixPro AZ251, AZ361, AZ262, AZ521
Condition => q{
$$self{Make}=~/Kodak/i and
$$valPt =~ /^MM\0\x2a\0\0\0\x08\0.\0\0/
},
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type8',
ProcessProc => \&ProcessKodakPatch,
ByteOrder => 'BigEndian',
Start => '$valuePtr + 8',
Base => '$start - 8',
},
},
{
Name => 'MakerNoteKodak8c',
# TIFF-format maker notes
Condition => q{
$$self{Make}=~/Kodak/i and
$$valPt =~ /^(MM\0\x2a\0\0\0\x08|II\x2a\0\x08\0\0\0)/
},
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type8',
ProcessProc => \&ProcessUnknown,
ByteOrder => 'Unknown',
Start => '$valuePtr + 8',
Base => '$start - 8',
},
},
{
Name => 'MakerNoteKodak9',
# test header and Kodak:DateTimeOriginal
Condition => '$$valPt =~ m{^IIII[\x02\x03]\0.{14}\d{4}/\d{2}/\d{2} }s',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type9',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNoteKodak10',
# yet another type of Kodak IFD-format maker notes:
# this type begins with a byte order indicator,
# followed immediately by the IFD
Condition => q{
$$self{Make}=~/Kodak/i and
$$valPt =~ /^(MM\0[\x02-\x7f]|II[\x02-\x7f]\0)/
},
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type10',
ProcessProc => \&ProcessUnknown,
ByteOrder => 'Unknown',
Start => '$valuePtr + 2',
},
},
{
Name => 'MakerNoteKodak11',
# these maker notes have a 4-byte entry count
# - written by the PixPro S-1 (Note: Make is "JK Imaging, Ltd.", so check Model for "Kodak")
Condition => q{
$$self{Model}=~/(Kodak|PixPro)/i and
$$valPt =~ /^II\x2a\0\x08\0\0\0.\0\0\0/s
},
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type11',
ProcessProc => \&ProcessKodakPatch,
ByteOrder => 'LittleEndian',
Start => '$valuePtr + 8',
Base => '$start - 8',
},
},
{
Name => 'MakerNoteKodak12',
# these maker notes have a 4-byte entry count
# - written by the PixPro AZ901 (Note: Make is "JK Imaging, Ltd.", so check Model for "Kodak")
Condition => q{
$$self{Model}=~/(Kodak|PixPro)/i and
$$valPt =~ /^MM\0\x2a\0\0\0\x08\0\0\0./s
},
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Type11',
ProcessProc => \&ProcessKodakPatch,
ByteOrder => 'BigEndian',
Start => '$valuePtr + 8',
Base => '$start - 8',
},
},
{
Name => 'MakerNoteKodakUnknown',
Condition => '$$self{Make}=~/Kodak/i and $$valPt!~/^AOC\0/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Kodak::Unknown',
ByteOrder => 'BigEndian',
},
},
{
Name => 'MakerNoteKyocera',
# (starts with "KYOCERA")
Condition => '$$valPt =~ /^KYOCERA/',
SubDirectory => {
TagTable => 'Image::ExifTool::Unknown::Main',
Start => '$valuePtr + 22',
Base => '$start + 2',
EntryBased => 1,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteMinolta',
Condition => q{
$$self{Make}=~/^(Konica Minolta|Minolta)/i and
$$valPt !~ /^(MINOL|CAMER|MLY0|KC|\+M\+M|\xd7)/
},
SubDirectory => {
TagTable => 'Image::ExifTool::Minolta::Main',
ByteOrder => 'Unknown',
},
},
{
# the DiMAGE E323 (MINOL) and E500 (CAMER), and some models
# of Mustek, Pentax, Ricoh and Vivitar (CAMER).
Name => 'MakerNoteMinolta2',
Condition => '$$valPt =~ /^(MINOL|CAMER)\0/ and $$self{OlympusCAMER} = 1',
SubDirectory => {
# these models use Olympus tags in the range 0x200-0x221 plus 0xf00
TagTable => 'Image::ExifTool::Olympus::Main',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
},
},
{
# /^MLY0/ - DiMAGE G400, G500, G530, G600
# /^KC/ - Revio KD-420Z, DiMAGE E203
# /^+M+M/ - DiMAGE E201
# /^\xd7/ - DiMAGE RD3000
Name => 'MakerNoteMinolta3',
Condition => '$$self{Make} =~ /^(Konica Minolta|Minolta)/i',
Binary => 1,
Notes => 'not EXIF-based',
},
{
Name => 'MakerNoteMotorola',
Condition => '$$valPt=~/^MOT\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Motorola::Main',
Start => '$valuePtr + 8',
Base => '$start - 8',
ByteOrder => 'Unknown',
},
},
{
# older Nikon maker notes
Name => 'MakerNoteNikon2',
Condition => '$$valPt=~/^Nikon\x00\x01/',
SubDirectory => {
TagTable => 'Image::ExifTool::Nikon::Type2',
Start => '$valuePtr + 8',
ByteOrder => 'LittleEndian',
},
},
{
# headerless Nikon maker notes
Name => 'MakerNoteNikon3',
Condition => '$$self{Make}=~/^NIKON/i',
SubDirectory => {
TagTable => 'Image::ExifTool::Nikon::Main',
ByteOrder => 'Unknown', # most are little-endian, but D1 is big
},
},
{
Name => 'MakerNoteNintendo',
# (starts with an IFD)
Condition => '$$self{Make} eq "Nintendo"',
SubDirectory => {
TagTable => 'Image::ExifTool::Nintendo::Main',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteOlympus',
# (if Make is 'SEIKO EPSON CORP.', starts with "EPSON\0")
# (if Make is 'OLYMPUS OPTICAL CO.,LTD' or 'OLYMPUS CORPORATION',
# starts with "OLYMP\0")
Condition => '$$valPt =~ /^(OLYMP|EPSON)\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Olympus::Main',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteOlympus2',
# new Olympus maker notes start with "OLYMPUS\0"
Condition => '$$valPt =~ /^OLYMPUS\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Olympus::Main',
Start => '$valuePtr + 12',
Base => '$start - 12',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteLeica',
# (starts with "LEICA\0\0\0")
Condition => '$$self{Make} eq "LEICA"',
SubDirectory => {
# many Leica models use the same format as Panasonic
TagTable => 'Image::ExifTool::Panasonic::Main',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteLeica2', # used by the M8
# (starts with "LEICA\0\0\0")
Condition => '$$self{Make} =~ /^Leica Camera AG/ and $$valPt =~ /^LEICA\0\0\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica2',
# (the offset base is different in JPEG and DNG images, but we
# can copy makernotes from one to the other, so we need special
# logic to decide which base to apply)
ProcessProc => \&FixLeicaBase,
Start => '$valuePtr + 8',
Base => '$start', # (- 8 for DNG images!)
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteLeica3', # used by the R8 and R9
# (starts with IFD)
Condition => q{
$$self{Make} =~ /^Leica Camera AG/ and $$valPt !~ /^LEICA/ and
$$self{Model} ne "S2" and $$self{Model} ne "LEICA M (Typ 240)"
},
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica3',
Start => '$valuePtr',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteLeica4', # used by the M9/M-Monochrom
# (M9 and M Monochrom start with "LEICA0\x03\0")
Condition => '$$self{Make} =~ /^Leica Camera AG/ and $$valPt =~ /^LEICA0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica4',
Start => '$valuePtr + 8',
Base => '$start - 8', # (yay! Leica fixed the M8 problem)
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteLeica5', # used by the X1/X2/X VARIO/T/X-U
# (X1 starts with "LEICA\0\x01\0", Make is "LEICA CAMERA AG")
# (X2 starts with "LEICA\0\x05\0", Make is "LEICA CAMERA AG")
# (X VARIO starts with "LEICA\0\x04\0", Make is "LEICA CAMERA AG")
# (T (Typ 701) starts with "LEICA\0\0x6", Make is "LEICA CAMERA AG")
# (X (Typ 113) starts with "LEICA\0\0x7", Make is "LEICA CAMERA AG")
# (X-U (Typ 113) starts with "LEICA\0\x10\0", Make is "LEICA CAMERA AG")
Condition => '$$valPt =~ /^LEICA\0[\x01\x04\x05\x06\x07\x10\x1a]\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica5',
Start => '$valuePtr + 8',
Base => '$start - 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteLeica6', # used by the S2, M (Typ 240) and S (Typ 006)
# (starts with "LEICA\0\x02\xff", Make is "Leica Camera AG", but test the
# model names separately because the maker notes data may not be loaded
# at the time this is tested if they are in a JPEG trailer. Also, this
# header is used by the M Monochrom (Type 246), with different offsets.)
Condition => q{
($$self{Make} eq 'Leica Camera AG' and ($$self{Model} eq 'S2' or
$$self{Model} eq 'LEICA M (Typ 240)' or $$self{Model} eq 'LEICA S (Typ 006)'))
},
DataTag => 'LeicaTrailer', # (generates fixup name for this tag)
LeicaTrailer => 1, # flag to special-case this tag in the Exif code
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica6',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
# NOTE: Leica uses absolute file offsets when this maker note is stored
# as a JPEG trailer -- this case is handled by ProcessLeicaTrailer in
# Panasonic.pm, and any "Base" defined here is ignored for this case.
# ExifTool may also create S2/M maker notes inside the APP1 segment when
# copying from other files, and for this the normal EXIF offsets are used,
# Base should not be defined!
},
},
{
Name => 'MakerNoteLeica7', # used by the M Monochrom (Typ 246)
# (starts with "LEICA\0\x02\xff", Make is "Leica Camera AG")
Condition => '$$valPt =~ /^LEICA\0\x02\xff/',
DataTag => 'LeicaTrailer', # (generates fixup name for this tag)
LeicaTrailer => 1, # flag to special-case this tag in the Exif code
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica6',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
Base => '-$base', # uses absolute file offsets (not based on TIFF header offset)
},
},
{
Name => 'MakerNoteLeica8', # used by the Q (Type 116)
# (Q (Typ 116) starts with "LEICA\0\x08\0", Make is "LEICA CAMERA AG")
# (SL (Typ 601) and CL start with "LEICA\0\x09\0", Make is "LEICA CAMERA AG")
Condition => '$$valPt =~ /^LEICA\0[\x08\x09]\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica5',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteLeica9', # used by the M10/S
# (M10 and S start with "LEICA0\x02\0")
Condition => '$$self{Make} =~ /^Leica Camera AG/ and $$valPt =~ /^LEICA\0\x02\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Leica9',
Start => '$valuePtr + 8',
Base => '$start - 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePanasonic',
# (starts with "Panasonic\0")
Condition => '$$valPt=~/^Panasonic/ and $$self{Model} ne "DC-FT7"',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Main',
Start => '$valuePtr + 12',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePanasonic2',
# (starts with "Panasonic\0")
Condition => '$$self{Make}=~/^Panasonic/ and $$valPt=~/^MKE/',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Type2',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNotePanasonic3', # (DC-FT7)
# (starts with "Panasonic\0")
Condition => '$$valPt=~/^Panasonic/',
SubDirectory => {
TagTable => 'Image::ExifTool::Panasonic::Main',
Start => '$valuePtr + 12',
Base => 12, # crazy!
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePentax',
# (starts with "AOC\0", but so does MakerNotePentax3)
# (also used by some Samsung models)
Condition => q{
$$valPt=~/^AOC\0/ and
$$self{Model} !~ /^PENTAX Optio ?[34]30RS\s*$/
},
SubDirectory => {
TagTable => 'Image::ExifTool::Pentax::Main',
# process as Unknown maker notes because the start offset and
# byte ordering are so variable
ProcessProc => \&ProcessUnknown,
# offsets can be totally whacky for Pentax maker notes,
# so attempt to fix the offset base if possible
FixBase => 1,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePentax2',
# (starts with an IFD)
# Casio-like maker notes used only by the Optio 330 and 430
Condition => '$$self{Make}=~/^Asahi/ and $$valPt!~/^AOC\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Pentax::Type2',
ProcessProc => \&ProcessUnknown,
FixBase => 1,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePentax3',
# (starts with "AOC\0", like the more common Pentax maker notes)
# Casio maker notes used only by the Optio 330RS and 430RS
Condition => '$$self{Make}=~/^Asahi/',
SubDirectory => {
TagTable => 'Image::ExifTool::Casio::Type2',
ProcessProc => \&ProcessUnknown,
FixBase => 1,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePentax4',
# (starts with 3 or 4 digits)
# HP2-like text-based maker notes used by Optio E20
Condition => '$$self{Make}=~/^PENTAX/ and $$valPt=~/^\d{3}/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Pentax::Type4',
Start => '$valuePtr',
ByteOrder => 'LittleEndian',
},
},
{
Name => 'MakerNotePentax5',
# (starts with "PENTAX \0")
# used by cameras such as the Q, Optio S1, RS1500 and WG-1
Condition => '$$valPt=~/^PENTAX \0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Pentax::Main',
Start => '$valuePtr + 10',
Base => '$start - 10',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePentax6',
# (starts with "S1\0\0\0\0\0\0\x0c\0\0\0")
Condition => '$$valPt=~/^S1\0{6}\x0c\0{3}/',
SubDirectory => {
TagTable => 'Image::ExifTool::Pentax::S1',
Start => '$valuePtr + 12',
Base => '$start - 12',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNotePhaseOne',
# Starts with: 'IIIITwaR' or 'IIIICwaR' (have seen both written by P25)
# (have also seen code which expects 'MMMMRawT')
Condition => q{
return undef unless $$valPt =~ /^(IIII.waR|MMMMRaw.)/s;
$self->OverrideFileType($$self{TIFF_TYPE} = 'IIQ') if $count > 1000000;
return 1;
},
NotIFD => 1,
IsPhaseOne => 1, # flag to rebuild these differently
SubDirectory => { TagTable => 'Image::ExifTool::PhaseOne::Main' },
PutFirst => 1, # place immediately after TIFF header
},
{
Name => 'MakerNoteReconyx',
Condition => q{
$$valPt =~ /^\x01\xf1([\x02\x03]\x00)?/ and
($1 or $$self{Make} eq "RECONYX")
},
SubDirectory => {
TagTable => 'Image::ExifTool::Reconyx::Main',
ByteOrder => 'Little-endian',
},
},
{
Name => 'MakerNoteReconyx2',
Condition => '$$valPt =~ /^RECONYXUF\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Reconyx::Type2',
ByteOrder => 'Little-endian',
},
},
{
Name => 'MakerNoteRicoh',
# (my test R50 image starts with " \x02\x01" - PH)
Condition => q{
$$self{Make} =~ /^(PENTAX )?RICOH/ and
$$valPt =~ /^(Ricoh| |MM\0\x2a|II\x2a\0)/i and
$$valPt !~ /^(MM\0\x2a\0\0\0\x08\0.\0\0|II\x2a\0\x08\0\0\0.\0\0\0)/s and
$$self{Model} ne 'RICOH WG-M1'
},
SubDirectory => {
TagTable => 'Image::ExifTool::Ricoh::Main',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteRicoh2',
# (the Ricoh HZ15 starts with "MM\0\x2a" and the Pentax XG-1 starts with "II\x2a\0",
# but an extra 2 bytes of padding after the IFD entry count prevents these from
# being processed as a standard IFD. Note that the offsets for the HZ15 are all
# zeros, but they seem to be mostly OK for the XG-1)
Condition => q{
$$self{Make} =~ /^(PENTAX )?RICOH/ and ($$self{Model} eq 'RICOH WG-M1' or
$$valPt =~ /^(MM\0\x2a\0\0\0\x08\0.\0\0|II\x2a\0\x08\0\0\0.\0\0\0)/s)
},
SubDirectory => {
TagTable => 'Image::ExifTool::Ricoh::Type2',
Start => '$valuePtr + 8',
Base => '$start - 8',
ByteOrder => 'Unknown',
ProcessProc => \&ProcessKodakPatch,
},
},
{
Name => 'MakerNoteRicohText',
Condition => '$$self{Make}=~/^RICOH/',
NotIFD => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Ricoh::Text',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSamsung1a',
# Samsung STMN maker notes WITHOUT PreviewImage
Condition => '$$valPt =~ /^STMN\d{3}.\0{4}/s',
Binary => 1,
Notes => 'Samsung "STMN" maker notes without PreviewImage',
},
{
Name => 'MakerNoteSamsung1b',
# Samsung STMN maker notes WITH PreviewImage
Condition => '$$valPt =~ /^STMN\d{3}/',
SubDirectory => {
TagTable => 'Image::ExifTool::Samsung::Main',
},
},
{
Name => 'MakerNoteSamsung2',
# Samsung EXIF-format maker notes
Condition => q{
$$self{Make} eq 'SAMSUNG' and ($$self{TIFF_TYPE} eq 'SRW' or
$$valPt=~/^(\0.\0\x01\0\x07\0{3}\x04|.\0\x01\0\x07\0\x04\0{3})0100/s)
},
SubDirectory => {
TagTable => 'Image::ExifTool::Samsung::Type2',
# Samsung is very inconsistent here, and uses absolute offsets for some
# models and relative offsets for others, so process as Unknown
ProcessProc => \&ProcessUnknown,
FixBase => 1,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSanyo',
# (starts with "SANYO\0")
Condition => '$$self{Make}=~/^SANYO/ and $$self{Model}!~/^(C4|J\d|S\d)\b/',
SubDirectory => {
TagTable => 'Image::ExifTool::Sanyo::Main',
Validate => '$val =~ /^SANYO/',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSanyoC4',
# The C4 offsets are wrong by 12, so they must be fixed
Condition => '$$self{Make}=~/^SANYO/ and $$self{Model}=~/^C4\b/',
SubDirectory => {
TagTable => 'Image::ExifTool::Sanyo::Main',
Validate => '$val =~ /^SANYO/',
Start => '$valuePtr + 8',
FixBase => 1,
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSanyoPatch',
# The J1, J2, J4, S1, S3 and S4 offsets are completely screwy
Condition => '$$self{Make}=~/^SANYO/',
SubDirectory => {
TagTable => 'Image::ExifTool::Sanyo::Main',
Validate => '$val =~ /^SANYO/',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
FixOffsets => 'Image::ExifTool::Sanyo::FixOffsets($valuePtr, $valEnd, $size, $tagID, $wFlag)',
},
},
{
Name => 'MakerNoteSigma',
# (starts with "SIGMA\0")
Condition => '$$self{Make}=~/^(SIGMA|FOVEON)/',
SubDirectory => {
TagTable => 'Image::ExifTool::Sigma::Main',
Validate => '$val =~ /^(SIGMA|FOVEON)/',
Start => '$valuePtr + 10',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSony',
# (starts with "SONY DSC \0" or "SONY CAM \0")
# (TF1 starts with "\0\0SONY PIC\0")
# (Hasselblad models start with "VHAB \0")
Condition => '$$valPt=~/^(SONY (DSC|CAM|MOBILE)|\0\0SONY PIC\0|VHAB \0)/',
SubDirectory => {
TagTable => 'Image::ExifTool::Sony::Main',
Start => '$valuePtr + 12',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSony2',
# (starts with "SONY PI\0" -- DSC-S650/S700/S750)
Condition => '$$valPt=~/^SONY PI\0/ and $$self{OlympusCAMER}=1',
SubDirectory => {
TagTable => 'Image::ExifTool::Olympus::Main',
Start => '$valuePtr + 12',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSony3',
# (starts with "PREMI\0" -- DSC-S45/S500)
Condition => '$$valPt=~/^(PREMI)\0/ and $$self{OlympusCAMER}=1',
SubDirectory => {
TagTable => 'Image::ExifTool::Olympus::Main',
Start => '$valuePtr + 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSony4',
# (starts with "SONY PIC\0" -- DSC-H200/J20/W370/W510, MHS-TS20)
Condition => '$$valPt=~/^SONY PIC\0/',
SubDirectory => { TagTable => 'Image::ExifTool::Sony::PIC' },
},
{
Name => 'MakerNoteSony5', # used in SR2 and ARW images
Condition => '$$self{Make}=~/^SONY/ and $$valPt!~/^\x01\x00/',
Condition => q{
($$self{Make}=~/^SONY/ or ($$self{Make}=~/^HASSELBLAD/ and
$$self{Model}=~/^(HV|Stellar|Lusso|Lunar)/)) and $$valPt!~/^\x01\x00/
},
SubDirectory => {
TagTable => 'Image::ExifTool::Sony::Main',
Start => '$valuePtr',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSonyEricsson',
Condition => '$$valPt =~ /^SEMC MS\0/',
SubDirectory => {
TagTable => 'Image::ExifTool::Sony::Ericsson',
Start => '$valuePtr + 20',
Base => '$start - 8',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteSonySRF',
Condition => '$$self{Make}=~/^SONY/',
SubDirectory => {
TagTable => 'Image::ExifTool::Sony::SRF',
Start => '$valuePtr',
ByteOrder => 'Unknown',
},
},
{
Name => 'MakerNoteUnknownText',
Condition => '$$valPt =~ /^[\x09\x0d\x0a\x20-\x7e]+\0*$/',
Notes => 'unknown text-based maker notes',
# show as binary if it is too long
ValueConv => 'length($val) > 64 ? \$val : $val',
ValueConvInv => '$val',
},
{
Name => 'MakerNoteUnknownBinary',
# "LSI1\0" - SilverFast
Condition => '$$valPt =~ /^LSI1\0/',
Notes => 'unknown binary maker notes',
Binary => 1,
},
{
Name => 'MakerNoteUnknown',
PossiblePreview => 1,
SubDirectory => {
TagTable => 'Image::ExifTool::Unknown::Main',
ProcessProc => \&ProcessUnknownOrPreview,
WriteProc => \&WriteUnknownOrPreview,
ByteOrder => 'Unknown',
FixBase => 2,
},
},
);
# insert writable properties so we can write our maker notes
my $tagInfo;
foreach $tagInfo (@Image::ExifTool::MakerNotes::Main) {
$$tagInfo{Writable} = 'undef';
$$tagInfo{Format} = 'undef', # (make sure we don't convert this when reading)
$$tagInfo{WriteGroup} = 'ExifIFD';
$$tagInfo{Groups} = { 1 => 'MakerNotes' };
next unless $$tagInfo{SubDirectory};
# make all SubDirectory tags block-writable
$$tagInfo{Binary} = 1,
$$tagInfo{MakerNotes} = 1;
}
#------------------------------------------------------------------------------
# Get normal offset of value data from end of maker note IFD
# Inputs: 0) ExifTool object reference
# Returns: Array: 0) relative flag (undef for no change)
# 1) normal offset from end of IFD to first value (empty if unknown)
# 2-N) other possible offsets used by some models
# Notes: Directory size should be validated before calling this routine
sub GetMakerNoteOffset($)
{
my $et = shift;
# figure out where we expect the value data based on camera type
my $make = $$et{Make};
my $model = $$et{Model};
my ($relative, @offsets);
# normally value data starts 4 bytes after end of directory, so this is the default.
# offsets of 0 and 4 are always allowed even if not specified,
# but the first offset specified is the one used when writing
if ($make =~ /^Canon/) {
push @offsets, ($model =~ /\b(20D|350D|REBEL XT|Kiss Digital N)\b/) ? 6 : 4;
# some Canon models (FV-M30, Optura50, Optura60) leave 24 unused bytes
# at the end of the IFD (2 spare IFD entries?)
push @offsets, 28 if $model =~ /\b(FV\b|OPTURA)/;
# some Canon PowerShot models leave 12 unused bytes
push @offsets, 16 if $model =~ /(PowerShot|IXUS|IXY)/;
} elsif ($make =~ /^CASIO/) {
# Casio AVI and MOV images use no padding, and their JPEG's use 4,
# except some models like the EX-S770,Z65,Z70,Z75 and Z700 which use 16,
# and the EX-Z35 which uses 2 (grrrr...)
push @offsets, $$et{FILE_TYPE} =~ /^(RIFF|MOV)$/ ? 0 : (4, 16, 2);
} elsif ($make =~ /^(General Imaging Co.|GEDSC IMAGING CORP.)/i) {
push @offsets, 0;
} elsif ($make =~ /^KYOCERA/) {
push @offsets, 12;
} elsif ($make =~ /^Leica Camera AG/) {
if ($model eq 'S2') {
# lots of empty space before first value in S2 images
push @offsets, 4, ($$et{FILE_TYPE} eq 'JPEG' ? 286 : 274);
} elsif ($model eq 'LEICA M MONOCHROM (Typ 246)') {
push @offsets, 4, 130;
} elsif ($model eq 'LEICA M (Typ 240)') {
push @offsets, 4, 118;
} elsif ($model =~ /^(R8|R9|M8)\b/) {
push @offsets, 6;
} else {
push @offsets, 4;
}
} elsif ($make =~ /^OLYMPUS/ and $model =~ /^E-(1|300|330)\b/) {
push @offsets, 16;
} elsif ($make =~ /^OLYMPUS/ and
# these Olympus models are just weird
$model =~ /^(C2500L|C-1Z?|C-5000Z|X-2|C720UZ|C725UZ|C150|C2Z|E-10|E-20|FerrariMODEL2003|u20D|u10D)\b/)
{
# no expected offset --> determine offset empirically via FixBase()
} elsif ($make =~ /^(Panasonic|JVC)\b/) {
push @offsets, 0;
} elsif ($make =~ /^SONY/) {
# earlier DSLR and "PREMI" models use an offset of 4
if ($model =~ /^(DSLR-.*|SLT-A(33|35|55V)|NEX-(3|5|C3|VG10E))$/ or
$$et{OlympusCAMER})
{
push @offsets, 4;
} else {
push @offsets, 0;
}
} elsif ($$et{TIFF_TYPE} eq 'SRW' and $make eq 'SAMSUNG' and $model eq 'EK-GN120') {
push @offsets, 40; # patch to read most of the maker notes, but breaks PreviewIFD
} elsif ($make eq 'FUJIFILM') {
# some models have offset of 6, so allow that too (A345,A350,A360,A370)
push @offsets, 4, 6;
} elsif ($make =~ /^TOSHIBA/) {
# similar to Canon, can also have 24 bytes of padding
push @offsets, 0, 24;
} elsif ($make =~ /^PENTAX/) {
push @offsets, 4;
# the Pentax addressing mode is determined automatically, but
# sometimes the algorithm gets it wrong, but Pentax always uses
# absolute addressing, so force it to be absolute
$relative = 0;
} elsif ($make =~ /^Konica Minolta/i) {
# patch for DiMAGE X50, Xg, Z2 and Z10
push @offsets, 4, -16;
} elsif ($make =~ /^Minolta/) {
# patch for DiMAGE 7, X20 and Z1
push @offsets, 4, -8, -12;
} else {
push @offsets, 4; # the normal offset
}
return ($relative, @offsets);
}
#------------------------------------------------------------------------------
# Get hash of value offsets / block sizes
# Inputs: 0) Data pointer, 1) offset to start of directory,
# 2) hash ref to return value pointers based on tag ID
# Returns: 0) hash reference: keys are offsets, values are block sizes
# 1) same thing, but with keys adjusted for value-based offsets
# Notes: Directory size should be validated before calling this routine
# - calculates MIN and MAX offsets in entry-based hash
sub GetValueBlocks($$;$)
{
my ($dataPt, $dirStart, $tagPtr) = @_;
my $numEntries = Get16u($dataPt, $dirStart);
my ($index, $valPtr, %valBlock, %valBlkAdj, $end);
for ($index=0; $index<$numEntries; ++$index) {
my $entry = $dirStart + 2 + 12 * $index;
my $format = Get16u($dataPt, $entry+2);
last if $format < 1 or $format > 13;
my $count = Get32u($dataPt, $entry+4);
my $size = $count * $Image::ExifTool::Exif::formatSize[$format];
next if $size <= 4;
$valPtr = Get32u($dataPt, $entry+8);
$tagPtr and $$tagPtr{Get16u($dataPt, $entry)} = $valPtr;
# save location and size of longest block at this offset
unless (defined $valBlock{$valPtr} and $valBlock{$valPtr} > $size) {
$valBlock{$valPtr} = $size;
}
# adjust for case of value-based offsets
$valPtr += 12 * $index;
unless (defined $valBlkAdj{$valPtr} and $valBlkAdj{$valPtr} > $size) {
$valBlkAdj{$valPtr} = $size;
my $end = $valPtr + $size;
if (defined $valBlkAdj{MIN}) {
# save minimum only if it has a value of 12 or greater
$valBlkAdj{MIN} = $valPtr if $valBlkAdj{MIN} < 12 or $valBlkAdj{MIN} > $valPtr;
$valBlkAdj{MAX} = $end if $valBlkAdj{MAX} > $end;
} else {
$valBlkAdj{MIN} = $valPtr;
$valBlkAdj{MAX} = $end;
}
}
}
return(\%valBlock, \%valBlkAdj);
}
#------------------------------------------------------------------------------
# Fix base for value offsets in maker notes IFD (if necessary)
# Inputs: 0) ExifTool object ref, 1) DirInfo hash ref
# Return: amount of base shift (and $dirInfo Base and DataPos are updated,
# FixedBy is set if offsets fixed, and Relative or EntryBased may be set)
sub FixBase($$)
{
local $_;
my ($et, $dirInfo) = @_;
# don't fix base if fixing offsets individually or if we don't want to fix them
return 0 if $$dirInfo{FixOffsets} or $$dirInfo{NoFixBase};
my $dataPt = $$dirInfo{DataPt};
my $dataPos = $$dirInfo{DataPos};
my $dirStart = $$dirInfo{DirStart} || 0;
my $entryBased = $$dirInfo{EntryBased};
my $dirName = $$dirInfo{DirName};
my $fixBase = $et->Options('FixBase');
my $setBase = (defined $fixBase and $fixBase ne '') ? 1 : 0;
my ($fix, $fixedBy, %tagPtr);
# get hash of value block positions
my ($valBlock, $valBlkAdj) = GetValueBlocks($dataPt, $dirStart, \%tagPtr);
return 0 unless %$valBlock;
# get sorted list of value offsets
my @valPtrs = sort { $a <=> $b } keys %$valBlock;
#
# handle special case of Canon maker notes with TIFF footer containing original offset
#
if ($$et{Make} =~ /^Canon/ and $$dirInfo{DirLen} > 8) {
my $footerPos = $dirStart + $$dirInfo{DirLen} - 8;
my $footer = substr($$dataPt, $footerPos, 8);
if ($footer =~ /^(II\x2a\0|MM\0\x2a)/ and # check for TIFF footer
substr($footer,0,2) eq GetByteOrder()) # validate byte ordering
{
my $oldOffset = Get32u(\$footer, 4);
my $newOffset = $dirStart + $dataPos;
if ($setBase) {
$fix = $fixBase;
} else {
$fix = $newOffset - $oldOffset;
return 0 unless $fix;
# Picasa and ACDSee have a bug where they update other offsets without
# updating the TIFF footer (PH - 2009/02/25), so test for this case:
# validate Canon maker note footer fix by checking offset of last value
my $maxPt = $valPtrs[-1] + $$valBlock{$valPtrs[-1]};
# compare to end of maker notes, taking 8-byte footer into account
my $endDiff = $dirStart + $$dirInfo{DirLen} - ($maxPt - $dataPos) - 8;
# ignore footer offset only if end difference is exactly correct
# (allow for possible padding byte, although I have never seen this)
if (not $endDiff or $endDiff == 1) {
$et->Warn('Canon maker note footer may be invalid (ignored)',1);
return 0;
}
}
$et->Warn("Adjusted $dirName base by $fix",1);
$$dirInfo{FixedBy} = $fix;
$$dirInfo{Base} += $fix;
$$dirInfo{DataPos} -= $fix;
return $fix;
}
}
#
# analyze value offsets to see if they need fixing. The first task is to determine
# the minimum valid offset used (this is tricky, because we have to weed out bad
# offsets written by some cameras)
#
my $minPt = $$dirInfo{MinOffset} = $valPtrs[0]; # if life were simple, this would be it
my $ifdLen = 2 + 12 * Get16u($$dirInfo{DataPt}, $dirStart);
my $ifdEnd = $dirStart + $ifdLen;
my ($relative, @offsets) = GetMakerNoteOffset($et);
my $makeDiff = $offsets[0];
my $verbose = $et->Options('Verbose');
my ($diff, $shift);
# calculate expected minimum value offset
my $expected = $dataPos + $ifdEnd + (defined $makeDiff ? $makeDiff : 4);
$debug and print "$expected expected\n";
# zero our counters
my ($countNeg12, $countZero, $countOverlap) = (0, 0, 0);
my ($valPtr, $last);
foreach $valPtr (@valPtrs) {
printf("%d - %d (%d)\n", $valPtr, $valPtr + $$valBlock{$valPtr},
$valPtr - ($last || 0)) if $debug;
if (defined $last) {
my $gap = $valPtr - $last;
if ($gap == 0 or $gap == 1) {
++$countZero;
} elsif ($gap == -12 and not $entryBased) {
# you get this when value offsets are relative to the IFD entry
++$countNeg12;
} elsif ($gap < 0) {
# any other negative difference indicates overlapping values
++$countOverlap if $valPtr; # (but ignore zero value pointers)
} elsif ($gap >= $ifdLen) {
# ignore previous minimum if we took a jump to near the expected value
# (some values can be stored before the IFD)
$minPt = $valPtr if abs($valPtr - $expected) <= 4;
}
# an offset less than 12 is surely garbage, so ignore it
$minPt = $valPtr if $minPt < 12;
}
$last = $valPtr + $$valBlock{$valPtr};
}
# could this IFD be using entry-based offsets?
if ((($countNeg12 > $countZero and $$valBlkAdj{MIN} >= $ifdLen - 2) or
($$valBlkAdj{MIN} == $ifdLen - 2 or $$valBlkAdj{MIN} == $ifdLen + 2)
) and $$valBlkAdj{MAX} <= $$dirInfo{DirLen}-2)
{
# looks like these offsets are entry-based, so use the offsets
# which have been correcting for individual entry position
$entryBased = 1;
$verbose and $et->Warn("$dirName offsets are entry-based",1);
} else {
# calculate offset difference from end of IFD to first value
$diff = ($minPt - $dataPos) - $ifdEnd;
$shift = 0;
$countOverlap and $et->Warn("Overlapping $dirName values",1);
if ($entryBased) {
$et->Warn("$dirName offsets do NOT look entry-based",1);
undef $entryBased;
undef $relative;
}
# use PrintIM tag to do special check for correct absolute offsets
if ($tagPtr{0xe00}) {
my $ptr = $tagPtr{0xe00} - $dataPos;
return 0 if $ptr > 0 and $ptr <= length($$dataPt) - 8 and
substr($$dataPt, $ptr, 8) eq "PrintIM\0";
}
# allow a range of reasonable differences for Unknown maker notes
if ($$dirInfo{FixBase} and $$dirInfo{FixBase} == 2) {
return 0 if $diff >=0 and $diff <= 24;
}
# ******** (used for testing to extract differences) ********
# $et->FoundTag('Diff', $diff);
# $et->FoundTag('MakeDiff',$makeDiff);
}
#
# handle entry-based offsets
#
if ($entryBased) {
$debug and print "--> entry-based!\n";
# most of my entry-based samples have first value immediately
# after last IFD entry (ie. no padding or next IFD pointer)
$makeDiff = 0;
push @offsets, 4; # but some do have a next IFD pointer
# corrected entry-based offsets are relative to start of first entry
my $expected = 12 * Get16u($$dirInfo{DataPt}, $dirStart);
$diff = $$valBlkAdj{MIN} - $expected;
# set up directory to read values with entry-based offsets
# (ignore everything and set base to start of first entry)
$shift = $dataPos + $dirStart + 2;
$$dirInfo{Base} += $shift;
$$dirInfo{DataPos} -= $shift;
$$dirInfo{EntryBased} = 1;
$$dirInfo{Relative} = 1; # entry-based offsets are relative
delete $$dirInfo{FixBase}; # no automatic base fix
undef $fixBase unless $setBase;
}
#
# return without doing shift if offsets look OK
#
unless ($setBase) {
# don't try to fix offsets for whacky cameras
return $shift unless defined $makeDiff;
# normal value data starts 4 bytes after IFD, but allow 0 or 4...
return $shift if $diff == 0 or $diff == 4;
# also check for allowed make-specific differences
foreach (@offsets) {
return $shift if $diff == $_;
}
}
#
# apply the fix, or issue a warning
#
# use default padding of 4 bytes unless already specified
$makeDiff = 4 unless defined $makeDiff;
$fix = $makeDiff - $diff; # assume standard diff for this make
if ($$dirInfo{FixBase}) {
# set flag if offsets are relative (base is at or above directory start)
if ($dataPos - $fix + $dirStart <= 0) {
$$dirInfo{Relative} = (defined $relative) ? $relative : 1;
}
if ($setBase) {
$fixedBy = $fixBase;
$fix += $fixBase;
}
} elsif (defined $fixBase) {
$fix = $fixBase if $fixBase ne '';
$fixedBy = $fix;
} else {
# print warning unless difference looks reasonable
if ($diff < 0 or $diff > 16 or ($diff & 0x01)) {
$et->Warn("Possibly incorrect maker notes offsets (fix by $fix?)",1);
}
# don't do the fix (but we already adjusted base if entry-based)
return $shift;
}
if (defined $fixedBy) {
$et->Warn("Adjusted $dirName base by $fixedBy",1);
$$dirInfo{FixedBy} = $fixedBy;
}
$$dirInfo{Base} += $fix;
$$dirInfo{DataPos} -= $fix;
return $fix + $shift;
}
#------------------------------------------------------------------------------
# Find start of IFD in unknown maker notes
# Inputs: 0) reference to directory information
# Returns: offset to IFD on success, undefined otherwise
# - dirInfo may contain TagInfo reference for tag associated with directory
# - on success, updates DirStart, DirLen, Base and DataPos in dirInfo
# - also sets Relative flag in dirInfo if offsets are relative to IFD
# Note: Changes byte ordering!
sub LocateIFD($$)
{
my ($et, $dirInfo) = @_;
my $dataPt = $$dirInfo{DataPt};
my $dirStart = $$dirInfo{DirStart} || 0;
# (ignore MakerNotes DirLen since sometimes this is incorrect)
my $size = $$dirInfo{DataLen} - $dirStart;
my $dirLen = defined $$dirInfo{DirLen} ? $$dirInfo{DirLen} : $size;
my $tagInfo = $$dirInfo{TagInfo};
my $ifdOffsetPos;
# the IFD should be within the first 32 bytes
# (Kyocera sets the current record at 22 bytes)
my ($firstTry, $lastTry) = (0, 32);
# make sure Base and DataPos are defined
$$dirInfo{Base} or $$dirInfo{Base} = 0;
$$dirInfo{DataPos} or $$dirInfo{DataPos} = 0;
#
# use tag information (if provided) to determine directory location
#
if ($tagInfo and $$tagInfo{SubDirectory}) {
my $subdir = $$tagInfo{SubDirectory};
unless ($$subdir{ProcessProc} and
($$subdir{ProcessProc} eq \&ProcessUnknown or
$$subdir{ProcessProc} eq \&ProcessUnknownOrPreview))
{
# look for the IFD at the "Start" specified in our SubDirectory information
my $valuePtr = $dirStart;
my $newStart = $dirStart;
if (defined $$subdir{Start}) {
#### eval Start ($valuePtr)
$newStart = eval($$subdir{Start});
}
if ($$subdir{Base}) {
# calculate subdirectory start relative to $base for eval
my $start = $newStart + $$dirInfo{DataPos};
my $base = $$dirInfo{Base} || 0;
#### eval Base ($start,$base)
my $baseShift = eval($$subdir{Base});
# shift directory base (note: we may do this again below
# if an OffsetPt is defined, but that doesn't matter since
# the base shift is relative to DataPos, which we set too)
$$dirInfo{Base} += $baseShift;
$$dirInfo{DataPos} -= $baseShift;
# this is a relative directory if Base depends on $start
if ($$subdir{Base} =~ /\$start\b/) {
$$dirInfo{Relative} = 1;
# hack to fix Leica quirk
if ($$subdir{ProcessProc} and $$subdir{ProcessProc} eq \&FixLeicaBase) {
my $oldStart = $$dirInfo{DirStart};
$$dirInfo{DirStart} = $newStart;
FixLeicaBase($et, $dirInfo);
$$dirInfo{DirStart} = $oldStart;
}
}
}
# add offset to the start of the directory if necessary
if ($$subdir{OffsetPt}) {
if ($$subdir{ByteOrder} =~ /^Little/i) {
SetByteOrder('II');
} elsif ($$subdir{ByteOrder} =~ /^Big/i) {
SetByteOrder('MM');
} else {
warn "Can't have variable byte ordering for SubDirectories using OffsetPt\n";
return undef;
}
#### eval OffsetPt ($valuePtr)
$ifdOffsetPos = eval($$subdir{OffsetPt}) - $dirStart;
}
# pinpoint position to look for this IFD
$firstTry = $lastTry = $newStart - $dirStart;
}
}
#
# scan for something that looks like an IFD
#
if ($dirLen >= 14 + $firstTry) { # minimum size for an IFD
my $offset;
IFD_TRY: for ($offset=$firstTry; $offset<=$lastTry; $offset+=2) {
last if $offset + 14 > $dirLen; # 14 bytes is minimum size for an IFD
my $pos = $dirStart + $offset;
#
# look for a standard TIFF header (Nikon uses it, others may as well),
#
if (SetByteOrder(substr($$dataPt, $pos, 2)) and
Get16u($dataPt, $pos + 2) == 0x2a)
{
$ifdOffsetPos = 4;
}
if (defined $ifdOffsetPos) {
# get pointer to IFD
my $ptr = Get32u($dataPt, $pos + $ifdOffsetPos);
if ($ptr >= $ifdOffsetPos + 4 and $ptr + $offset + 14 <= $dirLen) {
# shift directory start and shorten dirLen accordingly
$$dirInfo{DirStart} += $ptr + $offset;
$$dirInfo{DirLen} -= $ptr + $offset;
# shift pointer base to the start of the TIFF header
my $shift = $$dirInfo{DataPos} + $dirStart + $offset;
$$dirInfo{Base} += $shift;
$$dirInfo{DataPos} -= $shift;
$$dirInfo{Relative} = 1; # set "relative offsets" flag
return $ptr + $offset;
}
undef $ifdOffsetPos;
}
#
# look for a standard IFD (starts with 2-byte entry count)
#
my $num = Get16u($dataPt, $pos);
next unless $num;
# number of entries in an IFD should be between 1 and 255
if (!($num & 0xff)) {
# lower byte is zero -- byte order could be wrong
ToggleByteOrder();
$num >>= 8;
} elsif ($num & 0xff00) {
# upper byte isn't zero -- not an IFD
next;
}
my $bytesFromEnd = $size - ($offset + 2 + 12 * $num);
if ($bytesFromEnd < 4) {
next unless $bytesFromEnd == 2 or $bytesFromEnd == 0;
}
# do a quick validation of all format types
my $index;
for ($index=0; $index<$num; ++$index) {
my $entry = $pos + 2 + 12 * $index;
my $format = Get16u($dataPt, $entry+2);
my $count = Get32u($dataPt, $entry+4);
unless ($format) {
# patch for buggy Samsung NX200 JPEG MakerNotes entry count
if ($num == 23 and $index == 21 and $$et{Make} eq 'SAMSUNG') {
Set16u(21, $dataPt, $pos); # really 21 IFD entries!
$et->Warn('Fixed incorrect Makernote entry count', 1);
last;
}
# allow everything to be zero if not first entry
# because some manufacturers pad with null entries
next unless $count or $index == 0;
# patch for Canon EOS 40D firmware 1.0.4 bug: allow zero format for last entry
next if $index==$num-1 and $$et{Model}=~/EOS 40D/;
}
# patch for Sony cameras like the DSC-P10 that have invalid MakerNote entries
next if $num == 12 and $$et{Make} eq 'SONY' and $index >= 8;
# (would like to verify tag ID, but some manufactures don't
# sort entries in order of tag ID so we don't have much of
# a handle to verify this field)
# verify format
next IFD_TRY if $format < 1 or $format > 13;
# count must be reasonable (can't test for zero count because
# cameras like the 1DmkIII use this value)
next IFD_TRY if $count & 0xff000000;
# extra tests to avoid mis-identifying Samsung makernotes (GT-I9000, etc)
next unless $num == 1;
my $valueSize = $count * $Image::ExifTool::Exif::formatSize[$format];
if ($valueSize > 4) {
next IFD_TRY if $valueSize > $size;
my $valuePtr = Get32u($dataPt, $entry+8);
next IFD_TRY if $valuePtr > 0x10000;
}
}
$$dirInfo{DirStart} += $offset; # update directory start
$$dirInfo{DirLen} -= $offset;
return $offset; # success!!
}
}
return undef;
}
#------------------------------------------------------------------------------
# Fix base offset for Leica maker notes
# Inputs: 0) ExifTool object ref, 1) dirInfo ref, 2) tag table ref
# Returns: 1 on success, and updates $dirInfo if necessary for new directory
sub FixLeicaBase($$;$)
{
my ($et, $dirInfo, $tagTablePtr) = @_;
my $dataPt = $$dirInfo{DataPt};
my $dirStart = $$dirInfo{DirStart} || 0;
# get hash of value block positions
my ($valBlock, $valBlkAdj) = GetValueBlocks($dataPt, $dirStart);
if (%$valBlock) {
# get sorted list of value offsets
my @valPtrs = sort { $a <=> $b } keys %$valBlock;
my $numEntries = Get16u($dataPt, $dirStart);
my $diff = $valPtrs[0] - ($numEntries * 12 + 4);
if ($diff > 8) {
$$dirInfo{Base} -= 8;
$$dirInfo{DataPos} += 8;
}
}
my $success = 1;
if ($tagTablePtr) {
$success = Image::ExifTool::Exif::ProcessExif($et, $dirInfo, $tagTablePtr);
}
return $success;
}
#------------------------------------------------------------------------------
# Process Canon maker notes
# Inputs: 0) ExifTool object ref, 1) dirInfo ref, 2) tag table ref
# Returns: 1 on success
sub ProcessCanon($$$)
{
my ($et, $dirInfo, $tagTablePtr) = @_;
# identify Canon MakerNote footer in HtmlDump
# (this code moved from FixBase so it also works for Adobe MakN in DNG images)
if ($$et{HTML_DUMP} and $$dirInfo{DirLen} > 8) {
my $dataPos = $$dirInfo{DataPos};
my $dirStart = $$dirInfo{DirStart} || 0;
my $footerPos = $dirStart + $$dirInfo{DirLen} - 8;
my $footer = substr(${$$dirInfo{DataPt}}, $footerPos, 8);
if ($footer =~ /^(II\x2a\0|MM\0\x2a)/ and substr($footer,0,2) eq GetByteOrder()) {
my $oldOffset = Get32u(\$footer, 4);
my $newOffset = $dirStart + $dataPos;
my $str = sprintf('Original maker note offset: 0x%.4x', $oldOffset);
if ($oldOffset != $newOffset) {
$str .= sprintf("\nCurrent maker note offset: 0x%.4x", $newOffset);
}
my $filePos = ($$dirInfo{Base} || 0) + $dataPos + $footerPos;
$et->HDump($filePos, 8, '[Canon MakerNotes footer]', $str);
}
}
# process as normal
return Image::ExifTool::Exif::ProcessExif($et, $dirInfo, $tagTablePtr);
}
#------------------------------------------------------------------------------
# Process GE type 2 maker notes
# Inputs: 0) ExifTool object ref, 1) DirInfo ref, 2) tag table ref
# Returns: 1 on success
sub ProcessGE2($$$)
{
my ($et, $dirInfo, $tagTablePtr) = @_;
my $dataPt = $$dirInfo{DataPt} or return 0;
my $dirStart = $$dirInfo{DirStart} || 0;
# these maker notes are missing the IFD entry count, but they
# always have 25 entries, so write the entry count manually
Set16u(25, $dataPt, $dirStart);
return Image::ExifTool::Exif::ProcessExif($et, $dirInfo, $tagTablePtr);
}
#------------------------------------------------------------------------------
# Process broken Kodak type 8b maker notes
# Inputs: 0) ExifTool object ref, 1) DirInfo ref, 2) tag table ref
# Returns: 1 on success
sub ProcessKodakPatch($$$)
{
my ($et, $dirInfo, $tagTablePtr) = @_;
my $dataPt = $$dirInfo{DataPt} or return 0;
my $dirStart = $$dirInfo{DirStart} || 0;
# these maker notes have an 2 extra null bytes either before or after the entry count,
# so fix this by making a 2-byte entry count just before the first IFD entry
return 0 unless $$dirInfo{DirLen} >= 4;
my $t1 = Get16u($dataPt,$dirStart);
my $t2 = Get16u($dataPt,$dirStart+2);
Set16u($t1 || $t2, $dataPt, $dirStart+2);
$$dirInfo{DirStart} += 2;
return Image::ExifTool::Exif::ProcessExif($et, $dirInfo, $tagTablePtr);
}
#------------------------------------------------------------------------------
# Process unknown maker notes or PreviewImage
# Inputs: 0) ExifTool object ref, 1) dirInfo ref, 2) tag table ref
# Returns: 1 on success, and updates $dirInfo if necessary for new directory
sub ProcessUnknownOrPreview($$$)
{
my ($et, $dirInfo, $tagTablePtr) = @_;
my $dataPt = $$dirInfo{DataPt};
my $dirStart = $$dirInfo{DirStart};
my $dirLen = $$dirInfo{DirLen};
# check to see if this is a preview image
if ($dirLen > 6 and substr($$dataPt, $dirStart, 3) eq "\xff\xd8\xff") {
$et->VerboseDir('PreviewImage');
if ($$et{HTML_DUMP}) {
my $pos = $$dirInfo{DataPos} + $$dirInfo{Base} + $dirStart;
$et->HDump($pos, $dirLen, '(MakerNotes:PreviewImage data)', "Size: $dirLen bytes")
}
$et->FoundTag('PreviewImage', substr($$dataPt, $dirStart, $dirLen));
return 1;
}
return ProcessUnknown($et, $dirInfo, $tagTablePtr);
}
#------------------------------------------------------------------------------
# Write unknown maker notes or PreviewImage
# Inputs: 0) ExifTool object ref, 1) dirInfo ref, 2) tag table ref
# Returns: directory data, '' to delete, or undef on error
sub WriteUnknownOrPreview($$$)
{
my ($et, $dirInfo, $tagTablePtr) = @_;
my $dataPt = $$dirInfo{DataPt};
my $dirStart = $$dirInfo{DirStart};
my $dirLen = $$dirInfo{DirLen};
my $newVal;
# check to see if this is a preview image
if ($dirLen > 6 and substr($$dataPt, $dirStart, 3) eq "\xff\xd8\xff") {
if ($$et{NEW_VALUE}{$Image::ExifTool::Extra{PreviewImage}}) {
# write or delete new preview (if deleted, it can't currently be added back again)
$newVal = $et->GetNewValue('PreviewImage') || '';
if ($et->Options('Verbose') > 1) {
$et->VerboseValue("- MakerNotes:PreviewImage", substr($$dataPt, $dirStart, $dirLen));
$et->VerboseValue("+ MakerNotes:PreviewImage", $newVal) if $newVal;
}
++$$et{CHANGED};
} else {
$newVal = substr($$dataPt, $dirStart, $dirLen);
}
} else {
# rewrite MakerNote IFD
$newVal = Image::ExifTool::Exif::WriteExif($et, $dirInfo, $tagTablePtr);
}
return $newVal;
}
#------------------------------------------------------------------------------
# Process unknown maker notes assuming it is in EXIF IFD format
# Inputs: 0) ExifTool object ref, 1) dirInfo ref, 2) tag table ref
# Returns: 1 on success, and updates $dirInfo if necessary for new directory
sub ProcessUnknown($$$)
{
my ($et, $dirInfo, $tagTablePtr) = @_;
my $success = 0;
my $loc = LocateIFD($et, $dirInfo);
if (defined $loc) {
$$et{UnknownByteOrder} = GetByteOrder();
if ($et->Options('Verbose') > 1) {
my $out = $et->Options('TextOut');
my $indent = $$et{INDENT};
$indent =~ s/\| $/ /;
printf $out "${indent}Found IFD at offset 0x%.4x in maker notes:\n",
$$dirInfo{DirStart} + $$dirInfo{DataPos} + $$dirInfo{Base};
}
$success = Image::ExifTool::Exif::ProcessExif($et, $dirInfo, $tagTablePtr);
} else {
$$et{UnknownByteOrder} = ''; # indicates we tried but didn't set byte order
$et->Warn("Unrecognized $$dirInfo{DirName}", 1);
}
return $success;
}
1; # end
__END__
=head1 NAME
Image::ExifTool::MakerNotes - Read and write EXIF maker notes
=head1 SYNOPSIS
This module is required by Image::ExifTool.
=head1 DESCRIPTION
This module contains definitions required by Image::ExifTool to interpret
maker notes in EXIF information.
=head1 AUTHOR
Copyright 2003-2018, Phil Harvey (phil at owl.phy.queensu.ca)
This library is free software; you can redistribute it and/or modify it
under the same terms as Perl itself.
=head1 SEE ALSO
L<Image::ExifTool::TagNames(3pm)|Image::ExifTool::TagNames>,
L<Image::ExifTool(3pm)|Image::ExifTool>
=cut