Mercurial > geeqie
view src/format_raw.c @ 1701:2309e6150e8c
do not try to reload deleted image
deleted image is replaced by the next one from the list, keep
the old image displayed until the new one is set
author | nadvornik |
---|---|
date | Fri, 24 Jul 2009 12:56:54 +0000 |
parents | bc3f5c0432f6 |
children | 956aab097ea7 |
line wrap: on
line source
/* * Geeqie * (C) 2006 John Ellis * Copyright (C) 2008 - 2009 The Geeqie Team * * Authors: * Original version 2005 Lars Ellenberg, base on dcraw by David coffin. * * This software is released under the GNU General Public License (GNU GPL). * Please read the included file COPYING for more information. * This software comes with no warranty of any kind, use at your own risk! */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #ifndef HAVE_EXIV2 #include <stdio.h> #include <string.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <sys/mman.h> #include <glib.h> #include "intl.h" #include "main.h" #include "format_raw.h" #include "format_canon.h" #include "format_fuji.h" #include "format_nikon.h" #include "format_olympus.h" typedef struct _FormatRawEntry FormatRawEntry; struct _FormatRawEntry { const gchar *extension; FormatRawMatchType magic_type; const guint magic_offset; gconstpointer magic_pattern; const guint magic_length; const FormatRawExifType exif_type; FormatRawExifParseFunc exif_func; const gchar *description; FormatRawParseFunc func_parse; }; static FormatRawEntry format_raw_list[] = { #if DEBUG_RAW_TIFF FORMAT_RAW_DEBUG_TIFF, #endif FORMAT_RAW_CANON, FORMAT_RAW_FUJI, FORMAT_RAW_NIKON, FORMAT_RAW_OLYMPUS, FORMAT_RAW_PENTAX, FORMAT_RAW_SAMSUNG, { NULL, 0, 0, NULL, 0, 0, NULL, NULL, NULL } }; typedef struct _FormatExifEntry FormatExifEntry; struct _FormatExifEntry { FormatExifMatchType header_type; gconstpointer header_pattern; const guint header_length; const gchar *description; FormatExifParseFunc func_parse; }; static FormatExifEntry format_exif_list[] = { FORMAT_EXIF_CANON, FORMAT_EXIF_FUJI, FORMAT_EXIF_NIKON, FORMAT_EXIF_OLYMPUS, { 0, NULL, 0, NULL, NULL } }; static guint tiff_table(guchar *data, const guint len, guint offset, ExifByteOrder bo, guint tag, ExifFormatType type, guint *result_offset, guint *result_count) { guint count; guint i; if (len < offset + 2) return 0; if (type > EXIF_FORMAT_COUNT) return 0; count = exif_byte_get_int16(data + offset, bo); offset += 2; if (len < offset + count * 12 + 4) return 0; for (i = 0; i < count; i++) { guint segment; segment = offset + i * 12; if (exif_byte_get_int16(data + segment, bo) == tag && exif_byte_get_int16(data + segment + 2, bo) == type) { guint chunk_count; guint chunk_offset; guint chunk_length; chunk_count = exif_byte_get_int32(data + segment + 4, bo); chunk_length = ExifFormatList[type].size * chunk_count; if (chunk_length > 4) { chunk_offset = exif_byte_get_int32(data + segment + 8, bo); } else { chunk_offset = segment + 8; } if (chunk_offset + chunk_length <= len) { *result_offset = chunk_offset; *result_count = chunk_count; } return 0; } } return exif_byte_get_int32(data + offset + count * 12, bo); } static gboolean format_tiff_find_tag_data(guchar *data, const guint len, guint tag, ExifFormatType type, guint *result_offset, guint *result_count) { ExifByteOrder bo; guint offset; if (len < 8) return FALSE; if (memcmp(data, "II", 2) == 0) { bo = EXIF_BYTE_ORDER_INTEL; } else if (memcmp(data, "MM", 2) == 0) { bo = EXIF_BYTE_ORDER_MOTOROLA; } else { return FALSE; } if (exif_byte_get_int16(data + 2, bo) != 0x002A) { return FALSE; } offset = exif_byte_get_int32(data + 4, bo); while (offset != 0) { guint ro = 0; guint rc = 0; offset = tiff_table(data, len, offset, bo, tag, type, &ro, &rc); if (ro != 0) { *result_offset = ro; *result_count = rc; return TRUE; } } return FALSE; } static FormatRawEntry *format_raw_find(guchar *data, const guint len) { gint n; gboolean tiff; guint make_count = 0; guint make_offset = 0; tiff = (len > 8 && (memcmp(data, "II\x2a\x00", 4) == 0 || memcmp(data, "MM\x00\x2a", 4) == 0)); n = 0; while (format_raw_list[n].magic_pattern) { FormatRawEntry *entry = &format_raw_list[n]; switch (entry->magic_type) { case FORMAT_RAW_MATCH_MAGIC: if (entry->magic_length + entry->magic_offset <= len && memcmp(data + entry->magic_offset, entry->magic_pattern, entry->magic_length) == 0) { return entry; } break; case FORMAT_RAW_MATCH_TIFF_MAKE: if (tiff && make_offset == 0 && !format_tiff_find_tag_data(data, len, 0x10f, EXIF_FORMAT_STRING, &make_offset, &make_count)) { tiff = FALSE; } if (make_offset != 0 && make_count >= entry->magic_offset + entry->magic_length && memcmp(entry->magic_pattern, data + make_offset + entry->magic_offset, entry->magic_length) == 0) { return entry; } break; default: break; } n++; } return NULL; } static gboolean format_raw_parse(FormatRawEntry *entry, guchar *data, const guint len, guint *image_offset, guint *exif_offset) { guint io = 0; guint eo = 0; gboolean found; if (!entry || !entry->func_parse) return FALSE; DEBUG_1("RAW using file parser for %s", entry->description); found = entry->func_parse(data, len, &io, &eo); if (!found || io >= len - 4 || eo >= len) { return FALSE; } if (image_offset) *image_offset = io; if (exif_offset) *exif_offset = eo; return TRUE; } gboolean format_raw_img_exif_offsets(guchar *data, const guint len, guint *image_offset, guint *exif_offset) { FormatRawEntry *entry; if (!data || len < 1) return FALSE; entry = format_raw_find(data, len); if (!entry || !entry->func_parse) return FALSE; return format_raw_parse(entry, data, len, image_offset, exif_offset); } FormatRawExifType format_raw_exif_offset(guchar *data, const guint len, guint *exif_offset, FormatRawExifParseFunc *exif_parse_func) { FormatRawEntry *entry; if (!data || len < 1) return FALSE; entry = format_raw_find(data, len); if (!entry || !entry->func_parse) return FALSE; if (!format_raw_parse(entry, data, len, NULL, exif_offset)) return FORMAT_RAW_EXIF_NONE; if (entry->exif_type == FORMAT_RAW_EXIF_PROPRIETARY && exif_parse_func) { *exif_parse_func = entry->exif_func; } return entry->exif_type; } gboolean format_raw_img_exif_offsets_fd(gint fd, const gchar *path, guchar *header_data, const guint header_len, guint *image_offset, guint *exif_offset) { FormatRawEntry *entry; gpointer map_data = NULL; size_t map_len = 0; struct stat st; gboolean success; if (!header_data || fd < 0) return FALSE; /* given image pathname, first do simple (and fast) file extension test */ if (path) { const gchar *ext; gboolean match = FALSE; gint i; ext = strrchr(path, '.'); if (!ext) return FALSE; ext++; i = 0; while (!match && format_raw_list[i].magic_pattern) { if (format_raw_list[i].extension && g_ascii_strcasecmp(format_raw_list[i].extension, ext) == 0) { match = TRUE; } i++; } if (!match) return FALSE; DEBUG_1("RAW file parser extension match"); } /* FIXME: * when the target is a tiff file it should be mmaped prior to format_raw_find as * the make field data may not always be within header_data + header_len */ entry = format_raw_find(header_data, header_len); if (!entry || !entry->func_parse) return FALSE; if (fstat(fd, &st) == -1) { log_printf("Failed to stat file %d\n", fd); return FALSE; } map_len = st.st_size; map_data = mmap(0, map_len, PROT_READ, MAP_PRIVATE, fd, 0); if (map_data == MAP_FAILED) { log_printf("Failed to mmap file %d\n", fd); return FALSE; } success = format_raw_parse(entry, map_data, map_len, image_offset, exif_offset); if (munmap(map_data, map_len) == -1) { log_printf("Failed to unmap file %d\n", fd); } if (success && image_offset) { if (lseek(fd, *image_offset, SEEK_SET) != (off_t) *image_offset) { log_printf("Failed to seek to embedded image\n"); *image_offset = 0; if (*exif_offset) *exif_offset = 0; success = FALSE; } } return success; } static FormatExifEntry *format_exif_makernote_find(ExifData *exif, guchar *tiff, guint offset, guint size) { ExifItem *make; gint n; make = exif_get_item(exif, "Exif.Image.Make"); n = 0; while (format_exif_list[n].header_pattern) { switch (format_exif_list[n].header_type) { case FORMAT_EXIF_MATCH_MAKERNOTE: if (format_exif_list[n].header_length + offset < size && memcmp(tiff + offset, format_exif_list[n].header_pattern, format_exif_list[n].header_length) == 0) { return &format_exif_list[n]; } break; case FORMAT_EXIF_MATCH_MAKE: if (make && make->data_len >= format_exif_list[n].header_length && memcmp(make->data, format_exif_list[n].header_pattern, format_exif_list[n].header_length) == 0) { return &format_exif_list[n]; } break; } n++; } return FALSE; } gboolean format_exif_makernote_parse(ExifData *exif, guchar *tiff, guint offset, guint size, ExifByteOrder bo) { FormatExifEntry *entry; entry = format_exif_makernote_find(exif, tiff, offset, size); if (!entry || !entry->func_parse) return FALSE; DEBUG_1("EXIF using makernote parser for %s", entry->description); return entry->func_parse(exif, tiff, offset, size, bo); } /* *----------------------------------------------------------------------------- * Basic TIFF debugger, prints all IFD entries within tiff file *----------------------------------------------------------------------------- */ #if DEBUG_RAW_TIFF static guint format_debug_tiff_table(guchar *data, const guint len, guint offset, ExifByteOrder bo, gint level); static void format_debug_tiff_entry(guchar *data, const guint len, guint offset, ExifByteOrder bo, gint level) { guint tag; guint type; guint count; guint segment; guint seg_len; tag = exif_byte_get_int16(data + offset + EXIF_TIFD_OFFSET_TAG, bo); type = exif_byte_get_int16(data + offset + EXIF_TIFD_OFFSET_FORMAT, bo); count = exif_byte_get_int32(data + offset + EXIF_TIFD_OFFSET_COUNT, bo); seg_len = ExifFormatList[type].size * count; if (seg_len > 4) { segment = exif_byte_get_int32(data + offset + EXIF_TIFD_OFFSET_DATA, bo); if (segment + seg_len > len) return; } else { segment = offset + EXIF_TIFD_OFFSET_DATA; } log_printf("%*stag:0x%04X (%05d), type:%2d %9s, len:%6d [%02X %02X %02X %02X] @ offset:%d\n", level, "", tag, tag, type, (type < EXIF_FORMAT_COUNT) ? ExifFormatList[type].short_name : "???", count, data[segment], data[segment + 1], data[segment + 2], data[segment + 3], segment); if (tag == 0x8769 || tag == 0x14a) { gint i; log_printf("%*s~~~ found %s table\n", level, "", (tag == 0x14a) ? "subIFD" : "EXIF" ); for (i = 0; i < count; i++) { guint subset; subset = exif_byte_get_int32(data + segment + i * 4, bo); format_debug_tiff_table(data, len, subset, bo, level + 1); } } else if (tag == 0x8773 && type == EXIF_FORMAT_UNDEFINED) { log_printf("%*s~~~ found ICC color profile at offset %d, length %d\n", level, "", segment, seg_len); } else if (tag == 0x201 && (type == EXIF_FORMAT_LONG_UNSIGNED || type == EXIF_FORMAT_LONG)) { guint subset = exif_byte_get_int32(data + segment, bo); log_printf("%*s~~~ found jpeg data at offset %d\n", level, "", subset); } else if (tag == 0x202 && (type == EXIF_FORMAT_LONG_UNSIGNED || type == EXIF_FORMAT_LONG)) { guint subset = exif_byte_get_int32(data + segment, bo); log_printf("%*s~~~ found jpeg data length of %d\n", level, "", subset); } } static guint format_debug_tiff_table(guchar *data, const guint len, guint offset, ExifByteOrder bo, gint level) { guint count; guint i; if (level > EXIF_TIFF_MAX_LEVELS) return 0; if (len < offset + 2) return FALSE; count = exif_byte_get_int16(data + offset, bo); offset += 2; if (len < offset + count * EXIF_TIFD_SIZE + 4) return 0; log_printf("%*s== tiff table #%d has %d entries ==\n", level, "", level, count); for (i = 0; i < count; i++) { format_debug_tiff_entry(data, len, offset + i * EXIF_TIFD_SIZE, bo, level); } log_printf("%*s----------- end of #%d ------------\n", level, "", level); return exif_byte_get_int32(data + offset + count * EXIF_TIFD_SIZE, bo); } gboolean format_debug_tiff_raw(guchar *data, const guint len, guint *image_offset, guint *exif_offset) { ExifByteOrder bo; gint level; guint offset; if (len < 8) return FALSE; /* for debugging, we are more relaxed as to magic header */ if (memcmp(data, "II", 2) == 0) { bo = EXIF_BYTE_ORDER_INTEL; } else if (memcmp(data, "MM", 2) == 0) { bo = EXIF_BYTE_ORDER_MOTOROLA; } else { return FALSE; } log_printf("*** debug parsing tiff\n"); offset = exif_byte_get_int32(data + 4, bo); level = 0; while (offset && level < EXIF_TIFF_MAX_LEVELS) { offset = format_debug_tiff_table(data, len, offset, bo, 0); level++; } log_printf("*** end\n"); /* we are debugging, not trying to return any data */ return FALSE; } #endif #endif /* not HAVE_EXIV2 */ /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */