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