Merge branch 'master' into pdfium-experiment
This commit is contained in:
commit
67916e8ead
@ -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
|
||||||
|
@ -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*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -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
|
||||||
|
@ -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 );
|
||||||
|
@ -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 );
|
||||||
|
@ -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 );
|
||||||
|
|
||||||
|
@ -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 );
|
||||||
|
@ -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 );
|
||||||
|
@ -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" ),
|
||||||
|
@ -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 );
|
||||||
|
@ -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 );
|
||||||
|
@ -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 );
|
||||||
|
@ -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 ) ||
|
||||||
|
@ -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 )
|
||||||
|
@ -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));
|
||||||
|
@ -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 );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Loading…
x
Reference in New Issue
Block a user