Merge branch 'master' into add-thumbnail-geometry

This commit is contained in:
John Cupitt 2017-01-17 11:51:10 +00:00
commit fb88d037fc
29 changed files with 512 additions and 83 deletions

View File

@ -19,8 +19,12 @@
VIPS_LIBRARY_AGE VIPS_LIBRARY_AGE
- better support for bscale / bzero in fits images - better support for bscale / bzero in fits images
- deprecate vips_warn() / vips_info(); use g_warning() / g_info() instead - deprecate vips_warn() / vips_info(); use g_warning() / g_info() instead
- vipsthumbnail supports much fancier geometry strings - vipsthumbnail supports much fancier geometry strings, thanks tomasc
- vips_thumbnail() has new @size option - vips_thumbnail() has new @size option
- fix --vips-cache-max etc.
- add compute reordering, plus some new API to support it:
vips_reorder_margin_hint() and vips_reorder_prepare_many(), thanks
aferrero2707
8/12/16 started 8.4.5 8/12/16 started 8.4.5
- allow libgsf-1.14.26 to help centos, thanks tdiprima - allow libgsf-1.14.26 to help centos, thanks tdiprima

View File

@ -25,6 +25,8 @@ binding](http://www.vips.ecs.soton.ac.uk/supported/current/doc/html/libvips/usin
and a [command-line and a [command-line
interface](http://www.vips.ecs.soton.ac.uk/supported/current/doc/html/libvips/using-cli.html). interface](http://www.vips.ecs.soton.ac.uk/supported/current/doc/html/libvips/using-cli.html).
Bindings are available for [Ruby](https://rubygems.org/gems/ruby-vips), Bindings are available for [Ruby](https://rubygems.org/gems/ruby-vips),
[PHP](https://github.com/jcupitt/php-vips),
[Go](https://github.com/davidbyttow/govips),
JavaScript and others. There is full JavaScript and others. There is full
[documentation](http://www.vips.ecs.soton.ac.uk/supported/current/doc/html/libvips/index.html). [documentation](http://www.vips.ecs.soton.ac.uk/supported/current/doc/html/libvips/index.html).
There are several GUIs as well, see the [VIPS There are several GUIs as well, see the [VIPS

36
TODO
View File

@ -1,39 +1,3 @@
- </>
convert k2.jpg -resize "100x100<" x.jpg
downsize, ie. factor < 1
does not resize
convert x.jpg -resize "200x200<" y.jpg
upsize, ie. factor > 1
resizes
convert k2.jpg -resize "100x100>" x.jpg
downsize, ie. factor < 1
resizes
convert x.jpg -resize "200x200>" y.jpg
upsize, ie. factor > 1
does not resize
so: < means only resize if input size is less than, > means only resize if
input size is > than
awkward to fit inside vipsthumbnail.c, since we'd have to copy/paste all the
resize calcs
have to add another param to thumbnail ... up or down, only up, only down
- vipsdisp-tiny makes stripes if the image is not tagged correctly
is one of the colourspace functions not reading format?
- not sure about utf8 error messages on win - not sure about utf8 error messages on win
- strange: - strange:

View File

@ -64,13 +64,11 @@ EXTRA_DIST = \
CLEANFILES = CLEANFILES =
all-local: install-exec-hook:
while [ ! -f libvips.la ]; do sleep 1; done; sleep 1; \
echo '/* This file is autogenerated, do not edit. */' > soname.h && \ echo '/* This file is autogenerated, do not edit. */' > soname.h && \
source libvips.la && \ source libvips.la && \
echo "#define VIPS_SONAME \"$$dlname\"" >> soname.h && \ echo "#define VIPS_SONAME \"$$dlname\"" >> soname.h && \
( cmp -s soname.h include/vips/soname.h || \ cp soname.h $(DESTDIR)$(pkgincludedir) && \
cp soname.h include/vips ) && \
rm soname.h rm soname.h
-include $(INTROSPECTION_MAKEFILE) -include $(INTROSPECTION_MAKEFILE)

View File

@ -516,11 +516,10 @@ vips_arithmetic_gen( VipsRegion *or,
/* Prepare all input regions and make buffer pointers. /* Prepare all input regions and make buffer pointers.
*/ */
for( i = 0; ir[i]; i++ ) { if( vips_reorder_prepare_many( or->im, ir, r ) )
if( vips_region_prepare( ir[i], r ) )
return( -1 ); return( -1 );
for( i = 0; ir[i]; i++ )
p[i] = (VipsPel *) VIPS_REGION_ADDR( ir[i], r->left, r->top ); p[i] = (VipsPel *) VIPS_REGION_ADDR( ir[i], r->left, r->top );
}
p[i] = NULL; p[i] = NULL;
q = (VipsPel *) VIPS_REGION_ADDR( or, r->left, r->top ); q = (VipsPel *) VIPS_REGION_ADDR( or, r->left, r->top );

View File

@ -234,8 +234,7 @@ vips_colour_gen( VipsRegion *or,
int i, y; int i, y;
VipsPel *p[MAX_INPUT_IMAGES], *q; VipsPel *p[MAX_INPUT_IMAGES], *q;
for( i = 0; ir[i]; i++ ) if( vips_reorder_prepare_many( or->im, ir, r ) )
if( vips_region_prepare( ir[i], r ) )
return( -1 ); return( -1 );
VIPS_GATE_START( "vips_colour_gen: work" ); VIPS_GATE_START( "vips_colour_gen: work" );

View File

@ -92,11 +92,10 @@ vips_bandary_gen( VipsRegion *or, void *seq, void *a, void *b, gboolean *stop )
VipsPel *p[MAX_INPUT_IMAGES], *q; VipsPel *p[MAX_INPUT_IMAGES], *q;
int y, i; int y, i;
for( i = 0; i < bandary->n; i++ ) { if( vips_reorder_prepare_many( or->im, ir, r ) )
if( vips_region_prepare( ir[i], r ) )
return( -1 ); return( -1 );
for( i = 0; i < bandary->n; i++ )
p[i] = VIPS_REGION_ADDR( ir[i], r->left, r->top ); p[i] = VIPS_REGION_ADDR( ir[i], r->left, r->top );
}
p[i] = NULL; p[i] = NULL;
q = VIPS_REGION_ADDR( or, r->left, r->top ); q = VIPS_REGION_ADDR( or, r->left, r->top );

View File

@ -283,6 +283,9 @@ vips_blend_gen( VipsRegion *or, void *seq, void *client1, void *client2,
else { else {
/* Mix of set and clear ... ask for both then and else parts /* Mix of set and clear ... ask for both then and else parts
* and interleave. * and interleave.
*
* We can't use vips_reorder_prepare_many() since we always
* want the c image first.
*/ */
if( vips_region_prepare( ir[0], r ) || if( vips_region_prepare( ir[0], r ) ||
vips_region_prepare( ir[1], r ) ) vips_region_prepare( ir[1], r ) )

View File

@ -113,6 +113,9 @@ vips_conv_build( VipsObject *object )
g_assert_not_reached(); g_assert_not_reached();
} }
vips_reorder_margin_hint( convolution->out,
convolution->M->Xsize * convolution->M->Ysize );
return( 0 ); return( 0 );
} }

View File

@ -877,6 +877,9 @@ vips_convasep_build( VipsObject *object )
convolution->out->Xoffset = 0; convolution->out->Xoffset = 0;
convolution->out->Yoffset = 0; convolution->out->Yoffset = 0;
vips_reorder_margin_hint( convolution->out,
convolution->M->Xsize * convolution->M->Ysize );
return( 0 ); return( 0 );
} }

View File

@ -125,6 +125,9 @@ vips_correlation_build( VipsObject *object )
correlation->in_ready, correlation ) ) correlation->in_ready, correlation ) )
return( -1 ); return( -1 );
vips_reorder_margin_hint( correlation->out,
correlation->ref->Xsize * correlation->ref->Ysize );
return( 0 ); return( 0 );
} }

View File

@ -121,8 +121,7 @@ vips_sharpen_generate( VipsRegion *or,
int x, y; int x, y;
if( vips_region_prepare( in[0], r ) || if( vips_reorder_prepare_many( or->im, in, r ) )
vips_region_prepare( in[1], r ) )
return( -1 ); return( -1 );
VIPS_GATE_START( "vips_sharpen_generate: work" ); VIPS_GATE_START( "vips_sharpen_generate: work" );

View File

@ -649,12 +649,11 @@ process_region( VipsRegion *or, void *seq, void *a, void *b )
/* Prepare all input regions and make buffer pointers. /* Prepare all input regions and make buffer pointers.
*/ */
for( i = 0; ir[i]; i++ ) { if( vips_reorder_prepare_many( or->im, ir, &or->valid ) )
if( vips_region_prepare( ir[i], &or->valid ) )
return( -1 ); return( -1 );
for( i = 0; ir[i]; i++ )
p[i] = (PEL *) VIPS_REGION_ADDR( ir[i], p[i] = (PEL *) VIPS_REGION_ADDR( ir[i],
or->valid.left, or->valid.top ); or->valid.left, or->valid.top );
}
p[i] = NULL; p[i] = NULL;
q = (PEL *) VIPS_REGION_ADDR( or, or->valid.left, or->valid.top ); q = (PEL *) VIPS_REGION_ADDR( or, or->valid.left, or->valid.top );

View File

@ -160,7 +160,7 @@ vips_webp_writer_appendle( VipsWebPWriter *writer, uint32_t val, int n )
unsigned char buf[4]; unsigned char buf[4];
int i; int i;
g_assert( n < 4 ); g_assert( n <= 4 );
for( i = 0; i < n; i++ ) { for( i = 0; i < n; i++ ) {
buf[i] = (unsigned char) (val & 0xff); buf[i] = (unsigned char) (val & 0xff);
@ -237,7 +237,18 @@ write_webp( WebPPicture *pic, VipsImage *in,
WebPConfig config; WebPConfig config;
webp_import import; webp_import import;
if ( !WebPConfigPreset( &config, get_preset( preset ), Q ) ) { if( !WebPConfigInit( &config ) ) {
vips_error( "vips2webp",
"%s", _( "config version error" ) );
return( -1 );
}
/* These presets are only for lossy compression. There seems to be
* separate API for lossless or near-lossless, see
* WebPConfigLosslessPreset().
*/
if( !(lossless || near_lossless) &&
!WebPConfigPreset( &config, get_preset( preset ), Q ) ) {
vips_error( "vips2webp", "%s", _( "config version error" ) ); vips_error( "vips2webp", "%s", _( "config version error" ) );
return( -1 ); return( -1 );
} }

View File

@ -280,6 +280,8 @@ vips_hist_local_build( VipsObject *object )
local->out->Xoffset = 0; local->out->Xoffset = 0;
local->out->Yoffset = 0; local->out->Yoffset = 0;
vips_reorder_margin_hint( local->out, local->width * local->height );
return( 0 ); return( 0 );
} }

View File

@ -272,6 +272,8 @@ vips_stdif_build( VipsObject *object )
stdif->out->Xoffset = 0; stdif->out->Xoffset = 0;
stdif->out->Yoffset = 0; stdif->out->Yoffset = 0;
vips_reorder_margin_hint( stdif->out, stdif->width * stdif->height );
return( 0 ); return( 0 );
} }

View File

@ -506,6 +506,12 @@ VipsImage **vips_array_image_get( VipsArrayImage *array, int *n );
VipsImage **vips_value_get_array_image( const GValue *value, int *n ); VipsImage **vips_value_get_array_image( const GValue *value, int *n );
void vips_value_set_array_image( GValue *value, int n ); void vips_value_set_array_image( GValue *value, int n );
/* Defined in reorder.c, but really a function on image.
*/
int vips_reorder_prepare_many( VipsImage *image,
struct _VipsRegion **regions, VipsRect *r );
void vips_reorder_margin_hint( VipsImage *image, int margin );
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif /*__cplusplus*/ #endif /*__cplusplus*/

View File

@ -94,11 +94,6 @@ extern int vips__info;
*/ */
extern char *vips__disc_threshold; extern char *vips__disc_threshold;
/* Cache size settings.
*/
extern char *vips__cache_max;
extern char *vips__cache_max_mem;
extern char *vips__cache_max_files;
extern gboolean vips__cache_dump; extern gboolean vips__cache_dump;
extern gboolean vips__cache_trace; extern gboolean vips__cache_trace;
@ -255,6 +250,9 @@ int vips__foreign_convert_saveable( VipsImage *in, VipsImage **ready,
int vips__image_intize( VipsImage *in, VipsImage **out ); int vips__image_intize( VipsImage *in, VipsImage **out );
void vips__reorder_init( void );
int vips__reorder_set_input( VipsImage *image, VipsImage **in );
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif /*__cplusplus*/ #endif /*__cplusplus*/

View File

@ -1,6 +1,7 @@
noinst_LTLIBRARIES = libiofuncs.la noinst_LTLIBRARIES = libiofuncs.la
libiofuncs_la_SOURCES = \ libiofuncs_la_SOURCES = \
reorder.c \
vipsmarshal.h \ vipsmarshal.h \
vipsmarshal.c \ vipsmarshal.c \
type.c \ type.c \

View File

@ -67,9 +67,6 @@
/* Set by GOption from the command line, eg. "12m". /* Set by GOption from the command line, eg. "12m".
*/ */
char *vips__cache_max = NULL;
char *vips__cache_max_mem = NULL;
char *vips__cache_max_files = NULL;
gboolean vips__cache_dump = FALSE; gboolean vips__cache_dump = FALSE;
gboolean vips__cache_trace = FALSE; gboolean vips__cache_trace = FALSE;
@ -456,18 +453,6 @@ vips__cache_init( void )
vips_cache_table = g_hash_table_new( vips_cache_table = g_hash_table_new(
(GHashFunc) vips_operation_hash, (GHashFunc) vips_operation_hash,
(GEqualFunc) vips_operation_equal ); (GEqualFunc) vips_operation_equal );
if( vips__cache_max )
vips_cache_max =
vips__parse_size( vips__cache_max );
if( vips__cache_max_mem )
vips_cache_max_mem =
vips__parse_size( vips__cache_max_mem );
if( vips__cache_max_files )
vips_cache_max_files =
vips__parse_size( vips__cache_max_files );
} }
} }

