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