| /* |
| * EXIF metadata parser |
| * Copyright (c) 2013 Thilo Borgmann <thilo.borgmann _at_ mail.de> |
| * |
| * This file is part of FFmpeg. |
| * |
| * FFmpeg is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * FFmpeg is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with FFmpeg; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| */ |
| |
| /** |
| * @file |
| * EXIF metadata parser |
| * @author Thilo Borgmann <thilo.borgmann _at_ mail.de> |
| */ |
| |
| #include "exif.h" |
| #include "tiff_common.h" |
| |
| #define EXIF_TAG_NAME_LENGTH 32 |
| |
| struct exif_tag { |
| char name[EXIF_TAG_NAME_LENGTH]; |
| uint16_t id; |
| }; |
| |
| static const struct exif_tag tag_list[] = { // JEITA CP-3451 EXIF specification: |
| {"GPSVersionID", 0x00}, // <- Table 12 GPS Attribute Information |
| {"GPSLatitudeRef", 0x01}, |
| {"GPSLatitude", 0x02}, |
| {"GPSLongitudeRef", 0x03}, |
| {"GPSLongitude", 0x04}, |
| {"GPSAltitudeRef", 0x05}, |
| {"GPSAltitude", 0x06}, |
| {"GPSTimeStamp", 0x07}, |
| {"GPSSatellites", 0x08}, |
| {"GPSStatus", 0x09}, |
| {"GPSMeasureMode", 0x0A}, |
| {"GPSDOP", 0x0B}, |
| {"GPSSpeedRef", 0x0C}, |
| {"GPSSpeed", 0x0D}, |
| {"GPSTrackRef", 0x0E}, |
| {"GPSTrack", 0x0F}, |
| {"GPSImgDirectionRef", 0x10}, |
| {"GPSImgDirection", 0x11}, |
| {"GPSMapDatum", 0x12}, |
| {"GPSDestLatitudeRef", 0x13}, |
| {"GPSDestLatitude", 0x14}, |
| {"GPSDestLongitudeRef", 0x15}, |
| {"GPSDestLongitude", 0x16}, |
| {"GPSDestBearingRef", 0x17}, |
| {"GPSDestBearing", 0x18}, |
| {"GPSDestDistanceRef", 0x19}, |
| {"GPSDestDistance", 0x1A}, |
| {"GPSProcessingMethod", 0x1B}, |
| {"GPSAreaInformation", 0x1C}, |
| {"GPSDateStamp", 0x1D}, |
| {"GPSDifferential", 0x1E}, |
| {"ImageWidth", 0x100}, // <- Table 3 TIFF Rev. 6.0 Attribute Information Used in Exif |
| {"ImageLength", 0x101}, |
| {"BitsPerSample", 0x102}, |
| {"Compression", 0x103}, |
| {"PhotometricInterpretation", 0x106}, |
| {"Orientation", 0x112}, |
| {"SamplesPerPixel", 0x115}, |
| {"PlanarConfiguration", 0x11C}, |
| {"YCbCrSubSampling", 0x212}, |
| {"YCbCrPositioning", 0x213}, |
| {"XResolution", 0x11A}, |
| {"YResolution", 0x11B}, |
| {"ResolutionUnit", 0x128}, |
| {"StripOffsets", 0x111}, |
| {"RowsPerStrip", 0x116}, |
| {"StripByteCounts", 0x117}, |
| {"JPEGInterchangeFormat", 0x201}, |
| {"JPEGInterchangeFormatLength",0x202}, |
| {"TransferFunction", 0x12D}, |
| {"WhitePoint", 0x13E}, |
| {"PrimaryChromaticities", 0x13F}, |
| {"YCbCrCoefficients", 0x211}, |
| {"ReferenceBlackWhite", 0x214}, |
| {"DateTime", 0x132}, |
| {"ImageDescription", 0x10E}, |
| {"Make", 0x10F}, |
| {"Model", 0x110}, |
| {"Software", 0x131}, |
| {"Artist", 0x13B}, |
| {"Copyright", 0x8298}, |
| {"ExifVersion", 0x9000}, // <- Table 4 Exif IFD Attribute Information (1) |
| {"FlashpixVersion", 0xA000}, |
| {"ColorSpace", 0xA001}, |
| {"ComponentsConfiguration", 0x9101}, |
| {"CompressedBitsPerPixel", 0x9102}, |
| {"PixelXDimension", 0xA002}, |
| {"PixelYDimension", 0xA003}, |
| {"MakerNote", 0x927C}, |
| {"UserComment", 0x9286}, |
| {"RelatedSoundFile", 0xA004}, |
| {"DateTimeOriginal", 0x9003}, |
| {"DateTimeDigitized", 0x9004}, |
| {"SubSecTime", 0x9290}, |
| {"SubSecTimeOriginal", 0x9291}, |
| {"SubSecTimeDigitized", 0x9292}, |
| {"ImageUniqueID", 0xA420}, |
| {"ExposureTime", 0x829A}, // <- Table 5 Exif IFD Attribute Information (2) |
| {"FNumber", 0x829D}, |
| {"ExposureProgram", 0x8822}, |
| {"SpectralSensitivity", 0x8824}, |
| {"ISOSpeedRatings", 0x8827}, |
| {"OECF", 0x8828}, |
| {"ShutterSpeedValue", 0x9201}, |
| {"ApertureValue", 0x9202}, |
| {"BrightnessValue", 0x9203}, |
| {"ExposureBiasValue", 0x9204}, |
| {"MaxApertureValue", 0x9205}, |
| {"SubjectDistance", 0x9206}, |
| {"MeteringMode", 0x9207}, |
| {"LightSource", 0x9208}, |
| {"Flash", 0x9209}, |
| {"FocalLength", 0x920A}, |
| {"SubjectArea", 0x9214}, |
| {"FlashEnergy", 0xA20B}, |
| {"SpatialFrequencyResponse", 0xA20C}, |
| {"FocalPlaneXResolution", 0xA20E}, |
| {"FocalPlaneYResolution", 0xA20F}, |
| {"FocalPlaneResolutionUnit", 0xA210}, |
| {"SubjectLocation", 0xA214}, |
| {"ExposureIndex", 0xA215}, |
| {"SensingMethod", 0xA217}, |
| {"FileSource", 0xA300}, |
| {"SceneType", 0xA301}, |
| {"CFAPattern", 0xA302}, |
| {"CustomRendered", 0xA401}, |
| {"ExposureMode", 0xA402}, |
| {"WhiteBalance", 0xA403}, |
| {"DigitalZoomRatio", 0xA404}, |
| {"FocalLengthIn35mmFilm", 0xA405}, |
| {"SceneCaptureType", 0xA406}, |
| {"GainControl", 0xA407}, |
| {"Contrast", 0xA408}, |
| {"Saturation", 0xA409}, |
| {"Sharpness", 0xA40A}, |
| {"DeviceSettingDescription", 0xA40B}, |
| {"SubjectDistanceRange", 0xA40C} |
| // {"InteroperabilityIndex", 0x1}, // <- Table 13 Interoperability IFD Attribute Information |
| }; |
| |
| static const char *exif_get_tag_name(uint16_t id) |
| { |
| int i; |
| |
| for (i = 0; i < FF_ARRAY_ELEMS(tag_list); i++) { |
| if (tag_list[i].id == id) |
| return tag_list[i].name; |
| } |
| |
| return NULL; |
| } |
| |
| |
| static int exif_add_metadata(void *logctx, int count, int type, |
| const char *name, const char *sep, |
| GetByteContext *gb, int le, |
| AVDictionary **metadata) |
| { |
| switch(type) { |
| case 0: |
| av_log(logctx, AV_LOG_WARNING, |
| "Invalid TIFF tag type 0 found for %s with size %d\n", |
| name, count); |
| return 0; |
| case TIFF_DOUBLE : return ff_tadd_doubles_metadata(count, name, sep, gb, le, metadata); |
| case TIFF_SSHORT : return ff_tadd_shorts_metadata(count, name, sep, gb, le, 1, metadata); |
| case TIFF_SHORT : return ff_tadd_shorts_metadata(count, name, sep, gb, le, 0, metadata); |
| case TIFF_SBYTE : return ff_tadd_bytes_metadata(count, name, sep, gb, le, 1, metadata); |
| case TIFF_BYTE : |
| case TIFF_UNDEFINED: return ff_tadd_bytes_metadata(count, name, sep, gb, le, 0, metadata); |
| case TIFF_STRING : return ff_tadd_string_metadata(count, name, gb, le, metadata); |
| case TIFF_SRATIONAL: |
| case TIFF_RATIONAL : return ff_tadd_rational_metadata(count, name, sep, gb, le, metadata); |
| case TIFF_SLONG : |
| case TIFF_LONG : return ff_tadd_long_metadata(count, name, sep, gb, le, metadata); |
| default: |
| avpriv_request_sample(logctx, "TIFF tag type (%u)", type); |
| return 0; |
| }; |
| } |
| |
| |
| static int exif_decode_tag(void *logctx, GetByteContext *gbytes, int le, |
| int depth, AVDictionary **metadata) |
| { |
| int ret, cur_pos; |
| unsigned id, count; |
| enum TiffTypes type; |
| |
| if (depth > 2) { |
| return 0; |
| } |
| |
| ff_tread_tag(gbytes, le, &id, &type, &count, &cur_pos); |
| |
| if (!bytestream2_tell(gbytes)) { |
| bytestream2_seek(gbytes, cur_pos, SEEK_SET); |
| return 0; |
| } |
| |
| // read count values and add it metadata |
| // store metadata or proceed with next IFD |
| ret = ff_tis_ifd(id); |
| if (ret) { |
| ret = ff_exif_decode_ifd(logctx, gbytes, le, depth + 1, metadata); |
| } else { |
| const char *name = exif_get_tag_name(id); |
| char buf[7]; |
| |
| if (!name) { |
| name = buf; |
| snprintf(buf, sizeof(buf), "0x%04X", id); |
| } |
| |
| ret = exif_add_metadata(logctx, count, type, name, NULL, |
| gbytes, le, metadata); |
| } |
| |
| bytestream2_seek(gbytes, cur_pos, SEEK_SET); |
| |
| return ret; |
| } |
| |
| |
| int ff_exif_decode_ifd(void *logctx, GetByteContext *gbytes, |
| int le, int depth, AVDictionary **metadata) |
| { |
| int i, ret; |
| int entries; |
| |
| entries = ff_tget_short(gbytes, le); |
| |
| if (bytestream2_get_bytes_left(gbytes) < entries * 12) { |
| return AVERROR_INVALIDDATA; |
| } |
| |
| for (i = 0; i < entries; i++) { |
| if ((ret = exif_decode_tag(logctx, gbytes, le, depth, metadata)) < 0) { |
| return ret; |
| } |
| } |
| |
| // return next IDF offset or 0x000000000 or a value < 0 for failure |
| return ff_tget_long(gbytes, le); |
| } |
| |
| int avpriv_exif_decode_ifd(void *logctx, const uint8_t *buf, int size, |
| int le, int depth, AVDictionary **metadata) |
| { |
| GetByteContext gb; |
| |
| bytestream2_init(&gb, buf, size); |
| |
| return ff_exif_decode_ifd(logctx, &gb, le, depth, metadata); |
| } |