ExifInterface.java revision f83358389f0c4ea37a7e7d9e493857f99baf0440
114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon/*
214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * Copyright (C) 2007 The Android Open Source Project
314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon *
414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * Licensed under the Apache License, Version 2.0 (the "License");
514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * you may not use this file except in compliance with the License.
614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * You may obtain a copy of the License at
714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon *
814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon *      http://www.apache.org/licenses/LICENSE-2.0
914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon *
1014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * Unless required by applicable law or agreed to in writing, software
1114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * distributed under the License is distributed on an "AS IS" BASIS,
1214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * See the License for the specific language governing permissions and
1414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * limitations under the License.
1514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon */
1614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
1714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonpackage android.support.media;
1814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
1914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport android.content.res.AssetManager;
2014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport android.graphics.Bitmap;
2114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport android.graphics.BitmapFactory;
2214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport android.support.annotation.IntDef;
2314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport android.support.annotation.NonNull;
2414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport android.util.Log;
2514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport android.util.Pair;
2614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
2714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.BufferedInputStream;
2814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.ByteArrayInputStream;
2914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.Closeable;
3014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.DataInput;
3114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.DataInputStream;
3214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.EOFException;
3314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.File;
3414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.FileInputStream;
3514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.FileNotFoundException;
3614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.FileOutputStream;
3714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.FilterOutputStream;
3814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.IOException;
3914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.InputStream;
4014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.io.OutputStream;
4114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.lang.annotation.Retention;
4214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.lang.annotation.RetentionPolicy;
4314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.nio.ByteBuffer;
4414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.nio.ByteOrder;
4514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.nio.charset.Charset;
4614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.text.ParsePosition;
4714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.text.SimpleDateFormat;
4814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.Arrays;
4914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.Date;
5014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.HashMap;
5114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.HashSet;
5214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.Map;
5314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.Set;
5414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.TimeZone;
5514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.regex.Matcher;
5614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonimport java.util.regex.Pattern;
5714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
5814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon/**
5914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * This is a class for reading and writing Exif tags in a JPEG file or a RAW image file.
6014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * <p>
6114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * Supported formats are: JPEG, DNG, CR2, NEF, NRW, ARW, RW2, ORF, PEF, SRW and RAF.
6214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * <p>
6314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon * Attribute mutation is supported for JPEG image files.
6414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon */
6514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moonpublic class ExifInterface {
6614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG = "ExifInterface";
6714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final boolean DEBUG = false;
6814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
6914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // The Exif tag names. See Tiff 6.0 Section 3 and Section 8.
7014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
7114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ARTIST = "Artist";
7214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
7314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_BITS_PER_SAMPLE = "BitsPerSample";
7414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
7514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_COMPRESSION = "Compression";
7614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
7714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_COPYRIGHT = "Copyright";
7814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
7914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_DATETIME = "DateTime";
8014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
8114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_IMAGE_DESCRIPTION = "ImageDescription";
8214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
8314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_IMAGE_LENGTH = "ImageLength";
8414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
8514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_IMAGE_WIDTH = "ImageWidth";
8614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
8714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_JPEG_INTERCHANGE_FORMAT = "JPEGInterchangeFormat";
8814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
8914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_JPEG_INTERCHANGE_FORMAT_LENGTH = "JPEGInterchangeFormatLength";
9014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
9114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_MAKE = "Make";
9214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
9314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_MODEL = "Model";
9414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
9514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ORIENTATION = "Orientation";
9614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
9714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_PHOTOMETRIC_INTERPRETATION = "PhotometricInterpretation";
9814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
9914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_PLANAR_CONFIGURATION = "PlanarConfiguration";
10014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
10114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_PRIMARY_CHROMATICITIES = "PrimaryChromaticities";
10214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
10314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_REFERENCE_BLACK_WHITE = "ReferenceBlackWhite";
10414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
10514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RESOLUTION_UNIT = "ResolutionUnit";
10614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
10714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ROWS_PER_STRIP = "RowsPerStrip";
10814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
10914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SAMPLES_PER_PIXEL = "SamplesPerPixel";
11014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
11114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SOFTWARE = "Software";
11214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
11314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_STRIP_BYTE_COUNTS = "StripByteCounts";
11414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
11514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_STRIP_OFFSETS = "StripOffsets";
11614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
11714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_TRANSFER_FUNCTION = "TransferFunction";
11814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
11914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_WHITE_POINT = "WhitePoint";
12014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
12114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_X_RESOLUTION = "XResolution";
12214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
12314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_Y_CB_CR_COEFFICIENTS = "YCbCrCoefficients";
12414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
12514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_Y_CB_CR_POSITIONING = "YCbCrPositioning";
12614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
12714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_Y_CB_CR_SUB_SAMPLING = "YCbCrSubSampling";
12814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
12914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_Y_RESOLUTION = "YResolution";
13014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
13114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_APERTURE_VALUE = "ApertureValue";
13214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
13314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_BRIGHTNESS_VALUE = "BrightnessValue";
13414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
13514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_CFA_PATTERN = "CFAPattern";
13614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
13714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_COLOR_SPACE = "ColorSpace";
13814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
13914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_COMPONENTS_CONFIGURATION = "ComponentsConfiguration";
14014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
14114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_COMPRESSED_BITS_PER_PIXEL = "CompressedBitsPerPixel";
14214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
14314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_CONTRAST = "Contrast";
14414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
14514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_CUSTOM_RENDERED = "CustomRendered";
14614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
14714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_DATETIME_DIGITIZED = "DateTimeDigitized";
14814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
14914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_DATETIME_ORIGINAL = "DateTimeOriginal";
15014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
15114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_DEVICE_SETTING_DESCRIPTION = "DeviceSettingDescription";
15214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is double. */
15314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_DIGITAL_ZOOM_RATIO = "DigitalZoomRatio";
15414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
15514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_EXIF_VERSION = "ExifVersion";
15614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is double. */
15714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_EXPOSURE_BIAS_VALUE = "ExposureBiasValue";
15814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
15914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_EXPOSURE_INDEX = "ExposureIndex";
16014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
16114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_EXPOSURE_MODE = "ExposureMode";
16214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
16314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_EXPOSURE_PROGRAM = "ExposureProgram";
16414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is double. */
16514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_EXPOSURE_TIME = "ExposureTime";
16614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is double. */
16714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_F_NUMBER = "FNumber";
16814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
16914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FILE_SOURCE = "FileSource";
17014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
17114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FLASH = "Flash";
17214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
17314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FLASH_ENERGY = "FlashEnergy";
17414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
17514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FLASHPIX_VERSION = "FlashpixVersion";
17614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
17714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FOCAL_LENGTH = "FocalLength";
17814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
17914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FOCAL_LENGTH_IN_35MM_FILM = "FocalLengthIn35mmFilm";
18014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
18114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FOCAL_PLANE_RESOLUTION_UNIT = "FocalPlaneResolutionUnit";
18214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
18314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FOCAL_PLANE_X_RESOLUTION = "FocalPlaneXResolution";
18414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
18514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_FOCAL_PLANE_Y_RESOLUTION = "FocalPlaneYResolution";
18614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
18714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GAIN_CONTROL = "GainControl";
18814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
18914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ISO_SPEED_RATINGS = "ISOSpeedRatings";
19014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
19114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_IMAGE_UNIQUE_ID = "ImageUniqueID";
19214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
19314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_LIGHT_SOURCE = "LightSource";
19414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
19514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_MAKER_NOTE = "MakerNote";
19614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
19714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_MAX_APERTURE_VALUE = "MaxApertureValue";
19814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
19914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_METERING_MODE = "MeteringMode";
20014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
20114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_NEW_SUBFILE_TYPE = "NewSubfileType";
20214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
20314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_OECF = "OECF";
20414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
20514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_PIXEL_X_DIMENSION = "PixelXDimension";
20614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
20714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_PIXEL_Y_DIMENSION = "PixelYDimension";
20814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
20914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RELATED_SOUND_FILE = "RelatedSoundFile";
21014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
21114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SATURATION = "Saturation";
21214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
21314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SCENE_CAPTURE_TYPE = "SceneCaptureType";
21414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
21514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SCENE_TYPE = "SceneType";
21614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
21714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SENSING_METHOD = "SensingMethod";
21814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
21914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SHARPNESS = "Sharpness";
22014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
22114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SHUTTER_SPEED_VALUE = "ShutterSpeedValue";
22214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
22314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SPATIAL_FREQUENCY_RESPONSE = "SpatialFrequencyResponse";
22414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
22514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SPECTRAL_SENSITIVITY = "SpectralSensitivity";
22614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
22714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBFILE_TYPE = "SubfileType";
22814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
22914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBSEC_TIME = "SubSecTime";
23014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
23114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBSEC_TIME_DIGITIZED = "SubSecTimeDigitized";
23214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
23314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBSEC_TIME_ORIGINAL = "SubSecTimeOriginal";
23414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
23514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBJECT_AREA = "SubjectArea";
23614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is double. */
23714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBJECT_DISTANCE = "SubjectDistance";
23814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
23914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBJECT_DISTANCE_RANGE = "SubjectDistanceRange";
24014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
24114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_SUBJECT_LOCATION = "SubjectLocation";
24214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
24314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_USER_COMMENT = "UserComment";
24414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
24514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_WHITE_BALANCE = "WhiteBalance";
24614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
24714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * The altitude (in meters) based on the reference in TAG_GPS_ALTITUDE_REF.
24814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Type is rational.
24914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
25014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_ALTITUDE = "GPSAltitude";
25114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
25214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * 0 if the altitude is above sea level. 1 if the altitude is below sea
25314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * level. Type is int.
25414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
25514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_ALTITUDE_REF = "GPSAltitudeRef";
25614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
25714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_AREA_INFORMATION = "GPSAreaInformation";
25814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
25914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DOP = "GPSDOP";
26014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
26114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DATESTAMP = "GPSDateStamp";
26214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
26314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_BEARING = "GPSDestBearing";
26414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
26514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_BEARING_REF = "GPSDestBearingRef";
26614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
26714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_DISTANCE = "GPSDestDistance";
26814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
26914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_DISTANCE_REF = "GPSDestDistanceRef";
27014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
27114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_LATITUDE = "GPSDestLatitude";
27214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
27314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_LATITUDE_REF = "GPSDestLatitudeRef";
27414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
27514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_LONGITUDE = "GPSDestLongitude";
27614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
27714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DEST_LONGITUDE_REF = "GPSDestLongitudeRef";
27814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
27914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_DIFFERENTIAL = "GPSDifferential";
28014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
28114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_IMG_DIRECTION = "GPSImgDirection";
28214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
28314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_IMG_DIRECTION_REF = "GPSImgDirectionRef";
28414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. Format is "num1/denom1,num2/denom2,num3/denom3". */
28514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_LATITUDE = "GPSLatitude";
28614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
28714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_LATITUDE_REF = "GPSLatitudeRef";
28814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. Format is "num1/denom1,num2/denom2,num3/denom3". */
28914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_LONGITUDE = "GPSLongitude";
29014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
29114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_LONGITUDE_REF = "GPSLongitudeRef";
29214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
29314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_MAP_DATUM = "GPSMapDatum";
29414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
29514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_MEASURE_MODE = "GPSMeasureMode";
29614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. Name of GPS processing method used for location finding. */
29714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_PROCESSING_METHOD = "GPSProcessingMethod";
29814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
29914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_SATELLITES = "GPSSatellites";
30014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
30114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_SPEED = "GPSSpeed";
30214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
30314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_SPEED_REF = "GPSSpeedRef";
30414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
30514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_STATUS = "GPSStatus";
30614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. Format is "hh:mm:ss". */
30714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_TIMESTAMP = "GPSTimeStamp";
30814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is rational. */
30914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_TRACK = "GPSTrack";
31014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
31114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_TRACK_REF = "GPSTrackRef";
31214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
31314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_GPS_VERSION_ID = "GPSVersionID";
31414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is String. */
31514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_INTEROPERABILITY_INDEX = "InteroperabilityIndex";
31614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
31714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_THUMBNAIL_IMAGE_LENGTH = "ThumbnailImageLength";
31814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. */
31914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_THUMBNAIL_IMAGE_WIDTH = "ThumbnailImageWidth";
32014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. DNG Specification 1.4.0.0. Section 4 */
32114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_DNG_VERSION = "DNGVersion";
32214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. DNG Specification 1.4.0.0. Section 4 */
32314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_DEFAULT_CROP_SIZE = "DefaultCropSize";
32414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is undefined. See Olympus MakerNote tags in http://www.exiv2.org/tags-olympus.html. */
32514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ORF_THUMBNAIL_IMAGE = "ThumbnailImage";
32614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. See Olympus Camera Settings tags in http://www.exiv2.org/tags-olympus.html. */
32714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ORF_PREVIEW_IMAGE_START = "PreviewImageStart";
32814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. See Olympus Camera Settings tags in http://www.exiv2.org/tags-olympus.html. */
32914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ORF_PREVIEW_IMAGE_LENGTH = "PreviewImageLength";
33014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** Type is int. See Olympus Image Processing tags in http://www.exiv2.org/tags-olympus.html. */
33114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_ORF_ASPECT_FRAME = "AspectFrame";
33214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
33314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Type is int. See PanasonicRaw tags in
33414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/PanasonicRaw.html
33514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
33614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RW2_SENSOR_BOTTOM_BORDER = "SensorBottomBorder";
33714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
33814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Type is int. See PanasonicRaw tags in
33914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/PanasonicRaw.html
34014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
34114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RW2_SENSOR_LEFT_BORDER = "SensorLeftBorder";
34214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
34314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Type is int. See PanasonicRaw tags in
34414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/PanasonicRaw.html
34514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
34614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RW2_SENSOR_RIGHT_BORDER = "SensorRightBorder";
34714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
34814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Type is int. See PanasonicRaw tags in
34914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/PanasonicRaw.html
35014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
35114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RW2_SENSOR_TOP_BORDER = "SensorTopBorder";
35214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
35314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Type is int. See PanasonicRaw tags in
35414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/PanasonicRaw.html
35514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
35614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RW2_ISO = "ISO";
35714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
35814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Type is undefined. See PanasonicRaw tags in
35914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/PanasonicRaw.html
36014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
36114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final String TAG_RW2_JPG_FROM_RAW = "JpgFromRaw";
36214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
36314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
36414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Private tags used for pointing the other IFD offsets.
36514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * The types of the following tags are int.
36614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See JEITA CP-3451C Section 4.6.3: Exif-specific IFD.
36714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For SubIFD, see Note 1 of Adobe PageMaker® 6.0 TIFF Technical Notes.
36814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
36914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_EXIF_IFD_POINTER = "ExifIFDPointer";
37014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_GPS_INFO_IFD_POINTER = "GPSInfoIFDPointer";
37114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_INTEROPERABILITY_IFD_POINTER = "InteroperabilityIFDPointer";
37214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_SUB_IFD_POINTER = "SubIFDPointer";
37314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Proprietary pointer tags used for ORF files.
37414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See http://www.exiv2.org/tags-olympus.html
37514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_ORF_CAMERA_SETTINGS_IFD_POINTER = "CameraSettingsIFDPointer";
37614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_ORF_IMAGE_PROCESSING_IFD_POINTER = "ImageProcessingIFDPointer";
37714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
37814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Private tags used for thumbnail information.
37914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_HAS_THUMBNAIL = "HasThumbnail";
38014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_THUMBNAIL_OFFSET = "ThumbnailOffset";
38114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_THUMBNAIL_LENGTH = "ThumbnailLength";
38214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String TAG_THUMBNAIL_DATA = "ThumbnailData";
38314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int MAX_THUMBNAIL_SIZE = 512;
38414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
38514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Constants used for the Orientation Exif tag.
38614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_UNDEFINED = 0;
38714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_NORMAL = 1;
38814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_FLIP_HORIZONTAL = 2;  // left right reversed mirror
38914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_ROTATE_180 = 3;
39014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_FLIP_VERTICAL = 4;  // upside down mirror
39114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // flipped about top-left <--> bottom-right axis
39214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_TRANSPOSE = 5;
39314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_ROTATE_90 = 6;  // rotate 90 cw to right it
39414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // flipped about top-right <--> bottom-left axis
39514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_TRANSVERSE = 7;
39614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int ORIENTATION_ROTATE_270 = 8;  // rotate 270 to right it
39714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
39814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Constants used for white balance
39914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int WHITEBALANCE_AUTO = 0;
40014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public static final int WHITEBALANCE_MANUAL = 1;
40114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
40214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Maximum size for checking file type signature (see image_type_recognition_lite.cc)
40314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int SIGNATURE_CHECK_SIZE = 5000;
40414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
40514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte[] JPEG_SIGNATURE = new byte[] {(byte) 0xff, (byte) 0xd8, (byte) 0xff};
40614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String RAF_SIGNATURE = "FUJIFILMCCD-RAW";
40714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int RAF_OFFSET_TO_JPEG_IMAGE_OFFSET = 84;
40814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int RAF_INFO_SIZE = 160;
40914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int RAF_JPEG_LENGTH_VALUE_SIZE = 4;
41014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
41114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See http://fileformats.archiveteam.org/wiki/Olympus_ORF
41214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final short ORF_SIGNATURE_1 = 0x4f52;
41314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final short ORF_SIGNATURE_2 = 0x5352;
41414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // There are two formats for Olympus Makernote Headers. Each has different identifiers and
41514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // offsets to the actual data.
41614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See http://www.exiv2.org/makernote.html#R1
41714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte[] ORF_MAKER_NOTE_HEADER_1 = new byte[] {(byte) 0x4f, (byte) 0x4c,
41814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            (byte) 0x59, (byte) 0x4d, (byte) 0x50, (byte) 0x00}; // "OLYMP\0"
41914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte[] ORF_MAKER_NOTE_HEADER_2 = new byte[] {(byte) 0x4f, (byte) 0x4c,
42014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            (byte) 0x59, (byte) 0x4d, (byte) 0x50, (byte) 0x55, (byte) 0x53, (byte) 0x00,
42114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            (byte) 0x49, (byte) 0x49}; // "OLYMPUS\0II"
42214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int ORF_MAKER_NOTE_HEADER_1_SIZE = 8;
42314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int ORF_MAKER_NOTE_HEADER_2_SIZE = 12;
42414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
42514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See http://fileformats.archiveteam.org/wiki/RW2
42614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final short RW2_SIGNATURE = 0x0055;
42714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
42814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See http://fileformats.archiveteam.org/wiki/Pentax_PEF
42914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String PEF_SIGNATURE = "PENTAX";
43014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See http://www.exiv2.org/makernote.html#R11
43114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int PEF_MAKER_NOTE_SKIP_SIZE = 6;
43214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
43314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static SimpleDateFormat sFormatter;
43414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
43514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See Exchangeable image file format for digital still cameras: Exif version 2.2.
43614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // The following values are for parsing EXIF data area. There are tag groups in EXIF data area.
43714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // They are called "Image File Directory". They have multiple data formats to cover various
43814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // image metadata from GPS longitude to camera model name.
43914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
44014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Types of Exif byte alignments (see JEITA CP-3451C Section 4.5.2)
44114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final short BYTE_ALIGN_II = 0x4949;  // II: Intel order
44214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final short BYTE_ALIGN_MM = 0x4d4d;  // MM: Motorola order
44314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
44414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // TIFF Header Fixed Constant (see JEITA CP-3451C Section 4.5.2)
44514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte START_CODE = 0x2a; // 42
44614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_OFFSET = 8;
44714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
44814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Formats for the value in IFD entry (See TIFF 6.0 Section 2, "Image File Directory".)
44914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_BYTE = 1;
45014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_STRING = 2;
45114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_USHORT = 3;
45214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_ULONG = 4;
45314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_URATIONAL = 5;
45414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_SBYTE = 6;
45514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_UNDEFINED = 7;
45614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_SSHORT = 8;
45714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_SLONG = 9;
45814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_SRATIONAL = 10;
45914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_SINGLE = 11;
46014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_DOUBLE = 12;
46114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Format indicating a new IFD entry (See Adobe PageMaker® 6.0 TIFF Technical Notes, "New Tag")
46214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_FORMAT_IFD = 13;
46314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Names for the data formats for debugging purpose.
46414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final String[] IFD_FORMAT_NAMES = new String[] {
46514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            "", "BYTE", "STRING", "USHORT", "ULONG", "URATIONAL", "SBYTE", "UNDEFINED", "SSHORT",
46614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            "SLONG", "SRATIONAL", "SINGLE", "DOUBLE"
46714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
46814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Sizes of the components of each IFD value format
46914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int[] IFD_FORMAT_BYTES_PER_FORMAT = new int[] {
47014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            0, 1, 1, 2, 4, 8, 1, 1, 2, 4, 8, 4, 8, 1
47114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
47214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte[] EXIF_ASCII_PREFIX = new byte[] {
47314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            0x41, 0x53, 0x43, 0x49, 0x49, 0x0, 0x0, 0x0
47414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
47514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
47614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
47714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Constants used for Compression tag.
47814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For Value 1, 2, 32773, see TIFF 6.0 Spec Section 3: Bilevel Images, Compression
47914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For Value 6, see TIFF 6.0 Spec Section 22: JPEG Compression, Extensions to Existing Fields
48014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For Value 7, 8, 34892, see DNG Specification 1.4.0.0. Section 3, Compression
48114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
48214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int DATA_UNCOMPRESSED = 1;
48314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int DATA_HUFFMAN_COMPRESSED = 2;
48414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int DATA_JPEG = 6;
48514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int DATA_JPEG_COMPRESSED = 7;
48614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int DATA_DEFLATE_ZIP = 8;
48714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int DATA_PACK_BITS_COMPRESSED = 32773;
48814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int DATA_LOSSY_JPEG = 34892;
48914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
49014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
49114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Constants used for BitsPerSample tag.
49214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For RGB, see TIFF 6.0 Spec Section 6, Differences from Palette Color Images
49314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For Greyscale, see TIFF 6.0 Spec Section 4, Differences from Bilevel Images
49414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
49514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int[] BITS_PER_SAMPLE_RGB = new int[] { 8, 8, 8 };
49614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int[] BITS_PER_SAMPLE_GREYSCALE_1 = new int[] { 4 };
49714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int[] BITS_PER_SAMPLE_GREYSCALE_2 = new int[] { 8 };
49814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
49914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
50014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Constants used for PhotometricInterpretation tag.
50114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For White/Black, see Section 3, Color.
50214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See TIFF 6.0 Spec Section 22, Minimum Requirements for TIFF with JPEG Compression.
50314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
50414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int PHOTOMETRIC_INTERPRETATION_WHITE_IS_ZERO = 0;
50514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int PHOTOMETRIC_INTERPRETATION_BLACK_IS_ZERO = 1;
50614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int PHOTOMETRIC_INTERPRETATION_RGB = 2;
50714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int PHOTOMETRIC_INTERPRETATION_YCBCR = 6;
50814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
50914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
51014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Constants used for NewSubfileType tag.
51114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See TIFF 6.0 Spec Section 8
51214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * */
51314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int ORIGINAL_RESOLUTION_IMAGE = 0;
51414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int REDUCED_RESOLUTION_IMAGE = 1;
51514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
51614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // A class for indicating EXIF rational type.
51714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static class Rational {
51814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final long numerator;
51914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final long denominator;
52014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
52114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private Rational(long numerator, long denominator) {
52214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Handle erroneous case
52314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (denominator == 0) {
52414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                this.numerator = 0;
52514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                this.denominator = 1;
52614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return;
52714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
52814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.numerator = numerator;
52914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.denominator = denominator;
53014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
53114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
53214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
53314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public String toString() {
53414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return numerator + "/" + denominator;
53514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
53614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
53714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public double calculate() {
53814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return (double) numerator / denominator;
53914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
54014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
54114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
54214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // A class for indicating EXIF attribute.
54314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static class ExifAttribute {
54414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final int format;
54514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final int numberOfComponents;
54614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final byte[] bytes;
54714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
54814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private ExifAttribute(int format, int numberOfComponents, byte[] bytes) {
54914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.format = format;
55014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.numberOfComponents = numberOfComponents;
55114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.bytes = bytes;
55214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
55314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
55414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createUShort(int[] values, ByteOrder byteOrder) {
55514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final ByteBuffer buffer = ByteBuffer.wrap(
55614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    new byte[IFD_FORMAT_BYTES_PER_FORMAT[IFD_FORMAT_USHORT] * values.length]);
55714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            buffer.order(byteOrder);
55814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (int value : values) {
55914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putShort((short) value);
56014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
56114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_USHORT, values.length, buffer.array());
56214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
56314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
56414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createUShort(int value, ByteOrder byteOrder) {
56514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return createUShort(new int[] {value}, byteOrder);
56614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
56714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
56814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createULong(long[] values, ByteOrder byteOrder) {
56914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final ByteBuffer buffer = ByteBuffer.wrap(
57014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    new byte[IFD_FORMAT_BYTES_PER_FORMAT[IFD_FORMAT_ULONG] * values.length]);
57114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            buffer.order(byteOrder);
57214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (long value : values) {
57314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putInt((int) value);
57414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
57514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_ULONG, values.length, buffer.array());
57614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
57714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
57814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createULong(long value, ByteOrder byteOrder) {
57914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return createULong(new long[] {value}, byteOrder);
58014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
58114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
58214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createSLong(int[] values, ByteOrder byteOrder) {
58314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final ByteBuffer buffer = ByteBuffer.wrap(
58414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    new byte[IFD_FORMAT_BYTES_PER_FORMAT[IFD_FORMAT_SLONG] * values.length]);
58514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            buffer.order(byteOrder);
58614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (int value : values) {
58714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putInt(value);
58814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
58914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_SLONG, values.length, buffer.array());
59014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
59114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
59214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createSLong(int value, ByteOrder byteOrder) {
59314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return createSLong(new int[] {value}, byteOrder);
59414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
59514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
59614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createByte(String value) {
59714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Exception for GPSAltitudeRef tag
59814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value.length() == 1 && value.charAt(0) >= '0' && value.charAt(0) <= '1') {
59914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                final byte[] bytes = new byte[] { (byte) (value.charAt(0) - '0') };
60014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return new ExifAttribute(IFD_FORMAT_BYTE, bytes.length, bytes);
60114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
60214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final byte[] ascii = value.getBytes(ASCII);
60314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_BYTE, ascii.length, ascii);
60414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
60514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
60614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createString(String value) {
60714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final byte[] ascii = (value + '\0').getBytes(ASCII);
60814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_STRING, ascii.length, ascii);
60914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
61014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
61114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createURational(Rational[] values, ByteOrder byteOrder) {
61214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final ByteBuffer buffer = ByteBuffer.wrap(
61314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    new byte[IFD_FORMAT_BYTES_PER_FORMAT[IFD_FORMAT_URATIONAL] * values.length]);
61414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            buffer.order(byteOrder);
61514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (Rational value : values) {
61614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putInt((int) value.numerator);
61714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putInt((int) value.denominator);
61814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
61914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_URATIONAL, values.length, buffer.array());
62014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
62114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
62214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createURational(Rational value, ByteOrder byteOrder) {
62314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return createURational(new Rational[] {value}, byteOrder);
62414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
62514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
62614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createSRational(Rational[] values, ByteOrder byteOrder) {
62714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final ByteBuffer buffer = ByteBuffer.wrap(
62814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    new byte[IFD_FORMAT_BYTES_PER_FORMAT[IFD_FORMAT_SRATIONAL] * values.length]);
62914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            buffer.order(byteOrder);
63014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (Rational value : values) {
63114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putInt((int) value.numerator);
63214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putInt((int) value.denominator);
63314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
63414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_SRATIONAL, values.length, buffer.array());
63514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
63614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
63714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createSRational(Rational value, ByteOrder byteOrder) {
63814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return createSRational(new Rational[] {value}, byteOrder);
63914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
64014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
64114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createDouble(double[] values, ByteOrder byteOrder) {
64214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final ByteBuffer buffer = ByteBuffer.wrap(
64314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    new byte[IFD_FORMAT_BYTES_PER_FORMAT[IFD_FORMAT_DOUBLE] * values.length]);
64414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            buffer.order(byteOrder);
64514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (double value : values) {
64614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                buffer.putDouble(value);
64714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
64814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new ExifAttribute(IFD_FORMAT_DOUBLE, values.length, buffer.array());
64914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
65014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
65114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public static ExifAttribute createDouble(double value, ByteOrder byteOrder) {
65214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return createDouble(new double[] {value}, byteOrder);
65314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
65414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
65514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
65614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public String toString() {
65714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return "(" + IFD_FORMAT_NAMES[format] + ", data length:" + bytes.length + ")";
65814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
65914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
66014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private Object getValue(ByteOrder byteOrder) {
661f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes            ByteOrderedDataInputStream inputStream = null;
66214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            try {
663f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                inputStream = new ByteOrderedDataInputStream(bytes);
66414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                inputStream.setByteOrder(byteOrder);
66514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                switch (format) {
66614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_BYTE:
66714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SBYTE: {
66814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // Exception for GPSAltitudeRef tag
66914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        if (bytes.length == 1 && bytes[0] >= 0 && bytes[0] <= 1) {
67014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            return new String(new char[] { (char) (bytes[0] + '0') });
67114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
67214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return new String(bytes, ASCII);
67314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
67414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_UNDEFINED:
67514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_STRING: {
67614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        int index = 0;
67714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        if (numberOfComponents >= EXIF_ASCII_PREFIX.length) {
67814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            boolean same = true;
67914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            for (int i = 0; i < EXIF_ASCII_PREFIX.length; ++i) {
68014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                if (bytes[i] != EXIF_ASCII_PREFIX[i]) {
68114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                    same = false;
68214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                    break;
68314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                }
68414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            }
68514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            if (same) {
68614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                index = EXIF_ASCII_PREFIX.length;
68714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            }
68814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
68914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
69014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        StringBuilder stringBuilder = new StringBuilder();
69114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        while (index < numberOfComponents) {
69214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            int ch = bytes[index];
69314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            if (ch == 0) {
69414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                break;
69514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            }
69614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            if (ch >= 32) {
69714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                stringBuilder.append((char) ch);
69814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            } else {
69914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                stringBuilder.append('?');
70014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            }
70114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            ++index;
70214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
70314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return stringBuilder.toString();
70414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
70514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_USHORT: {
70614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final int[] values = new int[numberOfComponents];
70714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
70814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = inputStream.readUnsignedShort();
70914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
71014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
71114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
71214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_ULONG: {
71314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final long[] values = new long[numberOfComponents];
71414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
71514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = inputStream.readUnsignedInt();
71614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
71714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
71814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
71914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_URATIONAL: {
72014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final Rational[] values = new Rational[numberOfComponents];
72114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
72214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            final long numerator = inputStream.readUnsignedInt();
72314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            final long denominator = inputStream.readUnsignedInt();
72414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = new Rational(numerator, denominator);
72514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
72614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
72714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
72814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SSHORT: {
72914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final int[] values = new int[numberOfComponents];
73014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
73114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = inputStream.readShort();
73214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
73314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
73414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
73514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SLONG: {
73614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final int[] values = new int[numberOfComponents];
73714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
73814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = inputStream.readInt();
73914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
74014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
74114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
74214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SRATIONAL: {
74314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final Rational[] values = new Rational[numberOfComponents];
74414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
74514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            final long numerator = inputStream.readInt();
74614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            final long denominator = inputStream.readInt();
74714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = new Rational(numerator, denominator);
74814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
74914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
75014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
75114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SINGLE: {
75214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final double[] values = new double[numberOfComponents];
75314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
75414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = inputStream.readFloat();
75514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
75614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
75714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
75814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_DOUBLE: {
75914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final double[] values = new double[numberOfComponents];
76014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int i = 0; i < numberOfComponents; ++i) {
76114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            values[i] = inputStream.readDouble();
76214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
76314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return values;
76414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
76514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    default:
76614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return null;
76714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
76814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } catch (IOException e) {
76914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.w(TAG, "IOException occurred during reading a value", e);
77014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return null;
771f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes            } finally {
772f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                if (inputStream != null) {
773f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                    try {
774f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                        inputStream.close();
775f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                    } catch (IOException e) {
776f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                        Log.e(TAG, "IOException occurred while closing InputStream", e);
777f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                    }
778f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes                }
77914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
78014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
78114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
78214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public double getDoubleValue(ByteOrder byteOrder) {
78314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Object value = getValue(byteOrder);
78414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value == null) {
78514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("NULL can't be converted to a double value");
78614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
78714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof String) {
78814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return Double.parseDouble((String) value);
78914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
79014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof long[]) {
79114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                long[] array = (long[]) value;
79214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (array.length == 1) {
79314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return array[0];
79414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
79514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("There are more than one component");
79614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
79714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof int[]) {
79814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int[] array = (int[]) value;
79914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (array.length == 1) {
80014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return array[0];
80114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
80214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("There are more than one component");
80314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
80414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof double[]) {
80514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                double[] array = (double[]) value;
80614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (array.length == 1) {
80714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return array[0];
80814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
80914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("There are more than one component");
81014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
81114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof Rational[]) {
81214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Rational[] array = (Rational[]) value;
81314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (array.length == 1) {
81414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return array[0].calculate();
81514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
81614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("There are more than one component");
81714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
81814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new NumberFormatException("Couldn't find a double value");
81914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
82014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
82114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int getIntValue(ByteOrder byteOrder) {
82214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Object value = getValue(byteOrder);
82314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value == null) {
82414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("NULL can't be converted to a integer value");
82514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
82614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof String) {
82714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return Integer.parseInt((String) value);
82814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
82914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof long[]) {
83014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                long[] array = (long[]) value;
83114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (array.length == 1) {
83214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return (int) array[0];
83314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
83414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("There are more than one component");
83514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
83614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof int[]) {
83714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int[] array = (int[]) value;
83814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (array.length == 1) {
83914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return array[0];
84014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
84114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new NumberFormatException("There are more than one component");
84214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
84314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new NumberFormatException("Couldn't find a integer value");
84414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
84514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
84614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public String getStringValue(ByteOrder byteOrder) {
84714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Object value = getValue(byteOrder);
84814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value == null) {
84914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return null;
85014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
85114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof String) {
85214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return (String) value;
85314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
85414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
85514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final StringBuilder stringBuilder = new StringBuilder();
85614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof long[]) {
85714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                long[] array = (long[]) value;
85814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                for (int i = 0; i < array.length; ++i) {
85914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    stringBuilder.append(array[i]);
86014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (i + 1 != array.length) {
86114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        stringBuilder.append(",");
86214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
86314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
86414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return stringBuilder.toString();
86514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
86614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof int[]) {
86714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int[] array = (int[]) value;
86814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                for (int i = 0; i < array.length; ++i) {
86914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    stringBuilder.append(array[i]);
87014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (i + 1 != array.length) {
87114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        stringBuilder.append(",");
87214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
87314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
87414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return stringBuilder.toString();
87514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
87614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof double[]) {
87714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                double[] array = (double[]) value;
87814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                for (int i = 0; i < array.length; ++i) {
87914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    stringBuilder.append(array[i]);
88014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (i + 1 != array.length) {
88114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        stringBuilder.append(",");
88214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
88314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
88414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return stringBuilder.toString();
88514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
88614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value instanceof Rational[]) {
88714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Rational[] array = (Rational[]) value;
88814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                for (int i = 0; i < array.length; ++i) {
88914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    stringBuilder.append(array[i].numerator);
89014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    stringBuilder.append('/');
89114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    stringBuilder.append(array[i].denominator);
89214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (i + 1 != array.length) {
89314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        stringBuilder.append(",");
89414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
89514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
89614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return stringBuilder.toString();
89714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
89814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return null;
89914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
90014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
90114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int size() {
90214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return IFD_FORMAT_BYTES_PER_FORMAT[format] * numberOfComponents;
90314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
90414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
90514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
90614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // A class for indicating EXIF tag.
90714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static class ExifTag {
90814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final int number;
90914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final String name;
91014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final int primaryFormat;
91114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public final int secondaryFormat;
91214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
91314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private ExifTag(String name, int number, int format) {
91414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.name = name;
91514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.number = number;
91614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.primaryFormat = format;
91714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.secondaryFormat = -1;
91814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
91914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
92014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private ExifTag(String name, int number, int primaryFormat, int secondaryFormat) {
92114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.name = name;
92214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.number = number;
92314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.primaryFormat = primaryFormat;
92414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this.secondaryFormat = secondaryFormat;
92514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
92614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
92714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
92814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Primary image IFD TIFF tags (See JEITA CP-3451C Section 4.6.8 Tag Support Levels)
92914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] IFD_TIFF_TAGS = new ExifTag[] {
93014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // For below two, see TIFF 6.0 Spec Section 3: Bilevel Images.
93114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_NEW_SUBFILE_TYPE, 254, IFD_FORMAT_ULONG),
93214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBFILE_TYPE, 255, IFD_FORMAT_ULONG),
93314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_IMAGE_WIDTH, 256, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
93414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_IMAGE_LENGTH, 257, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
93514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_BITS_PER_SAMPLE, 258, IFD_FORMAT_USHORT),
93614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COMPRESSION, 259, IFD_FORMAT_USHORT),
93714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PHOTOMETRIC_INTERPRETATION, 262, IFD_FORMAT_USHORT),
93814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_IMAGE_DESCRIPTION, 270, IFD_FORMAT_STRING),
93914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_MAKE, 271, IFD_FORMAT_STRING),
94014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_MODEL, 272, IFD_FORMAT_STRING),
94114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_STRIP_OFFSETS, 273, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
94214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORIENTATION, 274, IFD_FORMAT_USHORT),
94314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SAMPLES_PER_PIXEL, 277, IFD_FORMAT_USHORT),
94414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ROWS_PER_STRIP, 278, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
94514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_STRIP_BYTE_COUNTS, 279, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
94614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_X_RESOLUTION, 282, IFD_FORMAT_URATIONAL),
94714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_RESOLUTION, 283, IFD_FORMAT_URATIONAL),
94814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PLANAR_CONFIGURATION, 284, IFD_FORMAT_USHORT),
94914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RESOLUTION_UNIT, 296, IFD_FORMAT_USHORT),
95014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_TRANSFER_FUNCTION, 301, IFD_FORMAT_USHORT),
95114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SOFTWARE, 305, IFD_FORMAT_STRING),
95214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DATETIME, 306, IFD_FORMAT_STRING),
95314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ARTIST, 315, IFD_FORMAT_STRING),
95414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_WHITE_POINT, 318, IFD_FORMAT_URATIONAL),
95514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PRIMARY_CHROMATICITIES, 319, IFD_FORMAT_URATIONAL),
95614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See Adobe PageMaker® 6.0 TIFF Technical Notes, Note 1.
95714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUB_IFD_POINTER, 330, IFD_FORMAT_ULONG),
95814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_JPEG_INTERCHANGE_FORMAT, 513, IFD_FORMAT_ULONG),
95914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_JPEG_INTERCHANGE_FORMAT_LENGTH, 514, IFD_FORMAT_ULONG),
96014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_CB_CR_COEFFICIENTS, 529, IFD_FORMAT_URATIONAL),
96114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_CB_CR_SUB_SAMPLING, 530, IFD_FORMAT_USHORT),
96214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_CB_CR_POSITIONING, 531, IFD_FORMAT_USHORT),
96314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_REFERENCE_BLACK_WHITE, 532, IFD_FORMAT_URATIONAL),
96414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COPYRIGHT, 33432, IFD_FORMAT_STRING),
96514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXIF_IFD_POINTER, 34665, IFD_FORMAT_ULONG),
96614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_INFO_IFD_POINTER, 34853, IFD_FORMAT_ULONG),
96714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // RW2 file tags
96814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/PanasonicRaw.html)
96914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RW2_SENSOR_TOP_BORDER, 4, IFD_FORMAT_ULONG),
97014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RW2_SENSOR_LEFT_BORDER, 5, IFD_FORMAT_ULONG),
97114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RW2_SENSOR_BOTTOM_BORDER, 6, IFD_FORMAT_ULONG),
97214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RW2_SENSOR_RIGHT_BORDER, 7, IFD_FORMAT_ULONG),
97314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RW2_ISO, 23, IFD_FORMAT_USHORT),
97414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RW2_JPG_FROM_RAW, 46, IFD_FORMAT_UNDEFINED)
97514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
97614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
97714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Primary image IFD Exif Private tags (See JEITA CP-3451C Section 4.6.8 Tag Support Levels)
97814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] IFD_EXIF_TAGS = new ExifTag[] {
97914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXPOSURE_TIME, 33434, IFD_FORMAT_URATIONAL),
98014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_F_NUMBER, 33437, IFD_FORMAT_URATIONAL),
98114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXPOSURE_PROGRAM, 34850, IFD_FORMAT_USHORT),
98214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SPECTRAL_SENSITIVITY, 34852, IFD_FORMAT_STRING),
98314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ISO_SPEED_RATINGS, 34855, IFD_FORMAT_USHORT),
98414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_OECF, 34856, IFD_FORMAT_UNDEFINED),
98514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXIF_VERSION, 36864, IFD_FORMAT_STRING),
98614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DATETIME_ORIGINAL, 36867, IFD_FORMAT_STRING),
98714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DATETIME_DIGITIZED, 36868, IFD_FORMAT_STRING),
98814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COMPONENTS_CONFIGURATION, 37121, IFD_FORMAT_UNDEFINED),
98914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COMPRESSED_BITS_PER_PIXEL, 37122, IFD_FORMAT_URATIONAL),
99014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SHUTTER_SPEED_VALUE, 37377, IFD_FORMAT_SRATIONAL),
99114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_APERTURE_VALUE, 37378, IFD_FORMAT_URATIONAL),
99214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_BRIGHTNESS_VALUE, 37379, IFD_FORMAT_SRATIONAL),
99314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXPOSURE_BIAS_VALUE, 37380, IFD_FORMAT_SRATIONAL),
99414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_MAX_APERTURE_VALUE, 37381, IFD_FORMAT_URATIONAL),
99514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBJECT_DISTANCE, 37382, IFD_FORMAT_URATIONAL),
99614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_METERING_MODE, 37383, IFD_FORMAT_USHORT),
99714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_LIGHT_SOURCE, 37384, IFD_FORMAT_USHORT),
99814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FLASH, 37385, IFD_FORMAT_USHORT),
99914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FOCAL_LENGTH, 37386, IFD_FORMAT_URATIONAL),
100014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBJECT_AREA, 37396, IFD_FORMAT_USHORT),
100114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_MAKER_NOTE, 37500, IFD_FORMAT_UNDEFINED),
100214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_USER_COMMENT, 37510, IFD_FORMAT_UNDEFINED),
100314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBSEC_TIME, 37520, IFD_FORMAT_STRING),
100414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBSEC_TIME_ORIGINAL, 37521, IFD_FORMAT_STRING),
100514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBSEC_TIME_DIGITIZED, 37522, IFD_FORMAT_STRING),
100614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FLASHPIX_VERSION, 40960, IFD_FORMAT_UNDEFINED),
100714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COLOR_SPACE, 40961, IFD_FORMAT_USHORT),
100814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PIXEL_X_DIMENSION, 40962, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
100914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PIXEL_Y_DIMENSION, 40963, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
101014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RELATED_SOUND_FILE, 40964, IFD_FORMAT_STRING),
101114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_INTEROPERABILITY_IFD_POINTER, 40965, IFD_FORMAT_ULONG),
101214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FLASH_ENERGY, 41483, IFD_FORMAT_URATIONAL),
101314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SPATIAL_FREQUENCY_RESPONSE, 41484, IFD_FORMAT_UNDEFINED),
101414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FOCAL_PLANE_X_RESOLUTION, 41486, IFD_FORMAT_URATIONAL),
101514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FOCAL_PLANE_Y_RESOLUTION, 41487, IFD_FORMAT_URATIONAL),
101614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FOCAL_PLANE_RESOLUTION_UNIT, 41488, IFD_FORMAT_USHORT),
101714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBJECT_LOCATION, 41492, IFD_FORMAT_USHORT),
101814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXPOSURE_INDEX, 41493, IFD_FORMAT_URATIONAL),
101914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SENSING_METHOD, 41495, IFD_FORMAT_USHORT),
102014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FILE_SOURCE, 41728, IFD_FORMAT_UNDEFINED),
102114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SCENE_TYPE, 41729, IFD_FORMAT_UNDEFINED),
102214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_CFA_PATTERN, 41730, IFD_FORMAT_UNDEFINED),
102314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_CUSTOM_RENDERED, 41985, IFD_FORMAT_USHORT),
102414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXPOSURE_MODE, 41986, IFD_FORMAT_USHORT),
102514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_WHITE_BALANCE, 41987, IFD_FORMAT_USHORT),
102614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DIGITAL_ZOOM_RATIO, 41988, IFD_FORMAT_URATIONAL),
102714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_FOCAL_LENGTH_IN_35MM_FILM, 41989, IFD_FORMAT_USHORT),
102814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SCENE_CAPTURE_TYPE, 41990, IFD_FORMAT_USHORT),
102914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GAIN_CONTROL, 41991, IFD_FORMAT_USHORT),
103014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_CONTRAST, 41992, IFD_FORMAT_USHORT),
103114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SATURATION, 41993, IFD_FORMAT_USHORT),
103214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SHARPNESS, 41994, IFD_FORMAT_USHORT),
103314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DEVICE_SETTING_DESCRIPTION, 41995, IFD_FORMAT_UNDEFINED),
103414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBJECT_DISTANCE_RANGE, 41996, IFD_FORMAT_USHORT),
103514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_IMAGE_UNIQUE_ID, 42016, IFD_FORMAT_STRING),
103614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DNG_VERSION, 50706, IFD_FORMAT_BYTE),
103714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DEFAULT_CROP_SIZE, 50720, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG)
103814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
103914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
104014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Primary image IFD GPS Info tags (See JEITA CP-3451C Section 4.6.8 Tag Support Levels)
104114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] IFD_GPS_TAGS = new ExifTag[] {
104214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_VERSION_ID, 0, IFD_FORMAT_BYTE),
104314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_LATITUDE_REF, 1, IFD_FORMAT_STRING),
104414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_LATITUDE, 2, IFD_FORMAT_URATIONAL),
104514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_LONGITUDE_REF, 3, IFD_FORMAT_STRING),
104614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_LONGITUDE, 4, IFD_FORMAT_URATIONAL),
104714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_ALTITUDE_REF, 5, IFD_FORMAT_BYTE),
104814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_ALTITUDE, 6, IFD_FORMAT_URATIONAL),
104914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_TIMESTAMP, 7, IFD_FORMAT_URATIONAL),
105014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_SATELLITES, 8, IFD_FORMAT_STRING),
105114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_STATUS, 9, IFD_FORMAT_STRING),
105214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_MEASURE_MODE, 10, IFD_FORMAT_STRING),
105314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DOP, 11, IFD_FORMAT_URATIONAL),
105414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_SPEED_REF, 12, IFD_FORMAT_STRING),
105514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_SPEED, 13, IFD_FORMAT_URATIONAL),
105614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_TRACK_REF, 14, IFD_FORMAT_STRING),
105714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_TRACK, 15, IFD_FORMAT_URATIONAL),
105814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_IMG_DIRECTION_REF, 16, IFD_FORMAT_STRING),
105914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_IMG_DIRECTION, 17, IFD_FORMAT_URATIONAL),
106014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_MAP_DATUM, 18, IFD_FORMAT_STRING),
106114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_LATITUDE_REF, 19, IFD_FORMAT_STRING),
106214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_LATITUDE, 20, IFD_FORMAT_URATIONAL),
106314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_LONGITUDE_REF, 21, IFD_FORMAT_STRING),
106414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_LONGITUDE, 22, IFD_FORMAT_URATIONAL),
106514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_BEARING_REF, 23, IFD_FORMAT_STRING),
106614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_BEARING, 24, IFD_FORMAT_URATIONAL),
106714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_DISTANCE_REF, 25, IFD_FORMAT_STRING),
106814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DEST_DISTANCE, 26, IFD_FORMAT_URATIONAL),
106914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_PROCESSING_METHOD, 27, IFD_FORMAT_UNDEFINED),
107014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_AREA_INFORMATION, 28, IFD_FORMAT_UNDEFINED),
107114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DATESTAMP, 29, IFD_FORMAT_STRING),
107214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_DIFFERENTIAL, 30, IFD_FORMAT_USHORT)
107314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
107414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Primary image IFD Interoperability tag (See JEITA CP-3451C Section 4.6.8 Tag Support Levels)
107514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] IFD_INTEROPERABILITY_TAGS = new ExifTag[] {
107614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_INTEROPERABILITY_INDEX, 1, IFD_FORMAT_STRING)
107714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
107814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // IFD Thumbnail tags (See JEITA CP-3451C Section 4.6.8 Tag Support Levels)
107914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] IFD_THUMBNAIL_TAGS = new ExifTag[] {
108014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // For below two, see TIFF 6.0 Spec Section 3: Bilevel Images.
108114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_NEW_SUBFILE_TYPE, 254, IFD_FORMAT_ULONG),
108214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUBFILE_TYPE, 255, IFD_FORMAT_ULONG),
108314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_THUMBNAIL_IMAGE_WIDTH, 256, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
108414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_THUMBNAIL_IMAGE_LENGTH, 257, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
108514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_BITS_PER_SAMPLE, 258, IFD_FORMAT_USHORT),
108614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COMPRESSION, 259, IFD_FORMAT_USHORT),
108714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PHOTOMETRIC_INTERPRETATION, 262, IFD_FORMAT_USHORT),
108814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_IMAGE_DESCRIPTION, 270, IFD_FORMAT_STRING),
108914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_MAKE, 271, IFD_FORMAT_STRING),
109014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_MODEL, 272, IFD_FORMAT_STRING),
109114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_STRIP_OFFSETS, 273, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
109214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORIENTATION, 274, IFD_FORMAT_USHORT),
109314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SAMPLES_PER_PIXEL, 277, IFD_FORMAT_USHORT),
109414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ROWS_PER_STRIP, 278, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
109514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_STRIP_BYTE_COUNTS, 279, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG),
109614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_X_RESOLUTION, 282, IFD_FORMAT_URATIONAL),
109714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_RESOLUTION, 283, IFD_FORMAT_URATIONAL),
109814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PLANAR_CONFIGURATION, 284, IFD_FORMAT_USHORT),
109914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_RESOLUTION_UNIT, 296, IFD_FORMAT_USHORT),
110014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_TRANSFER_FUNCTION, 301, IFD_FORMAT_USHORT),
110114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SOFTWARE, 305, IFD_FORMAT_STRING),
110214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DATETIME, 306, IFD_FORMAT_STRING),
110314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ARTIST, 315, IFD_FORMAT_STRING),
110414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_WHITE_POINT, 318, IFD_FORMAT_URATIONAL),
110514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_PRIMARY_CHROMATICITIES, 319, IFD_FORMAT_URATIONAL),
110614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See Adobe PageMaker® 6.0 TIFF Technical Notes, Note 1.
110714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUB_IFD_POINTER, 330, IFD_FORMAT_ULONG),
110814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_JPEG_INTERCHANGE_FORMAT, 513, IFD_FORMAT_ULONG),
110914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_JPEG_INTERCHANGE_FORMAT_LENGTH, 514, IFD_FORMAT_ULONG),
111014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_CB_CR_COEFFICIENTS, 529, IFD_FORMAT_URATIONAL),
111114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_CB_CR_SUB_SAMPLING, 530, IFD_FORMAT_USHORT),
111214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_Y_CB_CR_POSITIONING, 531, IFD_FORMAT_USHORT),
111314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_REFERENCE_BLACK_WHITE, 532, IFD_FORMAT_URATIONAL),
111414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COPYRIGHT, 33432, IFD_FORMAT_STRING),
111514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXIF_IFD_POINTER, 34665, IFD_FORMAT_ULONG),
111614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_INFO_IFD_POINTER, 34853, IFD_FORMAT_ULONG),
111714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DNG_VERSION, 50706, IFD_FORMAT_BYTE),
111814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_DEFAULT_CROP_SIZE, 50720, IFD_FORMAT_USHORT, IFD_FORMAT_ULONG)
111914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
112014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
112114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // RAF file tag (See piex.cc line 372)
112214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag TAG_RAF_IMAGE_SIZE =
112314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_STRIP_OFFSETS, 273, IFD_FORMAT_USHORT);
112414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
112514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // ORF file tags (See http://www.exiv2.org/tags-olympus.html)
112614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] ORF_MAKER_NOTE_TAGS = new ExifTag[] {
112714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_THUMBNAIL_IMAGE, 256, IFD_FORMAT_UNDEFINED),
112814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_CAMERA_SETTINGS_IFD_POINTER, 8224, IFD_FORMAT_ULONG),
112914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_IMAGE_PROCESSING_IFD_POINTER, 8256, IFD_FORMAT_ULONG)
113014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
113114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] ORF_CAMERA_SETTINGS_TAGS = new ExifTag[] {
113214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_PREVIEW_IMAGE_START, 257, IFD_FORMAT_ULONG),
113314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_PREVIEW_IMAGE_LENGTH, 258, IFD_FORMAT_ULONG)
113414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
113514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] ORF_IMAGE_PROCESSING_TAGS = new ExifTag[] {
113614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_ASPECT_FRAME, 4371, IFD_FORMAT_USHORT)
113714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
113814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // PEF file tag (See http://www.sno.phy.queensu.ca/~phil/exiftool/TagNames/Pentax.html)
113914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] PEF_TAGS = new ExifTag[] {
114014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_COLOR_SPACE, 55, IFD_FORMAT_USHORT)
114114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
114214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
114314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See JEITA CP-3451C Section 4.6.3: Exif-specific IFD.
114414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // The following values are used for indicating pointers to the other Image File Directories.
114514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
114614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Indices of Exif Ifd tag groups
114714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /** @hide */
114814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    @Retention(RetentionPolicy.SOURCE)
114914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    @IntDef({IFD_TYPE_PRIMARY, IFD_TYPE_EXIF, IFD_TYPE_GPS, IFD_TYPE_INTEROPERABILITY,
115014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            IFD_TYPE_THUMBNAIL, IFD_TYPE_PREVIEW, IFD_TYPE_ORF_MAKER_NOTE,
115114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            IFD_TYPE_ORF_CAMERA_SETTINGS, IFD_TYPE_ORF_IMAGE_PROCESSING, IFD_TYPE_PEF})
115214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public @interface IfdType {}
115314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
115414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_PRIMARY = 0;
115514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_EXIF = 1;
115614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_GPS = 2;
115714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_INTEROPERABILITY = 3;
115814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_THUMBNAIL = 4;
115914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_PREVIEW = 5;
116014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_ORF_MAKER_NOTE = 6;
116114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_ORF_CAMERA_SETTINGS = 7;
116214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_ORF_IMAGE_PROCESSING = 8;
116314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IFD_TYPE_PEF = 9;
116414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
116514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // List of Exif tag groups
116614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[][] EXIF_TAGS = new ExifTag[][] {
116714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            IFD_TIFF_TAGS, IFD_EXIF_TAGS, IFD_GPS_TAGS, IFD_INTEROPERABILITY_TAGS,
116814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            IFD_THUMBNAIL_TAGS, IFD_TIFF_TAGS, ORF_MAKER_NOTE_TAGS, ORF_CAMERA_SETTINGS_TAGS,
116914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ORF_IMAGE_PROCESSING_TAGS, PEF_TAGS
117014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
117114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // List of tags for pointing to the other image file directory offset.
117214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag[] EXIF_POINTER_TAGS = new ExifTag[] {
117314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_SUB_IFD_POINTER, 330, IFD_FORMAT_ULONG),
117414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_EXIF_IFD_POINTER, 34665, IFD_FORMAT_ULONG),
117514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_GPS_INFO_IFD_POINTER, 34853, IFD_FORMAT_ULONG),
117614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_INTEROPERABILITY_IFD_POINTER, 40965, IFD_FORMAT_ULONG),
117714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_CAMERA_SETTINGS_IFD_POINTER, 8224, IFD_FORMAT_BYTE),
117814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_ORF_IMAGE_PROCESSING_IFD_POINTER, 8256, IFD_FORMAT_BYTE)
117914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    };
118014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
118114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Tags for indicating the thumbnail offset and length
118214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag JPEG_INTERCHANGE_FORMAT_TAG =
118314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_JPEG_INTERCHANGE_FORMAT, 513, IFD_FORMAT_ULONG);
118414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final ExifTag JPEG_INTERCHANGE_FORMAT_LENGTH_TAG =
118514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            new ExifTag(TAG_JPEG_INTERCHANGE_FORMAT_LENGTH, 514, IFD_FORMAT_ULONG);
118614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
118714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Mappings from tag number to tag name and each item represents one IFD tag group.
118814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final HashMap[] sExifTagMapsForReading = new HashMap[EXIF_TAGS.length];
118914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Mappings from tag name to tag number and each item represents one IFD tag group.
119014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final HashMap[] sExifTagMapsForWriting = new HashMap[EXIF_TAGS.length];
119114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final HashSet<String> sTagSetForCompatibility = new HashSet<>(Arrays.asList(
119214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            TAG_F_NUMBER, TAG_DIGITAL_ZOOM_RATIO, TAG_EXPOSURE_TIME, TAG_SUBJECT_DISTANCE,
119314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            TAG_GPS_TIMESTAMP));
119414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Mappings from tag number to IFD type for pointer tags.
119514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final HashMap sExifPointerTagMap = new HashMap();
119614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
119714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See JPEG File Interchange Format Version 1.02.
119814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // The following values are defined for handling JPEG streams. In this implementation, we are
119914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // not only getting information from EXIF but also from some JPEG special segments such as
120014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // MARKER_COM for user comment and MARKER_SOFx for image width and height.
120114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
120214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final Charset ASCII = Charset.forName("US-ASCII");
120314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Identifier for EXIF APP1 segment in JPEG
120414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte[] IDENTIFIER_EXIF_APP1 = "Exif\0\0".getBytes(ASCII);
120514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // JPEG segment markers, that each marker consumes two bytes beginning with 0xff and ending with
120614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // the indicator. There is no SOF4, SOF8, SOF16 markers in JPEG and SOFx markers indicates start
120714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // of frame(baseline DCT) and the image size info exists in its beginning part.
120814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER = (byte) 0xff;
120914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOI = (byte) 0xd8;
121014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF0 = (byte) 0xc0;
121114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF1 = (byte) 0xc1;
121214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF2 = (byte) 0xc2;
121314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF3 = (byte) 0xc3;
121414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF5 = (byte) 0xc5;
121514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF6 = (byte) 0xc6;
121614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF7 = (byte) 0xc7;
121714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF9 = (byte) 0xc9;
121814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF10 = (byte) 0xca;
121914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF11 = (byte) 0xcb;
122014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF13 = (byte) 0xcd;
122114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF14 = (byte) 0xce;
122214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOF15 = (byte) 0xcf;
122314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_SOS = (byte) 0xda;
122414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_APP1 = (byte) 0xe1;
122514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_COM = (byte) 0xfe;
122614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final byte MARKER_EOI = (byte) 0xd9;
122714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
122814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Supported Image File Types
122914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_UNKNOWN = 0;
123014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_ARW = 1;
123114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_CR2 = 2;
123214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_DNG = 3;
123314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_JPEG = 4;
123414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_NEF = 5;
123514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_NRW = 6;
123614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_ORF = 7;
123714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_PEF = 8;
123814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_RAF = 9;
123914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_RW2 = 10;
124014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final int IMAGE_TYPE_SRW = 11;
124114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
124214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    static {
124314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sFormatter = new SimpleDateFormat("yyyy:MM:dd HH:mm:ss");
124414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sFormatter.setTimeZone(TimeZone.getTimeZone("UTC"));
124514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
124614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Build up the hash tables to look up Exif tags for reading Exif tags.
124714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int ifdType = 0; ifdType < EXIF_TAGS.length; ++ifdType) {
124814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            sExifTagMapsForReading[ifdType] = new HashMap();
124914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            sExifTagMapsForWriting[ifdType] = new HashMap();
125014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (ExifTag tag : EXIF_TAGS[ifdType]) {
125114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                sExifTagMapsForReading[ifdType].put(tag.number, tag);
125214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                sExifTagMapsForWriting[ifdType].put(tag.name, tag);
125314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
125414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
125514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
125614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Build up the hash table to look up Exif pointer tags.
125714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sExifPointerTagMap.put(EXIF_POINTER_TAGS[0].number, IFD_TYPE_PREVIEW); // 330
125814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sExifPointerTagMap.put(EXIF_POINTER_TAGS[1].number, IFD_TYPE_EXIF); // 34665
125914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sExifPointerTagMap.put(EXIF_POINTER_TAGS[2].number, IFD_TYPE_GPS); // 34853
126014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sExifPointerTagMap.put(EXIF_POINTER_TAGS[3].number, IFD_TYPE_INTEROPERABILITY); // 40965
126114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sExifPointerTagMap.put(EXIF_POINTER_TAGS[4].number, IFD_TYPE_ORF_CAMERA_SETTINGS); // 8224
126214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        sExifPointerTagMap.put(EXIF_POINTER_TAGS[5].number, IFD_TYPE_ORF_IMAGE_PROCESSING); // 8256
126314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
126414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
126514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private final String mFilename;
126614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private final AssetManager.AssetInputStream mAssetInputStream;
126714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mMimeType;
126814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private final HashMap[] mAttributes = new HashMap[EXIF_TAGS.length];
126914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private ByteOrder mExifByteOrder = ByteOrder.BIG_ENDIAN;
127014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean mHasThumbnail;
127114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // The following values used for indicating a thumbnail position.
127214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mThumbnailOffset;
127314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mThumbnailLength;
127414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private byte[] mThumbnailBytes;
127514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mThumbnailCompression;
127614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mExifOffset;
127714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mOrfMakerNoteOffset;
127814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mOrfThumbnailOffset;
127914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mOrfThumbnailLength;
128014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int mRw2JpgFromRawOffset;
128114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean mIsSupportedFile;
128214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
128314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Pattern to check non zero timestamp
128414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final Pattern sNonZeroTimePattern = Pattern.compile(".*[1-9].*");
128514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Pattern to check gps timestamp
128614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static final Pattern sGpsTimestampPattern =
128714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Pattern.compile("^([0-9][0-9]):([0-9][0-9]):([0-9][0-9])$");
128814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
128914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
129014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Reads Exif tags from the specified image file.
129114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
129214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public ExifInterface(String filename) throws IOException {
129314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (filename == null) {
129414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IllegalArgumentException("filename cannot be null");
129514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
129614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        FileInputStream in = null;
129714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mAssetInputStream = null;
129814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mFilename = filename;
129914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
130014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            in = new FileInputStream(filename);
130114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            loadAttributes(in);
130214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } finally {
130314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            closeQuietly(in);
130414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
130514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
130614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
130714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
130814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Reads Exif tags from the specified image input stream. Attribute mutation is not supported
130914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * for input streams. The given input stream will proceed its current position. Developers
131046ffc16f1c2e80e8fc401a5a440c3c3a847b4ba7Hyundo Moon     * should close the input stream after use. This constructor is not intended to be used with
131146ffc16f1c2e80e8fc401a5a440c3c3a847b4ba7Hyundo Moon     * an input stream that performs any networking operations.
131214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
131314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public ExifInterface(InputStream inputStream) throws IOException {
131414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (inputStream == null) {
131514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IllegalArgumentException("inputStream cannot be null");
131614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
131714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mFilename = null;
131814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (inputStream instanceof AssetManager.AssetInputStream) {
131914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAssetInputStream = (AssetManager.AssetInputStream) inputStream;
132014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else {
132114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAssetInputStream = null;
132214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
132314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        loadAttributes(inputStream);
132414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
132514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
132614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
132714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the EXIF attribute of the specified tag or {@code null} if there is no such tag in
132814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * the image file.
132914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
133014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param tag the name of the tag.
133114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
133214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private ExifAttribute getExifAttribute(String tag) {
133314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Retrieves all tag groups. The value from primary image tag group has a higher priority
133414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // than the value from the thumbnail tag group if there are more than one candidates.
133514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0; i < EXIF_TAGS.length; ++i) {
133614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Object value = mAttributes[i].get(tag);
133714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (value != null) {
133814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return (ExifAttribute) value;
133914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
134014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
134114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return null;
134214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
134314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
134414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
134514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the value of the specified tag or {@code null} if there
134614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * is no such tag in the image file.
134714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
134814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param tag the name of the tag.
134914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
135014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public String getAttribute(String tag) {
135114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute attribute = getExifAttribute(tag);
135214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (attribute != null) {
135314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (!sTagSetForCompatibility.contains(tag)) {
135414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return attribute.getStringValue(mExifByteOrder);
135514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
135614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (tag.equals(TAG_GPS_TIMESTAMP)) {
135714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Convert the rational values to the custom formats for backwards compatibility.
135814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (attribute.format != IFD_FORMAT_URATIONAL
135914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        && attribute.format != IFD_FORMAT_SRATIONAL) {
136014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "GPS Timestamp format is not rational. format=" + attribute.format);
136114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return null;
136214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
136314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Rational[] array = (Rational[]) attribute.getValue(mExifByteOrder);
136414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (array == null || array.length != 3) {
136514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Invalid GPS Timestamp array. array=" + Arrays.toString(array));
136614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return null;
136714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
136814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return String.format("%02d:%02d:%02d",
136914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        (int) ((float) array[0].numerator / array[0].denominator),
137014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        (int) ((float) array[1].numerator / array[1].denominator),
137114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        (int) ((float) array[2].numerator / array[2].denominator));
137214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
137314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            try {
137414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return Double.toString(attribute.getDoubleValue(mExifByteOrder));
137514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } catch (NumberFormatException e) {
137614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return null;
137714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
137814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
137914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return null;
138014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
138114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
138214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
138314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the integer value of the specified tag. If there is no such tag
138414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * in the image file or the value cannot be parsed as integer, return
138514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * <var>defaultValue</var>.
138614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
138714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param tag the name of the tag.
138814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param defaultValue the value to return if the tag is not available.
138914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
139014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public int getAttributeInt(String tag, int defaultValue) {
139114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute exifAttribute = getExifAttribute(tag);
139214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (exifAttribute == null) {
139314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return defaultValue;
139414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
139514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
139614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
139714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return exifAttribute.getIntValue(mExifByteOrder);
139814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (NumberFormatException e) {
139914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return defaultValue;
140014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
140114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
140214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
140314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
140414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the double value of the tag that is specified as rational or contains a
140514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * double-formatted value. If there is no such tag in the image file or the value cannot be
140614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * parsed as double, return <var>defaultValue</var>.
140714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
140814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param tag the name of the tag.
140914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param defaultValue the value to return if the tag is not available.
141014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
141114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public double getAttributeDouble(String tag, double defaultValue) {
141214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute exifAttribute = getExifAttribute(tag);
141314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (exifAttribute == null) {
141414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return defaultValue;
141514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
141614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
141714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
141814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return exifAttribute.getDoubleValue(mExifByteOrder);
141914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (NumberFormatException e) {
142014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return defaultValue;
142114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
142214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
142314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
142414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
142514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Set the value of the specified tag.
142614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
142714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param tag the name of the tag.
142814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param value the value of the tag.
142914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
143014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public void setAttribute(String tag, String value) {
143114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Convert the given value to rational values for backwards compatibility.
143214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (value != null && sTagSetForCompatibility.contains(tag)) {
143314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (tag.equals(TAG_GPS_TIMESTAMP)) {
143414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Matcher m = sGpsTimestampPattern.matcher(value);
143514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (!m.find()) {
143614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Invalid value for " + tag + " : " + value);
143714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return;
143814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
143914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                value = Integer.parseInt(m.group(1)) + "/1," + Integer.parseInt(m.group(2)) + "/1,"
144014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + Integer.parseInt(m.group(3)) + "/1";
144114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else {
144214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                try {
144314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    double doubleValue = Double.parseDouble(value);
144414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    value = (long) (doubleValue * 10000L) + "/10000";
144514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } catch (NumberFormatException e) {
144614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Invalid value for " + tag + " : " + value);
144714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return;
144814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
144914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
145014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
145114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
145214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0 ; i < EXIF_TAGS.length; ++i) {
145314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (i == IFD_TYPE_THUMBNAIL && !mHasThumbnail) {
145414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                continue;
145514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
145614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            final Object obj = sExifTagMapsForWriting[i].get(tag);
145714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (obj != null) {
145814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (value == null) {
145914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[i].remove(tag);
146014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    continue;
146114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
146214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                final ExifTag exifTag = (ExifTag) obj;
146314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Pair<Integer, Integer> guess = guessDataFormat(value);
146414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int dataFormat;
146514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (exifTag.primaryFormat == guess.first || exifTag.primaryFormat == guess.second) {
146614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataFormat = exifTag.primaryFormat;
146714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else if (exifTag.secondaryFormat != -1 && (exifTag.secondaryFormat == guess.first
146814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        || exifTag.secondaryFormat == guess.second)) {
146914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataFormat = exifTag.secondaryFormat;
147014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else if (exifTag.primaryFormat == IFD_FORMAT_BYTE
147114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        || exifTag.primaryFormat == IFD_FORMAT_UNDEFINED
147214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        || exifTag.primaryFormat == IFD_FORMAT_STRING) {
147314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataFormat = exifTag.primaryFormat;
147414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else {
147514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Given tag (" + tag + ") value didn't match with one of expected "
147614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + "formats: " + IFD_FORMAT_NAMES[exifTag.primaryFormat]
147714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + (exifTag.secondaryFormat == -1 ? "" : ", "
147814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + IFD_FORMAT_NAMES[exifTag.secondaryFormat]) + " (guess: "
147914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + IFD_FORMAT_NAMES[guess.first] + (guess.second == -1 ? "" : ", "
148014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + IFD_FORMAT_NAMES[guess.second]) + ")");
148114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    continue;
148214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
148314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                switch (dataFormat) {
148414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_BYTE: {
148514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag, ExifAttribute.createByte(value));
148614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
148714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
148814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_UNDEFINED:
148914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_STRING: {
149014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag, ExifAttribute.createString(value));
149114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
149214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
149314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_USHORT: {
149414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final String[] values = value.split(",");
149514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final int[] intArray = new int[values.length];
149614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int j = 0; j < values.length; ++j) {
149714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            intArray[j] = Integer.parseInt(values[j]);
149814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
149914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag,
150014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createUShort(intArray, mExifByteOrder));
150114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
150214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
150314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SLONG: {
150414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final String[] values = value.split(",");
150514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final int[] intArray = new int[values.length];
150614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int j = 0; j < values.length; ++j) {
150714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            intArray[j] = Integer.parseInt(values[j]);
150814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
150914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag,
151014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createSLong(intArray, mExifByteOrder));
151114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
151214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
151314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_ULONG: {
151414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final String[] values = value.split(",");
151514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final long[] longArray = new long[values.length];
151614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int j = 0; j < values.length; ++j) {
151714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            longArray[j] = Long.parseLong(values[j]);
151814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
151914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag,
152014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createULong(longArray, mExifByteOrder));
152114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
152214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
152314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_URATIONAL: {
152414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final String[] values = value.split(",");
152514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final Rational[] rationalArray = new Rational[values.length];
152614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int j = 0; j < values.length; ++j) {
152714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            final String[] numbers = values[j].split("/");
152814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            rationalArray[j] = new Rational(Long.parseLong(numbers[0]),
152914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                    Long.parseLong(numbers[1]));
153014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
153114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag,
153214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createURational(rationalArray, mExifByteOrder));
153314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
153414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
153514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SRATIONAL: {
153614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final String[] values = value.split(",");
153714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final Rational[] rationalArray = new Rational[values.length];
153814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int j = 0; j < values.length; ++j) {
153914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            final String[] numbers = values[j].split("/");
154014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            rationalArray[j] = new Rational(Long.parseLong(numbers[0]),
154114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                    Long.parseLong(numbers[1]));
154214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
154314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag,
154414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createSRational(rationalArray, mExifByteOrder));
154514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
154614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
154714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_DOUBLE: {
154814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final String[] values = value.split(",");
154914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        final double[] doubleArray = new double[values.length];
155014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        for (int j = 0; j < values.length; ++j) {
155114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            doubleArray[j] = Double.parseDouble(values[j]);
155214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
155314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[i].put(tag,
155414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createDouble(doubleArray, mExifByteOrder));
155514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
155614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
155714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    default:
155814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        Log.w(TAG, "Data format isn't one of expected formats: " + dataFormat);
155914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        continue;
156014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
156114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
156214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
156314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
156414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
156514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
156614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Update the values of the tags in the tag groups if any value for the tag already was stored.
156714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
156814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param tag the name of the tag.
156914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param value the value of the tag in a form of {@link ExifAttribute}.
157014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @return Returns {@code true} if updating is placed.
157114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
157214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean updateAttribute(String tag, ExifAttribute value) {
157314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        boolean updated = false;
157414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0 ; i < EXIF_TAGS.length; ++i) {
157514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mAttributes[i].containsKey(tag)) {
157614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[i].put(tag, value);
157714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                updated = true;
157814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
157914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
158014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return updated;
158114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
158214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
158314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
158414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Remove any values of the specified tag.
158514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
158614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param tag the name of the tag.
158714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
158814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void removeAttribute(String tag) {
158914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0 ; i < EXIF_TAGS.length; ++i) {
159014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[i].remove(tag);
159114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
159214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
159314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
159414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
159514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This function decides which parser to read the image data according to the given input stream
159614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * type and the content of the input stream. In each case, it reads the first three bytes to
159714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * determine whether the image data format is JPEG or not.
159814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
159914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void loadAttributes(@NonNull InputStream in) throws IOException {
160014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
160114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Initialize mAttributes.
160214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (int i = 0; i < EXIF_TAGS.length; ++i) {
160314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[i] = new HashMap();
160414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
160514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
160614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Check file type
160714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            in = new BufferedInputStream(in, SIGNATURE_CHECK_SIZE);
160814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mMimeType = getMimeType((BufferedInputStream) in);
160914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
161014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Create byte-ordered input stream
161114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ByteOrderedDataInputStream inputStream = new ByteOrderedDataInputStream(in);
161214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
161314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            switch (mMimeType) {
161414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_JPEG: {
161514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    getJpegAttributes(inputStream, 0, IFD_TYPE_PRIMARY); // 0 is offset
161614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
161714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
161814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_RAF: {
161914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    getRafAttributes(inputStream);
162014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
162114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
162214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_ORF: {
162314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    getOrfAttributes(inputStream);
162414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
162514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
162614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_RW2: {
162714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    getRw2Attributes(inputStream);
162814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
162914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
163014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_ARW:
163114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_CR2:
163214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_DNG:
163314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_NEF:
163414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_NRW:
163514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_PEF:
163614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_SRW:
163714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case IMAGE_TYPE_UNKNOWN: {
163814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    getRawAttributes(inputStream);
163914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
164014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
164114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                default: {
164214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
164314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
164414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
164514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Set thumbnail image offset and length
164614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            setThumbnailData(inputStream);
164714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mIsSupportedFile = true;
164814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (IOException e) {
164914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Ignore exceptions in order to keep the compatibility with the old versions of
165014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // ExifInterface.
165114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mIsSupportedFile = false;
165214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
165314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.w(TAG, "Invalid image: ExifInterface got an unsupported image format file"
165414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + "(ExifInterface supports JPEG and some RAW image formats only) "
165514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + "or a corrupted JPEG file to ExifInterface.", e);
165614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
165714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } finally {
165814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            addDefaultValuesForCompatibility();
165914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
166014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
166114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                printAttributes();
166214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
166314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
166414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
166514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
166614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Prints out attributes for debugging.
166714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void printAttributes() {
166814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0; i < mAttributes.length; ++i) {
166914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "The size of tag group[" + i + "]: " + mAttributes[i].size());
167014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (Map.Entry entry : (Set<Map.Entry>) mAttributes[i].entrySet()) {
167114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                final ExifAttribute tagValue = (ExifAttribute) entry.getValue();
167214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "tagName: " + entry.getKey() + ", tagType: " + tagValue.toString()
167314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + ", tagValue: '" + tagValue.getStringValue(mExifByteOrder) + "'");
167414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
167514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
167614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
167714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
167814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
167914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Save the tag data into the original image file. This is expensive because it involves
168014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * copying all the data from one file to another and deleting the old file and renaming the
168114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * other. It's best to use {@link #setAttribute(String,String)} to set all attributes to write
168214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * and make a single call rather than multiple calls for each attribute.
168314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * <p>
168414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This method is only supported for JPEG files.
168514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * </p>
168614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
168714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public void saveAttributes() throws IOException {
168814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mIsSupportedFile || mMimeType != IMAGE_TYPE_JPEG) {
168914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("ExifInterface only supports saving attributes on JPEG formats.");
169014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
169114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mFilename == null) {
169214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException(
169314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    "ExifInterface does not support saving attributes for the current input.");
169414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
169514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
169614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Keep the thumbnail in memory
169714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mThumbnailBytes = getThumbnail();
169814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
169914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        File tempFile = new File(mFilename + ".tmp");
170014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        File originalFile = new File(mFilename);
170114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!originalFile.renameTo(tempFile)) {
170214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Could not rename to " + tempFile.getAbsolutePath());
170314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
170414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
170514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        FileInputStream in = null;
170614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        FileOutputStream out = null;
170714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
170814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Save the new file.
170914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            in = new FileInputStream(tempFile);
171014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            out = new FileOutputStream(mFilename);
171114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            saveJpegAttributes(in, out);
171214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } finally {
171314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            closeQuietly(in);
171414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            closeQuietly(out);
171514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            tempFile.delete();
171614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
171714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
171814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Discard the thumbnail in memory
171914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mThumbnailBytes = null;
172014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
172114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
172214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
172314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns true if the image file has a thumbnail.
172414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
172514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public boolean hasThumbnail() {
172614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return mHasThumbnail;
172714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
172814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
172914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
173014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the JPEG compressed thumbnail inside the image file, or {@code null} if there is no
173114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * JPEG compressed thumbnail.
173214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * The returned data can be decoded using
173314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * {@link android.graphics.BitmapFactory#decodeByteArray(byte[],int,int)}
173414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
173514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public byte[] getThumbnail() {
173614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mThumbnailCompression == DATA_JPEG || mThumbnailCompression == DATA_JPEG_COMPRESSED) {
173714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return getThumbnailBytes();
173814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
173914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return null;
174014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
174114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
174214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
174314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the thumbnail bytes inside the image file, regardless of the compression type of the
174414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * thumbnail image.
174514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
174614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public byte[] getThumbnailBytes() {
174714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mHasThumbnail) {
174814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return null;
174914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
175014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mThumbnailBytes != null) {
175114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mThumbnailBytes;
175214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
175314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
175414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read the thumbnail.
175514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        InputStream in = null;
175614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
175714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mAssetInputStream != null) {
175814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                in = mAssetInputStream;
175914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (in.markSupported()) {
176014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    in.reset();
176114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else {
176214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.d(TAG, "Cannot read thumbnail from inputstream without mark/reset support");
176314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return null;
176414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
176514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mFilename != null) {
176614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                in = new FileInputStream(mFilename);
176714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
176814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (in == null) {
176914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Should not be reached this.
177014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new FileNotFoundException();
177114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
177214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (in.skip(mThumbnailOffset) != mThumbnailOffset) {
177314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Corrupted image");
177414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
177514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            byte[] buffer = new byte[mThumbnailLength];
177614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (in.read(buffer) != mThumbnailLength) {
177714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Corrupted image");
177814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
177914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mThumbnailBytes = buffer;
178014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return buffer;
178114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (IOException e) {
178214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Couldn't get a thumbnail image.
178314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "Encountered exception while getting thumbnail", e);
178414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } finally {
178514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            closeQuietly(in);
178614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
178714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return null;
178814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
178914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
179014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
179114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Creates and returns a Bitmap object of the thumbnail image based on the byte array and the
179214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * thumbnail compression value, or {@code null} if the compression type is unsupported.
179314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
179414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public Bitmap getThumbnailBitmap() {
179514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mHasThumbnail) {
179614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return null;
179714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else if (mThumbnailBytes == null) {
179814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mThumbnailBytes = getThumbnailBytes();
179914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
180014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
180114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mThumbnailCompression == DATA_JPEG || mThumbnailCompression == DATA_JPEG_COMPRESSED) {
180214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return BitmapFactory.decodeByteArray(mThumbnailBytes, 0, mThumbnailLength);
180314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else if (mThumbnailCompression == DATA_UNCOMPRESSED) {
180414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int[] rgbValues = new int[mThumbnailBytes.length / 3];
180514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            byte alpha = (byte) 0xff000000;
180614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (int i = 0; i < rgbValues.length; i++) {
180714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                rgbValues[i] = alpha + (mThumbnailBytes[3 * i] << 16)
180814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + (mThumbnailBytes[3 * i + 1] << 8) + mThumbnailBytes[3 * i + 2];
180914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
181014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
181114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute imageLengthAttribute =
181214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    (ExifAttribute) mAttributes[IFD_TYPE_THUMBNAIL].get(TAG_IMAGE_LENGTH);
181314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute imageWidthAttribute =
181414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    (ExifAttribute) mAttributes[IFD_TYPE_THUMBNAIL].get(TAG_IMAGE_WIDTH);
181514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (imageLengthAttribute != null && imageWidthAttribute != null) {
181614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int imageLength = imageLengthAttribute.getIntValue(mExifByteOrder);
181714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int imageWidth = imageWidthAttribute.getIntValue(mExifByteOrder);
181814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return Bitmap.createBitmap(
181914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        rgbValues, imageWidth, imageLength, Bitmap.Config.ARGB_8888);
182014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
182114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
182214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return null;
182314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
182414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
182514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
182614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns true if thumbnail image is JPEG Compressed, or false if either thumbnail image does
182714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * not exist or thumbnail image is uncompressed.
182814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
182914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public boolean isThumbnailCompressed() {
183014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return mThumbnailCompression == DATA_JPEG || mThumbnailCompression == DATA_JPEG_COMPRESSED;
183114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
183214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
183314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
183414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the offset and length of thumbnail inside the image file, or
183514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * {@code null} if there is no thumbnail.
183614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
183714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @return two-element array, the offset in the first value, and length in
183814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *         the second, or {@code null} if no thumbnail was found.
183914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
184014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public long[] getThumbnailRange() {
184114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mHasThumbnail) {
184214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return null;
184314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
184414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
184514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        long[] range = new long[2];
184614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        range[0] = mThumbnailOffset;
184714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        range[1] = mThumbnailLength;
184814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
184914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return range;
185014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
185114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
185214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
185314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Stores the latitude and longitude value in a float array. The first element is
185414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * the latitude, and the second element is the longitude. Returns false if the
185514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Exif tags are not available.
185614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
185714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public boolean getLatLong(float output[]) {
185814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String latValue = getAttribute(TAG_GPS_LATITUDE);
185914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String latRef = getAttribute(TAG_GPS_LATITUDE_REF);
186014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String lngValue = getAttribute(TAG_GPS_LONGITUDE);
186114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String lngRef = getAttribute(TAG_GPS_LONGITUDE_REF);
186214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
186314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (latValue != null && latRef != null && lngValue != null && lngRef != null) {
186414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            try {
186514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                output[0] = convertRationalLatLonToFloat(latValue, latRef);
186614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                output[1] = convertRationalLatLonToFloat(lngValue, lngRef);
186714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return true;
186814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } catch (IllegalArgumentException e) {
186914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.w(TAG, "Latitude/longitude values are not parseable. " +
187014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        String.format("latValue=%s, latRef=%s, lngValue=%s, lngRef=%s",
187114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                latValue, latRef, lngValue, lngRef));
187214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
187314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
187414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return false;
187514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
187614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
187714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
187814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Return the altitude in meters. If the exif tag does not exist, return
187914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * <var>defaultValue</var>.
188014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
188114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param defaultValue the value to return if the tag is not available.
188214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
188314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public double getAltitude(double defaultValue) {
188414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        double altitude = getAttributeDouble(TAG_GPS_ALTITUDE, -1);
188514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int ref = getAttributeInt(TAG_GPS_ALTITUDE_REF, -1);
188614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
188714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (altitude >= 0 && ref >= 0) {
188814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return (altitude * ((ref == 1) ? -1 : 1));
188914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else {
189014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return defaultValue;
189114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
189214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
189314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
189414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
189514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns number of milliseconds since Jan. 1, 1970, midnight local time.
189614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns -1 if the date time information if not available.
189714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @hide
189814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
189914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public long getDateTime() {
190014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String dateTimeString = getAttribute(TAG_DATETIME);
190114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (dateTimeString == null
190214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                || !sNonZeroTimePattern.matcher(dateTimeString).matches()) return -1;
190314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
190414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ParsePosition pos = new ParsePosition(0);
190514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
190614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // The exif field is in local time. Parsing it as if it is UTC will yield time
190714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // since 1/1/1970 local time
190814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Date datetime = sFormatter.parse(dateTimeString, pos);
190914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (datetime == null) return -1;
191014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            long msecs = datetime.getTime();
191114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
191214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            String subSecs = getAttribute(TAG_SUBSEC_TIME);
191314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (subSecs != null) {
191414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                try {
191514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    long sub = Long.parseLong(subSecs);
191614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    while (sub > 1000) {
191714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        sub /= 10;
191814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
191914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    msecs += sub;
192014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } catch (NumberFormatException e) {
192114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Ignored
192214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
192314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
192414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return msecs;
192514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (IllegalArgumentException e) {
192614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return -1;
192714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
192814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
192914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
193014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
193114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns number of milliseconds since Jan. 1, 1970, midnight UTC.
193214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns -1 if the date time information if not available.
193314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @hide
193414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
193514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    public long getGpsDateTime() {
193614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String date = getAttribute(TAG_GPS_DATESTAMP);
193714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String time = getAttribute(TAG_GPS_TIMESTAMP);
193814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (date == null || time == null
193914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                || (!sNonZeroTimePattern.matcher(date).matches()
194014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                && !sNonZeroTimePattern.matcher(time).matches())) {
194114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return -1;
194214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
194314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
194414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String dateTimeString = date + ' ' + time;
194514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
194614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ParsePosition pos = new ParsePosition(0);
194714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
194814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Date datetime = sFormatter.parse(dateTimeString, pos);
194914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (datetime == null) return -1;
195014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return datetime.getTime();
195114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (IllegalArgumentException e) {
195214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return -1;
195314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
195414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
195514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
195614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static float convertRationalLatLonToFloat(String rationalString, String ref) {
195714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
195814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            String [] parts = rationalString.split(",");
195914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
196014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            String [] pair;
196114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            pair = parts[0].split("/");
196214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            double degrees = Double.parseDouble(pair[0].trim())
196314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    / Double.parseDouble(pair[1].trim());
196414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
196514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            pair = parts[1].split("/");
196614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            double minutes = Double.parseDouble(pair[0].trim())
196714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    / Double.parseDouble(pair[1].trim());
196814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
196914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            pair = parts[2].split("/");
197014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            double seconds = Double.parseDouble(pair[0].trim())
197114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    / Double.parseDouble(pair[1].trim());
197214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
197314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            double result = degrees + (minutes / 60.0) + (seconds / 3600.0);
197414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if ((ref.equals("S") || ref.equals("W"))) {
197514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return (float) -result;
197614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
197714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return (float) result;
197814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (NumberFormatException | ArrayIndexOutOfBoundsException e) {
197914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Not valid
198014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IllegalArgumentException();
198114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
198214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
198314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
198414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Checks the type of image file
198514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int getMimeType(BufferedInputStream in) throws IOException {
198614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.mark(SIGNATURE_CHECK_SIZE);
198714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        byte[] signatureCheckBytes = new byte[SIGNATURE_CHECK_SIZE];
198814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (in.read(signatureCheckBytes) != SIGNATURE_CHECK_SIZE) {
198914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new EOFException();
199014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
199114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.reset();
199214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (isJpegFormat(signatureCheckBytes)) {
199314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return IMAGE_TYPE_JPEG;
199414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else if (isRafFormat(signatureCheckBytes)) {
199514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return IMAGE_TYPE_RAF;
199614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else if (isOrfFormat(signatureCheckBytes)) {
199714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return IMAGE_TYPE_ORF;
199814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else if (isRw2Format(signatureCheckBytes)) {
199914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return IMAGE_TYPE_RW2;
200014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
200114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Certain file formats (PEF) are identified in readImageFileDirectory()
200214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return IMAGE_TYPE_UNKNOWN;
200314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
200414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
200514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
200614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This method looks at the first 3 bytes to determine if this file is a JPEG file.
200714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See http://www.media.mit.edu/pia/Research/deepview/exif.html, "JPEG format and Marker"
200814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
200914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static boolean isJpegFormat(byte[] signatureCheckBytes) throws IOException {
201014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0; i < JPEG_SIGNATURE.length; i++) {
201114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (signatureCheckBytes[i] != JPEG_SIGNATURE[i]) {
201214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return false;
201314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
201414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
201514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return true;
201614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
201714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
201814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
201914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This method looks at the first 15 bytes to determine if this file is a RAF file.
202014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * There is no official specification for RAF files from Fuji, but there is an online archive of
202114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * image file specifications:
202214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://fileformats.archiveteam.org/wiki/Fujifilm_RAF
202314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
202414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean isRafFormat(byte[] signatureCheckBytes) throws IOException {
202514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        byte[] rafSignatureBytes = RAF_SIGNATURE.getBytes();
202614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0; i < rafSignatureBytes.length; i++) {
202714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (signatureCheckBytes[i] != rafSignatureBytes[i]) {
202814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return false;
202914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
203014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
203114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return true;
203214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
203314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
203414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
203514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * ORF has a similar structure to TIFF but it contains a different signature at the TIFF Header.
203614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This method looks at the 2 bytes following the Byte Order bytes to determine if this file is
203714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * an ORF file.
203814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * There is no official specification for ORF files from Olympus, but there is an online archive
203914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * of image file specifications:
204014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://fileformats.archiveteam.org/wiki/Olympus_ORF
204114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
204214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean isOrfFormat(byte[] signatureCheckBytes) throws IOException {
204314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ByteOrderedDataInputStream signatureInputStream =
204414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                new ByteOrderedDataInputStream(signatureCheckBytes);
204514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read byte order
204614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mExifByteOrder = readByteOrder(signatureInputStream);
204714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Set byte order
204814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        signatureInputStream.setByteOrder(mExifByteOrder);
204914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
205014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        short orfSignature = signatureInputStream.readShort();
2051f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes        signatureInputStream.close();
205214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return orfSignature == ORF_SIGNATURE_1 || orfSignature == ORF_SIGNATURE_2;
205314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
205414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
205514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
205614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * RW2 is TIFF-based, but stores 0x55 signature byte instead of 0x42 at the header
205714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See http://lclevy.free.fr/raw/
205814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
205914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean isRw2Format(byte[] signatureCheckBytes) throws IOException {
206014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ByteOrderedDataInputStream signatureInputStream =
206114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                new ByteOrderedDataInputStream(signatureCheckBytes);
206214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read byte order
206314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mExifByteOrder = readByteOrder(signatureInputStream);
206414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Set byte order
206514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        signatureInputStream.setByteOrder(mExifByteOrder);
206614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
206714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        short signatureByte = signatureInputStream.readShort();
2068f83358389f0c4ea37a7e7d9e493857f99baf0440Chris Banes        signatureInputStream.close();
206914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return signatureByte == RW2_SIGNATURE;
207014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
207114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
207214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
207314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Loads EXIF attributes from a JPEG input stream.
207414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
207514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param in The input stream that starts with the JPEG data.
207614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param jpegOffset The offset value in input stream for JPEG data.
207714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param imageType The image type from which to retrieve metadata. Use IFD_TYPE_PRIMARY for
207814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *                   primary image, IFD_TYPE_PREVIEW for preview image, and
207914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *                   IFD_TYPE_THUMBNAIL for thumbnail image.
208014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @throws IOException If the data contains invalid JPEG markers, offsets, or length values.
208114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
208214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void getJpegAttributes(ByteOrderedDataInputStream in, int jpegOffset, int imageType)
208314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
208414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // See JPEG File Interchange Format Specification, "JFIF Specification"
208514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (DEBUG) {
208614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "getJpegAttributes starting with: " + in);
208714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
208814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
208914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // JPEG uses Big Endian by default. See https://people.cs.umass.edu/~verts/cs32/endian.html
209014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.setByteOrder(ByteOrder.BIG_ENDIAN);
209114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
209214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Skip to JPEG data
209314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.seek(jpegOffset);
209414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int bytesRead = jpegOffset;
209514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
209614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        byte marker;
209714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if ((marker = in.readByte()) != MARKER) {
209814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid marker: " + Integer.toHexString(marker & 0xff));
209914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
210014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ++bytesRead;
210114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (in.readByte() != MARKER_SOI) {
210214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid marker: " + Integer.toHexString(marker & 0xff));
210314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
210414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ++bytesRead;
210514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        while (true) {
210614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            marker = in.readByte();
210714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (marker != MARKER) {
210814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Invalid marker:" + Integer.toHexString(marker & 0xff));
210914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
211014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ++bytesRead;
211114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            marker = in.readByte();
211214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
211314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "Found JPEG segment indicator: " + Integer.toHexString(marker & 0xff));
211414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
211514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ++bytesRead;
211614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
211714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // EOI indicates the end of an image and in case of SOS, JPEG image stream starts and
211814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // the image data will terminate right after.
211914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (marker == MARKER_EOI || marker == MARKER_SOS) {
212014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                break;
212114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
212214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int length = in.readUnsignedShort() - 2;
212314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            bytesRead += 2;
212414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
212514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "JPEG segment: " + Integer.toHexString(marker & 0xff) + " (length: "
212614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + (length + 2) + ")");
212714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
212814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (length < 0) {
212914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Invalid length");
213014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
213114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            switch (marker) {
213214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_APP1: {
213314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (DEBUG) {
213414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        Log.d(TAG, "MARKER_APP1");
213514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
213614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (length < 6) {
213714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // Skip if it's not an EXIF APP1 segment.
213814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
213914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
214014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    byte[] identifier = new byte[6];
214114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (in.read(identifier) != 6) {
214214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        throw new IOException("Invalid exif");
214314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
214414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    bytesRead += 6;
214514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    length -= 6;
214614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (!Arrays.equals(identifier, IDENTIFIER_EXIF_APP1)) {
214714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // Skip if it's not an EXIF APP1 segment.
214814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
214914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
215014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (length <= 0) {
215114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        throw new IOException("Invalid exif");
215214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
215314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (DEBUG) {
215414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        Log.d(TAG, "readExifSegment with a byte array (length: " + length + ")");
215514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
215614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Save offset values for createJpegThumbnailBitmap() function
215714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mExifOffset = bytesRead;
215814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
215914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    byte[] bytes = new byte[length];
216014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (in.read(bytes) != length) {
216114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        throw new IOException("Invalid exif");
216214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
216314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    bytesRead += length;
216414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    length = 0;
216514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
216614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    readExifSegment(bytes, imageType);
216714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
216814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
216914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
217014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_COM: {
217114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    byte[] bytes = new byte[length];
217214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (in.read(bytes) != length) {
217314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        throw new IOException("Invalid exif");
217414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
217514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    length = 0;
217614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (getAttribute(TAG_USER_COMMENT) == null) {
217714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[IFD_TYPE_EXIF].put(TAG_USER_COMMENT, ExifAttribute.createString(
217814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                new String(bytes, ASCII)));
217914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
218014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
218114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
218214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
218314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF0:
218414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF1:
218514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF2:
218614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF3:
218714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF5:
218814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF6:
218914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF7:
219014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF9:
219114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF10:
219214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF11:
219314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF13:
219414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF14:
219514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOF15: {
219614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (in.skipBytes(1) != 1) {
219714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        throw new IOException("Invalid SOFx");
219814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
219914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[imageType].put(TAG_IMAGE_LENGTH, ExifAttribute.createULong(
220014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            in.readUnsignedShort(), mExifByteOrder));
220114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[imageType].put(TAG_IMAGE_WIDTH, ExifAttribute.createULong(
220214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            in.readUnsignedShort(), mExifByteOrder));
220314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    length -= 5;
220414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
220514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
220614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
220714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                default: {
220814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
220914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
221014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
221114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (length < 0) {
221214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Invalid length");
221314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
221414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (in.skipBytes(length) != length) {
221514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Invalid JPEG segment");
221614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
221714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            bytesRead += length;
221814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
221914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Restore original byte order
222014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.setByteOrder(mExifByteOrder);
222114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
222214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
222314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void getRawAttributes(ByteOrderedDataInputStream in) throws IOException {
222414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Parse TIFF Headers. See JEITA CP-3451C Section 4.5.2. Table 1.
222514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        parseTiffHeaders(in, in.available());
222614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
222714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read TIFF image file directories. See JEITA CP-3451C Section 4.5.2. Figure 6.
222814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        readImageFileDirectory(in, IFD_TYPE_PRIMARY);
222914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
223014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Update ImageLength/Width tags for all image data.
223114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        updateImageSizeValues(in, IFD_TYPE_PRIMARY);
223214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        updateImageSizeValues(in, IFD_TYPE_PREVIEW);
223314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        updateImageSizeValues(in, IFD_TYPE_THUMBNAIL);
223414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
223514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Check if each image data is in valid position.
223614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        validateImages(in);
223714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
223814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mMimeType == IMAGE_TYPE_PEF) {
223914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // PEF files contain a MakerNote data, which contains the data for ColorSpace tag.
224014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See http://lclevy.free.fr/raw/ and piex.cc PefGetPreviewData()
224114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute makerNoteAttribute =
224214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    (ExifAttribute) mAttributes[IFD_TYPE_EXIF].get(TAG_MAKER_NOTE);
224314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (makerNoteAttribute != null) {
224414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Create an ordered DataInputStream for MakerNote
224514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ByteOrderedDataInputStream makerNoteDataInputStream =
224614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        new ByteOrderedDataInputStream(makerNoteAttribute.bytes);
224714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                makerNoteDataInputStream.setByteOrder(mExifByteOrder);
224814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
224914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Seek to MakerNote data
225014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                makerNoteDataInputStream.seek(PEF_MAKER_NOTE_SKIP_SIZE);
225114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
225214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Read IFD data from MakerNote
225314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                readImageFileDirectory(makerNoteDataInputStream, IFD_TYPE_PEF);
225414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
225514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Update ColorSpace tag
225614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ExifAttribute colorSpaceAttribute =
225714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        (ExifAttribute) mAttributes[IFD_TYPE_PEF].get(TAG_COLOR_SPACE);
225814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (colorSpaceAttribute != null) {
225914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[IFD_TYPE_EXIF].put(TAG_COLOR_SPACE, colorSpaceAttribute);
226014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
226114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
226214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
226314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
226414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
226514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
226614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * RAF files contains a JPEG and a CFA data.
226714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * The JPEG contains two images, a preview and a thumbnail, while the CFA contains a RAW image.
226814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This method looks at the first 160 bytes of a RAF file to retrieve the offset and length
226914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * values for the JPEG and CFA data.
227014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Using that data, it parses the JPEG data to retrieve the preview and thumbnail image data,
227114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * then parses the CFA metadata to retrieve the primary image length/width values.
227214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For data format details, see http://fileformats.archiveteam.org/wiki/Fujifilm_RAF
227314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
227414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void getRafAttributes(ByteOrderedDataInputStream in) throws IOException {
227514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Retrieve offset & length values
227614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.skipBytes(RAF_OFFSET_TO_JPEG_IMAGE_OFFSET);
227714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        byte[] jpegOffsetBytes = new byte[4];
227814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        byte[] cfaHeaderOffsetBytes = new byte[4];
227914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.read(jpegOffsetBytes);
228014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Skip JPEG length value since it is not needed
228114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.skipBytes(RAF_JPEG_LENGTH_VALUE_SIZE);
228214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.read(cfaHeaderOffsetBytes);
228314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int rafJpegOffset = ByteBuffer.wrap(jpegOffsetBytes).getInt();
228414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int rafCfaHeaderOffset = ByteBuffer.wrap(cfaHeaderOffsetBytes).getInt();
228514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
228614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Retrieve JPEG image metadata
228714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        getJpegAttributes(in, rafJpegOffset, IFD_TYPE_PREVIEW);
228814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
228914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Skip to CFA header offset.
229014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.seek(rafCfaHeaderOffset);
229114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
229214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Retrieve primary image length/width values, if TAG_RAF_IMAGE_SIZE exists
229314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        in.setByteOrder(ByteOrder.BIG_ENDIAN);
229414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int numberOfDirectoryEntry = in.readInt();
229514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (DEBUG) {
229614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "numberOfDirectoryEntry: " + numberOfDirectoryEntry);
229714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
229814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // CFA stores some metadata about the RAW image. Since CFA uses proprietary tags, can only
229914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // find and retrieve image size information tags, while skipping others.
230014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // See piex.cc RafGetDimension()
230114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0; i < numberOfDirectoryEntry; ++i) {
230214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int tagNumber = in.readUnsignedShort();
230314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int numberOfBytes = in.readUnsignedShort();
230414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (tagNumber == TAG_RAF_IMAGE_SIZE.number) {
230514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int imageLength = in.readShort();
230614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int imageWidth = in.readShort();
230714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ExifAttribute imageLengthAttribute =
230814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createUShort(imageLength, mExifByteOrder);
230914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ExifAttribute imageWidthAttribute =
231014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createUShort(imageWidth, mExifByteOrder);
231114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_LENGTH, imageLengthAttribute);
231214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_WIDTH, imageWidthAttribute);
231314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (DEBUG) {
231414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.d(TAG, "Updated to length: " + imageLength + ", width: " + imageWidth);
231514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
231614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return;
231714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
231814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            in.skipBytes(numberOfBytes);
231914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
232014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
232114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
232214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
232314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * ORF files contains a primary image data and a MakerNote data that contains preview/thumbnail
232414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * images. Both data takes the form of IFDs and can therefore be read with the
232514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * readImageFileDirectory() method.
232614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This method reads all the necessary data and updates the primary/preview/thumbnail image
232714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * information according to the GetOlympusPreviewImage() method in piex.cc.
232814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * For data format details, see the following:
232914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * http://fileformats.archiveteam.org/wiki/Olympus_ORF
233014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * https://libopenraw.freedesktop.org/wiki/Olympus_ORF
233114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
233214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void getOrfAttributes(ByteOrderedDataInputStream in) throws IOException {
233314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Retrieve primary image data
233414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Other Exif data will be located in the Makernote.
233514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        getRawAttributes(in);
233614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
233714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Additionally retrieve preview/thumbnail information from MakerNote tag, which contains
233814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // proprietary tags and therefore does not have offical documentation
233914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // See GetOlympusPreviewImage() in piex.cc & http://www.exiv2.org/tags-olympus.html
234014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute makerNoteAttribute =
234114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[IFD_TYPE_EXIF].get(TAG_MAKER_NOTE);
234214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (makerNoteAttribute != null) {
234314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Create an ordered DataInputStream for MakerNote
234414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ByteOrderedDataInputStream makerNoteDataInputStream =
234514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    new ByteOrderedDataInputStream(makerNoteAttribute.bytes);
234614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            makerNoteDataInputStream.setByteOrder(mExifByteOrder);
234714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
234814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // There are two types of headers for Olympus MakerNotes
234914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See http://www.exiv2.org/makernote.html#R1
235014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            byte[] makerNoteHeader1Bytes = new byte[ORF_MAKER_NOTE_HEADER_1.length];
235114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            makerNoteDataInputStream.readFully(makerNoteHeader1Bytes);
235214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            makerNoteDataInputStream.seek(0);
235314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            byte[] makerNoteHeader2Bytes = new byte[ORF_MAKER_NOTE_HEADER_2.length];
235414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            makerNoteDataInputStream.readFully(makerNoteHeader2Bytes);
235514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Skip the corresponding amount of bytes for each header type
235614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (Arrays.equals(makerNoteHeader1Bytes, ORF_MAKER_NOTE_HEADER_1)) {
235714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                makerNoteDataInputStream.seek(ORF_MAKER_NOTE_HEADER_1_SIZE);
235814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (Arrays.equals(makerNoteHeader2Bytes, ORF_MAKER_NOTE_HEADER_2)) {
235914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                makerNoteDataInputStream.seek(ORF_MAKER_NOTE_HEADER_2_SIZE);
236014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
236114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
236214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Read IFD data from MakerNote
236314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            readImageFileDirectory(makerNoteDataInputStream, IFD_TYPE_ORF_MAKER_NOTE);
236414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
236514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Retrieve & update preview image offset & length values
236614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute imageLengthAttribute = (ExifAttribute)
236714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[IFD_TYPE_ORF_CAMERA_SETTINGS].get(TAG_ORF_PREVIEW_IMAGE_START);
236814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute bitsPerSampleAttribute = (ExifAttribute)
236914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[IFD_TYPE_ORF_CAMERA_SETTINGS].get(TAG_ORF_PREVIEW_IMAGE_LENGTH);
237014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
237114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (imageLengthAttribute != null && bitsPerSampleAttribute != null) {
237214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[IFD_TYPE_PREVIEW].put(TAG_JPEG_INTERCHANGE_FORMAT,
237314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        imageLengthAttribute);
237414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[IFD_TYPE_PREVIEW].put(TAG_JPEG_INTERCHANGE_FORMAT_LENGTH,
237514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        bitsPerSampleAttribute);
237614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
237714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
237814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // TODO: Check this behavior in other ORF files
237914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Retrieve primary image length & width values
238014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See piex.cc GetOlympusPreviewImage()
238114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute aspectFrameAttribute = (ExifAttribute)
238214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[IFD_TYPE_ORF_IMAGE_PROCESSING].get(TAG_ORF_ASPECT_FRAME);
238314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (aspectFrameAttribute != null) {
238414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int[] aspectFrameValues = (int[]) aspectFrameAttribute.getValue(mExifByteOrder);
238514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (aspectFrameValues == null || aspectFrameValues.length != 4) {
238614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Invalid aspect frame values. frame="
238714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + Arrays.toString(aspectFrameValues));
238814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return;
238914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
239014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (aspectFrameValues[2] > aspectFrameValues[0] &&
239114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        aspectFrameValues[3] > aspectFrameValues[1]) {
239214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    int primaryImageWidth = aspectFrameValues[2] - aspectFrameValues[0] + 1;
239314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    int primaryImageLength = aspectFrameValues[3] - aspectFrameValues[1] + 1;
239414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Swap width & length values
239514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (primaryImageWidth < primaryImageLength) {
239614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        primaryImageWidth += primaryImageLength;
239714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        primaryImageLength = primaryImageWidth - primaryImageLength;
239814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        primaryImageWidth -= primaryImageLength;
239914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
240014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute primaryImageWidthAttribute =
240114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            ExifAttribute.createUShort(primaryImageWidth, mExifByteOrder);
240214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute primaryImageLengthAttribute =
240314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            ExifAttribute.createUShort(primaryImageLength, mExifByteOrder);
240414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
240514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_WIDTH, primaryImageWidthAttribute);
240614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_LENGTH, primaryImageLengthAttribute);
240714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
240814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
240914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
241014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
241114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
241214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // RW2 contains the primary image data in IFD0 and the preview and/or thumbnail image data in
241314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // the JpgFromRaw tag
241414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See https://libopenraw.freedesktop.org/wiki/Panasonic_RAW/ and piex.cc Rw2GetPreviewData()
241514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void getRw2Attributes(ByteOrderedDataInputStream in) throws IOException {
241614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Retrieve primary image data
241714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        getRawAttributes(in);
241814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
241914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Retrieve preview and/or thumbnail image data
242014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute jpgFromRawAttribute =
242114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[IFD_TYPE_PRIMARY].get(TAG_RW2_JPG_FROM_RAW);
242214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (jpgFromRawAttribute != null) {
242314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            getJpegAttributes(in, mRw2JpgFromRawOffset, IFD_TYPE_PREVIEW);
242414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
242514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
242614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Set ISO tag value if necessary
242714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute rw2IsoAttribute =
242814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[IFD_TYPE_PRIMARY].get(TAG_RW2_ISO);
242914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute exifIsoAttribute =
243014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[IFD_TYPE_EXIF].get(TAG_ISO_SPEED_RATINGS);
243114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (rw2IsoAttribute != null && exifIsoAttribute == null) {
243214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Place this attribute only if it doesn't exist
243314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_EXIF].put(TAG_ISO_SPEED_RATINGS, rw2IsoAttribute);
243414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
243514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
243614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
243714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Stores a new JPEG image with EXIF attributes into a given output stream.
243814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void saveJpegAttributes(InputStream inputStream, OutputStream outputStream)
243914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
244014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // See JPEG File Interchange Format Specification, "JFIF Specification"
244114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (DEBUG) {
244214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "saveJpegAttributes starting with (inputStream: " + inputStream
244314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    + ", outputStream: " + outputStream + ")");
244414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
244514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        DataInputStream dataInputStream = new DataInputStream(inputStream);
244614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ByteOrderedDataOutputStream dataOutputStream =
244714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                new ByteOrderedDataOutputStream(outputStream, ByteOrder.BIG_ENDIAN);
244814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (dataInputStream.readByte() != MARKER) {
244914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid marker");
245014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
245114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeByte(MARKER);
245214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (dataInputStream.readByte() != MARKER_SOI) {
245314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid marker");
245414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
245514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeByte(MARKER_SOI);
245614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
245714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Write EXIF APP1 segment
245814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeByte(MARKER);
245914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeByte(MARKER_APP1);
246014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        writeExifSegment(dataOutputStream, 6);
246114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
246214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        byte[] bytes = new byte[4096];
246314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
246414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        while (true) {
246514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            byte marker = dataInputStream.readByte();
246614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (marker != MARKER) {
246714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Invalid marker");
246814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
246914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            marker = dataInputStream.readByte();
247014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            switch (marker) {
247114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_APP1: {
247214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    int length = dataInputStream.readUnsignedShort() - 2;
247314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (length < 0) {
247414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        throw new IOException("Invalid length");
247514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
247614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    byte[] identifier = new byte[6];
247714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (length >= 6) {
247814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        if (dataInputStream.read(identifier) != 6) {
247914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            throw new IOException("Invalid exif");
248014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
248114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        if (Arrays.equals(identifier, IDENTIFIER_EXIF_APP1)) {
248214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            // Skip the original EXIF APP1 segment.
248314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            if (dataInputStream.skipBytes(length - 6) != length - 6) {
248414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                throw new IOException("Invalid length");
248514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            }
248614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            break;
248714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
248814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
248914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Copy non-EXIF APP1 segment.
249014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeByte(MARKER);
249114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeByte(marker);
249214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeUnsignedShort(length + 2);
249314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (length >= 6) {
249414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        length -= 6;
249514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        dataOutputStream.write(identifier);
249614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
249714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    int read;
249814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    while (length > 0 && (read = dataInputStream.read(
249914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            bytes, 0, Math.min(length, bytes.length))) >= 0) {
250014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        dataOutputStream.write(bytes, 0, read);
250114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        length -= read;
250214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
250314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
250414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
250514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_EOI:
250614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case MARKER_SOS: {
250714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeByte(MARKER);
250814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeByte(marker);
250914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Copy all the remaining data
251014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    copy(dataInputStream, dataOutputStream);
251114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return;
251214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
251314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                default: {
251414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Copy JPEG segment
251514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeByte(MARKER);
251614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeByte(marker);
251714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    int length = dataInputStream.readUnsignedShort();
251814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeUnsignedShort(length);
251914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    length -= 2;
252014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (length < 0) {
252114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        throw new IOException("Invalid length");
252214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
252314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    int read;
252414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    while (length > 0 && (read = dataInputStream.read(
252514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            bytes, 0, Math.min(length, bytes.length))) >= 0) {
252614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        dataOutputStream.write(bytes, 0, read);
252714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        length -= read;
252814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
252914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
253014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
253114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
253214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
253314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
253414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
253514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Reads the given EXIF byte area and save its tag data into attributes.
253614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void readExifSegment(byte[] exifBytes, int imageType) throws IOException {
253714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ByteOrderedDataInputStream dataInputStream =
253814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                new ByteOrderedDataInputStream(exifBytes);
253914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
254014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Parse TIFF Headers. See JEITA CP-3451C Section 4.5.2. Table 1.
254114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        parseTiffHeaders(dataInputStream, exifBytes.length);
254214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
254314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read TIFF image file directories. See JEITA CP-3451C Section 4.5.2. Figure 6.
254414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        readImageFileDirectory(dataInputStream, imageType);
254514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
254614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
254714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void addDefaultValuesForCompatibility() {
254814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // The value of DATETIME tag has the same value of DATETIME_ORIGINAL tag.
254914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        String valueOfDateTimeOriginal = getAttribute(TAG_DATETIME_ORIGINAL);
255014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (valueOfDateTimeOriginal != null) {
255114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(TAG_DATETIME,
255214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createString(valueOfDateTimeOriginal));
255314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
255414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
255514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Add the default value.
255614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (getAttribute(TAG_IMAGE_WIDTH) == null) {
255714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_WIDTH,
255814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
255914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
256014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (getAttribute(TAG_IMAGE_LENGTH) == null) {
256114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_LENGTH,
256214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
256314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
256414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (getAttribute(TAG_ORIENTATION) == null) {
256514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(TAG_ORIENTATION,
256614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
256714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
256814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (getAttribute(TAG_LIGHT_SOURCE) == null) {
256914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_EXIF].put(TAG_LIGHT_SOURCE,
257014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
257114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
257214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
257314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
257414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private ByteOrder readByteOrder(ByteOrderedDataInputStream dataInputStream)
257514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
257614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read byte order.
257714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        short byteOrder = dataInputStream.readShort();
257814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        switch (byteOrder) {
257914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            case BYTE_ALIGN_II:
258014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (DEBUG) {
258114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.d(TAG, "readExifSegment: Byte Align II");
258214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
258314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return ByteOrder.LITTLE_ENDIAN;
258414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            case BYTE_ALIGN_MM:
258514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (DEBUG) {
258614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.d(TAG, "readExifSegment: Byte Align MM");
258714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
258814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return ByteOrder.BIG_ENDIAN;
258914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            default:
259014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Invalid byte order: " + Integer.toHexString(byteOrder));
259114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
259214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
259314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
259414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void parseTiffHeaders(ByteOrderedDataInputStream dataInputStream,
259514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int exifBytesLength) throws IOException {
259614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read byte order
259714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        mExifByteOrder = readByteOrder(dataInputStream);
259814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Set byte order
259914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataInputStream.setByteOrder(mExifByteOrder);
260014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
260114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Check start code
260214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int startCode = dataInputStream.readUnsignedShort();
260314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mMimeType != IMAGE_TYPE_ORF && mMimeType != IMAGE_TYPE_RW2 && startCode != START_CODE) {
260414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid start code: " + Integer.toHexString(startCode));
260514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
260614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
260714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Read and skip to first ifd offset
260814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int firstIfdOffset = dataInputStream.readInt();
260914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (firstIfdOffset < 8 || firstIfdOffset >= exifBytesLength) {
261014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid first Ifd offset: " + firstIfdOffset);
261114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
261214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        firstIfdOffset -= 8;
261314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (firstIfdOffset > 0) {
261414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (dataInputStream.skipBytes(firstIfdOffset) != firstIfdOffset) {
261514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Couldn't jump to first Ifd: " + firstIfdOffset);
261614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
261714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
261814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
261914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
262014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Reads image file directory, which is a tag group in EXIF.
262114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void readImageFileDirectory(ByteOrderedDataInputStream dataInputStream,
262214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            @IfdType int ifdType) throws IOException {
262314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (dataInputStream.mPosition + 2 > dataInputStream.mLength) {
262414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Return if there is no data from the offset.
262514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return;
262614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
262714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // See TIFF 6.0 Section 2: TIFF Structure, Figure 1.
262814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        short numberOfDirectoryEntry = dataInputStream.readShort();
262914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (dataInputStream.mPosition + 12 * numberOfDirectoryEntry > dataInputStream.mLength) {
263014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Return if the size of entries is too big.
263114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return;
263214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
263314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
263414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (DEBUG) {
263514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "numberOfDirectoryEntry: " + numberOfDirectoryEntry);
263614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
263714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
263814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // See TIFF 6.0 Section 2: TIFF Structure, "Image File Directory".
263914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (short i = 0; i < numberOfDirectoryEntry; ++i) {
264014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int tagNumber = dataInputStream.readUnsignedShort();
264114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int dataFormat = dataInputStream.readUnsignedShort();
264214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int numberOfComponents = dataInputStream.readInt();
264314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Next four bytes is for data offset or value.
264414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            long nextEntryOffset = dataInputStream.peek() + 4;
264514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
264614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Look up a corresponding tag from tag number
264714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifTag tag = (ExifTag) sExifTagMapsForReading[ifdType].get(tagNumber);
264814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
264914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
265014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, String.format("ifdType: %d, tagNumber: %d, tagName: %s, dataFormat: %d, "
265114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + "numberOfComponents: %d", ifdType, tagNumber,
265214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        tag != null ? tag.name : null, dataFormat, numberOfComponents));
265314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
265414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
265514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            long byteCount = 0;
265614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            boolean valid = false;
265714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (tag == null) {
265814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.w(TAG, "Skip the tag entry since tag number is not defined: " + tagNumber);
265914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (dataFormat <= 0 || dataFormat >= IFD_FORMAT_BYTES_PER_FORMAT.length) {
266014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.w(TAG, "Skip the tag entry since data format is invalid: " + dataFormat);
266114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else {
266214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                byteCount = (long) numberOfComponents * IFD_FORMAT_BYTES_PER_FORMAT[dataFormat];
266314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (byteCount < 0 || byteCount > Integer.MAX_VALUE) {
266414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Skip the tag entry since the number of components is invalid: "
266514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + numberOfComponents);
266614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else {
266714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    valid = true;
266814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
266914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
267014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (!valid) {
267114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                dataInputStream.seek(nextEntryOffset);
267214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                continue;
267314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
267414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
267514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Read a value from data field or seek to the value offset which is stored in data
267614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // field if the size of the entry value is bigger than 4.
267714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (byteCount > 4) {
267814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int offset = dataInputStream.readInt();
267914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (DEBUG) {
268014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.d(TAG, "seek to data offset: " + offset);
268114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
268214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (mMimeType == IMAGE_TYPE_ORF) {
268314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (tag.name == TAG_MAKER_NOTE) {
268414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // Save offset value for reading thumbnail
268514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mOrfMakerNoteOffset = offset;
268614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    } else if (ifdType == IFD_TYPE_ORF_MAKER_NOTE
268714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            && tag.name == TAG_ORF_THUMBNAIL_IMAGE) {
268814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // Retrieve & update values for thumbnail offset and length values for ORF
268914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mOrfThumbnailOffset = offset;
269014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mOrfThumbnailLength = numberOfComponents;
269114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
269214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute compressionAttribute =
269314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createUShort(DATA_JPEG, mExifByteOrder);
269414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute jpegInterchangeFormatAttribute =
269514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createULong(mOrfThumbnailOffset, mExifByteOrder);
269614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute jpegInterchangeFormatLengthAttribute =
269714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                ExifAttribute.createULong(mOrfThumbnailLength, mExifByteOrder);
269814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
269914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[IFD_TYPE_THUMBNAIL].put(TAG_COMPRESSION, compressionAttribute);
270014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[IFD_TYPE_THUMBNAIL].put(TAG_JPEG_INTERCHANGE_FORMAT,
270114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                jpegInterchangeFormatAttribute);
270214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mAttributes[IFD_TYPE_THUMBNAIL].put(TAG_JPEG_INTERCHANGE_FORMAT_LENGTH,
270314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                jpegInterchangeFormatLengthAttribute);
270414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
270514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else if (mMimeType == IMAGE_TYPE_RW2) {
270614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (tag.name == TAG_RW2_JPG_FROM_RAW) {
270714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        mRw2JpgFromRawOffset = offset;
270814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
270914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
271014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (offset + byteCount <= dataInputStream.mLength) {
271114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataInputStream.seek(offset);
271214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else {
271314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Skip if invalid data offset.
271414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Skip the tag entry since data offset is invalid: " + offset);
271514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataInputStream.seek(nextEntryOffset);
271614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    continue;
271714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
271814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
271914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
272014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Recursively parse IFD when a IFD pointer tag appears.
272114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Object nextIfdType = sExifPointerTagMap.get(tagNumber);
272214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
272314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "nextIfdType: " + nextIfdType + " byteCount: " + byteCount);
272414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
272514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
272614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (nextIfdType != null) {
272714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                long offset = -1L;
272814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Get offset from data field
272914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                switch (dataFormat) {
273014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_USHORT: {
273114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        offset = dataInputStream.readUnsignedShort();
273214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
273314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
273414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SSHORT: {
273514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        offset = dataInputStream.readShort();
273614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
273714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
273814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_ULONG: {
273914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        offset = dataInputStream.readUnsignedInt();
274014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
274114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
274214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_SLONG:
274314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    case IFD_FORMAT_IFD: {
274414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        offset = dataInputStream.readInt();
274514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
274614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
274714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    default: {
274814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // Nothing to do
274914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        break;
275014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
275114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
275214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (DEBUG) {
275314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.d(TAG, String.format("Offset: %d, tagName: %s", offset, tag.name));
275414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
275514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (offset > 0L && offset < dataInputStream.mLength) {
275614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataInputStream.seek(offset);
275714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    readImageFileDirectory(dataInputStream, (int) nextIfdType);
275814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else {
275914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Skip jump into the IFD since its offset is invalid: " + offset);
276014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
276114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
276214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                dataInputStream.seek(nextEntryOffset);
276314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                continue;
276414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
276514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
276614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            byte[] bytes = new byte[(int) byteCount];
276714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            dataInputStream.readFully(bytes);
276814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute attribute = new ExifAttribute(dataFormat, numberOfComponents, bytes);
276914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[ifdType].put(tag.name, attribute);
277014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
277114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // DNG files have a DNG Version tag specifying the version of specifications that the
277214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // image file is following.
277314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See http://fileformats.archiveteam.org/wiki/DNG
277414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (tag.name == TAG_DNG_VERSION) {
277514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mMimeType = IMAGE_TYPE_DNG;
277614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
277714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
277814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // PEF files have a Make or Model tag that begins with "PENTAX" or a compression tag
277914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // that is 65535.
278014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See http://fileformats.archiveteam.org/wiki/Pentax_PEF
278114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (((tag.name == TAG_MAKE || tag.name == TAG_MODEL)
278214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    && attribute.getStringValue(mExifByteOrder).contains(PEF_SIGNATURE))
278314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    || (tag.name == TAG_COMPRESSION
278414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    && attribute.getIntValue(mExifByteOrder) == 65535)) {
278514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mMimeType = IMAGE_TYPE_PEF;
278614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
278714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
278814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Seek to next tag offset
278914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (dataInputStream.peek() != nextEntryOffset) {
279014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                dataInputStream.seek(nextEntryOffset);
279114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
279214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
279314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
279414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (dataInputStream.peek() + 4 <= dataInputStream.mLength) {
279514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int nextIfdOffset = dataInputStream.readInt();
279614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
279714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, String.format("nextIfdOffset: %d", nextIfdOffset));
279814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
279914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // The next IFD offset needs to be bigger than 8
280014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // since the first IFD offset is at least 8.
280114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (nextIfdOffset > 8 && nextIfdOffset < dataInputStream.mLength) {
280214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                dataInputStream.seek(nextIfdOffset);
280314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (mAttributes[IFD_TYPE_THUMBNAIL].isEmpty()) {
280414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Do not overwrite thumbnail IFD data if it alreay exists.
280514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    readImageFileDirectory(dataInputStream, IFD_TYPE_THUMBNAIL);
280614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else if (mAttributes[IFD_TYPE_PREVIEW].isEmpty()) {
280714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    readImageFileDirectory(dataInputStream, IFD_TYPE_PREVIEW);
280814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
280914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
281014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
281114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
281214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
281314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
281414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * JPEG compressed images do not contain IMAGE_LENGTH & IMAGE_WIDTH tags.
281514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This value uses JpegInterchangeFormat(JPEG data offset) value, and calls getJpegAttributes()
281614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * to locate SOF(Start of Frame) marker and update the image length & width values.
281714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See JEITA CP-3451C Table 5 and Section 4.8.1. B.
281814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
281914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void retrieveJpegImageSize(ByteOrderedDataInputStream in, int imageType)
282014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
282114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Check if image already has IMAGE_LENGTH & IMAGE_WIDTH values
282214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute imageLengthAttribute =
282314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[imageType].get(TAG_IMAGE_LENGTH);
282414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute imageWidthAttribute =
282514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[imageType].get(TAG_IMAGE_WIDTH);
282614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
282714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (imageLengthAttribute == null || imageWidthAttribute == null) {
282814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Find if offset for JPEG data exists
282914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute jpegInterchangeFormatAttribute =
283014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    (ExifAttribute) mAttributes[imageType].get(TAG_JPEG_INTERCHANGE_FORMAT);
283114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (jpegInterchangeFormatAttribute != null) {
283214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int jpegInterchangeFormat =
283314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        jpegInterchangeFormatAttribute.getIntValue(mExifByteOrder);
283414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
283514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Searches for SOF marker in JPEG data and updates IMAGE_LENGTH & IMAGE_WIDTH tags
283614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                getJpegAttributes(in, jpegInterchangeFormat, imageType);
283714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
283814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
283914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
284014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
284114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Sets thumbnail offset & length attributes based on JpegInterchangeFormat or StripOffsets tags
284214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void setThumbnailData(ByteOrderedDataInputStream in) throws IOException {
284314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        HashMap thumbnailData = mAttributes[IFD_TYPE_THUMBNAIL];
284414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
284514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute compressionAttribute =
284614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) thumbnailData.get(TAG_COMPRESSION);
284714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (compressionAttribute != null) {
284814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mThumbnailCompression = compressionAttribute.getIntValue(mExifByteOrder);
284914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            switch (mThumbnailCompression) {
285014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case DATA_JPEG: {
285114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    handleThumbnailFromJfif(in, thumbnailData);
285214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
285314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
285414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case DATA_UNCOMPRESSED:
285514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                case DATA_JPEG_COMPRESSED: {
285614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (isSupportedDataType(thumbnailData)) {
285714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        handleThumbnailFromStrips(in, thumbnailData);
285814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
285914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    break;
286014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
286114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
286214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else {
286314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Thumbnail data may not contain Compression tag value
286414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mThumbnailCompression = DATA_JPEG;
286514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            handleThumbnailFromJfif(in, thumbnailData);
286614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
286714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
286814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
286914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Check JpegInterchangeFormat(JFIF) tags to retrieve thumbnail offset & length values
287014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // and reads the corresponding bytes if stream does not support seek function
287114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void handleThumbnailFromJfif(ByteOrderedDataInputStream in, HashMap thumbnailData)
287214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
287314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute jpegInterchangeFormatAttribute =
287414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) thumbnailData.get(TAG_JPEG_INTERCHANGE_FORMAT);
287514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute jpegInterchangeFormatLengthAttribute =
287614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) thumbnailData.get(TAG_JPEG_INTERCHANGE_FORMAT_LENGTH);
287714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (jpegInterchangeFormatAttribute != null
287814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                && jpegInterchangeFormatLengthAttribute != null) {
287914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int thumbnailOffset = jpegInterchangeFormatAttribute.getIntValue(mExifByteOrder);
288014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int thumbnailLength = jpegInterchangeFormatLengthAttribute.getIntValue(mExifByteOrder);
288114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
288214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // The following code limits the size of thumbnail size not to overflow EXIF data area.
288314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            thumbnailLength = Math.min(thumbnailLength, in.available() - thumbnailOffset);
288414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mMimeType == IMAGE_TYPE_JPEG || mMimeType == IMAGE_TYPE_RAF
288514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    || mMimeType == IMAGE_TYPE_RW2) {
288614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                thumbnailOffset += mExifOffset;
288714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mMimeType == IMAGE_TYPE_ORF) {
288814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Update offset value since RAF files have IFD data preceding MakerNote data.
288914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                thumbnailOffset += mOrfMakerNoteOffset;
289014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
289114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
289214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "Setting thumbnail attributes with offset: " + thumbnailOffset
289314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + ", length: " + thumbnailLength);
289414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
289514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (thumbnailOffset > 0 && thumbnailLength > 0) {
289614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mHasThumbnail = true;
289714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mThumbnailOffset = thumbnailOffset;
289814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mThumbnailLength = thumbnailLength;
289914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (mFilename == null && mAssetInputStream == null) {
290014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Save the thumbnail in memory if the input doesn't support reading again.
290114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    byte[] thumbnailBytes = new byte[thumbnailLength];
290214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    in.seek(thumbnailOffset);
290314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    in.readFully(thumbnailBytes);
290414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mThumbnailBytes = thumbnailBytes;
290514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
290614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
290714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
290814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
290914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
291014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Check StripOffsets & StripByteCounts tags to retrieve thumbnail offset & length values
291114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void handleThumbnailFromStrips(ByteOrderedDataInputStream in, HashMap thumbnailData)
291214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
291314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute stripOffsetsAttribute =
291414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) thumbnailData.get(TAG_STRIP_OFFSETS);
291514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute stripByteCountsAttribute =
291614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) thumbnailData.get(TAG_STRIP_BYTE_COUNTS);
291714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
291814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (stripOffsetsAttribute != null && stripByteCountsAttribute != null) {
291914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            long[] stripOffsets =
292014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    (long[]) stripOffsetsAttribute.getValue(mExifByteOrder);
292114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            long[] stripByteCounts =
292214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    (long[]) stripByteCountsAttribute.getValue(mExifByteOrder);
292314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
292414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (stripOffsets == null) {
292514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.w(TAG, "stripOffsets should not be null.");
292614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return;
292714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
292814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (stripByteCounts == null) {
292914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.w(TAG, "stripByteCounts should not be null.");
293014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return;
293114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
293214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
293314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            long totalStripByteCount = 0;
293414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (long byteCount : stripByteCounts) {
293514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                totalStripByteCount += byteCount;
293614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
293714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
293814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Set thumbnail byte array data for non-consecutive strip bytes
293914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            byte[] totalStripBytes = new byte[(int) totalStripByteCount];
294014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
294114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int bytesRead = 0;
294214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int bytesAdded = 0;
294314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (int i = 0; i < stripOffsets.length; i++) {
294414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int stripOffset = (int) stripOffsets[i];
294514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int stripByteCount = (int) stripByteCounts[i];
294614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
294714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Skip to offset
294814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int skipBytes = stripOffset - bytesRead;
294914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (skipBytes < 0) {
295014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.d(TAG, "Invalid strip offset value");
295114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
295214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                in.seek(skipBytes);
295314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                bytesRead += skipBytes;
295414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
295514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Read strip bytes
295614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                byte[] stripBytes = new byte[stripByteCount];
295714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                in.read(stripBytes);
295814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                bytesRead += stripByteCount;
295914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
296014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Add bytes to array
296114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                System.arraycopy(stripBytes, 0, totalStripBytes, bytesAdded,
296214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        stripBytes.length);
296314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                bytesAdded += stripBytes.length;
296414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
296514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
296614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mHasThumbnail = true;
296714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mThumbnailBytes = totalStripBytes;
296814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mThumbnailLength = totalStripBytes.length;
296914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
297014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
297114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
297214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Check if thumbnail data type is currently supported or not
297314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean isSupportedDataType(HashMap thumbnailData) throws IOException {
297414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute bitsPerSampleAttribute =
297514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) thumbnailData.get(TAG_BITS_PER_SAMPLE);
297614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (bitsPerSampleAttribute != null) {
297714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int[] bitsPerSampleValue = (int[]) bitsPerSampleAttribute.getValue(mExifByteOrder);
297814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
297914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (Arrays.equals(BITS_PER_SAMPLE_RGB, bitsPerSampleValue)) {
298014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return true;
298114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
298214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
298314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // See DNG Specification 1.4.0.0. Section 3, Compression.
298414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mMimeType == IMAGE_TYPE_DNG) {
298514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ExifAttribute photometricInterpretationAttribute =
298614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        (ExifAttribute) thumbnailData.get(TAG_PHOTOMETRIC_INTERPRETATION);
298714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (photometricInterpretationAttribute != null) {
298814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    int photometricInterpretationValue
298914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            = photometricInterpretationAttribute.getIntValue(mExifByteOrder);
299014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if ((photometricInterpretationValue == PHOTOMETRIC_INTERPRETATION_BLACK_IS_ZERO
299114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            && Arrays.equals(bitsPerSampleValue, BITS_PER_SAMPLE_GREYSCALE_2))
299214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            || ((photometricInterpretationValue == PHOTOMETRIC_INTERPRETATION_YCBCR)
299314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            && (Arrays.equals(bitsPerSampleValue, BITS_PER_SAMPLE_RGB)))) {
299414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return true;
299514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    } else {
299614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // TODO: Add support for lossless Huffman JPEG data
299714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
299814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
299914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
300014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
300114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (DEBUG) {
300214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "Unsupported data type value");
300314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
300414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return false;
300514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
300614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
300714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Returns true if the image length and width values are <= 512.
300814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // See Section 4.8 of http://standardsproposals.bsigroup.com/Home/getPDF/567
300914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private boolean isThumbnail(HashMap map) throws IOException {
301014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute imageLengthAttribute = (ExifAttribute) map.get(TAG_IMAGE_LENGTH);
301114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute imageWidthAttribute = (ExifAttribute) map.get(TAG_IMAGE_WIDTH);
301214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
301314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (imageLengthAttribute != null && imageWidthAttribute != null) {
301414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int imageLengthValue = imageLengthAttribute.getIntValue(mExifByteOrder);
301514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int imageWidthValue = imageWidthAttribute.getIntValue(mExifByteOrder);
301614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (imageLengthValue <= MAX_THUMBNAIL_SIZE && imageWidthValue <= MAX_THUMBNAIL_SIZE) {
301714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return true;
301814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
301914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
302014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return false;
302114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
302214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
302314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Validate primary, preview, thumbnail image data by comparing image size
302414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void validateImages(InputStream in) throws IOException {
302514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Swap images based on size (primary > preview > thumbnail)
302614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        swapBasedOnImageSize(IFD_TYPE_PRIMARY, IFD_TYPE_PREVIEW);
302714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        swapBasedOnImageSize(IFD_TYPE_PRIMARY, IFD_TYPE_THUMBNAIL);
302814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        swapBasedOnImageSize(IFD_TYPE_PREVIEW, IFD_TYPE_THUMBNAIL);
302914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
303014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Check if image has PixelXDimension/PixelYDimension tags, which contain valid image
303114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // sizes, excluding padding at the right end or bottom end of the image to make sure that
303214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // the values are multiples of 64. See JEITA CP-3451C Table 5 and Section 4.8.1. B.
303314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute pixelXDimAttribute =
303414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[IFD_TYPE_EXIF].get(TAG_PIXEL_X_DIMENSION);
303514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute pixelYDimAttribute =
303614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[IFD_TYPE_EXIF].get(TAG_PIXEL_Y_DIMENSION);
303714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (pixelXDimAttribute != null && pixelYDimAttribute != null) {
303814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_WIDTH, pixelXDimAttribute);
303914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(TAG_IMAGE_LENGTH, pixelYDimAttribute);
304014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
304114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
304214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Check whether thumbnail image exists and whether preview image satisfies the thumbnail
304314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // image requirements
304414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mAttributes[IFD_TYPE_THUMBNAIL].isEmpty()) {
304514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (isThumbnail(mAttributes[IFD_TYPE_PREVIEW])) {
304614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[IFD_TYPE_THUMBNAIL] = mAttributes[IFD_TYPE_PREVIEW];
304714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[IFD_TYPE_PREVIEW] = new HashMap();
304814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
304914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
305014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
305114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Check if the thumbnail image satisfies the thumbnail size requirements
305214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!isThumbnail(mAttributes[IFD_TYPE_THUMBNAIL])) {
305314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "No image meets the size requirements of a thumbnail image.");
305414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
305514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
305614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
305714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
305814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * If image is uncompressed, ImageWidth/Length tags are used to store size info.
305914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * However, uncompressed images often store extra pixels around the edges of the final image,
306014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * which results in larger values for TAG_IMAGE_WIDTH and TAG_IMAGE_LENGTH tags.
306114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * This method corrects those tag values by checking first the values of TAG_DEFAULT_CROP_SIZE
306214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See DNG Specification 1.4.0.0. Section 4. (DefaultCropSize)
306314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
306414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * If image is a RW2 file, valid image sizes are stored in SensorBorder tags.
306514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * See tiff_parser.cc GetFullDimension32()
306614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * */
306714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void updateImageSizeValues(ByteOrderedDataInputStream in, int imageType)
306814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
306914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Uncompressed image valid image size values
307014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute defaultCropSizeAttribute =
307114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[imageType].get(TAG_DEFAULT_CROP_SIZE);
307214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // RW2 image valid image size values
307314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute topBorderAttribute =
307414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[imageType].get(TAG_RW2_SENSOR_TOP_BORDER);
307514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute leftBorderAttribute =
307614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[imageType].get(TAG_RW2_SENSOR_LEFT_BORDER);
307714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute bottomBorderAttribute =
307814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[imageType].get(TAG_RW2_SENSOR_BOTTOM_BORDER);
307914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute rightBorderAttribute =
308014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[imageType].get(TAG_RW2_SENSOR_RIGHT_BORDER);
308114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
308214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (defaultCropSizeAttribute != null) {
308314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Update for uncompressed image
308414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ExifAttribute defaultCropSizeXAttribute, defaultCropSizeYAttribute;
308514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (defaultCropSizeAttribute.format == IFD_FORMAT_URATIONAL) {
308614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Rational[] defaultCropSizeValue =
308714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        (Rational[]) defaultCropSizeAttribute.getValue(mExifByteOrder);
308814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (defaultCropSizeValue == null || defaultCropSizeValue.length != 2) {
308914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Invalid crop size values. cropSize="
309014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + Arrays.toString(defaultCropSizeValue));
309114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return;
309214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
309314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                defaultCropSizeXAttribute =
309414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createURational(defaultCropSizeValue[0], mExifByteOrder);
309514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                defaultCropSizeYAttribute =
309614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createURational(defaultCropSizeValue[1], mExifByteOrder);
309714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else {
309814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int[] defaultCropSizeValue =
309914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        (int[]) defaultCropSizeAttribute.getValue(mExifByteOrder);
310014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (defaultCropSizeValue == null || defaultCropSizeValue.length != 2) {
310114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    Log.w(TAG, "Invalid crop size values. cropSize="
310214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            + Arrays.toString(defaultCropSizeValue));
310314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return;
310414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
310514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                defaultCropSizeXAttribute =
310614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createUShort(defaultCropSizeValue[0], mExifByteOrder);
310714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                defaultCropSizeYAttribute =
310814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createUShort(defaultCropSizeValue[1], mExifByteOrder);
310914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
311014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[imageType].put(TAG_IMAGE_WIDTH, defaultCropSizeXAttribute);
311114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[imageType].put(TAG_IMAGE_LENGTH, defaultCropSizeYAttribute);
311214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else if (topBorderAttribute != null && leftBorderAttribute != null &&
311314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                bottomBorderAttribute != null && rightBorderAttribute != null) {
311414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Update for RW2 image
311514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int topBorderValue = topBorderAttribute.getIntValue(mExifByteOrder);
311614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int bottomBorderValue = bottomBorderAttribute.getIntValue(mExifByteOrder);
311714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int rightBorderValue = rightBorderAttribute.getIntValue(mExifByteOrder);
311814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int leftBorderValue = leftBorderAttribute.getIntValue(mExifByteOrder);
311914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (bottomBorderValue > topBorderValue && rightBorderValue > leftBorderValue) {
312014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int length = bottomBorderValue - topBorderValue;
312114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int width = rightBorderValue - leftBorderValue;
312214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ExifAttribute imageLengthAttribute =
312314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createUShort(length, mExifByteOrder);
312414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ExifAttribute imageWidthAttribute =
312514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        ExifAttribute.createUShort(width, mExifByteOrder);
312614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[imageType].put(TAG_IMAGE_LENGTH, imageLengthAttribute);
312714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[imageType].put(TAG_IMAGE_WIDTH, imageWidthAttribute);
312814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
312914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else {
313014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            retrieveJpegImageSize(in, imageType);
313114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
313214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
313314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
313414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Writes an Exif segment into the given output stream.
313514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private int writeExifSegment(ByteOrderedDataOutputStream dataOutputStream,
313614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int exifOffsetFromBeginning) throws IOException {
313714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // The following variables are for calculating each IFD tag group size in bytes.
313814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int[] ifdOffsets = new int[EXIF_TAGS.length];
313914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int[] ifdDataSizes = new int[EXIF_TAGS.length];
314014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
314114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Remove IFD pointer tags (we'll re-add it later.)
314214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (ExifTag tag : EXIF_POINTER_TAGS) {
314314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            removeAttribute(tag.name);
314414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
314514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Remove old thumbnail data
314614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        removeAttribute(JPEG_INTERCHANGE_FORMAT_TAG.name);
314714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        removeAttribute(JPEG_INTERCHANGE_FORMAT_LENGTH_TAG.name);
314814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
314914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Remove null value tags.
315014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int ifdType = 0; ifdType < EXIF_TAGS.length; ++ifdType) {
315114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (Object obj : mAttributes[ifdType].entrySet().toArray()) {
315214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                final Map.Entry entry = (Map.Entry) obj;
315314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (entry.getValue() == null) {
315414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    mAttributes[ifdType].remove(entry.getKey());
315514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
315614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
315714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
315814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
315914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Add IFD pointer tags. The next offset of primary image TIFF IFD will have thumbnail IFD
316014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // offset when there is one or more tags in the thumbnail IFD.
316114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mAttributes[IFD_TYPE_EXIF].isEmpty()) {
316214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(EXIF_POINTER_TAGS[1].name,
316314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
316414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
316514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mAttributes[IFD_TYPE_GPS].isEmpty()) {
316614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(EXIF_POINTER_TAGS[2].name,
316714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
316814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
316914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mAttributes[IFD_TYPE_INTEROPERABILITY].isEmpty()) {
317014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_EXIF].put(EXIF_POINTER_TAGS[3].name,
317114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
317214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
317314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mHasThumbnail) {
317414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_THUMBNAIL].put(JPEG_INTERCHANGE_FORMAT_TAG.name,
317514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(0, mExifByteOrder));
317614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_THUMBNAIL].put(JPEG_INTERCHANGE_FORMAT_LENGTH_TAG.name,
317714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(mThumbnailLength, mExifByteOrder));
317814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
317914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
318014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Calculate IFD group data area sizes. IFD group data area is assigned to save the entry
318114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // value which has a bigger size than 4 bytes.
318214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int i = 0; i < EXIF_TAGS.length; ++i) {
318314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int sum = 0;
318414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (Map.Entry entry : (Set<Map.Entry>) mAttributes[i].entrySet()) {
318514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                final ExifAttribute exifAttribute = (ExifAttribute) entry.getValue();
318614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                final int size = exifAttribute.size();
318714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (size > 4) {
318814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    sum += size;
318914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
319014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
319114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ifdDataSizes[i] += sum;
319214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
319314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
319414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Calculate IFD offsets.
319514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int position = 8;
319614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int ifdType = 0; ifdType < EXIF_TAGS.length; ++ifdType) {
319714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (!mAttributes[ifdType].isEmpty()) {
319814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ifdOffsets[ifdType] = position;
319914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                position += 2 + mAttributes[ifdType].size() * 12 + 4 + ifdDataSizes[ifdType];
320014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
320114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
320214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mHasThumbnail) {
320314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int thumbnailOffset = position;
320414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_THUMBNAIL].put(JPEG_INTERCHANGE_FORMAT_TAG.name,
320514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(thumbnailOffset, mExifByteOrder));
320614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mThumbnailOffset = exifOffsetFromBeginning + thumbnailOffset;
320714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            position += mThumbnailLength;
320814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
320914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
321014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Calculate the total size
321114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int totalSize = position + 8;  // eight bytes is for header part.
321214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (DEBUG) {
321314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "totalSize length: " + totalSize);
321414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (int i = 0; i < EXIF_TAGS.length; ++i) {
321514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, String.format("index: %d, offsets: %d, tag count: %d, data sizes: %d",
321614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        i, ifdOffsets[i], mAttributes[i].size(), ifdDataSizes[i]));
321714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
321814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
321914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
322014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Update IFD pointer tags with the calculated offsets.
322114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mAttributes[IFD_TYPE_EXIF].isEmpty()) {
322214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(EXIF_POINTER_TAGS[1].name,
322314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(ifdOffsets[IFD_TYPE_EXIF], mExifByteOrder));
322414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
322514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mAttributes[IFD_TYPE_GPS].isEmpty()) {
322614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_PRIMARY].put(EXIF_POINTER_TAGS[2].name,
322714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute.createULong(ifdOffsets[IFD_TYPE_GPS], mExifByteOrder));
322814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
322914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (!mAttributes[IFD_TYPE_INTEROPERABILITY].isEmpty()) {
323014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mAttributes[IFD_TYPE_EXIF].put(EXIF_POINTER_TAGS[3].name, ExifAttribute.createULong(
323114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ifdOffsets[IFD_TYPE_INTEROPERABILITY], mExifByteOrder));
323214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
323314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
323414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Write TIFF Headers. See JEITA CP-3451C Section 4.5.2. Table 1.
323514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeUnsignedShort(totalSize);
323614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.write(IDENTIFIER_EXIF_APP1);
323714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeShort(mExifByteOrder == ByteOrder.BIG_ENDIAN
323814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                ? BYTE_ALIGN_MM : BYTE_ALIGN_II);
323914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.setByteOrder(mExifByteOrder);
324014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeUnsignedShort(START_CODE);
324114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.writeUnsignedInt(IFD_OFFSET);
324214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
324314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Write IFD groups. See JEITA CP-3451C Section 4.5.8. Figure 9.
324414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        for (int ifdType = 0; ifdType < EXIF_TAGS.length; ++ifdType) {
324514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (!mAttributes[ifdType].isEmpty()) {
324614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // See JEITA CP-3451C Section 4.6.2: IFD structure.
324714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Write entry count
324814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                dataOutputStream.writeUnsignedShort(mAttributes[ifdType].size());
324914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
325014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Write entry info
325114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int dataOffset = ifdOffsets[ifdType] + 2 + mAttributes[ifdType].size() * 12 + 4;
325214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                for (Map.Entry entry : (Set<Map.Entry>) mAttributes[ifdType].entrySet()) {
325314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Convert tag name to tag number.
325414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    final ExifTag tag =
325514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            (ExifTag) sExifTagMapsForWriting[ifdType].get(entry.getKey());
325614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    final int tagNumber = tag.number;
325714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    final ExifAttribute attribute = (ExifAttribute) entry.getValue();
325814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    final int size = attribute.size();
325914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
326014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeUnsignedShort(tagNumber);
326114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeUnsignedShort(attribute.format);
326214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeInt(attribute.numberOfComponents);
326314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (size > 4) {
326414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        dataOutputStream.writeUnsignedInt(dataOffset);
326514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        dataOffset += size;
326614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    } else {
326714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        dataOutputStream.write(attribute.bytes);
326814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        // Fill zero up to 4 bytes
326914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        if (size < 4) {
327014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            for (int i = size; i < 4; ++i) {
327114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                                dataOutputStream.writeByte(0);
327214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                            }
327314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        }
327414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
327514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
327614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
327714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Write the next offset. It writes the offset of thumbnail IFD if there is one or
327814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // more tags in the thumbnail IFD when the current IFD is the primary image TIFF
327914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // IFD; Otherwise 0.
328014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (ifdType == 0 && !mAttributes[IFD_TYPE_THUMBNAIL].isEmpty()) {
328114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeUnsignedInt(ifdOffsets[IFD_TYPE_THUMBNAIL]);
328214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } else {
328314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataOutputStream.writeUnsignedInt(0);
328414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
328514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
328614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                // Write values of data field exceeding 4 bytes after the next offset.
328714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                for (Map.Entry entry : (Set<Map.Entry>) mAttributes[ifdType].entrySet()) {
328814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    ExifAttribute attribute = (ExifAttribute) entry.getValue();
328914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
329014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (attribute.bytes.length > 4) {
329114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        dataOutputStream.write(attribute.bytes, 0, attribute.bytes.length);
329214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
329314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
329414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
329514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
329614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
329714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Write thumbnail
329814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mHasThumbnail) {
329914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            dataOutputStream.write(getThumbnailBytes());
330014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
330114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
330214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Reset the byte order to big endian in order to write remaining parts of the JPEG file.
330314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        dataOutputStream.setByteOrder(ByteOrder.BIG_ENDIAN);
330414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
330514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return totalSize;
330614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
330714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
330814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
330914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Determines the data format of EXIF entry value.
331014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     *
331114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @param entryValue The value to be determined.
331214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * @return Returns two data formats gussed as a pair in integer. If there is no two candidate
331314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon               data formats for the given entry value, returns {@code -1} in the second of the pair.
331414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
331514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static Pair<Integer, Integer> guessDataFormat(String entryValue) {
331614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // See TIFF 6.0 Section 2, "Image File Directory".
331714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        // Take the first component if there are more than one component.
331814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (entryValue.contains(",")) {
331914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            String[] entryValues = entryValue.split(",");
332014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Pair<Integer, Integer> dataFormat = guessDataFormat(entryValues[0]);
332114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (dataFormat.first == IFD_FORMAT_STRING) {
332214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return dataFormat;
332314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
332414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            for (int i = 1; i < entryValues.length; ++i) {
332514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                final Pair<Integer, Integer> guessDataFormat = guessDataFormat(entryValues[i]);
332614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                int first = -1, second = -1;
332714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (guessDataFormat.first == dataFormat.first
332814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        || guessDataFormat.second == dataFormat.first) {
332914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    first = dataFormat.first;
333014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
333114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (dataFormat.second != -1 && (guessDataFormat.first == dataFormat.second
333214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        || guessDataFormat.second == dataFormat.second)) {
333314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    second = dataFormat.second;
333414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
333514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (first == -1 && second == -1) {
333614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return new Pair<>(IFD_FORMAT_STRING, -1);
333714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
333814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (first == -1) {
333914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataFormat = new Pair<>(second, -1);
334014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    continue;
334114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
334214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                if (second == -1) {
334314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    dataFormat = new Pair<>(first, -1);
334414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    continue;
334514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
334614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
334714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return dataFormat;
334814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
334914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
335014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (entryValue.contains("/")) {
335114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            String[] rationalNumber = entryValue.split("/");
335214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (rationalNumber.length == 2) {
335314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                try {
335414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    long numerator = Long.parseLong(rationalNumber[0]);
335514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    long denominator = Long.parseLong(rationalNumber[1]);
335614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (numerator < 0L || denominator < 0L) {
335714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return new Pair<>(IFD_FORMAT_SRATIONAL, -1);
335814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
335914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    if (numerator > Integer.MAX_VALUE || denominator > Integer.MAX_VALUE) {
336014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        return new Pair<>(IFD_FORMAT_URATIONAL, -1);
336114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    }
336214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    return new Pair<>(IFD_FORMAT_SRATIONAL, IFD_FORMAT_URATIONAL);
336314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                } catch (NumberFormatException e)  {
336414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    // Ignored
336514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                }
336614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
336714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new Pair<>(IFD_FORMAT_STRING, -1);
336814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
336914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
337014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Long longValue = Long.parseLong(entryValue);
337114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (longValue >= 0 && longValue <= 65535) {
337214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return new Pair<>(IFD_FORMAT_USHORT, IFD_FORMAT_ULONG);
337314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
337414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (longValue < 0) {
337514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return new Pair<>(IFD_FORMAT_SLONG, -1);
337614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
337714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new Pair<>(IFD_FORMAT_ULONG, -1);
337814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (NumberFormatException e) {
337914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Ignored
338014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
338114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        try {
338214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Double.parseDouble(entryValue);
338314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return new Pair<>(IFD_FORMAT_DOUBLE, -1);
338414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } catch (NumberFormatException e) {
338514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            // Ignored
338614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
338714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return new Pair<>(IFD_FORMAT_STRING, -1);
338814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
338914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
339014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // An input stream to parse EXIF data area, which can be written in either little or big endian
339114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // order.
339214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static class ByteOrderedDataInputStream extends InputStream implements DataInput {
339314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private static final ByteOrder LITTLE_ENDIAN = ByteOrder.LITTLE_ENDIAN;
339414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private static final ByteOrder BIG_ENDIAN = ByteOrder.BIG_ENDIAN;
339514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
339614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private DataInputStream mDataInputStream;
339714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private InputStream mInputStream;
339814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private ByteOrder mByteOrder = ByteOrder.BIG_ENDIAN;
339914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private final int mLength;
340014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private int mPosition;
340114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
340214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public ByteOrderedDataInputStream(InputStream in) throws IOException {
340314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mInputStream = in;
340414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mDataInputStream = new DataInputStream(in);
340514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mLength = mDataInputStream.available();
340614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition = 0;
340714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mDataInputStream.mark(mLength);
340814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
340914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
341014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public ByteOrderedDataInputStream(byte[] bytes) throws IOException {
341114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            this(new ByteArrayInputStream(bytes));
341214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
341314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
341414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void setByteOrder(ByteOrder byteOrder) {
341514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mByteOrder = byteOrder;
341614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
341714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
341814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void seek(long byteCount) throws IOException {
341914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > byteCount) {
342014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mPosition = 0;
342114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mDataInputStream.reset();
342214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mDataInputStream.mark(mLength);
342314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else {
342414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                byteCount -= mPosition;
342514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
342614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
342714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (skipBytes((int) byteCount) != (int) byteCount) {
342814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Couldn't seek up to the byteCount");
342914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
343014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
343114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
343214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int peek() {
343314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mPosition;
343414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
343514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
343614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
343714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int available() throws IOException {
343814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mDataInputStream.available();
343914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
344014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
344114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
344214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int read() throws IOException {
344314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ++mPosition;
344414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mDataInputStream.read();
344514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
344614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
344714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
344814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int readUnsignedByte() throws IOException {
344914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ++mPosition;
345014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mDataInputStream.readUnsignedByte();
345114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
345214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
345314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
345414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public String readLine() throws IOException {
345514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            Log.d(TAG, "Currently unsupported");
345614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return null;
345714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
345814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
345914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
346014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public boolean readBoolean() throws IOException {
346114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ++mPosition;
346214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mDataInputStream.readBoolean();
346314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
346414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
346514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
346614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public char readChar() throws IOException {
346714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += 2;
346814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mDataInputStream.readChar();
346914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
347014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
347114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
347214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public String readUTF() throws IOException {
347314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += 2;
347414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return mDataInputStream.readUTF();
347514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
347614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
347714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
347814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void readFully(byte[] buffer, int offset, int length) throws IOException {
347914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += length;
348014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > mLength) {
348114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
348214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
348314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mDataInputStream.read(buffer, offset, length) != length) {
348414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Couldn't read up to the length of buffer");
348514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
348614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
348714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
348814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
348914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void readFully(byte[] buffer) throws IOException {
349014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += buffer.length;
349114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > mLength) {
349214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
349314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
349414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mDataInputStream.read(buffer, 0, buffer.length) != buffer.length) {
349514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new IOException("Couldn't read up to the length of buffer");
349614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
349714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
349814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
349914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
350014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public byte readByte() throws IOException {
350114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            ++mPosition;
350214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > mLength) {
350314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
350414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
350514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch = mDataInputStream.read();
350614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (ch < 0) {
350714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
350814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
350914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return (byte) ch;
351014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
351114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
351214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
351314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public short readShort() throws IOException {
351414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += 2;
351514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > mLength) {
351614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
351714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
351814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch1 = mDataInputStream.read();
351914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch2 = mDataInputStream.read();
352014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if ((ch1 | ch2) < 0) {
352114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
352214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
352314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mByteOrder == LITTLE_ENDIAN) {
352414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return (short) ((ch2 << 8) + (ch1));
352514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mByteOrder == BIG_ENDIAN) {
352614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return (short) ((ch1 << 8) + (ch2));
352714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
352814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid byte order: " + mByteOrder);
352914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
353014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
353114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
353214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int readInt() throws IOException {
353314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += 4;
353414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > mLength) {
353514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
353614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
353714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch1 = mDataInputStream.read();
353814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch2 = mDataInputStream.read();
353914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch3 = mDataInputStream.read();
354014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch4 = mDataInputStream.read();
354114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if ((ch1 | ch2 | ch3 | ch4) < 0) {
354214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
354314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
354414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mByteOrder == LITTLE_ENDIAN) {
354514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return ((ch4 << 24) + (ch3 << 16) + (ch2 << 8) + ch1);
354614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mByteOrder == BIG_ENDIAN) {
354714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return ((ch1 << 24) + (ch2 << 16) + (ch3 << 8) + ch4);
354814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
354914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid byte order: " + mByteOrder);
355014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
355114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
355214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
355314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int skipBytes(int byteCount) throws IOException {
355414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int totalSkip = Math.min(byteCount, mLength - mPosition);
355514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int skipped = 0;
355614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            while (skipped < totalSkip) {
355714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                skipped += mDataInputStream.skipBytes(totalSkip - skipped);
355814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
355914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += skipped;
356014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return skipped;
356114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
356214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
356314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public int readUnsignedShort() throws IOException {
356414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += 2;
356514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > mLength) {
356614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
356714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
356814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch1 = mDataInputStream.read();
356914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch2 = mDataInputStream.read();
357014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if ((ch1 | ch2) < 0) {
357114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
357214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
357314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mByteOrder == LITTLE_ENDIAN) {
357414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return ((ch2 << 8) + (ch1));
357514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mByteOrder == BIG_ENDIAN) {
357614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return ((ch1 << 8) + (ch2));
357714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
357814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid byte order: " + mByteOrder);
357914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
358014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
358114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public long readUnsignedInt() throws IOException {
358214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return readInt() & 0xffffffffL;
358314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
358414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
358514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
358614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public long readLong() throws IOException {
358714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mPosition += 8;
358814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mPosition > mLength) {
358914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
359014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
359114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch1 = mDataInputStream.read();
359214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch2 = mDataInputStream.read();
359314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch3 = mDataInputStream.read();
359414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch4 = mDataInputStream.read();
359514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch5 = mDataInputStream.read();
359614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch6 = mDataInputStream.read();
359714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch7 = mDataInputStream.read();
359814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int ch8 = mDataInputStream.read();
359914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if ((ch1 | ch2 | ch3 | ch4 | ch5 | ch6 | ch7 | ch8) < 0) {
360014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw new EOFException();
360114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
360214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mByteOrder == LITTLE_ENDIAN) {
360314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return (((long) ch8 << 56) + ((long) ch7 << 48) + ((long) ch6 << 40)
360414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + ((long) ch5 << 32) + ((long) ch4 << 24) + ((long) ch3 << 16)
360514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + ((long) ch2 << 8) + (long) ch1);
360614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mByteOrder == BIG_ENDIAN) {
360714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                return (((long) ch1 << 56) + ((long) ch2 << 48) + ((long) ch3 << 40)
360814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + ((long) ch4 << 32) + ((long) ch5 << 24) + ((long) ch6 << 16)
360914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                        + ((long) ch7 << 8) + (long) ch8);
361014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
361114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throw new IOException("Invalid byte order: " + mByteOrder);
361214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
361314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
361414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
361514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public float readFloat() throws IOException {
361614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return Float.intBitsToFloat(readInt());
361714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
361814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
361914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        @Override
362014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public double readDouble() throws IOException {
362114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return Double.longBitsToDouble(readLong());
362214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
362314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
362414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
362514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // An output stream to write EXIF data area, which can be written in either little or big endian
362614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // order.
362714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static class ByteOrderedDataOutputStream extends FilterOutputStream {
362814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private final OutputStream mOutputStream;
362914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        private ByteOrder mByteOrder;
363014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
363114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public ByteOrderedDataOutputStream(OutputStream out, ByteOrder byteOrder) {
363214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            super(out);
363314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mOutputStream = out;
363414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mByteOrder = byteOrder;
363514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
363614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
363714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void setByteOrder(ByteOrder byteOrder) {
363814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mByteOrder = byteOrder;
363914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
364014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
364114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void write(byte[] bytes) throws IOException {
364214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mOutputStream.write(bytes);
364314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
364414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
364514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void write(byte[] bytes, int offset, int length) throws IOException {
364614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mOutputStream.write(bytes, offset, length);
364714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
364814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
364914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void writeByte(int val) throws IOException {
365014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            mOutputStream.write(val);
365114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
365214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
365314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void writeShort(short val) throws IOException {
365414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mByteOrder == ByteOrder.LITTLE_ENDIAN) {
365514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 0) & 0xFF);
365614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 8) & 0xFF);
365714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mByteOrder == ByteOrder.BIG_ENDIAN) {
365814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 8) & 0xFF);
365914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 0) & 0xFF);
366014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
366114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
366214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
366314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void writeInt(int val) throws IOException {
366414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (mByteOrder == ByteOrder.LITTLE_ENDIAN) {
366514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 0) & 0xFF);
366614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 8) & 0xFF);
366714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 16) & 0xFF);
366814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 24) & 0xFF);
366914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } else if (mByteOrder == ByteOrder.BIG_ENDIAN) {
367014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 24) & 0xFF);
367114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 16) & 0xFF);
367214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 8) & 0xFF);
367314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mOutputStream.write((val >>> 0) & 0xFF);
367414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
367514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
367614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
367714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void writeUnsignedShort(int val) throws IOException {
367814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            writeShort((short) val);
367914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
368014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
368114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        public void writeUnsignedInt(long val) throws IOException {
368214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            writeInt((int) val);
368314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
368414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
368514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
368614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    // Swaps image data based on image size
368714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private void swapBasedOnImageSize(@IfdType int firstIfdType, @IfdType int secondIfdType)
368814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            throws IOException {
368914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (mAttributes[firstIfdType].isEmpty() || mAttributes[secondIfdType].isEmpty()) {
369014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
369114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "Cannot perform swap since only one image data exists");
369214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
369314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            return;
369414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
369514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
369614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute firstImageLengthAttribute =
369714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[firstIfdType].get(TAG_IMAGE_LENGTH);
369814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute firstImageWidthAttribute =
369914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[firstIfdType].get(TAG_IMAGE_WIDTH);
370014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute secondImageLengthAttribute =
370114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[secondIfdType].get(TAG_IMAGE_LENGTH);
370214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        ExifAttribute secondImageWidthAttribute =
370314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                (ExifAttribute) mAttributes[secondIfdType].get(TAG_IMAGE_WIDTH);
370414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
370514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (firstImageLengthAttribute == null || firstImageWidthAttribute == null) {
370614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
370714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "First image does not contain valid size information");
370814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
370914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else if (secondImageLengthAttribute == null || secondImageWidthAttribute == null) {
371014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (DEBUG) {
371114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                Log.d(TAG, "Second image does not contain valid size information");
371214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
371314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        } else {
371414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int firstImageLengthValue = firstImageLengthAttribute.getIntValue(mExifByteOrder);
371514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int firstImageWidthValue = firstImageWidthAttribute.getIntValue(mExifByteOrder);
371614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int secondImageLengthValue = secondImageLengthAttribute.getIntValue(mExifByteOrder);
371714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            int secondImageWidthValue = secondImageWidthAttribute.getIntValue(mExifByteOrder);
371814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
371914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            if (firstImageLengthValue < secondImageLengthValue &&
372014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                    firstImageWidthValue < secondImageWidthValue) {
372114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                HashMap tempMap = mAttributes[firstIfdType];
372214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[firstIfdType] = mAttributes[secondIfdType];
372314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                mAttributes[secondIfdType] = tempMap;
372414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
372514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
372614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
372714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
372814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
372914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Closes 'closeable', ignoring any checked exceptions. Does nothing if 'closeable' is null.
373014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
373114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static void closeQuietly(Closeable closeable) {
373214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        if (closeable != null) {
373314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            try {
373414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                closeable.close();
373514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } catch (RuntimeException rethrown) {
373614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon                throw rethrown;
373714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            } catch (Exception ignored) {
373814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            }
373914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
374014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
374114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon
374214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    /**
374314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Copies all of the bytes from {@code in} to {@code out}. Neither stream is closed.
374414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     * Returns the total number of bytes transferred.
374514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon     */
374614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    private static int copy(InputStream in, OutputStream out) throws IOException {
374714ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int total = 0;
374814ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        byte[] buffer = new byte[8192];
374914ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        int c;
375014ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        while ((c = in.read(buffer)) != -1) {
375114ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            total += c;
375214ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon            out.write(buffer, 0, c);
375314ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        }
375414ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon        return total;
375514ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon    }
375614ea3100458369949fffba9978db2a51dde21bf2Hyundo Moon}
3757