| /* |
| * TIFF Common Routines |
| * 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 |
| * TIFF Common Routines |
| * @author Thilo Borgmann <thilo.borgmann _at_ mail.de> |
| */ |
| |
| #include "tiff_common.h" |
| |
| |
| int ff_tis_ifd(unsigned tag) |
| { |
| int i; |
| for (i = 0; i < FF_ARRAY_ELEMS(ifd_tags); i++) { |
| if (ifd_tags[i] == tag) { |
| return i + 1; |
| } |
| } |
| return 0; |
| } |
| |
| |
| unsigned ff_tget_short(GetByteContext *gb, int le) |
| { |
| return le ? bytestream2_get_le16(gb) : bytestream2_get_be16(gb); |
| } |
| |
| |
| unsigned ff_tget_long(GetByteContext *gb, int le) |
| { |
| return le ? bytestream2_get_le32(gb) : bytestream2_get_be32(gb); |
| } |
| |
| |
| double ff_tget_double(GetByteContext *gb, int le) |
| { |
| av_alias64 i = { .u64 = le ? bytestream2_get_le64(gb) : bytestream2_get_be64(gb)}; |
| return i.f64; |
| } |
| |
| |
| unsigned ff_tget(GetByteContext *gb, int type, int le) |
| { |
| switch (type) { |
| case TIFF_BYTE: return bytestream2_get_byte(gb); |
| case TIFF_SHORT: return ff_tget_short(gb, le); |
| case TIFF_LONG: return ff_tget_long(gb, le); |
| default: return UINT_MAX; |
| } |
| } |
| |
| static const char *auto_sep(int count, const char *sep, int i, int columns) |
| { |
| if (sep) |
| return i ? sep : ""; |
| if (i && i%columns) { |
| return ", "; |
| } else |
| return columns < count ? "\n" : ""; |
| } |
| |
| int ff_tadd_rational_metadata(int count, const char *name, const char *sep, |
| GetByteContext *gb, int le, AVDictionary **metadata) |
| { |
| AVBPrint bp; |
| char *ap; |
| int32_t nom, denom; |
| int i; |
| |
| if (count >= INT_MAX / sizeof(int64_t) || count <= 0) |
| return AVERROR_INVALIDDATA; |
| if (bytestream2_get_bytes_left(gb) < count * sizeof(int64_t)) |
| return AVERROR_INVALIDDATA; |
| |
| av_bprint_init(&bp, 10 * count, AV_BPRINT_SIZE_UNLIMITED); |
| |
| for (i = 0; i < count; i++) { |
| nom = ff_tget_long(gb, le); |
| denom = ff_tget_long(gb, le); |
| av_bprintf(&bp, "%s%7i:%-7i", auto_sep(count, sep, i, 4), nom, denom); |
| } |
| |
| if ((i = av_bprint_finalize(&bp, &ap))) { |
| return i; |
| } |
| if (!ap) { |
| return AVERROR(ENOMEM); |
| } |
| |
| av_dict_set(metadata, name, ap, AV_DICT_DONT_STRDUP_VAL); |
| |
| return 0; |
| } |
| |
| |
| int ff_tadd_long_metadata(int count, const char *name, const char *sep, |
| GetByteContext *gb, int le, AVDictionary **metadata) |
| { |
| AVBPrint bp; |
| char *ap; |
| int i; |
| |
| if (count >= INT_MAX / sizeof(int32_t) || count <= 0) |
| return AVERROR_INVALIDDATA; |
| if (bytestream2_get_bytes_left(gb) < count * sizeof(int32_t)) |
| return AVERROR_INVALIDDATA; |
| |
| av_bprint_init(&bp, 10 * count, AV_BPRINT_SIZE_UNLIMITED); |
| |
| for (i = 0; i < count; i++) { |
| av_bprintf(&bp, "%s%7i", auto_sep(count, sep, i, 8), ff_tget_long(gb, le)); |
| } |
| |
| if ((i = av_bprint_finalize(&bp, &ap))) { |
| return i; |
| } |
| if (!ap) { |
| return AVERROR(ENOMEM); |
| } |
| |
| av_dict_set(metadata, name, ap, AV_DICT_DONT_STRDUP_VAL); |
| |
| return 0; |
| } |
| |
| |
| int ff_tadd_doubles_metadata(int count, const char *name, const char *sep, |
| GetByteContext *gb, int le, AVDictionary **metadata) |
| { |
| AVBPrint bp; |
| char *ap; |
| int i; |
| |
| if (count >= INT_MAX / sizeof(int64_t) || count <= 0) |
| return AVERROR_INVALIDDATA; |
| if (bytestream2_get_bytes_left(gb) < count * sizeof(int64_t)) |
| return AVERROR_INVALIDDATA; |
| |
| av_bprint_init(&bp, 10 * count, 100 * count); |
| |
| for (i = 0; i < count; i++) { |
| av_bprintf(&bp, "%s%.15g", auto_sep(count, sep, i, 4), ff_tget_double(gb, le)); |
| } |
| |
| if ((i = av_bprint_finalize(&bp, &ap))) { |
| return i; |
| } |
| if (!ap) { |
| return AVERROR(ENOMEM); |
| } |
| |
| av_dict_set(metadata, name, ap, AV_DICT_DONT_STRDUP_VAL); |
| |
| return 0; |
| } |
| |
| |
| int ff_tadd_shorts_metadata(int count, const char *name, const char *sep, |
| GetByteContext *gb, int le, int is_signed, AVDictionary **metadata) |
| { |
| AVBPrint bp; |
| char *ap; |
| int i; |
| |
| if (count >= INT_MAX / sizeof(int16_t) || count <= 0) |
| return AVERROR_INVALIDDATA; |
| if (bytestream2_get_bytes_left(gb) < count * sizeof(int16_t)) |
| return AVERROR_INVALIDDATA; |
| |
| av_bprint_init(&bp, 10 * count, AV_BPRINT_SIZE_UNLIMITED); |
| |
| for (i = 0; i < count; i++) { |
| int v = is_signed ? (int16_t)ff_tget_short(gb, le) : ff_tget_short(gb, le); |
| av_bprintf(&bp, "%s%5i", auto_sep(count, sep, i, 8), v); |
| } |
| |
| if ((i = av_bprint_finalize(&bp, &ap))) { |
| return i; |
| } |
| if (!ap) { |
| return AVERROR(ENOMEM); |
| } |
| |
| av_dict_set(metadata, name, ap, AV_DICT_DONT_STRDUP_VAL); |
| |
| return 0; |
| } |
| |
| |
| int ff_tadd_bytes_metadata(int count, const char *name, const char *sep, |
| GetByteContext *gb, int le, int is_signed, AVDictionary **metadata) |
| { |
| AVBPrint bp; |
| char *ap; |
| int i; |
| |
| if (count >= INT_MAX / sizeof(int8_t) || count < 0) |
| return AVERROR_INVALIDDATA; |
| if (bytestream2_get_bytes_left(gb) < count * sizeof(int8_t)) |
| return AVERROR_INVALIDDATA; |
| |
| av_bprint_init(&bp, 10 * count, AV_BPRINT_SIZE_UNLIMITED); |
| |
| for (i = 0; i < count; i++) { |
| int v = is_signed ? (int8_t)bytestream2_get_byte(gb) : bytestream2_get_byte(gb); |
| av_bprintf(&bp, "%s%3i", auto_sep(count, sep, i, 16), v); |
| } |
| |
| if ((i = av_bprint_finalize(&bp, &ap))) { |
| return i; |
| } |
| if (!ap) { |
| return AVERROR(ENOMEM); |
| } |
| |
| av_dict_set(metadata, name, ap, AV_DICT_DONT_STRDUP_VAL); |
| |
| return 0; |
| } |
| |
| int ff_tadd_string_metadata(int count, const char *name, |
| GetByteContext *gb, int le, AVDictionary **metadata) |
| { |
| char *value; |
| |
| if (bytestream2_get_bytes_left(gb) < count || count < 0) |
| return AVERROR_INVALIDDATA; |
| |
| value = av_malloc(count + 1); |
| if (!value) |
| return AVERROR(ENOMEM); |
| |
| bytestream2_get_bufferu(gb, value, count); |
| value[count] = 0; |
| |
| av_dict_set(metadata, name, value, AV_DICT_DONT_STRDUP_VAL); |
| return 0; |
| } |
| |
| |
| int ff_tdecode_header(GetByteContext *gb, int *le, int *ifd_offset) |
| { |
| if (bytestream2_get_bytes_left(gb) < 8) { |
| return AVERROR_INVALIDDATA; |
| } |
| |
| *le = bytestream2_get_le16u(gb); |
| if (*le == AV_RB16("II")) { |
| *le = 1; |
| } else if (*le == AV_RB16("MM")) { |
| *le = 0; |
| } else { |
| return AVERROR_INVALIDDATA; |
| } |
| |
| if (ff_tget_short(gb, *le) != 42) { |
| return AVERROR_INVALIDDATA; |
| } |
| |
| *ifd_offset = ff_tget_long(gb, *le); |
| |
| return 0; |
| } |
| |
| |
| int ff_tread_tag(GetByteContext *gb, int le, unsigned *tag, unsigned *type, |
| unsigned *count, int *next) |
| { |
| int ifd_tag; |
| int valid_type; |
| |
| *tag = ff_tget_short(gb, le); |
| *type = ff_tget_short(gb, le); |
| *count = ff_tget_long (gb, le); |
| |
| ifd_tag = ff_tis_ifd(*tag); |
| valid_type = *type != 0 && *type < FF_ARRAY_ELEMS(type_sizes); |
| |
| *next = bytestream2_tell(gb) + 4; |
| |
| // check for valid type |
| if (!valid_type) { |
| return AVERROR_INVALIDDATA; |
| } |
| |
| // seek to offset if this is an IFD-tag or |
| // if count values do not fit into the offset value |
| if (ifd_tag || (*count > 4 || !(type_sizes[*type] * (*count) <= 4 || *type == TIFF_STRING))) { |
| bytestream2_seek(gb, ff_tget_long (gb, le), SEEK_SET); |
| } |
| |
| return 0; |
| } |