View File

@ -379,12 +379,23 @@ int
vips_image_pipeline_array( VipsImage *image, vips_image_pipeline_array( VipsImage *image,
VipsDemandStyle hint, VipsImage **in ) VipsDemandStyle hint, VipsImage **in )
{ {
/* This function can be called more than once per output image. For
* example, jpeg header load will call this once on ->out to set the
* default hint, then later call it again to connect the output image
* up to the real image.
*
* It's only ever called first time with in[0] == NULL and second time
* with a real value for @in.
*/
vips__demand_hint_array( image, hint, in ); vips__demand_hint_array( image, hint, in );
if( in[0] && if( in[0] &&
vips__image_copy_fields_array( image, in ) ) vips__image_copy_fields_array( image, in ) )
return( -1 ); return( -1 );
if( vips__reorder_set_input( image, in ) )
return( -1 );
return( 0 ); return( 0 );
} }

View File

@ -364,6 +364,10 @@ vips_init( const char *argv0 )
*/ */
vips__cache_init(); vips__cache_init();
/* Recomp reordering system.
*/
vips__reorder_init();
/* Start up packages. /* Start up packages.
*/ */
(void) vips_system_get_type(); (void) vips_system_get_type();
@ -614,6 +618,33 @@ vips_lib_version_cb( const gchar *option_name, const gchar *value,
exit( 0 ); exit( 0 );
} }
static gboolean
vips_cache_max_cb( const gchar *option_name, const gchar *value,
gpointer data, GError **error )
{
vips_cache_set_max( vips__parse_size( value ) );
return( TRUE );
}
static gboolean
vips_cache_max_memory_cb( const gchar *option_name, const gchar *value,
gpointer data, GError **error )
{
vips_cache_set_max_mem( vips__parse_size( value ) );
return( TRUE );
}
static gboolean
vips_cache_max_files_cb( const gchar *option_name, const gchar *value,
gpointer data, GError **error )
{
vips_cache_set_max_files( vips__parse_size( value ) );
return( TRUE );
}
static GOptionEntry option_entries[] = { static GOptionEntry option_entries[] = {
{ "vips-info", 0, G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG, { "vips-info", 0, G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
G_OPTION_ARG_CALLBACK, (gpointer) &vips_lib_info_cb, G_OPTION_ARG_CALLBACK, (gpointer) &vips_lib_info_cb,
@ -652,13 +683,13 @@ static GOptionEntry option_entries[] = {
G_OPTION_ARG_NONE, &vips__vector_enabled, G_OPTION_ARG_NONE, &vips__vector_enabled,
N_( "disable vectorised versions of operations" ), NULL }, N_( "disable vectorised versions of operations" ), NULL },
{ "vips-cache-max", 0, 0, { "vips-cache-max", 0, 0,
G_OPTION_ARG_STRING, &vips__cache_max, G_OPTION_ARG_CALLBACK, (gpointer) &vips_cache_max_cb,
N_( "cache at most N operations" ), "N" }, N_( "cache at most N operations" ), "N" },
{ "vips-cache-max-memory", 0, 0, { "vips-cache-max-memory", 0, 0,
G_OPTION_ARG_STRING, &vips__cache_max_mem, G_OPTION_ARG_CALLBACK, (gpointer) &vips_cache_max_memory_cb,
N_( "cache at most N bytes in memory" ), "N" }, N_( "cache at most N bytes in memory" ), "N" },
{ "vips-cache-max-files", 0, 0, { "vips-cache-max-files", 0, 0,
G_OPTION_ARG_STRING, &vips__cache_max_files, G_OPTION_ARG_CALLBACK, (gpointer) &vips_cache_max_files_cb,
N_( "allow at most N open files" ), "N" }, N_( "allow at most N open files" ), "N" },
{ "vips-cache-trace", 0, 0, { "vips-cache-trace", 0, 0,
G_OPTION_ARG_NONE, &vips__cache_trace, G_OPTION_ARG_NONE, &vips__cache_trace,

368
libvips/iofuncs/reorder.c Normal file
View File

@ -0,0 +1,368 @@
/* reorder.c ... manage reorder reordering
*
* 11/1/17
* - first version
*/
/*
This file is part of VIPS.
VIPS is free software; you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
02110-1301 USA
*/
/*
These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk
*/
/*
#define DEBUG
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif /*HAVE_CONFIG_H*/
#include <vips/intl.h>
#include <vips/vips.h>
#include <vips/internal.h>
#include <vips/debug.h>
/* Have one of these on every image, identified by a quark.
*/
typedef struct _VipsReorder {
/* The image we are attached to.
*/
VipsImage *image;
/* The direct inputs to this image, so a copy of the array that is
* passed to vips_image_pipeline_array(), and in the same order.
* NULL-terminated.
*
* Score is the priority we give to the inputs as we de-dupe the source
* arrays.
*
* The recomp order is the order we prepare regions in ... just make a
* range then sort by score.
*/
int n_inputs;
VipsImage **input;
int *score;
int *recomp_order;
/* Source images are images with no input images, so file load,
* vips_black(), etc. NULL-terminated array.
*
* The cumulative margin is the total margin that has been added to
* each source image up to this point in the pipeline.
*/
int n_sources;
VipsImage **source;
int *cumulative_margin;
} VipsReorder;
GQuark vips__image_reorder_quark = 0;
#ifdef DEBUG
static void
vips_reorder_print( VipsReorder *reorder )
{
int i;
printf( "vips_reorder_print: " );
vips_object_print_name( VIPS_OBJECT( reorder->image ) );
printf( "\n" );
printf( "n_inputs = %d\n", reorder->n_inputs );
printf( " n score order\n" );
for( i = 0; i < reorder->n_inputs; i++ ) {
printf( "%2d - %8d, %8d, ",
i, reorder->score[i], reorder->recomp_order[i] );
vips_object_print_name( VIPS_OBJECT( reorder->input[i] ) );
printf( "\n" );
}
printf( "n_sources = %d\n", reorder->n_sources );
printf( " n margin\n" );
for( i = 0; i < reorder->n_sources; i++ ) {
printf( "%2d - %8d, ",
i, reorder->cumulative_margin[i] );
vips_object_print_name( VIPS_OBJECT( reorder->source[i] ) );
printf( "\n" );
}
}
#endif /*DEBUG*/
static void
vips_reorder_destroy( VipsReorder *reorder )
{
VIPS_FREE( reorder->input );
VIPS_FREE( reorder->score );
VIPS_FREE( reorder->recomp_order );
VIPS_FREE( reorder->source );
VIPS_FREE( reorder->cumulative_margin );
VIPS_FREE( reorder );
}
static VipsReorder *
vips_reorder_get( VipsImage *image )
{
VipsReorder *reorder;
if( (reorder = g_object_get_qdata( G_OBJECT( image ),
vips__image_reorder_quark )) )
return( reorder );
reorder = VIPS_NEW( NULL, VipsReorder );
reorder->image = image;
reorder->n_inputs = 0;
reorder->input = NULL;
reorder->score = NULL;
reorder->recomp_order = NULL;
reorder->n_sources = 0;
reorder->source = NULL;
reorder->cumulative_margin = NULL;
g_object_set_qdata_full( G_OBJECT( image ), vips__image_reorder_quark,
reorder, (GDestroyNotify) vips_reorder_destroy );
return( reorder );
}
static int
vips_reorder_compare_score( const void *a, const void *b, void *arg )
{
int i1 = *((int *) a);
int i2 = *((int *) b);
VipsReorder *reorder = (VipsReorder *) arg;
return( reorder->score[i2] - reorder->score[i1] );
}
int
vips__reorder_set_input( VipsImage *image, VipsImage **in )
{
VipsReorder *reorder = vips_reorder_get( image );
int i;
int total;
/* We have to support being called more than once on the same image.
* Two cases:
*
* 1. in the first call, no images were set ... we throw away
* everything from the first call and try again. foreign can do this.
*
* 2. warn if the args were different and do nothing.
*/
if( reorder->source ) {
if( reorder->n_inputs == 0 )
reorder->n_sources = 0;
else {
for( i = 0; in[i]; i++ )
if( i >= reorder->n_inputs ||
in[i] != reorder->input[i] ) {
/* Should never happen.
*/
g_warning( "vips__reorder_set_input: "
"args differ\n" );
break;
}
return( 0 );
}
}
/* Make a copy of the input array.
*/
for( i = 0; in[i]; i++ )
;
reorder->n_inputs = i;
reorder->input = VIPS_ARRAY( NULL, reorder->n_inputs + 1, VipsImage * );
reorder->score = VIPS_ARRAY( NULL, reorder->n_inputs, int );
reorder->recomp_order = VIPS_ARRAY( NULL, reorder->n_inputs, int );
if( !reorder->input )
return( -1 );
if( reorder->n_inputs &&
(!reorder->score ||
!reorder->recomp_order) )
return( -1 );
for( i = 0; i < reorder->n_inputs; i++ ) {
reorder->input[i] = in[i];
reorder->score[i] = 0;
reorder->recomp_order[i] = i;
}
reorder->input[i] = NULL;
/* Find the total number of source images -- this gives an upper bound
* to the size of the unique source image array we will need.
*/
total = 0;
for( i = 0; i < reorder->n_inputs; i++ )
total += vips_reorder_get( reorder->input[i] )->n_sources;
/* No source images means this must itself be a source image, so it has
* a source image of itself.
*/
total = VIPS_MAX( 1, total );
reorder->source = VIPS_ARRAY( NULL, total + 1, VipsImage * );
reorder->cumulative_margin = VIPS_ARRAY( NULL, total, int );
if( !reorder->source ||
!reorder->cumulative_margin )
return( -1 );
/* Copy source images over, removing duplicates. If we find a
* duplicate, we have a reordering opportunity, and we adjust the
* scores of the two images containing the dupe.
*/
for( i = 0; i < reorder->n_inputs; i++ ) {
VipsReorder *input = vips_reorder_get( reorder->input[i] );
int j;
for( j = 0; j < input->n_sources; j++ ) {
int k;
/* Search for dupe.
*/
for( k = 0; k < reorder->n_sources; k++ )
if( reorder->source[k] == input->source[j] )
break;
if( k < reorder->n_sources ) {
/* Found a dupe. Does this new use of
* input->source[j] have a larger or smaller
* margin? Adjust the score to reflect the
* change, note the new max.
*/
reorder->score[i] +=
input->cumulative_margin[j] -
reorder->cumulative_margin[k];
reorder->cumulative_margin[k] = VIPS_MAX(
reorder->cumulative_margin[k],
input->cumulative_margin[j] );
}
else {
/* No dupe, just add to the table.
*/
reorder->source[reorder->n_sources] =
input->source[j];
reorder->cumulative_margin[reorder->n_sources] =
input->cumulative_margin[j];
reorder->n_sources += 1;
}
}
}
/* Sort recomp_order by score. qsort_r() is a GNU libc thing, don't use
* it.
*/
if( reorder->n_inputs > 1 )
g_qsort_with_data( reorder->recomp_order,
reorder->n_inputs,
sizeof( int ),
vips_reorder_compare_score, reorder );
/* No sources ... make one, us!
*/
if( reorder->n_inputs == 0 ) {
reorder->source[0] = image;
reorder->cumulative_margin[0] = 0;
reorder->n_sources = 1;
}
#ifdef DEBUG
vips_reorder_print( reorder );
#endif /*DEBUG*/
return( 0 );
}
/**
* vips_reorder_prepare_many:
* @image: the image that's being written
* @regions: the set of regions to prepare
* @r: the #VipsRect to prepare on each region
*
* vips_reorder_prepare_many() runs vips_region_prepare() on each region in
* @regions, requesting the pixels in @r.
*
* It tries to request the regions in the order which will cause least
* recomputation. This can give a large speedup, in some cases.
*
* See also: vips_region_prepare(), vips_reorder_margin_hint().
*
* Returns: 0 on success, or -1 on error.
*/
int
vips_reorder_prepare_many( VipsImage *image, VipsRegion **regions, VipsRect *r )
{
VipsReorder *reorder = vips_reorder_get( image );
int i;
for( i = 0; i < reorder->n_inputs; i++ ) {
g_assert( regions[i] );
if( vips_region_prepare( regions[reorder->recomp_order[i]], r ) )
return( -1 );
}
return( 0 );
}
/**
* vips_reorder_margin_hint:
* @image: the image to hint on
* @margin: the size of the margin this operation has added
*
* vips_reorder_margin_hint() sets a hint that @image contains a margin, that
* is, that each vips_region_prepare() on @image will request a slightly larger
* region from it's inputs. A good value for @margin is (width * height) for
* the window the operation uses.
*
* This information is used by vips_image_prepare_many() to attempt to reorder
* computations to minimise recomputation.
*
* See also: vips_image_prepare_many().
*/
void
vips_reorder_margin_hint( VipsImage *image, int margin )
{
VipsReorder *reorder = vips_reorder_get( image );
int i;
for( i = 0; i < reorder->n_sources; i++ )
reorder->cumulative_margin[i] += margin;
}
void
vips__reorder_init( void )
{
if( !vips__image_reorder_quark )
vips__image_reorder_quark =
g_quark_from_static_string( "vips-image-reorder" );
}

View File

@ -120,6 +120,9 @@ vips_morph_build( VipsObject *object )
g_assert_not_reached(); g_assert_not_reached();
} }
vips_reorder_margin_hint( morph->out,
morph->M->Xsize * morph->M->Ysize );
return( 0 ); return( 0 );
} }

View File

@ -391,6 +391,8 @@ vips_rank_build( VipsObject *object )
rank->out->Xoffset = 0; rank->out->Xoffset = 0;
rank->out->Yoffset = 0; rank->out->Yoffset = 0;
vips_reorder_margin_hint( rank->out, rank->width * rank->height );
return( 0 ); return( 0 );
} }

