rebuild exif on save

on save, rebuild the whole of the exif block from vips metadata ...
users can now alter tags by updating attached strings
This commit is contained in:
John Cupitt 2011-11-03 18:35:41 +00:00
parent 86fa10d474
commit 5f0db6a093
3 changed files with 181 additions and 123 deletions

44
TODO
View File

@ -65,50 +65,6 @@
- we have some serious exif problems
we attach exif as both the original exif data and as a set of broken-out
tags, such as 'exif-Orientation' ... the tags are attached as strings
when we write a jpg file, we parse the original exif block, update the
resolution and Orientation tags from metadata, and add to the output file
we need a way to update the exif block with all 'exif-' tags, but this is
hard because a tag is
tag
format (eg. short, rational, string)
components (for short, for example, you can have an array of 12)
data (byte buffer)
size (sizeof byte buffer)
many tags have extra meanings, for example Orientation means that the format
must be short and the data must be 1, 2, 3, 4, 5, 6, 7 or 8 for the 8
possible rotates and flips ... libexif does not expose any way to convert
strings to enum values safely, what do we do with "Bottom-right"
how do we expose this to bindings in a sane way, how do we accurately
convert the updated tag values back into correct exif when we reattach
additionally, exif tags can be in one of several ifd blocks, do we need to
add the tag to the correct block? how do we save that information?
oo! we have the original, of course
to reattach, get the entry from the original exif block, get the updated
tag, coerce the updated value back to the orginal format
save exif entries to tags as something like:
exif-Orientation: 1 (Top-left, Short, 1 component, 2 bytes)
so we can restore the value from the "1" without needing to look up i18n
strings
- vips_object_set_argument_from_string() needs more arg types - vips_object_set_argument_from_string() needs more arg types
must be some way to make this more automatic must be some way to make this more automatic

View File

