/* mnote-olympus-entry.c * * Copyright (c) 2002 Lutz Mueller <lutz@users.sourceforge.net> * * This library 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 of the License, or (at your option) any later version. * * This library 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 this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #include <config.h> #include "mnote-olympus-entry.h" #include <stdio.h> #include <stdlib.h> #include <string.h> #include <libexif/exif-format.h> #include <libexif/exif-utils.h> #include <libexif/exif-entry.h> #include <libexif/i18n.h> #define CF(format,target,v,maxlen) \ { \ if (format != target) { \ snprintf (v, maxlen, \ _("Invalid format '%s', " \ "expected '%s'."), \ exif_format_get_name (format), \ exif_format_get_name (target)); \ break; \ } \ } #define CF2(format,target1,target2,v,maxlen) \ { \ if ((format != target1) && (format != target2)) { \ snprintf (v, maxlen, \ _("Invalid format '%s', " \ "expected '%s' or '%s'."), \ exif_format_get_name (format), \ exif_format_get_name (target1), \ exif_format_get_name (target2)); \ break; \ } \ } #define CC(number,target,v,maxlen) \ { \ if (number != target) { \ snprintf (v, maxlen, \ _("Invalid number of components (%i, " \ "expected %i)."), (int) number, (int) target); \ break; \ } \ } #define CC2(number,t1,t2,v,maxlen) \ { \ if ((number < t1) || (number > t2)) { \ snprintf (v, maxlen, \ _("Invalid number of components (%i, " \ "expected %i or %i)."), (int) number, \ (int) t1, (int) t2); \ break; \ } \ } static const struct { ExifTag tag; ExifFormat fmt; struct { int index; const char *string; } elem[10]; } items[] = { #ifndef NO_VERBOSE_TAG_DATA { MNOTE_NIKON_TAG_LENSTYPE, EXIF_FORMAT_BYTE, { {0, N_("AF non D Lens")}, {1, N_("Manual")}, {2, N_("AF-D or AF-S Lens")}, {6, N_("AF-D G Lens")}, {10, N_("AF-D VR Lens")}, {0, NULL}}}, { MNOTE_NIKON_TAG_FLASHUSED, EXIF_FORMAT_BYTE, { {0, N_("Flash did not fire")}, {4, N_("Flash unit unknown")}, {7, N_("Flash is external")}, {9, N_("Flash is on Camera")}, {0, NULL}}}, { MNOTE_NIKON1_TAG_QUALITY, EXIF_FORMAT_SHORT, { {1, N_("VGA Basic")}, {2, N_("VGA Normal")}, {3, N_("VGA Fine")}, {4, N_("SXGA Basic")}, {5, N_("SXGA Normal")}, {6, N_("SXGA Fine")}, {10, N_("2 MPixel Basic")}, {11, N_("2 MPixel Normal")}, {12, N_("2 MPixel Fine")}, {0, NULL}}}, { MNOTE_NIKON1_TAG_COLORMODE, EXIF_FORMAT_SHORT, { {1, N_("Color")}, {2, N_("Monochrome")}, {0, NULL}}}, { MNOTE_NIKON1_TAG_IMAGEADJUSTMENT, EXIF_FORMAT_SHORT, { {0, N_("Normal")}, {1, N_("Bright+")}, {2, N_("Bright-")}, {3, N_("Contrast+")}, {4, N_("Contrast-")}, {0, NULL}}}, { MNOTE_NIKON1_TAG_CCDSENSITIVITY, EXIF_FORMAT_SHORT, { {0, N_("ISO80")}, {2, N_("ISO160")}, {4, N_("ISO320")}, {5, N_("ISO100")}, {0, NULL}}}, { MNOTE_NIKON1_TAG_WHITEBALANCE, EXIF_FORMAT_SHORT, { {0, N_("Auto")}, {1, N_("Preset")}, {2, N_("Daylight")}, {3, N_("Incandescence")}, {4, N_("Fluorescence")}, {5, N_("Cloudy")}, {6, N_("SpeedLight")}, {0, NULL}}}, { MNOTE_NIKON1_TAG_CONVERTER, EXIF_FORMAT_SHORT, { {0, N_("No Fisheye")}, {1, N_("Fisheye On")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_QUALITY, EXIF_FORMAT_SHORT, { {1, N_("SQ")}, {2, N_("HQ")}, {3, N_("SHQ")}, {4, N_("RAW")}, {5, N_("SQ1")}, {6, N_("SQ2")}, {17, N_("Standard")}, {529, N_("High")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_MACRO, EXIF_FORMAT_SHORT, { {0, N_("No")}, {1, N_("Yes")}, {2, N_("Super Macro")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_BWMODE, EXIF_FORMAT_SHORT, { {0, N_("No")}, {1, N_("Yes")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_ONETOUCHWB, EXIF_FORMAT_SHORT, { {0, N_("Off")}, {1, N_("On")}, {2, N_("On (Preset)")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_FLASHMODE, EXIF_FORMAT_SHORT, { {0, N_("Auto")}, {1, N_("Red-eye reduction")}, {2, N_("Fill")}, {3, N_("Off")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_FLASHDEVICE, EXIF_FORMAT_SHORT, { {0, N_("None")}, {1, N_("Internal")}, {4, N_("External")}, {5, N_("Internal + External")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_FOCUSRANGE, EXIF_FORMAT_SHORT, { {0, N_("Normal")}, {1, N_("Macro")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_MANFOCUS, EXIF_FORMAT_SHORT, { {0, N_("Auto")}, {1, N_("Manual")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_SHARPNESS, EXIF_FORMAT_SHORT, { {0, N_("Normal")}, {1, N_("Hard")}, {2, N_("Soft")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_EXTERNALFLASHBOUNCE, EXIF_FORMAT_SHORT, { {0, N_("No")}, {1, N_("Yes")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_CONTRAST, EXIF_FORMAT_SHORT, { {0, N_("Hard")}, {1, N_("Normal")}, {2, N_("Soft")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_PREVIEWIMAGEVALID, EXIF_FORMAT_LONG, { {0, N_("No")}, {1, N_("Yes")}, {0, NULL}}}, { MNOTE_OLYMPUS_TAG_CCDSCANMODE, EXIF_FORMAT_SHORT, { {0, N_("Interlaced")}, {1, N_("Progressive")}, {0, NULL}}}, { MNOTE_SANYO_TAG_SEQUENTIALSHOT, EXIF_FORMAT_SHORT, { {0, N_("None")}, {1, N_("Standard")}, {2, N_("Best")}, {3, N_("Adjust Exposure")}, {0, NULL}}}, { MNOTE_SANYO_TAG_RECORDSHUTTERRELEASE, EXIF_FORMAT_SHORT, { {0, N_("Record while down")}, {1, N_("Press start, press stop")}, {0, NULL}}}, { MNOTE_SANYO_TAG_RESAVED, EXIF_FORMAT_SHORT, { {0, N_("No")}, {1, N_("Yes")}, {0, NULL}}}, { MNOTE_SANYO_TAG_SCENESELECT, EXIF_FORMAT_SHORT, { {0, N_("Off")}, {1, N_("Sport")}, {2, N_("TV")}, {3, N_("Night")}, {4, N_("User 1")}, {5, N_("User 2")}, {6, N_("Lamp")}, {0, NULL}}}, { MNOTE_SANYO_TAG_SEQUENCESHOTINTERVAL, EXIF_FORMAT_SHORT, { {0, N_("5 frames/sec")}, {1, N_("10 frames/sec")}, {2, N_("15 frames/sec")}, {3, N_("20 frames/sec")}, {0, NULL}}}, #endif { 0, 0, { { 0, NULL } } } }; char * mnote_olympus_entry_get_value (MnoteOlympusEntry *entry, char *v, unsigned int maxlen) { char buf[30]; ExifLong vl; ExifShort vs = 0; ExifRational vr, vr2; ExifSRational vsr; int i, j; double r, b; if (!entry) return (NULL); memset (v, 0, maxlen); maxlen--; if ((!entry->data) && (entry->components > 0)) return (v); switch (entry->tag) { /* Nikon */ case MNOTE_NIKON_TAG_FIRMWARE: CF (entry->format, EXIF_FORMAT_UNDEFINED, v, maxlen); CC (entry->components, 4, v, maxlen); vl = exif_get_long (entry->data, entry->order); if ((vl & 0xF0F0F0F0) == 0x30303030) { memcpy (v, entry->data, MIN (maxlen, 4)); } else { snprintf (v, maxlen, "%04lx", (long unsigned int) vl); } break; case MNOTE_NIKON_TAG_ISO: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 2, v, maxlen); //vs = exif_get_short (entry->data, entry->order); vs = exif_get_short (entry->data + 2, entry->order); snprintf (v, maxlen, "ISO %hd", vs); break; case MNOTE_NIKON_TAG_ISO2: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 2, v, maxlen); //vs = exif_get_short (entry->data, entry->order); vs = exif_get_short (entry->data + 2, entry->order); snprintf (v, maxlen, "ISO2 %hd", vs); break; case MNOTE_NIKON_TAG_QUALITY: CF (entry->format, EXIF_FORMAT_ASCII, v, maxlen); //CC (entry->components, 8, v, maxlen); //vl = exif_get_long (entry->data , entry->order); //printf("-> 0x%04x\n",entry->data); //printf("-> 0x%s<\n",entry->data - 0); memcpy(v, entry->data, MIN(maxlen, entry->size)); //snprintf (v, maxlen, "%s<", ( entry->data - 9 ); break; case MNOTE_NIKON_TAG_COLORMODE: case MNOTE_NIKON_TAG_COLORMODE1: case MNOTE_NIKON_TAG_WHITEBALANCE: case MNOTE_NIKON_TAG_SHARPENING: case MNOTE_NIKON_TAG_FOCUSMODE: case MNOTE_NIKON_TAG_FLASHSETTING: case MNOTE_NIKON_TAG_ISOSELECTION: case MNOTE_NIKON_TAG_FLASHMODE: case MNOTE_NIKON_TAG_IMAGEADJUSTMENT: case MNOTE_NIKON_TAG_ADAPTER: case MNOTE_NIKON_TAG_SATURATION2: CF (entry->format, EXIF_FORMAT_ASCII, v, maxlen); memcpy(v, entry->data, MIN (maxlen, entry->size)); break; case MNOTE_NIKON_TAG_TOTALPICTURES: CF (entry->format, EXIF_FORMAT_LONG, v, maxlen); CC (entry->components, 1, v, maxlen); vl = exif_get_long (entry->data, entry->order); snprintf (v, maxlen, "%lu", (long unsigned int) vl ); break; case MNOTE_NIKON_TAG_LENS_FSTOPS: case MNOTE_NIKON_TAG_EXPOSUREDIFF: { unsigned char a,b,c,d; CF (entry->format, EXIF_FORMAT_UNDEFINED, v, maxlen); CC (entry->components, 4, v, maxlen); vl = exif_get_long (entry->data, entry->order); a = (vl>>24)&0xff; b = (vl>>16)&0xff; c = (vl>>8)&0xff; d = (vl)&0xff; snprintf (v, maxlen, "%.1f", c?(float)a*((float)b/(float)c):0 ); break; } case MNOTE_NIKON_TAG_FLASHEXPCOMPENSATION: case MNOTE_NIKON_TAG_FLASHEXPOSUREBRACKETVAL: CF (entry->format, EXIF_FORMAT_UNDEFINED, v, maxlen); CC (entry->components, 4, v, maxlen); vl = exif_get_long (entry->data, entry->order); snprintf (v, maxlen, "%.1f", ((long unsigned int) vl>>24)/6.0 ); break; case MNOTE_NIKON_TAG_SATURATION: case MNOTE_NIKON_TAG_WHITEBALANCEFINE: case MNOTE_NIKON_TAG_HUE: case MNOTE_OLYMPUS_TAG_SENSORTEMPERATURE: case MNOTE_OLYMPUS_TAG_LENSTEMPERATURE: CF (entry->format, EXIF_FORMAT_SSHORT, v, maxlen); CC (entry->components, 1, v, maxlen); vs = exif_get_short (entry->data, entry->order); snprintf (v, maxlen, "%hd", vs); break; case MNOTE_NIKON_TAG_WHITEBALANCERB: CF (entry->format, EXIF_FORMAT_RATIONAL, v, maxlen); CC (entry->components, 4, v, maxlen); vr = exif_get_rational (entry->data, entry->order); r = (double)vr.numerator / vr.denominator; vr = exif_get_rational (entry->data+8, entry->order); b = (double)vr.numerator / vr.denominator; //printf("numerator %li, denominator %li\n", vr.numerator, vr.denominator); snprintf (v, maxlen, _("Red Correction %f, Blue Correction %f"), r,b); break; case MNOTE_NIKON_TAG_MANUALFOCUSDISTANCE: CF (entry->format, EXIF_FORMAT_RATIONAL, v, maxlen); CC (entry->components, 1, v, maxlen); vr = exif_get_rational (entry->data, entry->order); if (vr.numerator) { r = (double)vr.numerator / vr.denominator; snprintf (v, maxlen, _("%2.2f meters"), r); } else { strncpy (v, _("No manual focus selection"), maxlen); } break; case MNOTE_NIKON_TAG_SENSORPIXELSIZE: CF (entry->format, EXIF_FORMAT_RATIONAL, v, maxlen); CC (entry->components, 2, v, maxlen); vr = exif_get_rational (entry->data, entry->order); vr2 = exif_get_rational (entry->data+8, entry->order); r = (double)vr.numerator / vr.denominator; b = (double)vr2.numerator / vr2.denominator; snprintf (v, maxlen, "%2.2f x %2.2f um", r, b); break; case MNOTE_NIKON_TAG_BRACKETING: CF2 (entry->format, EXIF_FORMAT_BYTE, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 1, v, maxlen); if (EXIF_FORMAT_SHORT == entry->format) { vs = exif_get_short (entry->data, entry->order); } else { vs = entry->data[0]; } snprintf (v, maxlen, "%hd", vs); break; case MNOTE_NIKON_TAG_AFFOCUSPOSITION: CF (entry->format, EXIF_FORMAT_UNDEFINED, v, maxlen); CC (entry->components, 4, v, maxlen); switch ( *( entry->data+1) ) { case 0: strncpy (v, _("AF Position: Center"), maxlen); break; case 1: strncpy (v, _("AF Position: Top"), maxlen); break; case 2: strncpy (v, _("AF Position: Bottom"), maxlen); break; case 3: strncpy (v, _("AF Position: Left"), maxlen); break; case 4: strncpy (v, _("AF Position: Right"), maxlen); break; case 5: strncpy (v, _("AF Position: Upper-left"), maxlen); break; case 6: strncpy (v, _("AF Position: Upper-right"), maxlen); break; case 7: strncpy (v, _("AF Position: Lower-left"), maxlen); break; case 8: strncpy (v, _("AF Position: Lower-right"), maxlen); break; case 9: strncpy (v, _("AF Position: Far Left"), maxlen); break; case 10: strncpy (v, _("AF Position: Far Right"), maxlen); break; default: strncpy (v, _("Unknown AF Position"), maxlen); } break; case MNOTE_OLYMPUS_TAG_FLASHDEVICE: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 2, v, maxlen); vs = exif_get_short(entry->data, entry->order); /* search for the tag */ for (i = 0; (items[i].tag && items[i].tag != entry->tag); i++) ; if (!items[i].tag) { snprintf (v, maxlen, _("Internal error (unknown value %hi)"), vs); break; } CF (entry->format, items[i].fmt, v, maxlen); /* find the value */ for (j = 0; items[i].elem[j].string && (items[i].elem[j].index < vs); j++); if (items[i].elem[j].index != vs) { snprintf (v, maxlen, _("Unknown value %hi"), vs); break; } strncpy (v, _(items[i].elem[j].string), maxlen); break; case MNOTE_OLYMPUS_TAG_DIGIZOOM: if (entry->format == EXIF_FORMAT_RATIONAL) { CC (entry->components, 1, v, maxlen); vr = exif_get_rational (entry->data, entry->order); r = (double)vr.numerator / vr.denominator; if (!vr.numerator) { strncpy (v, _("None"), maxlen); } else { snprintf (v, maxlen, "%2.2f", r); } break; } /* fall through to handle SHORT version of this tag */ case MNOTE_NIKON_TAG_LENSTYPE: case MNOTE_NIKON_TAG_FLASHUSED: case MNOTE_NIKON1_TAG_QUALITY: case MNOTE_NIKON1_TAG_COLORMODE: case MNOTE_NIKON1_TAG_IMAGEADJUSTMENT: case MNOTE_NIKON1_TAG_CCDSENSITIVITY: case MNOTE_NIKON1_TAG_WHITEBALANCE: case MNOTE_NIKON1_TAG_CONVERTER: case MNOTE_OLYMPUS_TAG_QUALITY: case MNOTE_OLYMPUS_TAG_MACRO: case MNOTE_OLYMPUS_TAG_BWMODE: case MNOTE_OLYMPUS_TAG_ONETOUCHWB: case MNOTE_OLYMPUS_TAG_FLASHMODE: case MNOTE_OLYMPUS_TAG_FOCUSRANGE: case MNOTE_OLYMPUS_TAG_MANFOCUS: case MNOTE_OLYMPUS_TAG_SHARPNESS: case MNOTE_OLYMPUS_TAG_EXTERNALFLASHBOUNCE: case MNOTE_OLYMPUS_TAG_CONTRAST: case MNOTE_OLYMPUS_TAG_PREVIEWIMAGEVALID: case MNOTE_OLYMPUS_TAG_CCDSCANMODE: case MNOTE_SANYO_TAG_SEQUENTIALSHOT: case MNOTE_SANYO_TAG_RECORDSHUTTERRELEASE: case MNOTE_SANYO_TAG_RESAVED: case MNOTE_SANYO_TAG_SCENESELECT: case MNOTE_SANYO_TAG_SEQUENCESHOTINTERVAL: CC (entry->components, 1, v, maxlen); switch (entry->format) { case EXIF_FORMAT_BYTE: case EXIF_FORMAT_UNDEFINED: vs = entry->data[0]; break; case EXIF_FORMAT_SHORT: vs = exif_get_short(entry->data, entry->order); break; default: vs = 0; break; } /* search for the tag */ for (i = 0; (items[i].tag && items[i].tag != entry->tag); i++) ; if (!items[i].tag) { snprintf (v, maxlen, _("Internal error (unknown value %hi)"), vs); break; } CF (entry->format, items[i].fmt, v, maxlen); /* find the value */ for (j = 0; items[i].elem[j].string && (items[i].elem[j].index < vs); j++); if (items[i].elem[j].index != vs) { snprintf (v, maxlen, _("Unknown value %hi"), vs); break; } strncpy (v, _(items[i].elem[j].string), maxlen); break; case MNOTE_OLYMPUS_TAG_NOISEREDUCTION: case MNOTE_SANYO_TAG_WIDERANGE: case MNOTE_SANYO_TAG_COLORADJUSTMENTMODE: case MNOTE_SANYO_TAG_QUICKSHOT: case MNOTE_SANYO_TAG_SELFTIMER: case MNOTE_SANYO_TAG_VOICEMEMO: case MNOTE_SANYO_TAG_FLICKERREDUCE: case MNOTE_SANYO_TAG_OPTICALZOOM: case MNOTE_SANYO_TAG_DIGITALZOOM: case MNOTE_SANYO_TAG_LIGHTSOURCESPECIAL: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 1, v, maxlen); vs = exif_get_short (entry->data, entry->order); switch (vs) { case 0: strncpy (v, _("Off"), maxlen); break; case 1: strncpy (v, _("On"), maxlen); break; default: strncpy (v, _("Unknown"), maxlen); break; } break; case MNOTE_NIKON_TAG_LENS: CF (entry->format, EXIF_FORMAT_RATIONAL, v, maxlen); CC (entry->components, 4, v, maxlen); { double c,d; unsigned long a,b; vr = exif_get_rational (entry->data, entry->order); a = vr.numerator / vr.denominator; vr = exif_get_rational (entry->data+8, entry->order); b = vr.numerator / vr.denominator; vr = exif_get_rational (entry->data+16, entry->order); c = (double)vr.numerator / vr.denominator; vr = exif_get_rational (entry->data+24, entry->order); d = (double)vr.numerator / vr.denominator; //printf("numerator %li, denominator %li\n", vr.numerator, vr.denominator); snprintf (v, maxlen, "%ld-%ldmm 1:%3.1f - %3.1f",a,b,c,d); } break; /* Olympus */ case MNOTE_OLYMPUS_TAG_MODE: CF (entry->format, EXIF_FORMAT_LONG, v, maxlen); CC (entry->components, 3, v, maxlen); vl = exif_get_long (entry->data, entry->order); switch (vl) { case 0: strncpy (v, _("normal"), maxlen); break; case 1: strncpy (v, _("unknown"), maxlen); break; case 2: strncpy (v, _("fast"), maxlen); break; case 3: strncpy (v, _("panorama"), maxlen); break; default: snprintf (v, maxlen, _("%li"), (long int) vl); } vl = exif_get_long (entry->data + 4, entry->order); snprintf (buf, sizeof (buf), "/%li/", (long int) vl); strncat (v, buf, maxlen - strlen (v)); vl = exif_get_long (entry->data + 4, entry->order); switch (vl) { case 1: strncat (v, _("left to right"), maxlen - strlen (v)); break; case 2: strncat (v, _("right to left"), maxlen - strlen (v)); break; case 3: strncat (v, _("bottom to top"), maxlen - strlen (v)); break; case 4: strncat (v, _("top to bottom"), maxlen - strlen (v)); break; default: snprintf (buf, sizeof (buf), _("%li"), (long int) vl); strncat (v, buf, maxlen - strlen (v)); } break; case MNOTE_OLYMPUS_TAG_LENSDISTORTION: CF (entry->format, EXIF_FORMAT_SSHORT, v, maxlen); CC (entry->components, 6, v, maxlen); for (i=0; i < (int)entry->components; ++i) { vs = exif_get_sshort (entry->data+2*i, entry->order); sprintf (buf, "%hd ", vs); strncat (v, buf, maxlen - strlen (v)); } break; case MNOTE_OLYMPUS_TAG_COLORCONTROL: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 6, v, maxlen); for (i=0; i < (int)entry->components; ++i) { vs = exif_get_short (entry->data+2*i, entry->order); sprintf (buf, "%hu ", vs); strncat (v, buf, maxlen - strlen (v)); } break; case MNOTE_OLYMPUS_TAG_VERSION: CF (entry->format, EXIF_FORMAT_ASCII, v, maxlen); CC2 (entry->components, 5, 8, v, maxlen); strncpy (v, (char *)entry->data, MIN (maxlen, entry->size)); break; case MNOTE_OLYMPUS_TAG_SERIALNUMBER2: CF (entry->format, EXIF_FORMAT_ASCII, v, maxlen); strncpy (v, (char *)entry->data, MIN (maxlen, entry->size)); break; case MNOTE_OLYMPUS_TAG_INFO: CF (entry->format, EXIF_FORMAT_ASCII, v, maxlen); CC2 (entry->components, 52, 60, v, maxlen); strncpy (v, (char *)entry->data, MIN (maxlen, entry->size)); break; case MNOTE_OLYMPUS_TAG_ID: CF (entry->format, EXIF_FORMAT_UNDEFINED, v, maxlen); CC (entry->components, 32, v, maxlen); strncpy (v, (char *)entry->data, MIN (maxlen, entry->size)); break; case MNOTE_OLYMPUS_TAG_UNKNOWN_4: CF (entry->format, EXIF_FORMAT_LONG, v, maxlen); CC (entry->components, 30, v, maxlen); /* TODO: display me */ break; case MNOTE_OLYMPUS_TAG_FOCUSDIST: CF (entry->format, EXIF_FORMAT_RATIONAL, v, maxlen); CC (entry->components, 1, v, maxlen); vr = exif_get_rational (entry->data, entry->order); if (vr.numerator == 0) { strncpy (v, _("Unknown"), maxlen); } else { unsigned long tmp = vr.numerator / vr.denominator; /* printf("numerator %li, denominator %li\n", vr.numerator, vr.denominator); */ snprintf (v, maxlen, "%li mm", tmp); } break; case MNOTE_OLYMPUS_TAG_WBALANCE: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 2, v, maxlen); vs = exif_get_short (entry->data, entry->order); switch (vs) { case 1: strncpy (v, _("Automatic"), maxlen); break; case 2: { ExifShort v2 = exif_get_short (entry->data + 2, entry->order); unsigned long colorTemp = 0; switch (v2) { case 2: colorTemp = 3000; break; case 3: colorTemp = 3700; break; case 4: colorTemp = 4000; break; case 5: colorTemp = 4500; break; case 6: colorTemp = 5500; break; case 7: colorTemp = 6500; break; case 9: colorTemp = 7500; break; } if (colorTemp) { snprintf (v, maxlen, _("Manual: %liK"), colorTemp); } else { strncpy (v, _("Manual: Unknown"), maxlen); } } break; case 3: strncpy (v, _("One-touch"), maxlen); break; default: strncpy (v, _("Unknown"), maxlen); break; } break; case MNOTE_OLYMPUS_TAG_REDBALANCE: case MNOTE_OLYMPUS_TAG_BLUEBALANCE: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); CC (entry->components, 2, v, maxlen); vs = exif_get_short (entry->data, entry->order); snprintf (v, maxlen, "%hu ", vs); vs = exif_get_short (entry->data + 2, entry->order); sprintf (buf, "%hu", vs); strncat (v, buf, maxlen - strlen (v)); break; case MNOTE_OLYMPUS_TAG_BLACKLEVEL: case MNOTE_NIKON_TAG_IMAGEBOUNDARY: CC (entry->components, 4, v, maxlen); /* Fall through to COLORMATRIX */ case MNOTE_OLYMPUS_TAG_COLORMATRIX: CF (entry->format, EXIF_FORMAT_SHORT, v, maxlen); if (entry->tag == MNOTE_OLYMPUS_TAG_COLORMATRIX) CC (entry->components, 9, v, maxlen); for (i=0; i < (int)entry->components; ++i) { vs = exif_get_short (entry->data+2*i, entry->order); sprintf (buf, "%hu ", vs); strncat (v, buf, maxlen - strlen (v)); } break; case MNOTE_NIKON1_TAG_FOCUS: case MNOTE_NIKON_TAG_DIGITALZOOM: case MNOTE_NIKON1_TAG_DIGITALZOOM: case MNOTE_OLYMPUS_TAG_FOCALPLANEDIAGONAL: CF (entry->format, EXIF_FORMAT_RATIONAL, v, maxlen); /* Fall through to default handler for display */ default: switch (entry->format) { case EXIF_FORMAT_ASCII: strncpy (v, (char *)entry->data, MIN (maxlen, entry->size)); break; case EXIF_FORMAT_SHORT: CC (entry->components, 1, v, maxlen); vs = exif_get_short (entry->data, entry->order); snprintf (v, maxlen, "%hu", vs); break; case EXIF_FORMAT_LONG: CC (entry->components, 1, v, maxlen); vl = exif_get_long (entry->data, entry->order); snprintf (v, maxlen, "%li", (long int) vl); break; case EXIF_FORMAT_RATIONAL: CC (entry->components, 1, v, maxlen); vr = exif_get_rational (entry->data, entry->order); if (!vr.denominator) { strncpy (v, _("Infinite"), maxlen); } else { r = (double)vr.numerator / vr.denominator; snprintf (v, maxlen, "%2.3f", r); } break; case EXIF_FORMAT_SRATIONAL: CC (entry->components, 1, v, maxlen); vsr = exif_get_srational (entry->data, entry->order); if (!vsr.denominator) { strncpy (v, _("Infinite"), maxlen); } else { r = (double)vsr.numerator / vsr.denominator; snprintf (v, maxlen, "%2.3f", r); } break; case EXIF_FORMAT_UNDEFINED: default: snprintf (v, maxlen, _("%i bytes unknown data: "), entry->size); for (i = 0; i < (int)entry->size; i++) { sprintf (buf, "%02x", entry->data[i]); strncat (v, buf, maxlen - strlen (v)); } break; } break; } return (v); }