View File

@ -540,6 +540,8 @@ vips_reduceh_build( VipsObject *object )
in, reduceh ) ) in, reduceh ) )
return( -1 ); return( -1 );
vips_reorder_margin_hint( resample->out, reduceh->n_point );
return( 0 ); return( 0 );
} }

View File

@ -813,6 +813,8 @@ vips_reducev_raw( VipsReducev *reducev, VipsImage *in )
in, reducev ) ) in, reducev ) )
return( -1 ); return( -1 );
vips_reorder_margin_hint( resample->out, reducev->n_point );
return( 0 ); return( 0 );
} }

View File

@ -109,6 +109,30 @@ typedef struct _VipsThumbnailClass {
G_DEFINE_ABSTRACT_TYPE( VipsThumbnail, vips_thumbnail, VIPS_TYPE_OPERATION ); G_DEFINE_ABSTRACT_TYPE( VipsThumbnail, vips_thumbnail, VIPS_TYPE_OPERATION );
static void
vips_thumbnail_dispose( GObject *gobject )
{
#ifdef DEBUG
printf( "vips_thumbnail_dispose: " );
vips_object_print_name( VIPS_OBJECT( gobject ) );
printf( "\n" );
#endif /*DEBUG*/
G_OBJECT_CLASS( vips_thumbnail_parent_class )->dispose( gobject );
}
static void
vips_thumbnail_finalize( GObject *gobject )
{
#ifdef DEBUG
printf( "vips_thumbnail_finalize: " );
vips_object_print_name( VIPS_OBJECT( gobject ) );
printf( "\n" );
#endif /*DEBUG*/
G_OBJECT_CLASS( vips_thumbnail_parent_class )->finalize( gobject );
}
/* Calculate the shrink factor, taking into account auto-rotate, the fit mode, /* Calculate the shrink factor, taking into account auto-rotate, the fit mode,
* and so on. * and so on.
*/ */
@ -483,6 +507,8 @@ vips_thumbnail_class_init( VipsThumbnailClass *class )
GObjectClass *gobject_class = G_OBJECT_CLASS( class ); GObjectClass *gobject_class = G_OBJECT_CLASS( class );
VipsObjectClass *vobject_class = VIPS_OBJECT_CLASS( class ); VipsObjectClass *vobject_class = VIPS_OBJECT_CLASS( class );
gobject_class->dispose = vips_thumbnail_dispose;
gobject_class->finalize = vips_thumbnail_finalize;
gobject_class->set_property = vips_object_set_property; gobject_class->set_property = vips_object_set_property;
gobject_class->get_property = vips_object_get_property; gobject_class->get_property = vips_object_get_property;

View File

@ -361,6 +361,10 @@ main( int argc, char **argv )
textdomain( GETTEXT_PACKAGE ); textdomain( GETTEXT_PACKAGE );
setlocale( LC_ALL, "" ); setlocale( LC_ALL, "" );
/* The operation cache is not useful for processing many files.
vips_cache_set_max( 0 );
*/
/* On Windows, argv is ascii-only .. use this to get a utf-8 version of /* On Windows, argv is ascii-only .. use this to get a utf-8 version of
* the args. * the args.
*/ */