better exif handling

- track ifd numbers for each item of exif
- be more careful about which ones we update
- don't update xres/yres in vips_shrink()
This commit is contained in:
John Cupitt 2012-11-16 10:03:31 +00:00
parent 717266ad2b
commit fcc33020bd
4 changed files with 71 additions and 61 deletions

View File

@ -20,6 +20,7 @@
- remove no threads option, glib no longer support it - remove no threads option, glib no longer support it
- better --help output for vips driver prog - better --help output for vips driver prog
- vipsthumbnail -o allows absolute file names - vipsthumbnail -o allows absolute file names
- much better exif handling for jpg images (thanks Gary)
14/11/12 started 7.30.6 14/11/12 started 7.30.6
- capture tiff warnings earlier - capture tiff warnings earlier

View File

@ -281,25 +281,28 @@ show_entry( ExifEntry *entry, void *client )
static void static void
show_ifd( ExifContent *content, void *client ) show_ifd( ExifContent *content, void *client )
{ {
int *ifd = (int *) client;
printf( "- ifd %d\n", *ifd );
exif_content_foreach_entry( content, show_entry, client ); exif_content_foreach_entry( content, show_entry, client );
printf( "-\n" );
*ifd += 1;
} }
void void
show_values( ExifData *data ) show_values( ExifData *data )
{ {
ExifByteOrder order; ExifByteOrder order;
int ifd;
order = exif_data_get_byte_order( data ); order = exif_data_get_byte_order( data );
printf( "EXIF tags in '%s' byte order\n", printf( "EXIF tags in '%s' byte order\n",
exif_byte_order_get_name( order ) ); exif_byte_order_get_name( order ) );
printf( "%-20.20s", "Tag" ); printf( "Title|Value|Format|Size\n" );
printf( "|" );
printf( "%-58.58s", "Value" );
printf( "\n" );
exif_data_foreach_content( data, show_ifd, NULL ); ifd = 0;
exif_data_foreach_content( data, show_ifd, &ifd );
if( data->size ) if( data->size )
printf( "contains thumbnail of %d bytes\n", data->size ); printf( "contains thumbnail of %d bytes\n", data->size );
@ -416,7 +419,9 @@ attach_exif_entry( ExifEntry *entry, VipsExif *ve )
char value_txt[256]; char value_txt[256];
VipsBuf value = VIPS_BUF_STATIC( value_txt ); VipsBuf value = VIPS_BUF_STATIC( value_txt );
vips_buf_appendf( &name, "exif-%s", exif_tag_get_title( entry->tag ) ); vips_buf_appendf( &name, "exif-ifd%d-%s",
exif_entry_get_ifd( entry ),
exif_tag_get_title( entry->tag ) );
vips_exif_to_s( ve->ed, entry, &value ); vips_exif_to_s( ve->ed, entry, &value );
/* Can't do anything sensible with the error return. /* Can't do anything sensible with the error return.
@ -432,29 +437,12 @@ attach_exif_content( ExifContent *content, VipsExif *ve )
(ExifContentForeachEntryFunc) attach_exif_entry, ve ); (ExifContentForeachEntryFunc) attach_exif_entry, ve );
} }
/* Just find the first occurence of the tag (is this correct?)
*/
static ExifEntry *
find_entry( ExifData *ed, ExifTag tag )
{
int i;
for( i = 0; i < EXIF_IFD_COUNT; i++ ) {
ExifEntry *entry;
if( (entry = exif_content_get_entry( ed->ifd[i], tag )) )
return( entry );
}
return( NULL );
}
static int static int
get_entry_rational( ExifData *ed, ExifTag tag, double *out ) get_entry_rational( ExifData *ed, int ifd, ExifTag tag, double *out )
{ {
ExifEntry *entry; ExifEntry *entry;
if( !(entry = find_entry( ed, tag )) || if( !(entry = exif_content_get_entry( ed->ifd[ifd], tag )) ||
entry->format != EXIF_FORMAT_RATIONAL || entry->format != EXIF_FORMAT_RATIONAL ||
entry->components != 1 ) entry->components != 1 )
return( -1 ); return( -1 );
@ -463,11 +451,11 @@ get_entry_rational( ExifData *ed, ExifTag tag, double *out )
} }
static int static int
get_entry_short( ExifData *ed, ExifTag tag, int *out ) get_entry_short( ExifData *ed, int ifd, ExifTag tag, int *out )
{ {
ExifEntry *entry; ExifEntry *entry;
if( !(entry = find_entry( ed, tag )) || if( !(entry = exif_content_get_entry( ed->ifd[ifd], tag )) ||
entry->format != EXIF_FORMAT_SHORT || entry->format != EXIF_FORMAT_SHORT ||
entry->components != 1 ) entry->components != 1 )
return( -1 ); return( -1 );
@ -481,14 +469,22 @@ set_vips_resolution( VipsImage *im, ExifData *ed )
double xres, yres; double xres, yres;
int unit; int unit;
if( get_entry_rational( ed, EXIF_TAG_X_RESOLUTION, &xres ) || /* The main image xres/yres are in ifd0. ifd1 has xres/yres of the
get_entry_rational( ed, EXIF_TAG_Y_RESOLUTION, &yres ) || * image thumbnail, if any.
get_entry_short( ed, EXIF_TAG_RESOLUTION_UNIT, &unit ) ) { */
if( get_entry_rational( ed, 0, EXIF_TAG_X_RESOLUTION, &xres ) ||
get_entry_rational( ed, 0, EXIF_TAG_Y_RESOLUTION, &yres ) ||
get_entry_short( ed, 0, EXIF_TAG_RESOLUTION_UNIT, &unit ) ) {
vips_warn( "VipsJpeg", vips_warn( "VipsJpeg",
"%s", _( "error reading resolution" ) ); "%s", _( "error reading resolution" ) );
return; return;
} }
#ifdef DEBUG
printf( "set_vips_resolution: seen exif tags "
"xres = %g, yres = %g, unit = %d\n", xres, yres, unit );
#endif /*DEBUG*/
switch( unit ) { switch( unit ) {
case 2: case 2:
/* In inches. /* In inches.
@ -701,6 +697,11 @@ read_jpeg_header( ReadJpeg *jpeg, VipsImage *out )
if( cinfo->saw_JFIF_marker && if( cinfo->saw_JFIF_marker &&
cinfo->X_density != 1U && cinfo->X_density != 1U &&
cinfo->Y_density != 1U ) { cinfo->Y_density != 1U ) {
#ifdef DEBUG
printf( "read_jpeg_header: seen jfif _density %d, %d\n",
cinfo->X_density, cinfo->Y_density );
#endif /*DEBUG*/
switch( cinfo->density_unit ) { switch( cinfo->density_unit ) {
case 1: case 1:
/* Pixels per inch. /* Pixels per inch.

View File

@ -80,6 +80,7 @@
/* /*
#define DEBUG #define DEBUG
#define VIPS_DEBUG
*/ */
#ifdef HAVE_CONFIG_H #ifdef HAVE_CONFIG_H
@ -282,6 +283,10 @@ vips_exif_set_double( ExifData *ed,
rational.numerator = value * scale; rational.numerator = value * scale;
rational.denominator = scale; rational.denominator = scale;
VIPS_DEBUG_MSG( "vips_exif_set_double: %g / %g\n",
(double) rational.numerator,
(double) rational.denominator );
exif_set_rational( entry->data + offset, bo, rational ); exif_set_rational( entry->data + offset, bo, rational );
} }
else if( entry->format == EXIF_FORMAT_SRATIONAL ) { else if( entry->format == EXIF_FORMAT_SRATIONAL ) {
@ -293,6 +298,9 @@ vips_exif_set_double( ExifData *ed,
rational.numerator = value * scale; rational.numerator = value * scale;
rational.denominator = scale; rational.denominator = scale;
VIPS_DEBUG_MSG( "vips_exif_set_double: %d / %d\n",
rational.numerator, rational.denominator );
exif_set_srational( entry->data + offset, bo, rational ); exif_set_srational( entry->data + offset, bo, rational );
} }
} }
@ -300,38 +308,26 @@ vips_exif_set_double( ExifData *ed,
typedef void (*write_fn)( ExifData *ed, typedef void (*write_fn)( ExifData *ed,
ExifEntry *entry, unsigned long component, void *data ); ExifEntry *entry, unsigned long component, void *data );
/* Write a component in a tag everywhere it appears. /* Write a tag. Update what's there, or make a new one.
*/ */
static int static int
write_tag( ExifData *ed, write_tag( ExifData *ed, int ifd,
ExifTag tag, ExifFormat format, write_fn fn, void *data ) ExifTag tag, ExifFormat format, write_fn fn, void *data )
{ {
int found;
int i;
found = 0;
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[ifd], tag )) ) {
entry->format == format ) { if( entry->format == format )
fn( ed, entry, 0, data ); fn( ed, entry, 0, data );
found = 1;
} }
} else {
if( !found ) {
/* There was no tag we could update ... make one in ifd[0].
*/
ExifEntry *entry;
entry = exif_entry_new(); entry = exif_entry_new();
/* tag must be set before calling exif_content_add_entry. /* tag must be set before calling exif_content_add_entry.
*/ */
entry->tag = tag; entry->tag = tag;
exif_content_add_entry( ed->ifd[0], entry ); exif_content_add_entry( ed->ifd[ifd], entry );
exif_entry_initialize( entry, tag ); exif_entry_initialize( entry, tag );
exif_entry_unref( entry ); exif_entry_unref( entry );
@ -351,6 +347,9 @@ set_exif_resolution( ExifData *ed, VipsImage *im )
char *p; char *p;
int unit; int unit;
VIPS_DEBUG_MSG( "set_exif_resolution: vips res of %g, %g\n",
im->Xres, im->Yres );
/* Default to inches, more progs support it. /* Default to inches, more progs support it.
*/ */
unit = 2; unit = 2;
@ -376,11 +375,14 @@ set_exif_resolution( ExifData *ed, VipsImage *im )
return( 0 ); return( 0 );
} }
if( write_tag( ed, EXIF_TAG_X_RESOLUTION, EXIF_FORMAT_RATIONAL, /* Main image xres/yres/unit are in ifd0. ifd1 has the thumbnail
* xres/yres/unit.
*/
if( write_tag( ed, 0, EXIF_TAG_X_RESOLUTION, EXIF_FORMAT_RATIONAL,
vips_exif_set_double, (void *) &xres ) || vips_exif_set_double, (void *) &xres ) ||
write_tag( ed, EXIF_TAG_Y_RESOLUTION, EXIF_FORMAT_RATIONAL, write_tag( ed, 0, EXIF_TAG_Y_RESOLUTION, EXIF_FORMAT_RATIONAL,
vips_exif_set_double, (void *) &yres ) || vips_exif_set_double, (void *) &yres ) ||
write_tag( ed, EXIF_TAG_RESOLUTION_UNIT, EXIF_FORMAT_SHORT, write_tag( ed, 0, EXIF_TAG_RESOLUTION_UNIT, EXIF_FORMAT_SHORT,
vips_exif_set_int, (void *) &unit ) ) { vips_exif_set_int, (void *) &unit ) ) {
vips_error( "VipsJpeg", vips_error( "VipsJpeg",
"%s", _( "error setting JPEG resolution" ) ); "%s", _( "error setting JPEG resolution" ) );
@ -453,7 +455,9 @@ vips_exif_update_entry( ExifEntry *entry, VipsExif *ve )
char name[256]; char name[256];
char *value; char *value;
vips_snprintf( name, 256, "exif-%s", exif_tag_get_title( entry->tag ) ); vips_snprintf( name, 256, "exif-ifd%d-%s",
exif_entry_get_ifd( entry ),
exif_tag_get_title( entry->tag ) );
if( vips_image_get_typeof( ve->image, name ) ) { if( vips_image_get_typeof( ve->image, name ) ) {
(void) vips_image_get_string( ve->image, name, &value ); (void) vips_image_get_string( ve->image, name, &value );
vips_exif_from_s( ve->ed, entry, value ); vips_exif_from_s( ve->ed, entry, value );
@ -539,7 +543,7 @@ write_exif( Write *write )
data_length = idl; data_length = idl;
#ifdef DEBUG #ifdef DEBUG
printf( "VipsJpeg: attaching %zd bytes of EXIF\n", data_length ); printf( "write_exif: attaching %zd bytes of EXIF\n", data_length );
#endif /*DEBUG*/ #endif /*DEBUG*/
exif_data_free( ed ); exif_data_free( ed );
@ -554,7 +558,7 @@ write_exif( Write *write )
return( -1 ); return( -1 );
#ifdef DEBUG #ifdef DEBUG
printf( "VipsJpeg: attaching %zd bytes of EXIF\n", printf( "write_exif: attaching %zd bytes of EXIF\n",
data_length ); data_length );
#endif /*DEBUG*/ #endif /*DEBUG*/
@ -580,7 +584,7 @@ write_xmp( Write *write )
return( -1 ); return( -1 );
#ifdef DEBUG #ifdef DEBUG
printf( "VipsJpeg: attaching %zd bytes of XMP\n", printf( "write_xmp: attaching %zd bytes of XMP\n",
data_length ); data_length );
#endif /*DEBUG*/ #endif /*DEBUG*/
@ -677,7 +681,7 @@ write_profile_file( Write *write, const char *profile )
(JOCTET *) write->profile_bytes, write->profile_length ); (JOCTET *) write->profile_bytes, write->profile_length );
#ifdef DEBUG #ifdef DEBUG
printf( "VipsJpeg: attached profile \"%s\"\n", profile ); printf( "write_profile_file: attached profile \"%s\"\n", profile );
#endif /*DEBUG*/ #endif /*DEBUG*/
return( 0 ); return( 0 );
@ -696,7 +700,7 @@ write_profile_meta( Write *write )
write_profile_data( &write->cinfo, data, data_length ); write_profile_data( &write->cinfo, data, data_length );
#ifdef DEBUG #ifdef DEBUG
printf( "VipsJpeg: attached %zd byte profile from VIPS header\n", printf( "write_profile_meta: attached %zd byte profile from header\n",
data_length ); data_length );
#endif /*DEBUG*/ #endif /*DEBUG*/

View File

@ -34,6 +34,8 @@
* - redone as a class * - redone as a class
* - warn about non-int shrinks * - warn about non-int shrinks
* - some tuning .. tried an int coordinate path, not worthwhile * - some tuning .. tried an int coordinate path, not worthwhile
* 16/11/12
* - don't change xres/yres, see comment below
*/ */
/* /*
@ -336,11 +338,13 @@ vips_shrink_build( VipsObject *object )
VIPS_DEMAND_STYLE_ANY, resample->in, NULL ); VIPS_DEMAND_STYLE_ANY, resample->in, NULL );
/* Size output. Note: we round the output width down! /* Size output. Note: we round the output width down!
*
* Don't change xres/yres, leave that to the application layer. For
* example, vipsthumbnail knows the true shrink factor (including the
* fractional part), we just see the integer part here.
*/ */
resample->out->Xsize = resample->in->Xsize / shrink->xshrink; resample->out->Xsize = resample->in->Xsize / shrink->xshrink;
resample->out->Ysize = resample->in->Ysize / shrink->yshrink; resample->out->Ysize = resample->in->Ysize / shrink->yshrink;
resample->out->Xres = resample->in->Xres / shrink->xshrink;
resample->out->Yres = resample->in->Yres / shrink->yshrink;
if( resample->out->Xsize <= 0 || if( resample->out->Xsize <= 0 ||
resample->out->Ysize <= 0 ) { resample->out->Ysize <= 0 ) {
vips_error( class->nickname, vips_error( class->nickname,