#------------------------------------------------------------------------------ # File: MinoltaRaw.pm # # Description: Read/write Konica-Minolta RAW (MRW) meta information # # Revisions: 03/11/2006 - P. Harvey Split out from Minolta.pm # # References: 1) http://www.cybercom.net/~dcoffin/dcraw/ # 2) http://www.chauveau-central.net/mrw-format/ # 3) Igal Milchtaich private communication (A100) #------------------------------------------------------------------------------ package Image::ExifTool::MinoltaRaw; use strict; use vars qw($VERSION); use Image::ExifTool qw(:DataAccess :Utils); use Image::ExifTool::Minolta; $VERSION = '1.15'; sub ProcessMRW($$;$); sub WriteMRW($$;$); # Minolta MRW tags %Image::ExifTool::MinoltaRaw::Main = ( GROUPS => { 0 => 'MakerNotes', 2 => 'Camera' }, PROCESS_PROC => \&Image::ExifTool::MinoltaRaw::ProcessMRW, WRITE_PROC => \&Image::ExifTool::MinoltaRaw::WriteMRW, NOTES => 'These tags are used in Minolta RAW format (MRW) images.', "\0TTW" => { # TIFF Tags Name => 'MinoltaTTW', SubDirectory => { TagTable => 'Image::ExifTool::Exif::Main', # this EXIF information starts with a TIFF header ProcessProc => \&Image::ExifTool::ProcessTIFF, WriteProc => \&Image::ExifTool::WriteTIFF, }, }, "\0PRD" => { # Raw Picture Dimensions Name => 'MinoltaPRD', SubDirectory => { TagTable => 'Image::ExifTool::MinoltaRaw::PRD' }, }, "\0WBG" => { # White Balance Gains Name => 'MinoltaWBG', SubDirectory => { TagTable => 'Image::ExifTool::MinoltaRaw::WBG' }, }, "\0RIF" => { # Requested Image Format Name => 'MinoltaRIF', SubDirectory => { TagTable => 'Image::ExifTool::MinoltaRaw::RIF' }, }, # "\0CSA" is padding ); # Minolta MRW PRD information (ref 2) %Image::ExifTool::MinoltaRaw::PRD = ( PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, WRITE_PROC => \&Image::ExifTool::WriteBinaryData, CHECK_PROC => \&Image::ExifTool::CheckBinaryData, WRITABLE => 1, GROUPS => { 0 => 'MakerNotes', 2 => 'Camera' }, FIRST_ENTRY => 0, 0 => { Name => 'FirmwareID', Format => 'string[8]', }, 8 => { Name => 'SensorHeight', Format => 'int16u', }, 10 => { Name => 'SensorWidth', Format => 'int16u', }, 12 => { Name => 'ImageHeight', Format => 'int16u', }, 14 => { Name => 'ImageWidth', Format => 'int16u', }, 16 => { Name => 'RawDepth', Format => 'int8u', }, 17 => { Name => 'BitDepth', Format => 'int8u', }, 18 => { Name => 'StorageMethod', Format => 'int8u', PrintConv => { 82 => 'Padded', 89 => 'Linear', }, }, 23 => { Name => 'BayerPattern', Format => 'int8u', PrintConv => { # 0 - seen in some Sony A850 ARW images 1 => 'RGGB', 4 => 'GBRG', }, }, ); # Minolta MRW WBG information (ref 2) %Image::ExifTool::MinoltaRaw::WBG = ( PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, WRITE_PROC => \&Image::ExifTool::WriteBinaryData, CHECK_PROC => \&Image::ExifTool::CheckBinaryData, WRITABLE => 1, GROUPS => { 0 => 'MakerNotes', 2 => 'Camera' }, FIRST_ENTRY => 0, 0 => { Name => 'WBScale', Format => 'int8u[4]', }, 4 => [ { Condition => '$$self{Model} =~ /DiMAGE A200\b/', Name => 'WB_GBRGLevels', Format => 'int16u[4]', Notes => 'DiMAGE A200', }, { Name => 'WB_RGGBLevels', Format => 'int16u[4]', Notes => 'other models', }, ], ); # Minolta MRW RIF information (ref 2) %Image::ExifTool::MinoltaRaw::RIF = ( PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, WRITE_PROC => \&Image::ExifTool::WriteBinaryData, CHECK_PROC => \&Image::ExifTool::CheckBinaryData, WRITABLE => 1, GROUPS => { 0 => 'MakerNotes', 2 => 'Image' }, FIRST_ENTRY => 0, 1 => { Name => 'Saturation', Format => 'int8s', }, 2 => { Name => 'Contrast', Format => 'int8s', }, 3 => { Name => 'Sharpness', Format => 'int8s', }, 4 => { Name => 'WBMode', PrintConv => 'Image::ExifTool::MinoltaRaw::ConvertWBMode($val)', }, 5 => { Name => 'ProgramMode', PrintConv => { 0 => 'None', 1 => 'Portrait', 2 => 'Text', 3 => 'Night Portrait', 4 => 'Sunset', 5 => 'Sports', # have seen these values in Sony ARW images: - PH # 7, 128, 129, 160 }, }, 6 => { Name => 'ISOSetting', RawConv => '$val == 255 ? undef : $val', PrintConv => { #3 0 => 'Auto', 48 => 100, 56 => 200, 64 => 400, 72 => 800, 80 => 1600, 174 => '80 (Zone Matching Low)', 184 => '200 (Zone Matching High)', OTHER => sub { my ($val, $inv) = @_; return int(2 ** (($val-48)/8) * 100 + 0.5) unless $inv; # (must check for zero below in inverse conversion) return 48 + 8*log($val/100)/log(2) if Image::ExifTool::IsFloat($val) and $val != 0; return undef; }, }, #ValueConv => '2 ** (($val-48)/8) * 100', #ValueConvInv => '48 + 8*log($val/100)/log(2)', #PrintConv => 'int($val + 0.5)', #PrintConvInv => '$val', }, 7 => [ { Name => 'ColorMode', Condition => '$$self{Make} !~ /^SONY/', Priority => 0, Writable => 'int32u', PrintConv => \%Image::ExifTool::Minolta::minoltaColorMode, }, { #3 Name => 'ColorMode', Condition => '$$self{Model} eq "DSLR-A100"', Writable => 'int32u', Notes => 'Sony A100', Priority => 0, PrintHex => 1, PrintConv => \%Image::ExifTool::Minolta::sonyColorMode, }, ], # NOTE: some of these WB_RBLevels may apply to other models too... 8 => { #3 Name => 'WB_RBLevelsTungsten', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', Notes => 'these WB_RBLevels currently decoded only for the Sony A100', }, 12 => { #3 Name => 'WB_RBLevelsDaylight', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 16 => { #3 Name => 'WB_RBLevelsCloudy', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 20 => { #3 Name => 'WB_RBLevelsCoolWhiteF', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 24 => { #3 Name => 'WB_RBLevelsFlash', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 28 => { #3 Name => 'WB_RBLevelsUnknown', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', Unknown => 1, }, 32 => { #3 Name => 'WB_RBLevelsShade', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 36 => { #3 Name => 'WB_RBLevelsDaylightF', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 40 => { #3 Name => 'WB_RBLevelsDayWhiteF', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 44 => { #3 Name => 'WB_RBLevelsWhiteF', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int16u[2]', }, 56 => { Name => 'ColorFilter', Condition => '$$self{Make} !~ /^SONY/', Format => 'int8s', Notes => 'Minolta models', }, 57 => 'BWFilter', 58 => { Name => 'ZoneMatching', Condition => '$$self{Make} !~ /^SONY/', Priority => 0, Notes => 'Minolta models', PrintConv => { 0 => 'ISO Setting Used', 1 => 'High Key', 2 => 'Low Key', }, }, 59 => { Name => 'Hue', Format => 'int8s', }, 60 => { Name => 'ColorTemperature', Condition => '$$self{Make} !~ /^SONY/', Notes => 'Minolta models', ValueConv => '$val * 100', ValueConvInv => '$val / 100', }, 74 => { #3 Name => 'ZoneMatching', Condition => '$$self{Make} =~ /^SONY/', Priority => 0, Notes => 'Sony models', PrintConv => { 0 => 'ISO Setting Used', 1 => 'High Key', 2 => 'Low Key', }, }, 76 => { #3 Name => 'ColorTemperature', Condition => '$$self{Model} eq "DSLR-A100"', Notes => 'A100', ValueConv => '$val * 100', ValueConvInv => '$val / 100', PrintConv => '$val ? $val : "Auto"', PrintConvInv => '$val=~/Auto/i ? 0 : $val', }, 77 => { #3 Name => 'ColorFilter', Condition => '$$self{Model} eq "DSLR-A100"', Notes => 'A100', }, 78 => { #3 Name => 'ColorTemperature', Condition => '$$self{Model} =~ /^DSLR-A(200|700)$/', Notes => 'A200 and A700', ValueConv => '$val * 100', ValueConvInv => '$val / 100', PrintConv => '$val ? $val : "Auto"', PrintConvInv => '$val=~/Auto/i ? 0 : $val', }, 79 => { #3 Name => 'ColorFilter', Condition => '$$self{Model} =~ /^DSLR-A(200|700)$/', Notes => 'A200 and A700', }, 80 => { #3 Name => 'RawDataLength', Condition => '$$self{Model} eq "DSLR-A100"', Format => 'int32u', Notes => 'A100', Writable => 0, }, ); #------------------------------------------------------------------------------ # PrintConv for WBMode sub ConvertWBMode($) { my $val = shift; my %mrwWB = ( 0 => 'Auto', 1 => 'Daylight', 2 => 'Cloudy', 3 => 'Tungsten', 4 => 'Flash/Fluorescent', 5 => 'Fluorescent', 6 => 'Shade', 7 => 'User 1', 8 => 'User 2', 9 => 'User 3', 10 => 'Temperature', ); my $lo = $val & 0x0f; my $wbstr = $mrwWB{$lo} || "Unknown ($lo)"; my $hi = $val >> 4; $wbstr .= ' (' . ($hi - 8) . ')' if $hi >= 6 and $hi <=12; return $wbstr; } #------------------------------------------------------------------------------ # Write MRW directory (eg. in ARW images) # Inputs: 0) ExifTool object ref, 1) dirInfo ref, 2) optional tag table ref # Returns: new MRW data or undef on error sub WriteMRW($$;$) { my ($et, $dirInfo, $tagTablePtr) = @_; $et or return 1; # allow dummy access my $buff = ''; $$dirInfo{OutFile} = \$buff; ProcessMRW($et, $dirInfo, $tagTablePtr) > 0 or undef $buff; return $buff; } #------------------------------------------------------------------------------ # Read or write Minolta MRW file # Inputs: 0) ExifTool object ref, 1) dirInfo ref, 2) optional tag table ref # Returns: 1 on success, 0 if this wasn't a valid MRW file, or -1 on write error # Notes: File pointer must be set to start of MRW in RAF upon entry sub ProcessMRW($$;$) { my ($et, $dirInfo, $tagTablePtr) = @_; my $raf = $$dirInfo{RAF}; my $outfile = $$dirInfo{OutFile}; my $verbose = $et->Options('Verbose'); my $out = $et->Options('TextOut'); my ($data, $err, $outBuff); if ($$dirInfo{DataPt}) { # make a RAF object for MRW information extracted from other file types $raf = new File::RandomAccess($$dirInfo{DataPt}); # MRW information in DNG images may not start at beginning of data block $raf->Seek($$dirInfo{DirStart}, 0) if $$dirInfo{DirStart}; } $raf->Read($data,8) == 8 or return 0; # "\0MRM" for big-endian (MRW images), and # "\0MRI" for little-endian (MRWInfo in ARW images) $data =~ /^\0MR([MI])/ or return 0; my $hdr = "\0MR$1"; SetByteOrder($1 . $1); $et->SetFileType(); $tagTablePtr = GetTagTable('Image::ExifTool::MinoltaRaw::Main'); if ($outfile) { $et->InitWriteDirs('TIFF'); # use same write dirs as TIFF $outBuff = ''; } my $pos = $raf->Tell(); my $offset = Get32u(\$data, 4) + $pos; my $rtnVal = 1; $verbose and printf $out " [MRW Data Offset: 0x%x]\n", $offset; # loop through MRW segments (ref 1) while ($pos < $offset) { $raf->Read($data,8) == 8 or $err = 1, last; $pos += 8; my $tag = substr($data, 0, 4); my $len = Get32u(\$data, 4); if ($verbose) { print $out "MRW ",$et->Printable($tag)," segment ($len bytes):\n"; if ($verbose > 2) { $raf->Read($data,$len) == $len and $raf->Seek($pos,0) or $err = 1, last; $et->VerboseDump(\$data); } } my $tagInfo = $et->GetTagInfo($tagTablePtr, $tag); if ($tagInfo and $$tagInfo{SubDirectory}) { my $subTable = GetTagTable($tagInfo->{SubDirectory}->{TagTable}); my $buff; # save shift for values stored with wrong base offset $$et{MRW_WrongBase} = -($raf->Tell()); $raf->Read($buff, $len) == $len or $err = 1, last; my %subdirInfo = ( DataPt => \$buff, DataLen => $len, DataPos => $pos, DirStart => 0, DirLen => $len, DirName => $$tagInfo{Name}, Parent => 'MRW', NoTiffEnd => 1, # no end-of-TIFF check ); if ($outfile) { my $writeProc = $tagInfo->{SubDirectory}->{WriteProc}; my $val = $et->WriteDirectory(\%subdirInfo, $subTable, $writeProc); if (defined $val and length $val) { # pad to an even 4 bytes (can't hurt, and it seems to be the standard) $val .= "\0" x (4 - (length($val) & 0x03)) if length($val) & 0x03; $outBuff .= $tag . Set32u(length $val) . $val; } elsif (not defined $val) { $outBuff .= $data . $buff; # copy over original information } } else { my $processProc = $tagInfo->{SubDirectory}->{ProcessProc}; $et->ProcessDirectory(\%subdirInfo, $subTable, $processProc); } } elsif ($outfile) { # add this segment to the output buffer my $buff; $raf->Read($buff, $len) == $len or $err = 1, last; $outBuff .= $data . $buff; } else { # skip this segment $raf->Seek($pos+$len, 0) or $err = 1, last; } $pos += $len; } $pos == $offset or $err = 1; # meta information length check if ($outfile) { # write the file header then the buffered meta information Write($outfile, $hdr, Set32u(length $outBuff), $outBuff) or $rtnVal = -1; # copy over image data while ($raf->Read($outBuff, 65536)) { Write($outfile, $outBuff) or $rtnVal = -1; } # Sony IDC utility corrupts MRWInfo when writing ARW images, # so make this a minor error for these images $err and $et->Error("MRW format error", $$et{TIFF_TYPE} eq 'ARW'); } else { $err and $et->Warn("MRW format error"); } return $rtnVal; } 1; # end __END__ =head1 NAME Image::ExifTool::MinoltaRaw - Read/write Konica-Minolta RAW (MRW) information =head1 SYNOPSIS This module is loaded automatically by Image::ExifTool when required. =head1 DESCRIPTION This module contains definitions required by Image::ExifTool to read and write Konica-Minolta RAW (MRW) images. =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 REFERENCES =over 4 =item L =item L =back =head1 SEE ALSO L, L, L =cut