shun-iwasawa fdbab5
/*-------------------------------------------------------------
shun-iwasawa fdbab5
tiio_jpg_exif.cpp
shun-iwasawa fdbab5
Based on source code of a public domain software "Exif Jpeg header manipulation
shun-iwasawa fdbab5
tool (jhead)" by Matthias Wandel.
shun-iwasawa fdbab5
For now it is used only for obtaining resolution values.
shun-iwasawa fdbab5
http://www.sentex.net/~mwandel/jhead/
shun-iwasawa fdbab5
-------------------------------------------------------------*/
shun-iwasawa fdbab5
#include "tiio_jpg_exif.h"
shun-iwasawa fdbab5
#include <iostream></iostream>
shun-iwasawa fdbab5
#include <string.h></string.h>
shun-iwasawa fdbab5
shun-iwasawa fdbab5
// for debug
shun-iwasawa fdbab5
#define ShowTags 0
shun-iwasawa fdbab5
#define DumpExifMap 0
shun-iwasawa fdbab5
shun-iwasawa fdbab5
namespace {
shun-iwasawa fdbab5
shun-iwasawa fdbab5
typedef unsigned char uchar;
shun-iwasawa fdbab5
const int BytesPerFormat[] = {0, 1, 1, 2, 4, 8, 1, 1, 2, 4, 8, 4, 8};
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
// Describes tag values
shun-iwasawa fdbab5
const int TAG_X_RESOLUTION    = 0x011A;
shun-iwasawa fdbab5
const int TAG_Y_RESOLUTION    = 0x011B;
shun-iwasawa fdbab5
const int TAG_RESOLUTION_UNIT = 0x0128;
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
const int TAG_INTEROP_INDEX          = 0x0001;
shun-iwasawa fdbab5
const int TAG_INTEROP_VERSION        = 0x0002;
shun-iwasawa fdbab5
const int TAG_IMAGE_WIDTH            = 0x0100;
shun-iwasawa fdbab5
const int TAG_IMAGE_LENGTH           = 0x0101;
shun-iwasawa fdbab5
const int TAG_BITS_PER_SAMPLE        = 0x0102;
shun-iwasawa fdbab5
const int TAG_COMPRESSION            = 0x0103;
shun-iwasawa fdbab5
const int TAG_PHOTOMETRIC_INTERP     = 0x0106;
shun-iwasawa fdbab5
const int TAG_FILL_ORDER             = 0x010A;
shun-iwasawa fdbab5
const int TAG_DOCUMENT_NAME          = 0x010D;
shun-iwasawa fdbab5
const int TAG_IMAGE_DESCRIPTION      = 0x010E;
shun-iwasawa fdbab5
const int TAG_MAKE                   = 0x010F;
shun-iwasawa fdbab5
const int TAG_MODEL                  = 0x0110;
shun-iwasawa fdbab5
const int TAG_SRIP_OFFSET            = 0x0111;
shun-iwasawa fdbab5
const int TAG_ORIENTATION            = 0x0112;
shun-iwasawa fdbab5
const int TAG_SAMPLES_PER_PIXEL      = 0x0115;
shun-iwasawa fdbab5
const int TAG_ROWS_PER_STRIP         = 0x0116;
shun-iwasawa fdbab5
const int TAG_STRIP_BYTE_COUNTS      = 0x0117;
shun-iwasawa fdbab5
const int TAG_PLANAR_CONFIGURATION   = 0x011C;
shun-iwasawa fdbab5
const int TAG_TRANSFER_FUNCTION      = 0x012D;
shun-iwasawa fdbab5
const int TAG_SOFTWARE               = 0x0131;
shun-iwasawa fdbab5
const int TAG_DATETIME               = 0x0132;
shun-iwasawa fdbab5
const int TAG_ARTIST                 = 0x013B;
shun-iwasawa fdbab5
const int TAG_WHITE_POINT            = 0x013E;
shun-iwasawa fdbab5
const int TAG_PRIMARY_CHROMATICITIES = 0x013F;
shun-iwasawa fdbab5
const int TAG_TRANSFER_RANGE         = 0x0156;
shun-iwasawa fdbab5
const int TAG_JPEG_PROC              = 0x0200;
shun-iwasawa fdbab5
const int TAG_THUMBNAIL_OFFSET       = 0x0201;
shun-iwasawa fdbab5
const int TAG_THUMBNAIL_LENGTH       = 0x0202;
shun-iwasawa fdbab5
const int TAG_Y_CB_CR_COEFFICIENTS   = 0x0211;
shun-iwasawa fdbab5
const int TAG_Y_CB_CR_SUB_SAMPLING   = 0x0212;
shun-iwasawa fdbab5
const int TAG_Y_CB_CR_POSITIONING    = 0x0213;
shun-iwasawa fdbab5
const int TAG_REFERENCE_BLACK_WHITE  = 0x0214;
shun-iwasawa fdbab5
const int TAG_RELATED_IMAGE_WIDTH    = 0x1001;
shun-iwasawa fdbab5
const int TAG_RELATED_IMAGE_LENGTH   = 0x1002;
shun-iwasawa fdbab5
const int TAG_CFA_REPEAT_PATTERN_DIM = 0x828D;
shun-iwasawa fdbab5
const int TAG_CFA_PATTERN1           = 0x828E;
shun-iwasawa fdbab5
const int TAG_BATTERY_LEVEL          = 0x828F;
shun-iwasawa fdbab5
const int TAG_COPYRIGHT              = 0x8298;
shun-iwasawa fdbab5
const int TAG_EXPOSURETIME           = 0x829A;
shun-iwasawa fdbab5
const int TAG_FNUMBER                = 0x829D;
shun-iwasawa fdbab5
const int TAG_IPTC_NAA               = 0x83BB;
shun-iwasawa fdbab5
const int TAG_EXIF_OFFSET            = 0x8769;
shun-iwasawa fdbab5
const int TAG_INTER_COLOR_PROFILE    = 0x8773;
shun-iwasawa fdbab5
const int TAG_EXPOSURE_PROGRAM       = 0x8822;
shun-iwasawa fdbab5
const int TAG_SPECTRAL_SENSITIVITY   = 0x8824;
shun-iwasawa fdbab5
const int TAG_GPSINFO                = 0x8825;
shun-iwasawa fdbab5
const int TAG_ISO_EQUIVALENT         = 0x8827;
shun-iwasawa fdbab5
const int TAG_OECF                   = 0x8828;
shun-iwasawa fdbab5
const int TAG_EXIF_VERSION           = 0x9000;
shun-iwasawa fdbab5
const int TAG_DATETIME_ORIGINAL      = 0x9003;
shun-iwasawa fdbab5
const int TAG_DATETIME_DIGITIZED     = 0x9004;
shun-iwasawa fdbab5
const int TAG_COMPONENTS_CONFIG      = 0x9101;
shun-iwasawa fdbab5
const int TAG_CPRS_BITS_PER_PIXEL    = 0x9102;
shun-iwasawa fdbab5
const int TAG_SHUTTERSPEED           = 0x9201;
shun-iwasawa fdbab5
const int TAG_APERTURE               = 0x9202;
shun-iwasawa fdbab5
const int TAG_BRIGHTNESS_VALUE       = 0x9203;
shun-iwasawa fdbab5
const int TAG_EXPOSURE_BIAS          = 0x9204;
shun-iwasawa fdbab5
const int TAG_MAXAPERTURE            = 0x9205;
shun-iwasawa fdbab5
const int TAG_SUBJECT_DISTANCE       = 0x9206;
shun-iwasawa fdbab5
const int TAG_METERING_MODE          = 0x9207;
shun-iwasawa fdbab5
const int TAG_LIGHT_SOURCE           = 0x9208;
shun-iwasawa fdbab5
const int TAG_FLASH                  = 0x9209;
shun-iwasawa fdbab5
const int TAG_FOCALLENGTH            = 0x920A;
shun-iwasawa fdbab5
const int TAG_SUBJECTAREA            = 0x9214;
shun-iwasawa fdbab5
const int TAG_MAKER_NOTE             = 0x927C;
shun-iwasawa fdbab5
const int TAG_USERCOMMENT            = 0x9286;
shun-iwasawa fdbab5
const int TAG_SUBSEC_TIME            = 0x9290;
shun-iwasawa fdbab5
const int TAG_SUBSEC_TIME_ORIG       = 0x9291;
shun-iwasawa fdbab5
const int TAG_SUBSEC_TIME_DIG        = 0x9292;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
const int TAG_WINXP_TITLE = 0x9c9b;  // Windows XP - not part of exif standard.
shun-iwasawa fdbab5
const int TAG_WINXP_COMMENT =
shun-iwasawa fdbab5
    0x9c9c;                           // Windows XP - not part of exif standard.
shun-iwasawa fdbab5
const int TAG_WINXP_AUTHOR = 0x9c9d;  // Windows XP - not part of exif standard.
shun-iwasawa fdbab5
const int TAG_WINXP_KEYWORDS =
shun-iwasawa fdbab5
    0x9c9e;  // Windows XP - not part of exif standard.
shun-iwasawa fdbab5
const int TAG_WINXP_SUBJECT =
shun-iwasawa fdbab5
    0x9c9f;  // Windows XP - not part of exif standard.
shun-iwasawa fdbab5
shun-iwasawa fdbab5
const int TAG_FLASH_PIX_VERSION  = 0xA000;
shun-iwasawa fdbab5
const int TAG_COLOR_SPACE        = 0xA001;
shun-iwasawa fdbab5
const int TAG_PIXEL_X_DIMENSION  = 0xA002;
shun-iwasawa fdbab5
const int TAG_PIXEL_Y_DIMENSION  = 0xA003;
shun-iwasawa fdbab5
const int TAG_RELATED_AUDIO_FILE = 0xA004;
shun-iwasawa fdbab5
const int TAG_INTEROP_OFFSET     = 0xA005;
shun-iwasawa fdbab5
const int TAG_FLASH_ENERGY       = 0xA20B;
shun-iwasawa fdbab5
const int TAG_SPATIAL_FREQ_RESP  = 0xA20C;
shun-iwasawa fdbab5
const int TAG_FOCAL_PLANE_XRES   = 0xA20E;
shun-iwasawa fdbab5
const int TAG_FOCAL_PLANE_YRES   = 0xA20F;
shun-iwasawa fdbab5
const int TAG_FOCAL_PLANE_UNITS  = 0xA210;
shun-iwasawa fdbab5
const int TAG_SUBJECT_LOCATION   = 0xA214;
shun-iwasawa fdbab5
const int TAG_EXPOSURE_INDEX     = 0xA215;
shun-iwasawa fdbab5
const int TAG_SENSING_METHOD     = 0xA217;
shun-iwasawa fdbab5
const int TAG_FILE_SOURCE        = 0xA300;
shun-iwasawa fdbab5
const int TAG_SCENE_TYPE         = 0xA301;
shun-iwasawa fdbab5
const int TAG_CFA_PATTERN        = 0xA302;
shun-iwasawa fdbab5
const int TAG_CUSTOM_RENDERED    = 0xA401;
shun-iwasawa fdbab5
const int TAG_EXPOSURE_MODE      = 0xA402;
shun-iwasawa fdbab5
const int TAG_WHITEBALANCE       = 0xA403;
shun-iwasawa fdbab5
const int TAG_DIGITALZOOMRATIO   = 0xA404;
shun-iwasawa fdbab5
const int TAG_FOCALLENGTH_35MM   = 0xA405;
shun-iwasawa fdbab5
const int TAG_SCENE_CAPTURE_TYPE = 0xA406;
shun-iwasawa fdbab5
const int TAG_GAIN_CONTROL       = 0xA407;
shun-iwasawa fdbab5
const int TAG_CONTRAST           = 0xA408;
shun-iwasawa fdbab5
const int TAG_SATURATION         = 0xA409;
shun-iwasawa fdbab5
const int TAG_SHARPNESS          = 0xA40A;
shun-iwasawa fdbab5
const int TAG_DISTANCE_RANGE     = 0xA40C;
shun-iwasawa fdbab5
const int TAG_IMAGE_UNIQUE_ID    = 0xA420;
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
shun-iwasawa fdbab5
typedef struct {
shun-iwasawa fdbab5
  unsigned short Tag;
shun-iwasawa fdbab5
  char *Desc;
shun-iwasawa fdbab5
} TagTable_t;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
const TagTable_t TagTable[] = {
shun-iwasawa fdbab5
    {TAG_X_RESOLUTION, "XResolution"},
shun-iwasawa fdbab5
    {TAG_Y_RESOLUTION, "YResolution"},
shun-iwasawa fdbab5
    {TAG_RESOLUTION_UNIT, "ResolutionUnit"},
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
    {TAG_INTEROP_INDEX, "InteropIndex"},
shun-iwasawa fdbab5
    {TAG_INTEROP_VERSION, "InteropVersion"},
shun-iwasawa fdbab5
    {TAG_IMAGE_WIDTH, "ImageWidth"},
shun-iwasawa fdbab5
    {TAG_IMAGE_LENGTH, "ImageLength"},
shun-iwasawa fdbab5
    {TAG_BITS_PER_SAMPLE, "BitsPerSample"},
shun-iwasawa fdbab5
    {TAG_COMPRESSION, "Compression"},
shun-iwasawa fdbab5
    {TAG_PHOTOMETRIC_INTERP, "PhotometricInterpretation"},
shun-iwasawa fdbab5
    {TAG_FILL_ORDER, "FillOrder"},
shun-iwasawa fdbab5
    {TAG_DOCUMENT_NAME, "DocumentName"},
shun-iwasawa fdbab5
    {TAG_IMAGE_DESCRIPTION, "ImageDescription"},
shun-iwasawa fdbab5
    {TAG_MAKE, "Make"},
shun-iwasawa fdbab5
    {TAG_MODEL, "Model"},
shun-iwasawa fdbab5
    {TAG_SRIP_OFFSET, "StripOffsets"},
shun-iwasawa fdbab5
    {TAG_ORIENTATION, "Orientation"},
shun-iwasawa fdbab5
    {TAG_SAMPLES_PER_PIXEL, "SamplesPerPixel"},
shun-iwasawa fdbab5
    {TAG_ROWS_PER_STRIP, "RowsPerStrip"},
shun-iwasawa fdbab5
    {TAG_STRIP_BYTE_COUNTS, "StripByteCounts"},
shun-iwasawa fdbab5
    {TAG_PLANAR_CONFIGURATION, "PlanarConfiguration"},
shun-iwasawa fdbab5
    {TAG_TRANSFER_FUNCTION, "TransferFunction"},
shun-iwasawa fdbab5
    {TAG_SOFTWARE, "Software"},
shun-iwasawa fdbab5
    {TAG_DATETIME, "DateTime"},
shun-iwasawa fdbab5
    {TAG_ARTIST, "Artist"},
shun-iwasawa fdbab5
    {TAG_WHITE_POINT, "WhitePoint"},
shun-iwasawa fdbab5
    {TAG_PRIMARY_CHROMATICITIES, "PrimaryChromaticities"},
shun-iwasawa fdbab5
    {TAG_TRANSFER_RANGE, "TransferRange"},
shun-iwasawa fdbab5
    {TAG_JPEG_PROC, "JPEGProc"},
shun-iwasawa fdbab5
    {TAG_THUMBNAIL_OFFSET, "ThumbnailOffset"},
shun-iwasawa fdbab5
    {TAG_THUMBNAIL_LENGTH, "ThumbnailLength"},
shun-iwasawa fdbab5
    {TAG_Y_CB_CR_COEFFICIENTS, "YCbCrCoefficients"},
shun-iwasawa fdbab5
    {TAG_Y_CB_CR_SUB_SAMPLING, "YCbCrSubSampling"},
shun-iwasawa fdbab5
    {TAG_Y_CB_CR_POSITIONING, "YCbCrPositioning"},
shun-iwasawa fdbab5
    {TAG_REFERENCE_BLACK_WHITE, "ReferenceBlackWhite"},
shun-iwasawa fdbab5
    {TAG_RELATED_IMAGE_WIDTH, "RelatedImageWidth"},
shun-iwasawa fdbab5
    {TAG_RELATED_IMAGE_LENGTH, "RelatedImageLength"},
shun-iwasawa fdbab5
    {TAG_CFA_REPEAT_PATTERN_DIM, "CFARepeatPatternDim"},
shun-iwasawa fdbab5
    {TAG_CFA_PATTERN1, "CFAPattern"},
shun-iwasawa fdbab5
    {TAG_BATTERY_LEVEL, "BatteryLevel"},
shun-iwasawa fdbab5
    {TAG_COPYRIGHT, "Copyright"},
shun-iwasawa fdbab5
    {TAG_EXPOSURETIME, "ExposureTime"},
shun-iwasawa fdbab5
    {TAG_FNUMBER, "FNumber"},
shun-iwasawa fdbab5
    {TAG_IPTC_NAA, "IPTC/NAA"},
shun-iwasawa fdbab5
    {TAG_EXIF_OFFSET, "ExifOffset"},
shun-iwasawa fdbab5
    {TAG_INTER_COLOR_PROFILE, "InterColorProfile"},
shun-iwasawa fdbab5
    {TAG_EXPOSURE_PROGRAM, "ExposureProgram"},
shun-iwasawa fdbab5
    {TAG_SPECTRAL_SENSITIVITY, "SpectralSensitivity"},
shun-iwasawa fdbab5
    {TAG_GPSINFO, "GPS Dir offset"},
shun-iwasawa fdbab5
    {TAG_ISO_EQUIVALENT, "ISOSpeedRatings"},
shun-iwasawa fdbab5
    {TAG_OECF, "OECF"},
shun-iwasawa fdbab5
    {TAG_EXIF_VERSION, "ExifVersion"},
shun-iwasawa fdbab5
    {TAG_DATETIME_ORIGINAL, "DateTimeOriginal"},
shun-iwasawa fdbab5
    {TAG_DATETIME_DIGITIZED, "DateTimeDigitized"},
shun-iwasawa fdbab5
    {TAG_COMPONENTS_CONFIG, "ComponentsConfiguration"},
shun-iwasawa fdbab5
    {TAG_CPRS_BITS_PER_PIXEL, "CompressedBitsPerPixel"},
shun-iwasawa fdbab5
    {TAG_SHUTTERSPEED, "ShutterSpeedValue"},
shun-iwasawa fdbab5
    {TAG_APERTURE, "ApertureValue"},
shun-iwasawa fdbab5
    {TAG_BRIGHTNESS_VALUE, "BrightnessValue"},
shun-iwasawa fdbab5
    {TAG_EXPOSURE_BIAS, "ExposureBiasValue"},
shun-iwasawa fdbab5
    {TAG_MAXAPERTURE, "MaxApertureValue"},
shun-iwasawa fdbab5
    {TAG_SUBJECT_DISTANCE, "SubjectDistance"},
shun-iwasawa fdbab5
    {TAG_METERING_MODE, "MeteringMode"},
shun-iwasawa fdbab5
    {TAG_LIGHT_SOURCE, "LightSource"},
shun-iwasawa fdbab5
    {TAG_FLASH, "Flash"},
shun-iwasawa fdbab5
    {TAG_FOCALLENGTH, "FocalLength"},
shun-iwasawa fdbab5
    {TAG_MAKER_NOTE, "MakerNote"},
shun-iwasawa fdbab5
    {TAG_USERCOMMENT, "UserComment"},
shun-iwasawa fdbab5
    {TAG_SUBSEC_TIME, "SubSecTime"},
shun-iwasawa fdbab5
    {TAG_SUBSEC_TIME_ORIG, "SubSecTimeOriginal"},
shun-iwasawa fdbab5
    {TAG_SUBSEC_TIME_DIG, "SubSecTimeDigitized"},
shun-iwasawa fdbab5
    {TAG_WINXP_TITLE, "Windows-XP Title"},
shun-iwasawa fdbab5
    {TAG_WINXP_COMMENT, "Windows-XP comment"},
shun-iwasawa fdbab5
    {TAG_WINXP_AUTHOR, "Windows-XP author"},
shun-iwasawa fdbab5
    {TAG_WINXP_KEYWORDS, "Windows-XP keywords"},
shun-iwasawa fdbab5
    {TAG_WINXP_SUBJECT, "Windows-XP subject"},
shun-iwasawa fdbab5
    {TAG_FLASH_PIX_VERSION, "FlashPixVersion"},
shun-iwasawa fdbab5
    {TAG_COLOR_SPACE, "ColorSpace"},
shun-iwasawa fdbab5
    {TAG_PIXEL_X_DIMENSION, "ExifImageWidth"},
shun-iwasawa fdbab5
    {TAG_PIXEL_Y_DIMENSION, "ExifImageLength"},
shun-iwasawa fdbab5
    {TAG_RELATED_AUDIO_FILE, "RelatedAudioFile"},
shun-iwasawa fdbab5
    {TAG_INTEROP_OFFSET, "InteroperabilityOffset"},
shun-iwasawa fdbab5
    {TAG_FLASH_ENERGY, "FlashEnergy"},
shun-iwasawa fdbab5
    {TAG_SPATIAL_FREQ_RESP, "SpatialFrequencyResponse"},
shun-iwasawa fdbab5
    {TAG_FOCAL_PLANE_XRES, "FocalPlaneXResolution"},
shun-iwasawa fdbab5
    {TAG_FOCAL_PLANE_YRES, "FocalPlaneYResolution"},
shun-iwasawa fdbab5
    {TAG_FOCAL_PLANE_UNITS, "FocalPlaneResolutionUnit"},
shun-iwasawa fdbab5
    {TAG_SUBJECT_LOCATION, "SubjectLocation"},
shun-iwasawa fdbab5
    {TAG_EXPOSURE_INDEX, "ExposureIndex"},
shun-iwasawa fdbab5
    {TAG_SENSING_METHOD, "SensingMethod"},
shun-iwasawa fdbab5
    {TAG_FILE_SOURCE, "FileSource"},
shun-iwasawa fdbab5
    {TAG_SCENE_TYPE, "SceneType"},
shun-iwasawa fdbab5
    {TAG_CFA_PATTERN, "CFA Pattern"},
shun-iwasawa fdbab5
    {TAG_CUSTOM_RENDERED, "CustomRendered"},
shun-iwasawa fdbab5
    {TAG_EXPOSURE_MODE, "ExposureMode"},
shun-iwasawa fdbab5
    {TAG_WHITEBALANCE, "WhiteBalance"},
shun-iwasawa fdbab5
    {TAG_DIGITALZOOMRATIO, "DigitalZoomRatio"},
shun-iwasawa fdbab5
    {TAG_FOCALLENGTH_35MM, "FocalLengthIn35mmFilm"},
shun-iwasawa fdbab5
    {TAG_SUBJECTAREA, "SubjectArea"},
shun-iwasawa fdbab5
    {TAG_SCENE_CAPTURE_TYPE, "SceneCaptureType"},
shun-iwasawa fdbab5
    {TAG_GAIN_CONTROL, "GainControl"},
shun-iwasawa fdbab5
    {TAG_CONTRAST, "Contrast"},
shun-iwasawa fdbab5
    {TAG_SATURATION, "Saturation"},
shun-iwasawa fdbab5
    {TAG_SHARPNESS, "Sharpness"},
shun-iwasawa fdbab5
    {TAG_DISTANCE_RANGE, "SubjectDistanceRange"},
shun-iwasawa fdbab5
    {TAG_IMAGE_UNIQUE_ID, "ImageUniqueId"},
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
};
shun-iwasawa fdbab5
shun-iwasawa fdbab5
const int TAG_TABLE_SIZE = (sizeof(TagTable) / sizeof(TagTable_t));
shun-iwasawa fdbab5
shun-iwasawa fdbab5
const int TRUE  = 1;
shun-iwasawa fdbab5
const int FALSE = 0;
shun-iwasawa fdbab5
}
shun-iwasawa fdbab5
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
// Convert a 16 bit unsigned value from file's native byte order
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
int JpgExifReader::Get16u(void *Short) {
shun-iwasawa fdbab5
  if (MotorolaOrder) {
shun-iwasawa fdbab5
    return (((uchar *)Short)[0] << 8) | ((uchar *)Short)[1];
shun-iwasawa fdbab5
  } else {
shun-iwasawa fdbab5
    return (((uchar *)Short)[1] << 8) | ((uchar *)Short)[0];
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
}
shun-iwasawa fdbab5
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
// Convert a 32 bit signed value from file's native byte order
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
int JpgExifReader::Get32s(void *Long) {
shun-iwasawa fdbab5
  if (MotorolaOrder) {
shun-iwasawa fdbab5
    return (((char *)Long)[0] << 24) | (((uchar *)Long)[1] << 16) |
shun-iwasawa fdbab5
           (((uchar *)Long)[2] << 8) | (((uchar *)Long)[3] << 0);
shun-iwasawa fdbab5
  } else {
shun-iwasawa fdbab5
    return (((char *)Long)[3] << 24) | (((uchar *)Long)[2] << 16) |
shun-iwasawa fdbab5
           (((uchar *)Long)[1] << 8) | (((uchar *)Long)[0] << 0);
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
}
shun-iwasawa fdbab5
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
// Convert a 32 bit unsigned value from file's native byte order
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
unsigned JpgExifReader::Get32u(void *Long) {
shun-iwasawa fdbab5
  return (unsigned)Get32s(Long) & 0xffffffff;
shun-iwasawa fdbab5
}
shun-iwasawa fdbab5
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
// Display a number as one of its many formats
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
void JpgExifReader::PrintFormatNumber(void *ValuePtr, int Format,
shun-iwasawa fdbab5
                                      int ByteCount) {
shun-iwasawa fdbab5
  int s, n;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  for (n = 0; n < 16; n++) {
shun-iwasawa fdbab5
    switch (Format) {
shun-iwasawa fdbab5
    case FMT_SBYTE:
shun-iwasawa fdbab5
    case FMT_BYTE:
shun-iwasawa fdbab5
      printf("%02x", *(uchar *)ValuePtr);
shun-iwasawa fdbab5
      s = 1;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
    case FMT_USHORT:
shun-iwasawa fdbab5
      printf("%d", Get16u(ValuePtr));
shun-iwasawa fdbab5
      s = 2;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
    case FMT_ULONG:
shun-iwasawa fdbab5
    case FMT_SLONG:
shun-iwasawa fdbab5
      printf("%d", Get32s(ValuePtr));
shun-iwasawa fdbab5
      s = 4;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
    case FMT_SSHORT:
shun-iwasawa fdbab5
      printf("%hd", (signed short)Get16u(ValuePtr));
shun-iwasawa fdbab5
      s = 2;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
    case FMT_URATIONAL:
shun-iwasawa fdbab5
      printf("%u/%u", Get32s(ValuePtr), Get32s(4 + (char *)ValuePtr));
shun-iwasawa fdbab5
      s = 8;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case FMT_SRATIONAL:
shun-iwasawa fdbab5
      printf("%d/%d", Get32s(ValuePtr), Get32s(4 + (char *)ValuePtr));
shun-iwasawa fdbab5
      s = 8;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case FMT_SINGLE:
shun-iwasawa fdbab5
      printf("%f", (double)*(float *)ValuePtr);
shun-iwasawa fdbab5
      s = 8;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
    case FMT_DOUBLE:
shun-iwasawa fdbab5
      printf("%f", *(double *)ValuePtr);
shun-iwasawa fdbab5
      s = 8;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
    default:
shun-iwasawa fdbab5
      printf("Unknown format %d:", Format);
shun-iwasawa fdbab5
      return;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
    ByteCount -= s;
shun-iwasawa fdbab5
    if (ByteCount <= 0) break;
shun-iwasawa fdbab5
    printf(", ");
shun-iwasawa fdbab5
    ValuePtr = (void *)((char *)ValuePtr + s);
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
  if (n >= 16) printf("...");
shun-iwasawa fdbab5
}
shun-iwasawa fdbab5
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
// Evaluate number, be it int, rational, or float from directory.
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
double JpgExifReader::ConvertAnyFormat(void *ValuePtr, int Format) {
shun-iwasawa fdbab5
  double Value;
shun-iwasawa fdbab5
  Value = 0;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  switch (Format) {
shun-iwasawa fdbab5
  case FMT_SBYTE:
shun-iwasawa fdbab5
    Value = *(signed char *)ValuePtr;
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
  case FMT_BYTE:
shun-iwasawa fdbab5
    Value = *(uchar *)ValuePtr;
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  case FMT_USHORT:
shun-iwasawa fdbab5
    Value = Get16u(ValuePtr);
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
  case FMT_ULONG:
shun-iwasawa fdbab5
    Value = Get32u(ValuePtr);
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  case FMT_URATIONAL:
shun-iwasawa fdbab5
  case FMT_SRATIONAL: {
shun-iwasawa fdbab5
    int Num, Den;
shun-iwasawa fdbab5
    Num = Get32s(ValuePtr);
shun-iwasawa fdbab5
    Den = Get32s(4 + (char *)ValuePtr);
shun-iwasawa fdbab5
    if (Den == 0) {
shun-iwasawa fdbab5
      Value = 0;
shun-iwasawa fdbab5
    } else {
shun-iwasawa fdbab5
      if (Format == FMT_SRATIONAL) {
shun-iwasawa fdbab5
        Value = (double)Num / Den;
shun-iwasawa fdbab5
      } else {
shun-iwasawa fdbab5
        Value = (double)(unsigned)Num / (double)(unsigned)Den;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  case FMT_SSHORT:
shun-iwasawa fdbab5
    Value = (signed short)Get16u(ValuePtr);
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
  case FMT_SLONG:
shun-iwasawa fdbab5
    Value = Get32s(ValuePtr);
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  // Not sure if this is correct (never seen float used in Exif format)
shun-iwasawa fdbab5
  case FMT_SINGLE:
shun-iwasawa fdbab5
    Value = (double)*(float *)ValuePtr;
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
  case FMT_DOUBLE:
shun-iwasawa fdbab5
    Value = *(double *)ValuePtr;
shun-iwasawa fdbab5
    break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  default:
shun-iwasawa fdbab5
    std::cout << "Illegal format code " << Format << " in Exif header"
shun-iwasawa fdbab5
              << std::endl;
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
  return Value;
shun-iwasawa fdbab5
}
shun-iwasawa fdbab5
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
shun-iwasawa fdbab5
void JpgExifReader::process_EXIF(unsigned char *ExifSection,
shun-iwasawa fdbab5
                                 unsigned int length) {
shun-iwasawa fdbab5
  unsigned int FirstOffset;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
  FocalplaneXRes       = 0;
shun-iwasawa fdbab5
  FocalplaneUnits      = 0;
shun-iwasawa fdbab5
  ExifImageWidth       = 0;
shun-iwasawa fdbab5
  NumOrientations      = 0;
shun-iwasawa fdbab5
  DirWithThumbnailPtrs = NULL;
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  if (ShowTags) printf("Exif header %u bytes long\n", length);
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  {  // Check the EXIF header component
shun-iwasawa fdbab5
    static uchar ExifHeader[] = "Exif\0\0";
shun-iwasawa fdbab5
    if (memcmp(ExifSection + 2, ExifHeader, 6)) {
shun-iwasawa fdbab5
      std::cout << "Incorrect Exif header" << std::endl;
shun-iwasawa fdbab5
      return;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  if (memcmp(ExifSection + 8, "II", 2) == 0) {
shun-iwasawa fdbab5
    if (ShowTags) printf("Exif section in Intel order\n");
shun-iwasawa fdbab5
    MotorolaOrder = 0;
shun-iwasawa fdbab5
  } else {
shun-iwasawa fdbab5
    if (memcmp(ExifSection + 8, "MM", 2) == 0) {
shun-iwasawa fdbab5
      if (ShowTags) printf("Exif section in Motorola order\n");
shun-iwasawa fdbab5
      MotorolaOrder = 1;
shun-iwasawa fdbab5
    } else {
shun-iwasawa fdbab5
      std::cout << "Invalid Exif alignment marker." << std::endl;
shun-iwasawa fdbab5
      return;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  // Check the next value for correctness.
shun-iwasawa fdbab5
  if (Get16u(ExifSection + 10) != 0x2a) {
shun-iwasawa fdbab5
    std::cout << "Invalid Exif start (1)" << std::endl;
shun-iwasawa fdbab5
    return;
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  FirstOffset = Get32u(ExifSection + 12);
shun-iwasawa fdbab5
  if (FirstOffset < 8 || FirstOffset > 16) {
shun-iwasawa fdbab5
    if (FirstOffset < 16 || FirstOffset > length - 16) {
shun-iwasawa fdbab5
      std::cout << "invalid offset for first Exif IFD value" << std::endl;
shun-iwasawa fdbab5
      return;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
    // Usually set to 8, but other values valid too.
shun-iwasawa fdbab5
    std::cout << "Suspicious offset of first Exif IFD value" << std::endl;
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  // First directory starts 16 bytes in.  All offset are relative to 8 bytes in.
shun-iwasawa fdbab5
  ProcessExifDir(ExifSection + 8 + FirstOffset, ExifSection + 8, length - 8, 0);
shun-iwasawa fdbab5
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
  ImageInfo.ThumbnailAtEnd =
shun-iwasawa fdbab5
      ImageInfo.ThumbnailOffset >= ImageInfo.LargestExifOffset ? TRUE : FALSE;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  if (DumpExifMap) {
shun-iwasawa fdbab5
    unsigned a, b;
shun-iwasawa fdbab5
    printf("Map: %05d- End of exif\n", length - 8);
shun-iwasawa fdbab5
    for (a = 0; a < length - 8; a += 10) {
shun-iwasawa fdbab5
      printf("Map: %05d ", a);
shun-iwasawa fdbab5
      for (b = 0; b < 10; b++) printf(" %02x", *(ExifSection + 8 + a + b));
shun-iwasawa fdbab5
      printf("\n");
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  // Compute the CCD width, in millimeters.
shun-iwasawa fdbab5
  if (FocalplaneXRes != 0 && ExifImageWidth != 0) {
shun-iwasawa fdbab5
    // Note: With some cameras, its not possible to compute this correctly
shun-iwasawa fdbab5
    // because
shun-iwasawa fdbab5
    // they don't adjust the indicated focal plane resolution units when using
shun-iwasawa fdbab5
    // less
shun-iwasawa fdbab5
    // than maximum resolution, so the CCDWidth value comes out too small.
shun-iwasawa fdbab5
    // Nothing
shun-iwasawa fdbab5
    // that Jhad can do about it - its a camera problem.
shun-iwasawa fdbab5
    ImageInfo.CCDWidth =
shun-iwasawa fdbab5
        (float)(ExifImageWidth * FocalplaneUnits / FocalplaneXRes);
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if (ImageInfo.FocalLength && ImageInfo.FocalLength35mmEquiv == 0) {
shun-iwasawa fdbab5
      // Compute 35 mm equivalent focal length based on sensor geometry if we
shun-iwasawa fdbab5
      // haven't
shun-iwasawa fdbab5
      // already got it explicitly from a tag.
shun-iwasawa fdbab5
      ImageInfo.FocalLength35mmEquiv =
shun-iwasawa fdbab5
          (int)(ImageInfo.FocalLength / ImageInfo.CCDWidth * 36 + 0.5);
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
}
shun-iwasawa fdbab5
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
// Process one of the nested EXIF directories.
shun-iwasawa fdbab5
//--------------------------------------------------------------------------
shun-iwasawa fdbab5
void JpgExifReader::ProcessExifDir(unsigned char *DirStart,
shun-iwasawa fdbab5
                                   unsigned char *OffsetBase,
shun-iwasawa fdbab5
                                   unsigned ExifLength, int NestingLevel) {
shun-iwasawa fdbab5
  int de;
shun-iwasawa fdbab5
  int a;
shun-iwasawa fdbab5
  int NumDirEntries;
shun-iwasawa fdbab5
  unsigned ThumbnailOffset = 0;
shun-iwasawa fdbab5
  unsigned ThumbnailSize   = 0;
shun-iwasawa fdbab5
  char IndentString[25];
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  if (NestingLevel > 4) {
shun-iwasawa fdbab5
    std::cout << "Maximum Exif directory nesting exceeded (corrupt Exif header)"
shun-iwasawa fdbab5
              << std::endl;
shun-iwasawa fdbab5
    return;
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  memset(IndentString, ' ', 25);
shun-iwasawa fdbab5
  IndentString[NestingLevel * 4] = '\0';
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  NumDirEntries = Get16u(DirStart);
shun-iwasawa fdbab5
#define DIR_ENTRY_ADDR(Start, Entry) (Start + 2 + 12 * (Entry))
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  {
shun-iwasawa fdbab5
    unsigned char *DirEnd;
shun-iwasawa fdbab5
    DirEnd = DIR_ENTRY_ADDR(DirStart, NumDirEntries);
shun-iwasawa fdbab5
    if (DirEnd + 4 > (OffsetBase + ExifLength)) {
shun-iwasawa fdbab5
      if (DirEnd + 2 == OffsetBase + ExifLength ||
shun-iwasawa fdbab5
          DirEnd == OffsetBase + ExifLength) {
shun-iwasawa fdbab5
        // Version 1.3 of jhead would truncate a bit too much.
shun-iwasawa fdbab5
        // This also caught later on as well.
shun-iwasawa fdbab5
      } else {
shun-iwasawa fdbab5
        std::cout << "Illegally sized Exif subdirectory (" << NumDirEntries
shun-iwasawa fdbab5
                  << "entries)" << std::endl;
shun-iwasawa fdbab5
        return;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
    if (DumpExifMap) {
shun-iwasawa fdbab5
      printf("Map: %05u-%05u: Directory\n", (int)(DirStart - OffsetBase),
shun-iwasawa fdbab5
             (int)(DirEnd + 4 - OffsetBase));
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  if (ShowTags) {
shun-iwasawa fdbab5
    printf("(dir has %d entries)\n", NumDirEntries);
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  for (de = 0; de < NumDirEntries; de++) {
shun-iwasawa fdbab5
    int Tag, Format, Components;
shun-iwasawa fdbab5
    unsigned char *ValuePtr;
shun-iwasawa fdbab5
    int ByteCount;
shun-iwasawa fdbab5
    unsigned char *DirEntry;
shun-iwasawa fdbab5
    DirEntry = DIR_ENTRY_ADDR(DirStart, de);
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    Tag        = Get16u(DirEntry);
shun-iwasawa fdbab5
    Format     = Get16u(DirEntry + 2);
shun-iwasawa fdbab5
    Components = Get32u(DirEntry + 4);
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if ((Format - 1) >= NUM_FORMATS) {
shun-iwasawa fdbab5
      // (-1) catches illegal zero case as unsigned underflows to positive
shun-iwasawa fdbab5
      // large.
shun-iwasawa fdbab5
      std::cout << "Illegal number format " << Format << " for tag " << Tag
shun-iwasawa fdbab5
                << " in Exif" << std::endl;
shun-iwasawa fdbab5
      continue;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if ((unsigned)Components > 0x10000) {
shun-iwasawa fdbab5
      std::cout << "Too many components " << Components << " for tag " << Tag
shun-iwasawa fdbab5
                << " in Exif";
shun-iwasawa fdbab5
      continue;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    ByteCount = Components * BytesPerFormat[Format];
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if (ByteCount > 4) {
shun-iwasawa fdbab5
      unsigned OffsetVal;
shun-iwasawa fdbab5
      OffsetVal = Get32u(DirEntry + 8);
shun-iwasawa fdbab5
      // If its bigger than 4 bytes, the dir entry contains an offset.
shun-iwasawa fdbab5
      if (OffsetVal + ByteCount > ExifLength) {
shun-iwasawa fdbab5
        // Bogus pointer offset and / or bytecount value
shun-iwasawa fdbab5
        std::cout << "Illegal value pointer for tag " << Tag << " in Exif";
shun-iwasawa fdbab5
        continue;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      ValuePtr = OffsetBase + OffsetVal;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
      if (OffsetVal > ImageInfo.LargestExifOffset) {
shun-iwasawa fdbab5
        ImageInfo.LargestExifOffset = OffsetVal;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      if (DumpExifMap) {
shun-iwasawa fdbab5
        printf("Map: %05u-%05u:   Data for tag %04x\n", OffsetVal,
shun-iwasawa fdbab5
               OffsetVal + ByteCount, Tag);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
    } else {
shun-iwasawa fdbab5
      // 4 bytes or less and value is in the dir entry itself
shun-iwasawa fdbab5
      ValuePtr = DirEntry + 8;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
    if (Tag == TAG_MAKER_NOTE) {
shun-iwasawa fdbab5
      if (ShowTags) {
shun-iwasawa fdbab5
        printf("%s    Maker note: ", IndentString);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      // ProcessMakerNote(ValuePtr, ByteCount, OffsetBase, ExifLength);
shun-iwasawa fdbab5
      continue;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if (ShowTags) {
shun-iwasawa fdbab5
      // Show tag name
shun-iwasawa fdbab5
      for (a = 0;; a++) {
shun-iwasawa fdbab5
        if (a >= TAG_TABLE_SIZE) {
shun-iwasawa fdbab5
          printf("%s    Unknown Tag %04x Value = ", IndentString, Tag);
shun-iwasawa fdbab5
          break;
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
        if (TagTable[a].Tag == Tag) {
shun-iwasawa fdbab5
          printf("%s    %s = ", IndentString, TagTable[a].Desc);
shun-iwasawa fdbab5
          break;
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      // Show tag value.
shun-iwasawa fdbab5
      switch (Format) {
shun-iwasawa fdbab5
      case FMT_BYTE:
shun-iwasawa fdbab5
        if (ByteCount > 1) {
shun-iwasawa fdbab5
          printf("%.*ls\n", ByteCount / 2, (wchar_t *)ValuePtr);
shun-iwasawa fdbab5
        } else {
shun-iwasawa fdbab5
          PrintFormatNumber(ValuePtr, Format, ByteCount);
shun-iwasawa fdbab5
          printf("\n");
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
        break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      case FMT_UNDEFINED:
shun-iwasawa fdbab5
      // Undefined is typically an ascii string.
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      case FMT_STRING:
shun-iwasawa fdbab5
        // String arrays printed without function call (different from int
shun-iwasawa fdbab5
        // arrays)
shun-iwasawa fdbab5
        {
shun-iwasawa fdbab5
          int NoPrint = 0;
shun-iwasawa fdbab5
          printf("\"");
shun-iwasawa fdbab5
          for (a = 0; a < ByteCount; a++) {
shun-iwasawa fdbab5
            if (ValuePtr[a] >= 32) {
shun-iwasawa fdbab5
              putchar(ValuePtr[a]);
shun-iwasawa fdbab5
              NoPrint = 0;
shun-iwasawa fdbab5
            } else {
shun-iwasawa fdbab5
              // Avoiding indicating too many unprintable characters of
shun-iwasawa fdbab5
              // proprietary
shun-iwasawa fdbab5
              // bits of binary information this program may not know how to
shun-iwasawa fdbab5
              // parse.
shun-iwasawa fdbab5
              if (!NoPrint && a != ByteCount - 1) {
shun-iwasawa fdbab5
                putchar('?');
shun-iwasawa fdbab5
                NoPrint = 1;
shun-iwasawa fdbab5
              }
shun-iwasawa fdbab5
            }
shun-iwasawa fdbab5
          }
shun-iwasawa fdbab5
          printf("\"\n");
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
        break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      default:
shun-iwasawa fdbab5
        // Handle arrays of numbers later (will there ever be?)
shun-iwasawa fdbab5
        PrintFormatNumber(ValuePtr, Format, ByteCount);
shun-iwasawa fdbab5
        printf("\n");
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
    // Extract useful components of tag
shun-iwasawa fdbab5
    switch (Tag) {
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
    case TAG_MAKE:
shun-iwasawa fdbab5
      strncpy(ImageInfo.CameraMake, (char *)ValuePtr,
shun-iwasawa fdbab5
              ByteCount < 31 ? ByteCount : 31);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_MODEL:
shun-iwasawa fdbab5
      strncpy(ImageInfo.CameraModel, (char *)ValuePtr,
shun-iwasawa fdbab5
              ByteCount < 39 ? ByteCount : 39);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_DATETIME_ORIGINAL:
shun-iwasawa fdbab5
      // If we get a DATETIME_ORIGINAL, we use that one.
shun-iwasawa fdbab5
      strncpy(ImageInfo.DateTime, (char *)ValuePtr, 19);
shun-iwasawa fdbab5
    // Fallthru...
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_DATETIME_DIGITIZED:
shun-iwasawa fdbab5
    case TAG_DATETIME:
shun-iwasawa fdbab5
      if (!isdigit(static_cast<unsigned char="">(ImageInfo.DateTime[0]))) {</unsigned>
shun-iwasawa fdbab5
        // If we don't already have a DATETIME_ORIGINAL, use whatever
shun-iwasawa fdbab5
        // time fields we may have.
shun-iwasawa fdbab5
        strncpy(ImageInfo.DateTime, (char *)ValuePtr, 19);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      if (ImageInfo.numDateTimeTags >= MAX_DATE_COPIES) {
shun-iwasawa fdbab5
        std::cout << "More than " << MAX_DATE_COPIES
shun-iwasawa fdbab5
                  << " date fields in Exif.  This is nuts" << std::endl;
shun-iwasawa fdbab5
        break;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      ImageInfo.DateTimeOffsets[ImageInfo.numDateTimeTags++] =
shun-iwasawa fdbab5
          (char *)ValuePtr - (char *)OffsetBase;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_WINXP_COMMENT:
shun-iwasawa fdbab5
      if (ImageInfo.Comments[0]) {  // We already have a jpeg comment.
shun-iwasawa fdbab5
        // Already have a comment (probably windows comment), skip this one.
shun-iwasawa fdbab5
        if (ShowTags)
shun-iwasawa fdbab5
          printf("Windows XP commend and other comment in header\n");
shun-iwasawa fdbab5
        break;  // Already have a windows comment, skip this one.
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      if (ByteCount > 1) {
shun-iwasawa fdbab5
        if (ByteCount > MAX_COMMENT_SIZE) ByteCount = MAX_COMMENT_SIZE;
shun-iwasawa fdbab5
        memcpy(ImageInfo.Comments, ValuePtr, ByteCount);
shun-iwasawa fdbab5
        ImageInfo.CommentWidthchars = ByteCount / 2;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_USERCOMMENT:
shun-iwasawa fdbab5
      if (ImageInfo.Comments[0]) {  // We already have a jpeg comment.
shun-iwasawa fdbab5
        // Already have a comment (probably windows comment), skip this one.
shun-iwasawa fdbab5
        if (ShowTags) printf("Multiple comments in exif header\n");
shun-iwasawa fdbab5
        break;  // Already have a windows comment, skip this one.
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      // Comment is often padded with trailing spaces.  Remove these first.
shun-iwasawa fdbab5
      for (a = ByteCount;;) {
shun-iwasawa fdbab5
        a--;
shun-iwasawa fdbab5
        if ((ValuePtr)[a] == ' ') {
shun-iwasawa fdbab5
          (ValuePtr)[a] = '\0';
shun-iwasawa fdbab5
        } else {
shun-iwasawa fdbab5
          break;
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
        if (a == 0) break;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      // Copy the comment
shun-iwasawa fdbab5
      {
shun-iwasawa fdbab5
        int msiz                   = ExifLength - (ValuePtr - OffsetBase);
shun-iwasawa fdbab5
        if (msiz > ByteCount) msiz = ByteCount;
shun-iwasawa fdbab5
        if (msiz > MAX_COMMENT_SIZE - 1) msiz = MAX_COMMENT_SIZE - 1;
shun-iwasawa fdbab5
        if (msiz > 5 && memcmp(ValuePtr, "ASCII", 5) == 0) {
shun-iwasawa fdbab5
          for (a = 5; a < 10 && a < msiz; a++) {
shun-iwasawa fdbab5
            int c = (ValuePtr)[a];
shun-iwasawa fdbab5
            if (c != '\0' && c != ' ') {
shun-iwasawa fdbab5
              strncpy(ImageInfo.Comments, (char *)ValuePtr + a, msiz - a);
shun-iwasawa fdbab5
              break;
shun-iwasawa fdbab5
            }
shun-iwasawa fdbab5
          }
shun-iwasawa fdbab5
        } else {
shun-iwasawa fdbab5
          strncpy(ImageInfo.Comments, (char *)ValuePtr, msiz);
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_FNUMBER:
shun-iwasawa fdbab5
      // Simplest way of expressing aperture, so I trust it the most.
shun-iwasawa fdbab5
      // (overwrite previously computd value if there is one)
shun-iwasawa fdbab5
      ImageInfo.ApertureFNumber = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_APERTURE:
shun-iwasawa fdbab5
    case TAG_MAXAPERTURE:
shun-iwasawa fdbab5
      // More relevant info always comes earlier, so only use this field if we
shun-iwasawa fdbab5
      // don't
shun-iwasawa fdbab5
      // have appropriate aperture information yet.
shun-iwasawa fdbab5
      if (ImageInfo.ApertureFNumber == 0) {
shun-iwasawa fdbab5
        ImageInfo.ApertureFNumber =
shun-iwasawa fdbab5
            (float)exp(ConvertAnyFormat(ValuePtr, Format) * log(2) * 0.5);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_FOCALLENGTH:
shun-iwasawa fdbab5
      // Nice digital cameras actually save the focal length as a function
shun-iwasawa fdbab5
      // of how farthey are zoomed in.
shun-iwasawa fdbab5
      ImageInfo.FocalLength = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_SUBJECT_DISTANCE:
shun-iwasawa fdbab5
      // Inidcates the distacne the autofocus camera is focused to.
shun-iwasawa fdbab5
      // Tends to be less accurate as distance increases.
shun-iwasawa fdbab5
      ImageInfo.Distance = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_EXPOSURETIME:
shun-iwasawa fdbab5
      // Simplest way of expressing exposure time, so I trust it most.
shun-iwasawa fdbab5
      // (overwrite previously computd value if there is one)
shun-iwasawa fdbab5
      ImageInfo.ExposureTime = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_SHUTTERSPEED:
shun-iwasawa fdbab5
      // More complicated way of expressing exposure time, so only use
shun-iwasawa fdbab5
      // this value if we don't already have it from somewhere else.
shun-iwasawa fdbab5
      if (ImageInfo.ExposureTime == 0) {
shun-iwasawa fdbab5
        ImageInfo.ExposureTime =
shun-iwasawa fdbab5
            (float)(1 / exp(ConvertAnyFormat(ValuePtr, Format) * log(2)));
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_FLASH:
shun-iwasawa fdbab5
      ImageInfo.FlashUsed = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_ORIENTATION:
shun-iwasawa fdbab5
      if (NumOrientations >= 2) {
shun-iwasawa fdbab5
        // Can have another orientation tag for the thumbnail, but if there's
shun-iwasawa fdbab5
        // a third one, things are strange.
shun-iwasawa fdbab5
        std::cout << "More than two orientation in Exif" << std::endl;
shun-iwasawa fdbab5
        break;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      OrientationPtr[NumOrientations]       = ValuePtr;
shun-iwasawa fdbab5
      OrientationNumFormat[NumOrientations] = Format;
shun-iwasawa fdbab5
      if (NumOrientations == 0) {
shun-iwasawa fdbab5
        ImageInfo.Orientation = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      if (ImageInfo.Orientation < 0 || ImageInfo.Orientation > 8) {
shun-iwasawa fdbab5
        std::cout << "Undefined rotation value " << ImageInfo.Orientation
shun-iwasawa fdbab5
                  << " in Exif" << std::endl;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      NumOrientations += 1;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_PIXEL_Y_DIMENSION:
shun-iwasawa fdbab5
    case TAG_PIXEL_X_DIMENSION:
shun-iwasawa fdbab5
      // Use largest of height and width to deal with images that have been
shun-iwasawa fdbab5
      // rotated to portrait format.
shun-iwasawa fdbab5
      a = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      if (ExifImageWidth < a) ExifImageWidth = a;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_FOCAL_PLANE_XRES:
shun-iwasawa fdbab5
      FocalplaneXRes = ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_FOCAL_PLANE_UNITS:
shun-iwasawa fdbab5
      switch ((int)ConvertAnyFormat(ValuePtr, Format)) {
shun-iwasawa fdbab5
      case 1:
shun-iwasawa fdbab5
        FocalplaneUnits = 25.4;
shun-iwasawa fdbab5
        break;  // inch
shun-iwasawa fdbab5
      case 2:
shun-iwasawa fdbab5
        // According to the information I was using, 2 means meters.
shun-iwasawa fdbab5
        // But looking at the Cannon powershot's files, inches is the only
shun-iwasawa fdbab5
        // sensible value.
shun-iwasawa fdbab5
        FocalplaneUnits = 25.4;
shun-iwasawa fdbab5
        break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      case 3:
shun-iwasawa fdbab5
        FocalplaneUnits = 10;
shun-iwasawa fdbab5
        break;  // centimeter
shun-iwasawa fdbab5
      case 4:
shun-iwasawa fdbab5
        FocalplaneUnits = 1;
shun-iwasawa fdbab5
        break;  // millimeter
shun-iwasawa fdbab5
      case 5:
shun-iwasawa fdbab5
        FocalplaneUnits = .001;
shun-iwasawa fdbab5
        break;  // micrometer
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_EXPOSURE_BIAS:
shun-iwasawa fdbab5
      ImageInfo.ExposureBias = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_WHITEBALANCE:
shun-iwasawa fdbab5
      ImageInfo.Whitebalance = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_LIGHT_SOURCE:
shun-iwasawa fdbab5
      ImageInfo.LightSource = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_METERING_MODE:
shun-iwasawa fdbab5
      ImageInfo.MeteringMode = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_EXPOSURE_PROGRAM:
shun-iwasawa fdbab5
      ImageInfo.ExposureProgram = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_EXPOSURE_INDEX:
shun-iwasawa fdbab5
      if (ImageInfo.ISOequivalent == 0) {
shun-iwasawa fdbab5
        // Exposure index and ISO equivalent are often used interchangeably,
shun-iwasawa fdbab5
        // so we will do the same in jhead.
shun-iwasawa fdbab5
        // http://photography.about.com/library/glossary/bldef_ei.htm
shun-iwasawa fdbab5
        ImageInfo.ISOequivalent = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_EXPOSURE_MODE:
shun-iwasawa fdbab5
      ImageInfo.ExposureMode = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_ISO_EQUIVALENT:
shun-iwasawa fdbab5
      ImageInfo.ISOequivalent = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_DIGITALZOOMRATIO:
shun-iwasawa fdbab5
      ImageInfo.DigitalZoomRatio = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_THUMBNAIL_OFFSET:
shun-iwasawa fdbab5
      ThumbnailOffset      = (unsigned)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      DirWithThumbnailPtrs = DirStart;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_THUMBNAIL_LENGTH:
shun-iwasawa fdbab5
      ThumbnailSize = (unsigned)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      ImageInfo.ThumbnailSizeOffset = ValuePtr - OffsetBase;
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_EXIF_OFFSET:
shun-iwasawa fdbab5
      if (ShowTags) printf("%s    Exif Dir:", IndentString);
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_INTEROP_OFFSET:
shun-iwasawa fdbab5
      if (Tag == TAG_INTEROP_OFFSET && ShowTags)
shun-iwasawa fdbab5
        printf("%s    Interop Dir:", IndentString);
shun-iwasawa fdbab5
      {
shun-iwasawa fdbab5
        unsigned char *SubdirStart;
shun-iwasawa fdbab5
        SubdirStart = OffsetBase + Get32u(ValuePtr);
shun-iwasawa fdbab5
        if (SubdirStart < OffsetBase || SubdirStart > OffsetBase + ExifLength) {
shun-iwasawa fdbab5
          std::cout << "Illegal Exif or interop ofset directory link"
shun-iwasawa fdbab5
                    << std::endl;
shun-iwasawa fdbab5
        } else {
shun-iwasawa fdbab5
          ProcessExifDir(SubdirStart, OffsetBase, ExifLength, NestingLevel + 1);
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
        continue;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_GPSINFO:
shun-iwasawa fdbab5
      if (ShowTags) printf("%s    GPS info dir:", IndentString);
shun-iwasawa fdbab5
      {
shun-iwasawa fdbab5
        unsigned char *SubdirStart;
shun-iwasawa fdbab5
        SubdirStart = OffsetBase + Get32u(ValuePtr);
shun-iwasawa fdbab5
        if (SubdirStart < OffsetBase || SubdirStart > OffsetBase + ExifLength) {
shun-iwasawa fdbab5
          std::cout << "Illegal GPS directory link in Exif" << std::endl;
shun-iwasawa fdbab5
        } else {
shun-iwasawa fdbab5
          // ProcessGpsInfo(SubdirStart, OffsetBase, ExifLength);
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
        continue;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_FOCALLENGTH_35MM:
shun-iwasawa fdbab5
      // The focal length equivalent 35 mm is a 2.2 tag (defined as of April
shun-iwasawa fdbab5
      // 2002)
shun-iwasawa fdbab5
      // if its present, use it to compute equivalent focal length instead of
shun-iwasawa fdbab5
      // computing it from sensor geometry and actual focal length.
shun-iwasawa fdbab5
      ImageInfo.FocalLength35mmEquiv =
shun-iwasawa fdbab5
          (unsigned)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_DISTANCE_RANGE:
shun-iwasawa fdbab5
      // Three possible standard values:
shun-iwasawa fdbab5
      //   1 = macro, 2 = close, 3 = distant
shun-iwasawa fdbab5
      ImageInfo.DistanceRange = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_X_RESOLUTION:
shun-iwasawa fdbab5
      if (NestingLevel ==
shun-iwasawa fdbab5
          0) {  // Only use the values from the top level directory
shun-iwasawa fdbab5
        ImageInfo.xResolution = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
        // if yResolution has not been set, use the value of xResolution
shun-iwasawa fdbab5
        if (ImageInfo.yResolution == 0.0)
shun-iwasawa fdbab5
          ImageInfo.yResolution = ImageInfo.xResolution;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_Y_RESOLUTION:
shun-iwasawa fdbab5
      if (NestingLevel ==
shun-iwasawa fdbab5
          0) {  // Only use the values from the top level directory
shun-iwasawa fdbab5
        ImageInfo.yResolution = (float)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
        // if xResolution has not been set, use the value of yResolution
shun-iwasawa fdbab5
        if (ImageInfo.xResolution == 0.0)
shun-iwasawa fdbab5
          ImageInfo.xResolution = ImageInfo.yResolution;
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    case TAG_RESOLUTION_UNIT:
shun-iwasawa fdbab5
      if (NestingLevel ==
shun-iwasawa fdbab5
          0) {  // Only use the values from the top level directory
shun-iwasawa fdbab5
        ImageInfo.ResolutionUnit = (int)ConvertAnyFormat(ValuePtr, Format);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      break;
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
#ifdef ReadAllTags
shun-iwasawa fdbab5
  {
shun-iwasawa fdbab5
    // In addition to linking to subdirectories via exif tags,
shun-iwasawa fdbab5
    // there's also a potential link to another directory at the end of each
shun-iwasawa fdbab5
    // directory.  this has got to be the result of a committee!
shun-iwasawa fdbab5
    unsigned char *SubdirStart;
shun-iwasawa fdbab5
    unsigned Offset;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if (DIR_ENTRY_ADDR(DirStart, NumDirEntries) + 4 <=
shun-iwasawa fdbab5
        OffsetBase + ExifLength) {
shun-iwasawa fdbab5
      Offset = Get32u(DirStart + 2 + 12 * NumDirEntries);
shun-iwasawa fdbab5
      if (Offset) {
shun-iwasawa fdbab5
        SubdirStart = OffsetBase + Offset;
shun-iwasawa fdbab5
        if (SubdirStart > OffsetBase + ExifLength || SubdirStart < OffsetBase) {
shun-iwasawa fdbab5
          if (SubdirStart > OffsetBase &&
shun-iwasawa fdbab5
              SubdirStart < OffsetBase + ExifLength + 20) {
shun-iwasawa fdbab5
            // Jhead 1.3 or earlier would crop the whole directory!
shun-iwasawa fdbab5
            // As Jhead produces this form of format incorrectness,
shun-iwasawa fdbab5
            // I'll just let it pass silently
shun-iwasawa fdbab5
            if (ShowTags)
shun-iwasawa fdbab5
              printf("Thumbnail removed with Jhead 1.3 or earlier\n");
shun-iwasawa fdbab5
          } else {
shun-iwasawa fdbab5
            std::cout << "Illegal subdirectory link in Exif header"
shun-iwasawa fdbab5
                      << std::endl;
shun-iwasawa fdbab5
          }
shun-iwasawa fdbab5
        } else {
shun-iwasawa fdbab5
          if (SubdirStart <= OffsetBase + ExifLength) {
shun-iwasawa fdbab5
            if (ShowTags) printf("%s    Continued directory ", IndentString);
shun-iwasawa fdbab5
            ProcessExifDir(SubdirStart, OffsetBase, ExifLength,
shun-iwasawa fdbab5
                           NestingLevel + 1);
shun-iwasawa fdbab5
          }
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
        if (Offset > ImageInfo.LargestExifOffset) {
shun-iwasawa fdbab5
          ImageInfo.LargestExifOffset = Offset;
shun-iwasawa fdbab5
        }
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
    } else {
shun-iwasawa fdbab5
      // The exif header ends before the last next directory pointer.
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
  if (ThumbnailOffset) {
shun-iwasawa fdbab5
    ImageInfo.ThumbnailAtEnd = FALSE;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if (DumpExifMap) {
shun-iwasawa fdbab5
      printf("Map: %05d-%05d: Thumbnail\n", ThumbnailOffset,
shun-iwasawa fdbab5
             ThumbnailOffset + ThumbnailSize);
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
shun-iwasawa fdbab5
    if (ThumbnailOffset <= ExifLength) {
shun-iwasawa fdbab5
      if (ThumbnailSize > ExifLength - ThumbnailOffset) {
shun-iwasawa fdbab5
        // If thumbnail extends past exif header, only save the part that
shun-iwasawa fdbab5
        // actually exists.  Canon's EOS viewer utility will do this - the
shun-iwasawa fdbab5
        // thumbnail extracts ok with this hack.
shun-iwasawa fdbab5
        ThumbnailSize = ExifLength - ThumbnailOffset;
shun-iwasawa fdbab5
        if (ShowTags) printf("Thumbnail incorrectly placed in header\n");
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
      // The thumbnail pointer appears to be valid.  Store it.
shun-iwasawa fdbab5
      ImageInfo.ThumbnailOffset = ThumbnailOffset;
shun-iwasawa fdbab5
      ImageInfo.ThumbnailSize   = ThumbnailSize;
shun-iwasawa fdbab5
shun-iwasawa fdbab5
      if (ShowTags) {
shun-iwasawa fdbab5
        printf("Thumbnail size: %u bytes\n", ThumbnailSize);
shun-iwasawa fdbab5
      }
shun-iwasawa fdbab5
    }
shun-iwasawa fdbab5
  }
shun-iwasawa fdbab5
#endif
shun-iwasawa fdbab5
}