Merge branch 'master' into pdfium-experiment

This commit is contained in:
John Cupitt 2018-04-05 11:17:48 +01:00
commit 67916e8ead
16 changed files with 143 additions and 25 deletions

View File

@ -9,9 +9,13 @@
- add vips_rotate() ... a convenience method for vips_similarity() - add vips_rotate() ... a convenience method for vips_similarity()
- svgload was missing is_a [lovell] - svgload was missing is_a [lovell]
- better header sniffing for small files - better header sniffing for small files
- drop incompatible ICC profiles before save
- better hasalpha rules
- create funcs always make MULTIBAND (ie. no alpha)
12/3/18 started 8.6.4 12/3/18 started 8.6.4
- better fitting of fonts with overhanging edges, thanks Adrià - better fitting of fonts with overhanging edges, thanks Adrià
- lower stack use in radsave to help musl [Jacob Thrane Lund]
12/2/18 started 8.6.3 12/2/18 started 8.6.3
- use pkg-config to find libjpeg, if we can - use pkg-config to find libjpeg, if we can

View File

@ -1226,6 +1226,36 @@ vips_icc_ac2rc( VipsImage *in, VipsImage **out, const char *profile_filename )
return( 0 ); return( 0 );
} }
/* TRUE if a profile is compatible with an image.
*/
gboolean
vips_icc_is_compatible_profile( VipsImage *image,
void *data, size_t data_length )
{
cmsHPROFILE profile;
if( !(profile = cmsOpenProfileFromMem( data, data_length )) ) {
g_warning( "%s", _( "corrupt profile" ) );
return( FALSE );
}
if( vips_image_expected_bands( image ) !=
vips_icc_profile_needs_bands( profile ) ) {
VIPS_FREEF( cmsCloseProfile, profile );
g_warning( "%s",
_( "profile incompatible with image" ) );
return( FALSE );
}
if( vips_image_expected_sig( image ) != cmsGetColorSpace( profile ) ) {
VIPS_FREEF( cmsCloseProfile, profile );
g_warning( "%s",
_( "profile colourspace differs from image" ) );
return( FALSE );
}
return( TRUE );
}
#else /*!HAVE_LCMS2*/ #else /*!HAVE_LCMS2*/
#include <vips/vips.h> #include <vips/vips.h>
@ -1245,6 +1275,13 @@ vips_icc_ac2rc( VipsImage *in, VipsImage **out, const char *profile_filename )
return( -1 ); return( -1 );
} }
gboolean
vips_icc_is_compatible_profile( VipsImage *image,
void *data, size_t data_length )
{
return( TRUE );
}
#endif /*HAVE_LCMS*/ #endif /*HAVE_LCMS*/
/** /**

View File

@ -499,7 +499,7 @@ vips_bandjoin_const1( VipsImage *in, VipsImage **out, double c, ... )
return( result ); return( result );
} }
/* vips_addalpha: /* vips_addalpha: (method)
* @in: input image * @in: input image
* @out: (out): output image * @out: (out): output image
* @...: %NULL-terminated list of optional named arguments * @...: %NULL-terminated list of optional named arguments

View File

@ -15,6 +15,9 @@
* - gtkdoc * - gtkdoc
* 31/10/11 * 31/10/11
* - redo as a class * - redo as a class
* 3/4/18
* - always write MULTIBAND, otherwise when we join up these things it'll
* look like we have an alpha
*/ */
/* /*
@ -97,8 +100,7 @@ vips_black_build( VipsObject *object )
vips_image_init_fields( create->out, vips_image_init_fields( create->out,
black->width, black->height, black->bands, black->width, black->height, black->bands,
VIPS_FORMAT_UCHAR, VIPS_CODING_NONE, VIPS_FORMAT_UCHAR, VIPS_CODING_NONE,
black->bands == 1 ? VIPS_INTERPRETATION_MULTIBAND,
VIPS_INTERPRETATION_B_W : VIPS_INTERPRETATION_MULTIBAND,
1.0, 1.0 ); 1.0, 1.0 );
vips_image_pipelinev( create->out, vips_image_pipelinev( create->out,
VIPS_DEMAND_STYLE_ANY, NULL ); VIPS_DEMAND_STYLE_ANY, NULL );

View File

@ -134,7 +134,8 @@ vips_gaussmat_build( VipsObject *object )
vips_image_init_fields( create->out, vips_image_init_fields( create->out,
width, height, 1, width, height, 1,
VIPS_FORMAT_DOUBLE, VIPS_CODING_NONE, VIPS_INTERPRETATION_B_W, VIPS_FORMAT_DOUBLE, VIPS_CODING_NONE,
VIPS_INTERPRETATION_MULTIBAND,
1.0, 1.0 ); 1.0, 1.0 );
vips_image_pipelinev( create->out, vips_image_pipelinev( create->out,
VIPS_DEMAND_STYLE_ANY, NULL ); VIPS_DEMAND_STYLE_ANY, NULL );

View File

@ -134,7 +134,7 @@ vips_gaussnoise_build( VipsObject *object )
vips_image_init_fields( create->out, vips_image_init_fields( create->out,
gaussnoise->width, gaussnoise->height, 1, gaussnoise->width, gaussnoise->height, 1,
VIPS_FORMAT_FLOAT, VIPS_CODING_NONE, VIPS_FORMAT_FLOAT, VIPS_CODING_NONE,
VIPS_INTERPRETATION_B_W, 1.0, 1.0 ); VIPS_INTERPRETATION_MULTIBAND, 1.0, 1.0 );
vips_image_pipelinev( create->out, vips_image_pipelinev( create->out,
VIPS_DEMAND_STYLE_ANY, NULL ); VIPS_DEMAND_STYLE_ANY, NULL );

View File

@ -153,7 +153,8 @@ vips_logmat_build( VipsObject *object )
vips_image_init_fields( create->out, vips_image_init_fields( create->out,
width, height, 1, width, height, 1,
VIPS_FORMAT_DOUBLE, VIPS_CODING_NONE, VIPS_INTERPRETATION_B_W, VIPS_FORMAT_DOUBLE, VIPS_CODING_NONE,
VIPS_INTERPRETATION_MULTIBAND,
1.0, 1.0 ); 1.0, 1.0 );
vips_image_pipelinev( create->out, vips_image_pipelinev( create->out,
VIPS_DEMAND_STYLE_ANY, NULL ); VIPS_DEMAND_STYLE_ANY, NULL );

View File

@ -249,7 +249,7 @@ vips_perlin_build( VipsObject *object )
vips_image_init_fields( create->out, vips_image_init_fields( create->out,
perlin->width, perlin->height, 1, perlin->width, perlin->height, 1,
perlin->uchar ? VIPS_FORMAT_UCHAR : VIPS_FORMAT_FLOAT, perlin->uchar ? VIPS_FORMAT_UCHAR : VIPS_FORMAT_FLOAT,
VIPS_CODING_NONE, VIPS_INTERPRETATION_B_W, VIPS_CODING_NONE, VIPS_INTERPRETATION_MULTIBAND,
1.0, 1.0 ); 1.0, 1.0 );
vips_image_pipelinev( create->out, vips_image_pipelinev( create->out,
VIPS_DEMAND_STYLE_ANY, NULL ); VIPS_DEMAND_STYLE_ANY, NULL );

View File

@ -109,10 +109,9 @@ vips_point_build( VipsObject *object )
return( -1 ); return( -1 );
in = t[2]; in = t[2];
/* uchar mode always does B_W. We don't want FOURIER or /* We don't want FOURIER or whatever in this case.
* whatever in this case.
*/ */
in->Type = VIPS_INTERPRETATION_B_W; in->Type = VIPS_INTERPRETATION_MULTIBAND;
} }
if( vips_image_write( in, create->out ) ) if( vips_image_write( in, create->out ) )
@ -137,7 +136,7 @@ vips_point_class_init( VipsPointClass *class )
class->point = NULL; class->point = NULL;
class->min = -1.0; class->min = -1.0;
class->max = 1.0; class->max = 1.0;
class->interpretation = VIPS_INTERPRETATION_B_W; class->interpretation = VIPS_INTERPRETATION_MULTIBAND;
VIPS_ARG_INT( class, "width", 2, VIPS_ARG_INT( class, "width", 2,
_( "Width" ), _( "Width" ),

View File

@ -379,7 +379,8 @@ vips_text_build( VipsObject *object )
vips_image_init_fields( create->out, vips_image_init_fields( create->out,
text->bitmap.width, text->bitmap.rows, 1, text->bitmap.width, text->bitmap.rows, 1,
VIPS_FORMAT_UCHAR, VIPS_CODING_NONE, VIPS_INTERPRETATION_B_W, VIPS_FORMAT_UCHAR, VIPS_CODING_NONE,
VIPS_INTERPRETATION_MULTIBAND,
1.0, 1.0 ); 1.0, 1.0 );
vips_image_pipelinev( create->out, vips_image_pipelinev( create->out,
VIPS_DEMAND_STYLE_ANY, NULL ); VIPS_DEMAND_STYLE_ANY, NULL );

View File

@ -281,7 +281,8 @@ vips_worley_build( VipsObject *object )
vips_image_init_fields( create->out, vips_image_init_fields( create->out,
worley->width, worley->height, 1, worley->width, worley->height, 1,
VIPS_FORMAT_FLOAT, VIPS_CODING_NONE, VIPS_INTERPRETATION_B_W, VIPS_FORMAT_FLOAT, VIPS_CODING_NONE,
VIPS_INTERPRETATION_MULTIBAND,
1.0, 1.0 ); 1.0, 1.0 );
vips_image_pipelinev( create->out, vips_image_pipelinev( create->out,
VIPS_DEMAND_STYLE_ANY, NULL ); VIPS_DEMAND_STYLE_ANY, NULL );

View File

@ -22,6 +22,8 @@
* - META_SEQ support moved here * - META_SEQ support moved here
* 5/3/18 * 5/3/18
* - block _start if one start fails, see #893 * - block _start if one start fails, see #893
* 1/4/18
* - drop incompatible ICC profiles before save
*/ */
/* /*
@ -1504,6 +1506,21 @@ vips__foreign_convert_saveable( VipsImage *in, VipsImage **ready,
in = out; in = out;
} }
/* Some format libraries, like libpng, will throw a hard error if the
* profile is inappropriate for this image type. With profiles inherited
* from a source image, this can happen all the time, so we
* want to just drop the profile in this case.
*/
if( vips_image_get_typeof( in, VIPS_META_ICC_NAME ) ) {
void *data;
size_t length;
if( !vips_image_get_blob( in, VIPS_META_ICC_NAME,
&data, &length ) &&
!vips_icc_is_compatible_profile( in, data, length ) )
vips_image_remove( in, VIPS_META_ICC_NAME );
}
*ready = in; *ready = in;
return( 0 ); return( 0 );

View File

@ -19,6 +19,8 @@
* - add buffer save functions * - add buffer save functions
* 28/2/17 * 28/2/17
* - use dbuf for buffer output * - use dbuf for buffer output
* 4/4/17
* - reduce stack use to help musl
*/ */
/* /*
@ -885,10 +887,12 @@ rle_scanline_write( COLR *scanline, int width,
} }
} }
/* Write a single scanline. /* Write a single scanline. buffer is at least MAX_LINE bytes and is used to
* construct the RLE scanline. Don't allocate this on the stack so we don't
* die too horribly on small-stack libc.
*/ */
static int static int
scanline_write( COLR *scanline, int width, FILE *fp ) scanline_write( unsigned char *buffer, COLR *scanline, int width, FILE *fp )
{ {
if( width < MINELEN || if( width < MINELEN ||
width > MAXELEN ) width > MAXELEN )
@ -898,11 +902,12 @@ scanline_write( COLR *scanline, int width, FILE *fp )
else { else {
/* An RLE scanline. /* An RLE scanline.
*/ */
unsigned char buffer[MAX_LINE];
int length; int length;
rle_scanline_write( scanline, width, buffer, &length ); rle_scanline_write( scanline, width, buffer, &length );
g_assert( length <= MAX_LINE );
return( fwrite( buffer, 1, length, fp ) - length ); return( fwrite( buffer, 1, length, fp ) - length );
} }
} }
@ -1290,12 +1295,23 @@ static int
vips2rad_put_data_block( VipsRegion *region, VipsRect *area, void *a ) vips2rad_put_data_block( VipsRegion *region, VipsRect *area, void *a )
{ {
Write *write = (Write *) a; Write *write = (Write *) a;
size_t size;
unsigned char *buffer;
int i; int i;
/* You have to seek back after a write.
*/
buffer = vips_dbuf_get_write( &write->dbuf, &size );
vips_dbuf_seek( &write->dbuf, 0, SEEK_SET );
g_assert( size >= MAX_LINE );
for( i = 0; i < area->height; i++ ) { for( i = 0; i < area->height; i++ ) {
VipsPel *p = VIPS_REGION_ADDR( region, 0, area->top + i ); VipsPel *p = VIPS_REGION_ADDR( region, 0, area->top + i );
if( scanline_write( (COLR *) p, area->width, write->fout ) ) if( scanline_write( buffer,
(COLR *) p, area->width, write->fout ) )
return( -1 ); return( -1 );
} }
@ -1329,6 +1345,11 @@ vips__rad_save( VipsImage *in, const char *filename )
write->filename = vips_strdup( NULL, filename ); write->filename = vips_strdup( NULL, filename );
write->fout = vips__file_open_write( filename, FALSE ); write->fout = vips__file_open_write( filename, FALSE );
/* scanline_write() needs a buffer to write compressed scanlines to.
* We use the dbuf ... why not.
*/
vips_dbuf_allocate( &write->dbuf, MAX_LINE );
if( !write->filename || if( !write->filename ||
!write->fout || !write->fout ||
vips2rad_put_header( write ) || vips2rad_put_header( write ) ||

View File

@ -992,7 +992,8 @@ write_vips( Write *write,
/* If we're an intel byte order CPU and this is a 16bit image, we need /* If we're an intel byte order CPU and this is a 16bit image, we need
* to swap bytes. * to swap bytes.
*/ */
if( bit_depth > 8 && !vips_amiMSBfirst() ) if( bit_depth > 8 &&
!vips_amiMSBfirst() )
png_set_swap( write->pPng ); png_set_swap( write->pPng );
if( interlace ) if( interlace )

View File

@ -177,6 +177,8 @@ int vips_icc_export( VipsImage *in, VipsImage **out, ... )
__attribute__((sentinel)); __attribute__((sentinel));
int vips_icc_ac2rc( VipsImage *in, VipsImage **out, int vips_icc_ac2rc( VipsImage *in, VipsImage **out,
const char *profile_filename ); const char *profile_filename );
gboolean vips_icc_is_compatible_profile( VipsImage *image,
void *data, size_t data_length );
int vips_dE76( VipsImage *left, VipsImage *right, VipsImage **out, ... ) int vips_dE76( VipsImage *left, VipsImage *right, VipsImage **out, ... )
__attribute__((sentinel)); __attribute__((sentinel));

View File

@ -12,8 +12,10 @@
* - vips_image_write() does not ref input for non-partial images * - vips_image_write() does not ref input for non-partial images
* 29/10/16 * 29/10/16
* - add vips_image_hasalpha() * - add vips_image_hasalpha()
* 11/10/1 * 11/10/17
* - more severing for vips_image_write() * - more severing for vips_image_write()
* 3/4/18
* - better rules for hasalpha
*/ */
/* /*
@ -2866,18 +2868,47 @@ vips_image_ispartial( VipsImage *image )
* vips_image_hasalpha: (method) * vips_image_hasalpha: (method)
* @image: image to check * @image: image to check
* *
* libvips assumes an image has an alpha if it has two bands (ie. it is a * Look at an image's interpretation and see if it has extra alpha bands. For
* monochrome image with an extra band), if it has four bands (unless it's been * example, a 4-band #VIPS_INTERPRETATION_RGB would, but a six-band
* tagged as CMYK), or if it has more than four bands. * #VIPS_INTERPRETATION_MULTIBAND would not.
* *
* Return %TRUE if @image has an alpha channel. * Return %TRUE if @image has an alpha channel.
*/ */
gboolean gboolean
vips_image_hasalpha( VipsImage *image ) vips_image_hasalpha( VipsImage *image )
{ {
return( image->Bands == 2 || /* The result of hasalpha is used to turn on things like
(image->Bands == 4 && image->Type != VIPS_INTERPRETATION_CMYK) || * premultiplication, so we are rather conservative about when we
image->Bands > 4 ); * signal this. We don't want to premultiply things that should not be
* premultiplied.
*/
switch( image->Type ) {
case VIPS_INTERPRETATION_B_W:
case VIPS_INTERPRETATION_GREY16:
return( image->Bands > 1 );
case VIPS_INTERPRETATION_RGB:
case VIPS_INTERPRETATION_CMC:
case VIPS_INTERPRETATION_LCH:
case VIPS_INTERPRETATION_LABS:
case VIPS_INTERPRETATION_sRGB:
case VIPS_INTERPRETATION_YXY:
case VIPS_INTERPRETATION_XYZ:
case VIPS_INTERPRETATION_LAB:
case VIPS_INTERPRETATION_RGB16:
case VIPS_INTERPRETATION_scRGB:
case VIPS_INTERPRETATION_HSV:
return( image->Bands > 3 );
case VIPS_INTERPRETATION_CMYK:
return( image->Bands > 4 );
default:
/* We can't really infer anything about bands from things like
* HISTOGRAM or FOURIER.
*/
return( FALSE );
}
} }
/** /**