seems to work

This commit is contained in:
John Cupitt 2014-10-17 13:17:44 +01:00
parent 14d7ce1e91
commit 0a51536162
3 changed files with 68 additions and 78 deletions

View File

@ -16,6 +16,7 @@
- rename VIPS_ANGLE_180 as VIPS_ANGLE_D180 etc. to help python - rename VIPS_ANGLE_180 as VIPS_ANGLE_D180 etc. to help python
- remove cimg support, we have a gmic plugin now instead - remove cimg support, we have a gmic plugin now instead
- add support for vips8 plugins - add support for vips8 plugins
- added "autorotate" option to jpeg load
8/10/14 started 7.40.11 8/10/14 started 7.40.11
- rework extra band handling for colour functions - rework extra band handling for colour functions

View File

@ -1853,6 +1853,7 @@ vips_tiffsave( VipsImage *in, const char *filename, ... )
* *
* @shrink: shrink by this much on load * @shrink: shrink by this much on load
* @fail: fail on warnings * @fail: fail on warnings
* @autorotate: use exif Orientation tag to rotate the image during load
* *
* Read a JPEG file into a VIPS image. It can read most 8-bit JPEG images, * Read a JPEG file into a VIPS image. It can read most 8-bit JPEG images,
* including CMYK and YCbCr. * including CMYK and YCbCr.
@ -1861,10 +1862,15 @@ vips_tiffsave( VipsImage *in, const char *filename, ... )
* are 1, 2, 4 and 8. Shrinking during read is very much faster than * are 1, 2, 4 and 8. Shrinking during read is very much faster than
* decompressing the whole image and then shrinking later. * decompressing the whole image and then shrinking later.
* *
* Setting @fail to true makes the JPEG reader fail on any warnings. * Setting @fail to %TRUE makes the JPEG reader fail on any warnings.
* This can be useful for detecting truncated files, for example. Normally * This can be useful for detecting truncated files, for example. Normally
* reading these produces a warning, but no fatal error. * reading these produces a warning, but no fatal error.
* *
* Setting @autorotate to %TRUE will make the loader interpret the EXIF
* Orientation field and automatically rotate the image appropriately during
* load. After rotation, the Orientation tag will be removed to prevent
* accidental double-rotation.
*
* Example: * Example:
* *
* |[ * |[
@ -1892,8 +1898,7 @@ vips_tiffsave( VipsImage *in, const char *filename, ... )
* "jpeg-thumbnail-data". See vips_image_get_blob(). * "jpeg-thumbnail-data". See vips_image_get_blob().
* *
* This function only reads the image header and does not decompress any pixel * This function only reads the image header and does not decompress any pixel
* data. Decompression only occurs when pixels are accessed by some other * data. Decompression only occurs when pixels are accessed.
* function.
* *
* See also: vips_jpegload_buffer(), vips_image_new_from_file(). * See also: vips_jpegload_buffer(), vips_image_new_from_file().
* *

View File

@ -133,12 +133,6 @@
typedef struct _ReadJpeg { typedef struct _ReadJpeg {
VipsImage *out; VipsImage *out;
/* The raw image size: out may have width and height swapped if we are
* autorotating.
*/
int width;
int height;
/* Shrink by this much during load. 1, 2, 4, 8. /* Shrink by this much during load. 1, 2, 4, 8.
*/ */
int shrink; int shrink;
@ -167,7 +161,8 @@ typedef struct _ReadJpeg {
*/ */
int y_pos; int y_pos;
/* Use exif tags to automatically rotate and flip image. /* Use Orientation exif tag to automatically rotate and flip image
* during load.
*/ */
gboolean autorotate; gboolean autorotate;
} ReadJpeg; } ReadJpeg;
@ -933,20 +928,6 @@ read_jpeg_header( ReadJpeg *jpeg, VipsImage *out )
(VipsCallbackFn) vips_free, data, data_length ); (VipsCallbackFn) vips_free, data, data_length );
} }
/* Swap width and height if we're going to rotate this image. Keep the
* unrotated dimensions around too.
*/
jpeg->width = out->Xsize;
jpeg->height = out->Ysize;
if( jpeg->autorotate ) {
VipsAngle angle = get_angle( out );
if( angle == VIPS_ANGLE_D90 ||
angle == VIPS_ANGLE_D270 )
VIPS_SWAP( int, out->Xsize, out->Ysize )
}
return( 0 ); return( 0 );
} }
@ -1101,36 +1082,11 @@ read_jpeg_image( ReadJpeg *jpeg, VipsImage *out )
return( 0 ); return( 0 );
} }
/* Read a JPEG file into a VIPS image. /* Read the jpeg from file or buffer.
*/ */
int static int
vips__jpeg_read_file( const char *filename, VipsImage *out, vips__jpeg_read( ReadJpeg *jpeg, VipsImage *out, gboolean header_only )
gboolean header_only, int shrink, gboolean fail, gboolean readbehind,
gboolean autorotate )
{ {
ReadJpeg *jpeg;
int result;
if( !(jpeg = readjpeg_new( out,
shrink, fail, readbehind, autorotate )) )
return( -1 );
/* Here for longjmp() from vips__new_error_exit() during startup.
*/
if( setjmp( jpeg->eman.jmp ) ) {
(void) readjpeg_free( jpeg );
return( -1 );
}
/* Set input to file.
*/
if( readjpeg_file( jpeg, filename ) ) {
(void) readjpeg_free( jpeg );
return( -1 );
}
/* Need to read in APP1 (EXIF metadata), APP2 (ICC profile), APP13 /* Need to read in APP1 (EXIF metadata), APP2 (ICC profile), APP13
* (photoshop IPCT). * (photoshop IPCT).
*/ */
@ -1140,15 +1096,59 @@ vips__jpeg_read_file( const char *filename, VipsImage *out,
/* Convert! /* Convert!
*/ */
if( header_only ) if( header_only ) {
result = read_jpeg_header( jpeg, out ); if( read_jpeg_header( jpeg, out ) )
else return( -1 );
result = read_jpeg_image( jpeg, out );
/* Don't call readjpeg_free(), we're probably still live. /* Swap width and height if we're going to rotate this image.
*/ */
if( jpeg->autorotate ) {
VipsAngle angle = get_angle( out );
return( result ); if( angle == VIPS_ANGLE_D90 ||
angle == VIPS_ANGLE_D270 )
VIPS_SWAP( int, out->Xsize, out->Ysize );
/* We won't be returning an orientation tag.
*/
(void) vips_image_remove( out, ORIENTATION );
}
}
else {
if( read_jpeg_image( jpeg, out ) )
return( -1 );
}
return( 0 );
}
/* Read a JPEG file into a VIPS image.
*/
int
vips__jpeg_read_file( const char *filename, VipsImage *out,
gboolean header_only, int shrink, gboolean fail, gboolean readbehind,
gboolean autorotate )
{
ReadJpeg *jpeg;
if( !(jpeg = readjpeg_new( out,
shrink, fail, readbehind, autorotate )) )
return( -1 );
/* Here for longjmp() from vips__new_error_exit() during startup.
*/
if( setjmp( jpeg->eman.jmp ) )
return( -1 );
/* Set input to file.
*/
if( readjpeg_file( jpeg, filename ) )
return( -1 );
if( vips__jpeg_read( jpeg, out, header_only ) )
return( -1 );
return( 0 );
} }
/* Just like the above, but we read from a memory buffer. /* Just like the above, but we read from a memory buffer.
@ -1329,38 +1329,22 @@ vips__jpeg_read_buffer( void *buf, size_t len, VipsImage *out,
gboolean autorotate ) gboolean autorotate )
{ {
ReadJpeg *jpeg; ReadJpeg *jpeg;
int result;
if( !(jpeg = readjpeg_new( out, if( !(jpeg = readjpeg_new( out,
shrink, fail, readbehind, autorotate )) ) shrink, fail, readbehind, autorotate )) )
return( -1 ); return( -1 );
if( setjmp( jpeg->eman.jmp ) ) { if( setjmp( jpeg->eman.jmp ) )
(void) readjpeg_free( jpeg );
return( -1 ); return( -1 );
}
/* Set input to buffer. /* Set input to buffer.
*/ */
readjpeg_buffer( jpeg, buf, len ); readjpeg_buffer( jpeg, buf, len );
/* Need to read in APP1 (EXIF metadata) and APP2 (ICC profile). if( vips__jpeg_read( jpeg, out, header_only ) )
*/ return( -1 );
jpeg_save_markers( &jpeg->cinfo, JPEG_APP0 + 1, 0xffff );
jpeg_save_markers( &jpeg->cinfo, JPEG_APP0 + 2, 0xffff );
/* Convert! return( 0 );
*/
if( header_only )
result = read_jpeg_header( jpeg, out );
else
result = read_jpeg_image( jpeg, out );
/* Don't call readjpeg_free(), we're probably still live.
*/
return( result );
} }
int int