@ -37,6 +37,8 @@
* - added im_bufjpeg2vips() * - added im_bufjpeg2vips()
* 12/10/2011 * 12/10/2011
* - read XMP data * - read XMP data
* 3/11/11
* - attach exif tags as coded values
*/ */
/* /*
@ -353,7 +355,7 @@ typedef struct _VipsExif {
VipsImage *image; VipsImage *image;
ExifData *ed; ExifData *ed;
} VipsExif; } VipsExif;
static void static void
attach_exif_entry( ExifEntry *entry, VipsExif *ve ) attach_exif_entry( ExifEntry *entry, VipsExif *ve )
{ {

View File

@ -44,6 +44,8 @@
* - write XMP data * - write XMP data
* 18/10/2011 * 18/10/2011
* - update Orientation as well * - update Orientation as well
* 3/11/11
* - rebuild exif tags from coded metadata values
*/ */
/* /*
@ -75,6 +77,7 @@
/* /*
#define DEBUG_VERBOSE #define DEBUG_VERBOSE
#define DEBUG #define DEBUG
#define VIPS_DEBUG
*/ */
#ifdef HAVE_CONFIG_H #ifdef HAVE_CONFIG_H
@ -136,6 +139,7 @@ im_vips2mimejpeg( IMAGE *in, int qfac )
#include <vips/vips.h> #include <vips/vips.h>
#include <vips/internal.h> #include <vips/internal.h>
#include <vips/debug.h>
#include <vips/buf.h> #include <vips/buf.h>
/* jpeglib includes jconfig.h, which can define HAVE_STDLIB_H ... which we /* jpeglib includes jconfig.h, which can define HAVE_STDLIB_H ... which we
@ -477,42 +481,109 @@ write_new( IMAGE *in )
#ifdef HAVE_EXIF #ifdef HAVE_EXIF
static void static void
write_rational( ExifEntry *entry, ExifByteOrder bo, void *data ) vips_exif_set_int( ExifData *ed,
ExifEntry *entry, unsigned long component, void *data )
{ {
ExifRational *v = (ExifRational *) data; int value = *((int *) data);
exif_set_rational( entry->data, bo, *v ); ExifByteOrder bo;
size_t sizeof_component;
size_t offset = component;
if( entry->components <= component ) {
VIPS_DEBUG_MSG( "vips_exif_set_int: too few components\n" );
return;
}
/* Wait until after the component check to make sure we cant get /0.
*/
bo = exif_data_get_byte_order( ed );
sizeof_component = entry->size / entry->components;
offset = component * sizeof_component;
VIPS_DEBUG_MSG( "vips_exif_set_int: %s = %d\n",
exif_tag_get_title( entry->tag ), value );
if( entry->format == EXIF_FORMAT_SHORT )
exif_set_short( entry->data + offset, bo, value );
else if( entry->format == EXIF_FORMAT_SSHORT )
exif_set_sshort( entry->data + offset, bo, value );
else if( entry->format == EXIF_FORMAT_LONG )
exif_set_long( entry->data + offset, bo, value );
else if( entry->format == EXIF_FORMAT_SLONG )
exif_set_slong( entry->data + offset, bo, value );
} }
static void static void
write_short( ExifEntry *entry, ExifByteOrder bo, void *data ) vips_exif_set_double( ExifData *ed,
ExifEntry *entry, unsigned long component, void *data )
{ {
ExifShort *v = (ExifShort *) data; double value = *((double *) data);
exif_set_short( entry->data, bo, *v ); ExifByteOrder bo;
size_t sizeof_component;
size_t offset;
if( entry->components <= component ) {
VIPS_DEBUG_MSG( "vips_exif_set_int: too few components\n" );
return;
}
/* Wait until after the component check to make sure we cant get /0.
*/
bo = exif_data_get_byte_order( ed );
sizeof_component = entry->size / entry->components;
offset = component * sizeof_component;
VIPS_DEBUG_MSG( "vips_exif_set_double: %s = %g\n",
exif_tag_get_title( entry->tag ), value );
if( entry->format == EXIF_FORMAT_RATIONAL ) {
ExifRational rational;
unsigned int scale;
/* We scale up to fill uint32, then set that as the
* denominator. Try to avoid generating 0.
*/
scale = (int) ((UINT_MAX - 1000) / value);
scale = scale == 0 ? 1 : scale;
rational.numerator = value * scale;
rational.denominator = scale;
exif_set_rational( entry->data + offset, bo, rational );
}
else if( entry->format == EXIF_FORMAT_SRATIONAL ) {
ExifSRational rational;
int scale;
scale = (int) ((INT_MAX - 1000) / value);
scale = scale == 0 ? 1 : scale;
rational.numerator = value * scale;
rational.denominator = scale;
exif_set_srational( entry->data + offset, bo, rational );
}
} }
typedef void (*write_fn)( ExifEntry *, ExifByteOrder, void * ); typedef void (*write_fn)( ExifData *ed,
ExifEntry *entry, unsigned long component, void *data );
/* Write a component in a tag everywhere it appears.
*/
static int static int
write_tag( ExifData *ed, ExifTag tag, ExifFormat f, write_fn fn, void *data ) write_tag( ExifData *ed,
ExifTag tag, ExifFormat format, write_fn fn, void *data )
{ {
ExifByteOrder bo;
int found; int found;
int i; int i;
bo = exif_data_get_byte_order( ed );
/* Need to set the tag in all sections which have it :-(
*/
found = 0; found = 0;
for( i = 0; i < EXIF_IFD_COUNT; i++ ) { for( i = 0; i < EXIF_IFD_COUNT; i++ ) {
ExifEntry *entry; ExifEntry *entry;
if( (entry = exif_content_get_entry( ed->ifd[i], tag )) && if( (entry = exif_content_get_entry( ed->ifd[i], tag )) &&
entry->format == f && entry->format == format ) {
entry->components == 1 ) { fn( ed, entry, 0, data );
fn( entry, bo, data );
found = 1; found = 1;
} }
} }
@ -532,18 +603,20 @@ write_tag( ExifData *ed, ExifTag tag, ExifFormat f, write_fn fn, void *data )
exif_entry_initialize( entry, tag ); exif_entry_initialize( entry, tag );
exif_entry_unref( entry ); exif_entry_unref( entry );
fn( entry, bo, data ); fn( ed, entry, 0, data );
} }
return( 0 ); return( 0 );
} }
/* This is different, we set the xres/yres from the vips header rather than
* from the exif tags on the image metadata.
*/
static int static int
set_exif_resolution( ExifData *ed, IMAGE *im ) set_exif_resolution( ExifData *ed, IMAGE *im )
{ {
double xres, yres; double xres, yres;
ExifRational xres_rational, yres_rational; int unit;
ExifShort unit;
/* Always save as inches - more progs support it for read. /* Always save as inches - more progs support it for read.
*/ */
@ -551,19 +624,12 @@ set_exif_resolution( ExifData *ed, IMAGE *im )
yres = im->Yres * 25.4; yres = im->Yres * 25.4;
unit = 2; unit = 2;
/* Wow, how dumb, fix this.
*/
xres_rational.numerator = xres * 100000;
xres_rational.denominator = 100000;
yres_rational.numerator = yres * 100000;
yres_rational.denominator = 100000;
if( write_tag( ed, EXIF_TAG_X_RESOLUTION, EXIF_FORMAT_RATIONAL, if( write_tag( ed, EXIF_TAG_X_RESOLUTION, EXIF_FORMAT_RATIONAL,
write_rational, &xres_rational ) || vips_exif_set_double, (void *) &xres ) ||
write_tag( ed, EXIF_TAG_Y_RESOLUTION, EXIF_FORMAT_RATIONAL, write_tag( ed, EXIF_TAG_Y_RESOLUTION, EXIF_FORMAT_RATIONAL,
write_rational, &yres_rational ) || vips_exif_set_double, (void *) &yres ) ||
write_tag( ed, EXIF_TAG_RESOLUTION_UNIT, EXIF_FORMAT_SHORT, write_tag( ed, EXIF_TAG_RESOLUTION_UNIT, EXIF_FORMAT_SHORT,
write_short, &unit ) ) { vips_exif_set_int, (void *) &unit ) ) {
im_error( "im_jpeg2vips", im_error( "im_jpeg2vips",
"%s", _( "error setting JPEG resolution" ) ); "%s", _( "error setting JPEG resolution" ) );
return( -1 ); return( -1 );
@ -572,61 +638,95 @@ set_exif_resolution( ExifData *ed, IMAGE *im )
return( 0 ); return( 0 );
} }
static void * /* See also vips_exif_to_s() ... keep in sync.
update_orientation_cb( VipsImage *in, const char *name, GValue *value, void *a ) */
static void
vips_exif_from_s( ExifData *ed, ExifEntry *entry, const char *value )
{ {
/* Sadly these strings are subject to i18n by libexif. We should unsigned long i;
* attach EXIF data as binary and leave interpretaion to the UI argh. const char *p;
if( entry->format != EXIF_FORMAT_SHORT &&
entry->format != EXIF_FORMAT_SSHORT &&
entry->format != EXIF_FORMAT_LONG &&
entry->format != EXIF_FORMAT_SLONG &&
entry->format != EXIF_FORMAT_RATIONAL &&
entry->format != EXIF_FORMAT_SRATIONAL )
return;
if( entry->components >= 10 )
return;
/* Skip any leading spaces.
*/ */
static const char *orientations[] = { p = value;
"", while( *p == ' ' )
"Top-left", "Top-right", "Bottom-right", p += 1;
"Bottom-left", "Left-top", "Right-top",
"Right-bottom", "Left-bottom",
NULL
};
ExifData *ed = (ExifData *) a; for( i = 0; i < entry->components; i++ ) {
if( entry->format == EXIF_FORMAT_SHORT ||
entry->format == EXIF_FORMAT_SSHORT ||
entry->format == EXIF_FORMAT_LONG ||
entry->format == EXIF_FORMAT_SLONG ) {
int value = atof( p );
int i; vips_exif_set_int( ed, entry, i, &value );
ExifShort orientation; }
else if( entry->format == EXIF_FORMAT_RATIONAL ||
entry->format == EXIF_FORMAT_SRATIONAL ) {
double value = g_ascii_strtod( p, NULL );
if( vips_isprefix( "exif-Orientation", name ) ) { vips_exif_set_double( ed, entry, i, &value );
const char *string;
if( !(string = vips_value_get_ref_string( value, NULL )) ) {
vips_warn( "im_jpeg2vips",
"%s", _( "exif-Orientation is not a string" ) );
return( NULL );
} }
orientation = 0; /* Skip to the next set of spaces, then to the beginning of
for( i = 0; orientations[i]; i++ ) * the next item.
if( strlen( orientations[i] ) > 0 && */
vips_isprefix( orientations[i], string ) ) { while( *p && *p != ' ' )
orientation = i; p += 1;
break; while( *p == ' ' )
} p += 1;
if( orientation == 0 ) { if( !*p )
vips_warn( "im_jpeg2vips", break;
"%s", _( "unknown JPEG orientation" ) );
return( NULL );
}
if( write_tag( ed, EXIF_TAG_ORIENTATION, EXIF_FORMAT_SHORT,
write_short, &orientation ) ) {
vips_warn( "im_jpeg2vips",
"%s", _( "error setting JPEG orientation" ) );
return( NULL );
}
#ifdef DEBUG
printf( "im_vips2jpeg: updated orientation\n" );
#endif /*DEBUG*/
} }
return( NULL );
} }
typedef struct _VipsExif {
VipsImage *image;
ExifData *ed;
} VipsExif;
static void
vips_exif_update_entry( ExifEntry *entry, VipsExif *ve )
{
char name[256];
char *value;
im_snprintf( name, 256, "exif-%s", exif_tag_get_title( entry->tag ) );
if( !vips_image_get_string( ve->image, name, &value ) )
vips_exif_from_s( ve->ed, entry, value );
}
static void
vips_exif_update_content( ExifContent *content, VipsExif *ve )
{
exif_content_foreach_entry( content,
(ExifContentForeachEntryFunc) vips_exif_update_entry, ve );
}
static void
vips_exif_update( ExifData *ed, VipsImage *image )
{
VipsExif ve;
VIPS_DEBUG_MSG( "vips_exif_update: \n" );
ve.image = image;
ve.ed = ed;
exif_data_foreach_content( ed,
(ExifDataForeachContentFunc) vips_exif_update_content, &ve );
}
#endif /*HAVE_EXIF*/ #endif /*HAVE_EXIF*/
static int static int
@ -662,11 +762,11 @@ write_exif( Write *write )
exif_data_fix( ed ); exif_data_fix( ed );
} }
/* Update EXIF orientation from VIPS. /* Update EXIF tags from the image metadata.
*/ */
(void) vips_image_map( write->in, update_orientation_cb, ed ); vips_exif_update( ed, write->in );
/* Update EXIF resolution from VIPS. /* Update EXIF resolution from the vips image header..
*/ */
if( set_exif_resolution( ed, write->in ) ) { if( set_exif_resolution( ed, write->in ) ) {
exif_data_free( ed ); exif_data_free( ed );