Merge pull request #1592 from kleisauke/issue-703
Fix the pixel shift within reduce (#703)
This commit is contained in:
commit
75632a5641
@ -79,16 +79,15 @@ typedef VipsInterpolate VipsInterpolateBicubic;
|
|||||||
typedef VipsInterpolateClass VipsInterpolateBicubicClass;
|
typedef VipsInterpolateClass VipsInterpolateBicubicClass;
|
||||||
|
|
||||||
/* Precalculated interpolation matrices. int (used for pel
|
/* Precalculated interpolation matrices. int (used for pel
|
||||||
* sizes up to short), and double (for all others). We go to
|
* sizes up to short), and double (for all others).
|
||||||
* scale + 1 so we can round-to-nearest safely.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* We could keep a large set of 2d 4x4 matricies, but this actually
|
/* We could keep a large set of 2d 4x4 matricies, but this actually
|
||||||
* works out slower since for many resizes the thing will no longer
|
* works out slower since for many resizes the thing will no longer
|
||||||
* fit in L1.
|
* fit in L1.
|
||||||
*/
|
*/
|
||||||
static int vips_bicubic_matrixi[VIPS_TRANSFORM_SCALE + 1][4];
|
static int vips_bicubic_matrixi[VIPS_TRANSFORM_SCALE][4];
|
||||||
static double vips_bicubic_matrixf[VIPS_TRANSFORM_SCALE + 1][4];
|
static double vips_bicubic_matrixf[VIPS_TRANSFORM_SCALE][4];
|
||||||
|
|
||||||
/* We need C linkage for this.
|
/* We need C linkage for this.
|
||||||
*/
|
*/
|
||||||
@ -498,19 +497,13 @@ static void
|
|||||||
vips_interpolate_bicubic_interpolate( VipsInterpolate *interpolate,
|
vips_interpolate_bicubic_interpolate( VipsInterpolate *interpolate,
|
||||||
void *out, VipsRegion *in, double x, double y )
|
void *out, VipsRegion *in, double x, double y )
|
||||||
{
|
{
|
||||||
/* Find the mask index. We round-to-nearest, so we need to generate
|
/* Find the mask index.
|
||||||
* indexes in 0 to VIPS_TRANSFORM_SCALE, 2^n + 1 values. We multiply
|
|
||||||
* by 2 more than we need to, add one, mask, then shift down again to
|
|
||||||
* get the extra range.
|
|
||||||
*/
|
*/
|
||||||
const int sx = x * VIPS_TRANSFORM_SCALE * 2;
|
const int sx = x * VIPS_TRANSFORM_SCALE;
|
||||||
const int sy = y * VIPS_TRANSFORM_SCALE * 2;
|
const int sy = y * VIPS_TRANSFORM_SCALE;
|
||||||
|
|
||||||
const int six = sx & (VIPS_TRANSFORM_SCALE * 2 - 1);
|
const int tx = sx & (VIPS_TRANSFORM_SCALE - 1);
|
||||||
const int siy = sy & (VIPS_TRANSFORM_SCALE * 2 - 1);
|
const int ty = sy & (VIPS_TRANSFORM_SCALE - 1);
|
||||||
|
|
||||||
const int tx = (six + 1) >> 1;
|
|
||||||
const int ty = (siy + 1) >> 1;
|
|
||||||
|
|
||||||
/* We know x/y are always positive, so we can just (int) them.
|
/* We know x/y are always positive, so we can just (int) them.
|
||||||
*/
|
*/
|
||||||
@ -643,7 +636,7 @@ vips_interpolate_bicubic_class_init( VipsInterpolateBicubicClass *iclass )
|
|||||||
|
|
||||||
/* Build the tables of pre-computed coefficients.
|
/* Build the tables of pre-computed coefficients.
|
||||||
*/
|
*/
|
||||||
for( int x = 0; x < VIPS_TRANSFORM_SCALE + 1; x++ ) {
|
for( int x = 0; x < VIPS_TRANSFORM_SCALE; x++ ) {
|
||||||
calculate_coefficients_catmull( vips_bicubic_matrixf[x],
|
calculate_coefficients_catmull( vips_bicubic_matrixf[x],
|
||||||
(float) x / VIPS_TRANSFORM_SCALE );
|
(float) x / VIPS_TRANSFORM_SCALE );
|
||||||
|
|
||||||
|
@ -6,6 +6,8 @@
|
|||||||
* - rename xshrink -> hshrink for greater consistency
|
* - rename xshrink -> hshrink for greater consistency
|
||||||
* 9/9/16
|
* 9/9/16
|
||||||
* - add @centre option
|
* - add @centre option
|
||||||
|
* 6/6/20 kleisauke
|
||||||
|
* - deprecate @centre option, it's now always on
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -78,14 +80,10 @@
|
|||||||
* Optional arguments:
|
* Optional arguments:
|
||||||
*
|
*
|
||||||
* * @kernel: #VipsKernel to use to interpolate (default: lanczos3)
|
* * @kernel: #VipsKernel to use to interpolate (default: lanczos3)
|
||||||
* * @centre: %gboolean use centre rather than corner sampling convention
|
|
||||||
*
|
*
|
||||||
* Reduce @in vertically by a float factor. The pixels in @out are
|
* Reduce @in vertically by a float factor. The pixels in @out are
|
||||||
* interpolated with a 1D mask generated by @kernel.
|
* interpolated with a 1D mask generated by @kernel.
|
||||||
*
|
*
|
||||||
* Set @centre to use centre rather than corner sampling convention. Centre
|
|
||||||
* convention can be useful to match the behaviour of other systems.
|
|
||||||
*
|
|
||||||
* This is a very low-level operation: see vips_resize() for a more
|
* This is a very low-level operation: see vips_resize() for a more
|
||||||
* convenient way to resize images.
|
* convenient way to resize images.
|
||||||
*
|
*
|
||||||
@ -107,14 +105,10 @@
|
|||||||
* Optional arguments:
|
* Optional arguments:
|
||||||
*
|
*
|
||||||
* * @kernel: #VipsKernel to use to interpolate (default: lanczos3)
|
* * @kernel: #VipsKernel to use to interpolate (default: lanczos3)
|
||||||
* * @centre: %gboolean use centre rather than corner sampling convention
|
|
||||||
*
|
*
|
||||||
* Reduce @in horizontally by a float factor. The pixels in @out are
|
* Reduce @in horizontally by a float factor. The pixels in @out are
|
||||||
* interpolated with a 1D mask generated by @kernel.
|
* interpolated with a 1D mask generated by @kernel.
|
||||||
*
|
*
|
||||||
* Set @centre to use centre rather than corner sampling convention. Centre
|
|
||||||
* convention can be useful to match the behaviour of other systems.
|
|
||||||
*
|
|
||||||
* This is a very low-level operation: see vips_resize() for a more
|
* This is a very low-level operation: see vips_resize() for a more
|
||||||
* convenient way to resize images.
|
* convenient way to resize images.
|
||||||
*
|
*
|
||||||
@ -136,7 +130,7 @@ typedef struct _VipsReduce {
|
|||||||
*/
|
*/
|
||||||
VipsKernel kernel;
|
VipsKernel kernel;
|
||||||
|
|
||||||
/* Use centre rather than corner sampling convention.
|
/* Deprecated.
|
||||||
*/
|
*/
|
||||||
gboolean centre;
|
gboolean centre;
|
||||||
|
|
||||||
@ -152,18 +146,16 @@ vips_reduce_build( VipsObject *object )
|
|||||||
VipsResample *resample = VIPS_RESAMPLE( object );
|
VipsResample *resample = VIPS_RESAMPLE( object );
|
||||||
VipsReduce *reduce = (VipsReduce *) object;
|
VipsReduce *reduce = (VipsReduce *) object;
|
||||||
VipsImage **t = (VipsImage **)
|
VipsImage **t = (VipsImage **)
|
||||||
vips_object_local_array( object, 3 );
|
vips_object_local_array( object, 2 );
|
||||||
|
|
||||||
if( VIPS_OBJECT_CLASS( vips_reduce_parent_class )->build( object ) )
|
if( VIPS_OBJECT_CLASS( vips_reduce_parent_class )->build( object ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
if( vips_reducev( resample->in, &t[0], reduce->vshrink,
|
if( vips_reducev( resample->in, &t[0], reduce->vshrink,
|
||||||
"kernel", reduce->kernel,
|
"kernel", reduce->kernel,
|
||||||
"centre", reduce->centre,
|
|
||||||
NULL ) ||
|
NULL ) ||
|
||||||
vips_reduceh( t[0], &t[1], reduce->hshrink,
|
vips_reduceh( t[0], &t[1], reduce->hshrink,
|
||||||
"kernel", reduce->kernel,
|
"kernel", reduce->kernel,
|
||||||
"centre", reduce->centre,
|
|
||||||
NULL ) ||
|
NULL ) ||
|
||||||
vips_image_write( t[1], resample->out ) )
|
vips_image_write( t[1], resample->out ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -210,13 +202,6 @@ vips_reduce_class_init( VipsReduceClass *class )
|
|||||||
G_STRUCT_OFFSET( VipsReduce, kernel ),
|
G_STRUCT_OFFSET( VipsReduce, kernel ),
|
||||||
VIPS_TYPE_KERNEL, VIPS_KERNEL_LANCZOS3 );
|
VIPS_TYPE_KERNEL, VIPS_KERNEL_LANCZOS3 );
|
||||||
|
|
||||||
VIPS_ARG_BOOL( class, "centre", 7,
|
|
||||||
_( "Centre" ),
|
|
||||||
_( "Use centre sampling convention" ),
|
|
||||||
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
|
||||||
G_STRUCT_OFFSET( VipsReduce, centre ),
|
|
||||||
FALSE );
|
|
||||||
|
|
||||||
/* The old names .. now use h and v everywhere.
|
/* The old names .. now use h and v everywhere.
|
||||||
*/
|
*/
|
||||||
VIPS_ARG_DOUBLE( class, "xshrink", 8,
|
VIPS_ARG_DOUBLE( class, "xshrink", 8,
|
||||||
@ -233,6 +218,15 @@ vips_reduce_class_init( VipsReduceClass *class )
|
|||||||
G_STRUCT_OFFSET( VipsReduce, vshrink ),
|
G_STRUCT_OFFSET( VipsReduce, vshrink ),
|
||||||
1.0, 1000000.0, 1.0 );
|
1.0, 1000000.0, 1.0 );
|
||||||
|
|
||||||
|
/* We used to let people pick centre or corner, but it's automatic now.
|
||||||
|
*/
|
||||||
|
VIPS_ARG_BOOL( class, "centre", 7,
|
||||||
|
_( "Centre" ),
|
||||||
|
_( "Use centre sampling convention" ),
|
||||||
|
VIPS_ARGUMENT_OPTIONAL_INPUT | VIPS_ARGUMENT_DEPRECATED,
|
||||||
|
G_STRUCT_OFFSET( VipsReduce, centre ),
|
||||||
|
FALSE );
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -252,14 +246,10 @@ vips_reduce_init( VipsReduce *reduce )
|
|||||||
* Optional arguments:
|
* Optional arguments:
|
||||||
*
|
*
|
||||||
* * @kernel: #VipsKernel to use to interpolate (default: lanczos3)
|
* * @kernel: #VipsKernel to use to interpolate (default: lanczos3)
|
||||||
* * @centre: %gboolean use centre rather than corner sampling convention
|
|
||||||
*
|
*
|
||||||
* Reduce @in by a pair of factors with a pair of 1D kernels. This
|
* Reduce @in by a pair of factors with a pair of 1D kernels. This
|
||||||
* will not work well for shrink factors greater than three.
|
* will not work well for shrink factors greater than three.
|
||||||
*
|
*
|
||||||
* Set @centre to use centre rather than corner sampling convention. Centre
|
|
||||||
* convention can be useful to match the behaviour of other systems.
|
|
||||||
*
|
|
||||||
* This is a very low-level operation: see vips_resize() for a more
|
* This is a very low-level operation: see vips_resize() for a more
|
||||||
* convenient way to resize images.
|
* convenient way to resize images.
|
||||||
*
|
*
|
||||||
|
@ -8,6 +8,10 @@
|
|||||||
* - rename xshrink as hshrink for consistency
|
* - rename xshrink as hshrink for consistency
|
||||||
* 9/9/16
|
* 9/9/16
|
||||||
* - add @centre option
|
* - add @centre option
|
||||||
|
* 6/6/20 kleisauke
|
||||||
|
* - deprecate @centre option, it's now always on
|
||||||
|
* - fix pixel shift
|
||||||
|
* - remove unnecessary round-to-nearest behaviour
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -66,20 +70,23 @@ typedef struct _VipsReduceh {
|
|||||||
*/
|
*/
|
||||||
VipsKernel kernel;
|
VipsKernel kernel;
|
||||||
|
|
||||||
/* Use centre rather than corner sampling convention.
|
|
||||||
*/
|
|
||||||
gboolean centre;
|
|
||||||
|
|
||||||
/* Number of points in kernel.
|
/* Number of points in kernel.
|
||||||
*/
|
*/
|
||||||
int n_point;
|
int n_point;
|
||||||
|
|
||||||
/* Precalculated interpolation matrices. int (used for pel
|
/* Horizontal displacement.
|
||||||
* sizes up to short), and double (for all others). We go to
|
|
||||||
* scale + 1 so we can round-to-nearest safely.
|
|
||||||
*/
|
*/
|
||||||
int *matrixi[VIPS_TRANSFORM_SCALE + 1];
|
double hoffset;
|
||||||
double *matrixf[VIPS_TRANSFORM_SCALE + 1];
|
|
||||||
|
/* Precalculated interpolation matrices. int (used for pel
|
||||||
|
* sizes up to short), and double (for all others).
|
||||||
|
*/
|
||||||
|
int *matrixi[VIPS_TRANSFORM_SCALE];
|
||||||
|
double *matrixf[VIPS_TRANSFORM_SCALE];
|
||||||
|
|
||||||
|
/* Deprecated.
|
||||||
|
*/
|
||||||
|
gboolean centre;
|
||||||
|
|
||||||
} VipsReduceh;
|
} VipsReduceh;
|
||||||
|
|
||||||
@ -194,13 +201,11 @@ reduceh_signed_int_tab( VipsReduceh *reduceh,
|
|||||||
for( int z = 0; z < bands; z++ ) {
|
for( int z = 0; z < bands; z++ ) {
|
||||||
int sum;
|
int sum;
|
||||||
|
|
||||||
sum = reduce_sum<T, int>( in, bands, cx, n );
|
sum = reduce_sum<T, int>( in + z, bands, cx, n );
|
||||||
sum = signed_fixed_round( sum );
|
sum = signed_fixed_round( sum );
|
||||||
sum = VIPS_CLIP( min_value, sum, max_value );
|
sum = VIPS_CLIP( min_value, sum, max_value );
|
||||||
|
|
||||||
out[z] = sum;
|
out[z] = sum;
|
||||||
|
|
||||||
in += 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -216,10 +221,8 @@ reduceh_float_tab( VipsReduceh *reduceh,
|
|||||||
const T* restrict in = (T *) pin;
|
const T* restrict in = (T *) pin;
|
||||||
const int n = reduceh->n_point;
|
const int n = reduceh->n_point;
|
||||||
|
|
||||||
for( int z = 0; z < bands; z++ ) {
|
for( int z = 0; z < bands; z++ )
|
||||||
out[z] = reduce_sum<T, double>( in, bands, cx, n );
|
out[z] = reduce_sum<T, double>( in + z, bands, cx, n );
|
||||||
in += 1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 32-bit int output needs a double intermediate.
|
/* 32-bit int output needs a double intermediate.
|
||||||
@ -238,10 +241,8 @@ reduceh_unsigned_int32_tab( VipsReduceh *reduceh,
|
|||||||
for( int z = 0; z < bands; z++ ) {
|
for( int z = 0; z < bands; z++ ) {
|
||||||
double sum;
|
double sum;
|
||||||
|
|
||||||
sum = reduce_sum<T, double>( in, bands, cx, n );
|
sum = reduce_sum<T, double>( in + z, bands, cx, n );
|
||||||
out[z] = VIPS_CLIP( 0, sum, max_value );
|
out[z] = VIPS_CLIP( 0, sum, max_value );
|
||||||
|
|
||||||
in += 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -258,11 +259,9 @@ reduceh_signed_int32_tab( VipsReduceh *reduceh,
|
|||||||
for( int z = 0; z < bands; z++ ) {
|
for( int z = 0; z < bands; z++ ) {
|
||||||
double sum;
|
double sum;
|
||||||
|
|
||||||
sum = reduce_sum<T, double>( in, bands, cx, n );
|
sum = reduce_sum<T, double>( in + z, bands, cx, n );
|
||||||
sum = VIPS_CLIP( min_value, sum, max_value );
|
sum = VIPS_CLIP( min_value, sum, max_value );
|
||||||
out[z] = sum;
|
out[z] = sum;
|
||||||
|
|
||||||
in += 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -282,11 +281,8 @@ reduceh_notab( VipsReduceh *reduceh,
|
|||||||
|
|
||||||
vips_reduce_make_mask( cx, reduceh->kernel, reduceh->hshrink, x );
|
vips_reduce_make_mask( cx, reduceh->kernel, reduceh->hshrink, x );
|
||||||
|
|
||||||
for( int z = 0; z < bands; z++ ) {
|
for( int z = 0; z < bands; z++ )
|
||||||
out[z] = reduce_sum<T, double>( in, bands, cx, n );
|
out[z] = reduce_sum<T, double>( in + z, bands, cx, n );
|
||||||
|
|
||||||
in += 1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Tried a vector path (see reducev) but it was slower. The vectors for
|
/* Tried a vector path (see reducev) but it was slower. The vectors for
|
||||||
@ -315,18 +311,16 @@ vips_reduceh_gen( VipsRegion *out_region, void *seq,
|
|||||||
r->width, r->height, r->left, r->top );
|
r->width, r->height, r->left, r->top );
|
||||||
#endif /*DEBUG*/
|
#endif /*DEBUG*/
|
||||||
|
|
||||||
s.left = r->left * reduceh->hshrink;
|
s.left = r->left * reduceh->hshrink - reduceh->hoffset;
|
||||||
s.top = r->top;
|
s.top = r->top;
|
||||||
s.width = r->width * reduceh->hshrink + reduceh->n_point;
|
s.width = r->width * reduceh->hshrink + reduceh->n_point;
|
||||||
s.height = r->height;
|
s.height = r->height;
|
||||||
if( reduceh->centre )
|
|
||||||
s.width += 1;
|
|
||||||
if( vips_region_prepare( ir, &s ) )
|
if( vips_region_prepare( ir, &s ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
VIPS_GATE_START( "vips_reduceh_gen: work" );
|
VIPS_GATE_START( "vips_reduceh_gen: work" );
|
||||||
|
|
||||||
for( int y = 0; y < r->height; y ++ ) {
|
for( int y = 0; y < r->height; y++ ) {
|
||||||
VipsPel *p0;
|
VipsPel *p0;
|
||||||
VipsPel *q;
|
VipsPel *q;
|
||||||
|
|
||||||
@ -334,9 +328,8 @@ vips_reduceh_gen( VipsRegion *out_region, void *seq,
|
|||||||
|
|
||||||
q = VIPS_REGION_ADDR( out_region, r->left, r->top + y );
|
q = VIPS_REGION_ADDR( out_region, r->left, r->top + y );
|
||||||
|
|
||||||
X = r->left * reduceh->hshrink;
|
X = (r->left + 0.5) * reduceh->hshrink - 0.5 -
|
||||||
if( reduceh->centre )
|
reduceh->hoffset;
|
||||||
X += 0.5;
|
|
||||||
|
|
||||||
/* We want p0 to be the start (ie. x == 0) of the input
|
/* We want p0 to be the start (ie. x == 0) of the input
|
||||||
* scanline we are reading from. We can then calculate the p we
|
* scanline we are reading from. We can then calculate the p we
|
||||||
@ -351,11 +344,10 @@ vips_reduceh_gen( VipsRegion *out_region, void *seq,
|
|||||||
ir->valid.left * ps;
|
ir->valid.left * ps;
|
||||||
|
|
||||||
for( int x = 0; x < r->width; x++ ) {
|
for( int x = 0; x < r->width; x++ ) {
|
||||||
int ix = (int) X;
|
const int ix = (int) X;
|
||||||
VipsPel *p = p0 + ix * ps;
|
VipsPel *p = p0 + ix * ps;
|
||||||
const int sx = X * VIPS_TRANSFORM_SCALE * 2;
|
const int sx = X * VIPS_TRANSFORM_SCALE;
|
||||||
const int six = sx & (VIPS_TRANSFORM_SCALE * 2 - 1);
|
const int tx = sx & (VIPS_TRANSFORM_SCALE - 1);
|
||||||
const int tx = (six + 1) >> 1;
|
|
||||||
const int *cxi = reduceh->matrixi[tx];
|
const int *cxi = reduceh->matrixi[tx];
|
||||||
const double *cxf = reduceh->matrixf[tx];
|
const double *cxf = reduceh->matrixf[tx];
|
||||||
|
|
||||||
@ -441,7 +433,7 @@ vips_reduceh_build( VipsObject *object )
|
|||||||
vips_object_local_array( object, 2 );
|
vips_object_local_array( object, 2 );
|
||||||
|
|
||||||
VipsImage *in;
|
VipsImage *in;
|
||||||
int width;
|
double width, extra_pixels;
|
||||||
|
|
||||||
if( VIPS_OBJECT_CLASS( vips_reduceh_parent_class )->build( object ) )
|
if( VIPS_OBJECT_CLASS( vips_reduceh_parent_class )->build( object ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -457,8 +449,6 @@ vips_reduceh_build( VipsObject *object )
|
|||||||
if( reduceh->hshrink == 1 )
|
if( reduceh->hshrink == 1 )
|
||||||
return( vips_image_write( in, resample->out ) );
|
return( vips_image_write( in, resample->out ) );
|
||||||
|
|
||||||
/* Build the tables of pre-computed coefficients.
|
|
||||||
*/
|
|
||||||
reduceh->n_point =
|
reduceh->n_point =
|
||||||
vips_reduce_get_points( reduceh->kernel, reduceh->hshrink );
|
vips_reduce_get_points( reduceh->kernel, reduceh->hshrink );
|
||||||
g_info( "reduceh: %d point mask", reduceh->n_point );
|
g_info( "reduceh: %d point mask", reduceh->n_point );
|
||||||
@ -467,7 +457,29 @@ vips_reduceh_build( VipsObject *object )
|
|||||||
"%s", _( "reduce factor too large" ) );
|
"%s", _( "reduce factor too large" ) );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
for( int x = 0; x < VIPS_TRANSFORM_SCALE + 1; x++ ) {
|
|
||||||
|
/* Output size. We need to always round to nearest, so round(), not
|
||||||
|
* rint().
|
||||||
|
*/
|
||||||
|
width = VIPS_ROUND_UINT(
|
||||||
|
(double) resample->in->Xsize / reduceh->hshrink );
|
||||||
|
|
||||||
|
/* How many pixels we are inventing in the input, -ve for
|
||||||
|
* discarding.
|
||||||
|
*/
|
||||||
|
extra_pixels =
|
||||||
|
width * reduceh->hshrink - resample->in->Xsize;
|
||||||
|
|
||||||
|
/* If we are rounding down, we are not using some input
|
||||||
|
* pixels. We need to move the origin *inside* the input image
|
||||||
|
* by half that distance so that we discard pixels equally
|
||||||
|
* from left and right.
|
||||||
|
*/
|
||||||
|
reduceh->hoffset = (1 + extra_pixels) / 2.0;
|
||||||
|
|
||||||
|
/* Build the tables of pre-computed coefficients.
|
||||||
|
*/
|
||||||
|
for( int x = 0; x < VIPS_TRANSFORM_SCALE; x++ ) {
|
||||||
reduceh->matrixf[x] =
|
reduceh->matrixf[x] =
|
||||||
VIPS_ARRAY( object, reduceh->n_point, double );
|
VIPS_ARRAY( object, reduceh->n_point, double );
|
||||||
reduceh->matrixi[x] =
|
reduceh->matrixi[x] =
|
||||||
@ -499,15 +511,10 @@ vips_reduceh_build( VipsObject *object )
|
|||||||
in = t[0];
|
in = t[0];
|
||||||
|
|
||||||
/* Add new pixels around the input so we can interpolate at the edges.
|
/* Add new pixels around the input so we can interpolate at the edges.
|
||||||
* In centre mode, we read 0.5 pixels more to the right, so we must
|
|
||||||
* enlarge a little further.
|
|
||||||
*/
|
*/
|
||||||
width = in->Xsize + reduceh->n_point - 1;
|
|
||||||
if( reduceh->centre )
|
|
||||||
width += 1;
|
|
||||||
if( vips_embed( in, &t[1],
|
if( vips_embed( in, &t[1],
|
||||||
reduceh->n_point / 2 - 1, 0,
|
reduceh->n_point / 2 - 1, 0,
|
||||||
width, in->Ysize,
|
in->Xsize + reduceh->n_point, in->Ysize,
|
||||||
"extend", VIPS_EXTEND_COPY,
|
"extend", VIPS_EXTEND_COPY,
|
||||||
(void *) NULL ) )
|
(void *) NULL ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -524,8 +531,7 @@ vips_reduceh_build( VipsObject *object )
|
|||||||
* example, vipsthumbnail knows the true reduce factor (including the
|
* example, vipsthumbnail knows the true reduce factor (including the
|
||||||
* fractional part), we just see the integer part here.
|
* fractional part), we just see the integer part here.
|
||||||
*/
|
*/
|
||||||
resample->out->Xsize = VIPS_ROUND_UINT(
|
resample->out->Xsize = width;
|
||||||
resample->in->Xsize / reduceh->hshrink );
|
|
||||||
if( resample->out->Xsize <= 0 ) {
|
if( resample->out->Xsize <= 0 ) {
|
||||||
vips_error( object_class->nickname,
|
vips_error( object_class->nickname,
|
||||||
"%s", _( "image has shrunk to nothing" ) );
|
"%s", _( "image has shrunk to nothing" ) );
|
||||||
@ -574,20 +580,13 @@ vips_reduceh_class_init( VipsReducehClass *reduceh_class )
|
|||||||
G_STRUCT_OFFSET( VipsReduceh, hshrink ),
|
G_STRUCT_OFFSET( VipsReduceh, hshrink ),
|
||||||
1, 1000000, 1 );
|
1, 1000000, 1 );
|
||||||
|
|
||||||
VIPS_ARG_ENUM( reduceh_class, "kernel", 3,
|
VIPS_ARG_ENUM( reduceh_class, "kernel", 4,
|
||||||
_( "Kernel" ),
|
_( "Kernel" ),
|
||||||
_( "Resampling kernel" ),
|
_( "Resampling kernel" ),
|
||||||
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
||||||
G_STRUCT_OFFSET( VipsReduceh, kernel ),
|
G_STRUCT_OFFSET( VipsReduceh, kernel ),
|
||||||
VIPS_TYPE_KERNEL, VIPS_KERNEL_LANCZOS3 );
|
VIPS_TYPE_KERNEL, VIPS_KERNEL_LANCZOS3 );
|
||||||
|
|
||||||
VIPS_ARG_BOOL( reduceh_class, "centre", 7,
|
|
||||||
_( "Centre" ),
|
|
||||||
_( "Use centre sampling convention" ),
|
|
||||||
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
|
||||||
G_STRUCT_OFFSET( VipsReduceh, centre ),
|
|
||||||
FALSE );
|
|
||||||
|
|
||||||
/* Old name.
|
/* Old name.
|
||||||
*/
|
*/
|
||||||
VIPS_ARG_DOUBLE( reduceh_class, "xshrink", 3,
|
VIPS_ARG_DOUBLE( reduceh_class, "xshrink", 3,
|
||||||
@ -597,6 +596,15 @@ vips_reduceh_class_init( VipsReducehClass *reduceh_class )
|
|||||||
G_STRUCT_OFFSET( VipsReduceh, hshrink ),
|
G_STRUCT_OFFSET( VipsReduceh, hshrink ),
|
||||||
1, 1000000, 1 );
|
1, 1000000, 1 );
|
||||||
|
|
||||||
|
/* We used to let people pick centre or corner, but it's automatic now.
|
||||||
|
*/
|
||||||
|
VIPS_ARG_BOOL( reduceh_class, "centre", 7,
|
||||||
|
_( "Centre" ),
|
||||||
|
_( "Use centre sampling convention" ),
|
||||||
|
VIPS_ARGUMENT_OPTIONAL_INPUT | VIPS_ARGUMENT_DEPRECATED,
|
||||||
|
G_STRUCT_OFFSET( VipsReduceh, centre ),
|
||||||
|
FALSE );
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -17,6 +17,11 @@
|
|||||||
* - add @centre option
|
* - add @centre option
|
||||||
* 7/3/17
|
* 7/3/17
|
||||||
* - add a seq line cache
|
* - add a seq line cache
|
||||||
|
* 6/6/20 kleisauke
|
||||||
|
* - deprecate @centre option, it's now always on
|
||||||
|
* - fix pixel shift
|
||||||
|
* - speed up the mask construction for uchar/ushort images
|
||||||
|
* - remove unnecessary round-to-nearest behaviour
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -104,30 +109,33 @@ typedef struct _VipsReducev {
|
|||||||
*/
|
*/
|
||||||
VipsKernel kernel;
|
VipsKernel kernel;
|
||||||
|
|
||||||
/* Use centre rather than corner sampling convention.
|
|
||||||
*/
|
|
||||||
gboolean centre;
|
|
||||||
|
|
||||||
/* Number of points in kernel.
|
/* Number of points in kernel.
|
||||||
*/
|
*/
|
||||||
int n_point;
|
int n_point;
|
||||||
|
|
||||||
/* Precalculated interpolation matrices. int (used for pel
|
/* Vertical displacement.
|
||||||
* sizes up to short), and double (for all others). We go to
|
|
||||||
* scale + 1 so we can round-to-nearest safely.
|
|
||||||
*/
|
*/
|
||||||
int *matrixi[VIPS_TRANSFORM_SCALE + 1];
|
double voffset;
|
||||||
double *matrixf[VIPS_TRANSFORM_SCALE + 1];
|
|
||||||
|
/* Precalculated interpolation matrices. int (used for pel
|
||||||
|
* sizes up to short), and double (for all others).
|
||||||
|
*/
|
||||||
|
int *matrixi[VIPS_TRANSFORM_SCALE];
|
||||||
|
double *matrixf[VIPS_TRANSFORM_SCALE];
|
||||||
|
|
||||||
/* And another set for orc: we want 2.6 precision.
|
/* And another set for orc: we want 2.6 precision.
|
||||||
*/
|
*/
|
||||||
int *matrixo[VIPS_TRANSFORM_SCALE + 1];
|
int *matrixo[VIPS_TRANSFORM_SCALE];
|
||||||
|
|
||||||
/* The passes we generate for this mask.
|
/* The passes we generate for this mask.
|
||||||
*/
|
*/
|
||||||
int n_pass;
|
int n_pass;
|
||||||
Pass pass[MAX_PASS];
|
Pass pass[MAX_PASS];
|
||||||
|
|
||||||
|
/* Deprecated.
|
||||||
|
*/
|
||||||
|
gboolean centre;
|
||||||
|
|
||||||
} VipsReducev;
|
} VipsReducev;
|
||||||
|
|
||||||
typedef VipsResampleClass VipsReducevClass;
|
typedef VipsResampleClass VipsReducevClass;
|
||||||
@ -146,7 +154,7 @@ vips_reducev_finalize( GObject *gobject )
|
|||||||
for( int i = 0; i < reducev->n_pass; i++ )
|
for( int i = 0; i < reducev->n_pass; i++ )
|
||||||
VIPS_FREEF( vips_vector_free, reducev->pass[i].vector );
|
VIPS_FREEF( vips_vector_free, reducev->pass[i].vector );
|
||||||
reducev->n_pass = 0;
|
reducev->n_pass = 0;
|
||||||
for( int i = 0; i < VIPS_TRANSFORM_SCALE + 1; i++ ) {
|
for( int i = 0; i < VIPS_TRANSFORM_SCALE; i++ ) {
|
||||||
VIPS_FREE( reducev->matrixf[i] );
|
VIPS_FREE( reducev->matrixf[i] );
|
||||||
VIPS_FREE( reducev->matrixi[i] );
|
VIPS_FREE( reducev->matrixi[i] );
|
||||||
VIPS_FREE( reducev->matrixo[i] );
|
VIPS_FREE( reducev->matrixo[i] );
|
||||||
@ -531,25 +539,24 @@ vips_reducev_gen( VipsRegion *out_region, void *vseq,
|
|||||||
#endif /*DEBUG*/
|
#endif /*DEBUG*/
|
||||||
|
|
||||||
s.left = r->left;
|
s.left = r->left;
|
||||||
s.top = r->top * reducev->vshrink;
|
s.top = r->top * reducev->vshrink - reducev->voffset;
|
||||||
s.width = r->width;
|
s.width = r->width;
|
||||||
s.height = r->height * reducev->vshrink + reducev->n_point;
|
s.height = r->height * reducev->vshrink + reducev->n_point;
|
||||||
if( reducev->centre )
|
|
||||||
s.height += 1;
|
|
||||||
if( vips_region_prepare( ir, &s ) )
|
if( vips_region_prepare( ir, &s ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
VIPS_GATE_START( "vips_reducev_gen: work" );
|
VIPS_GATE_START( "vips_reducev_gen: work" );
|
||||||
|
|
||||||
for( int y = 0; y < r->height; y ++ ) {
|
double Y = (r->top + 0.5) * reducev->vshrink - 0.5 -
|
||||||
|
reducev->voffset;
|
||||||
|
|
||||||
|
for( int y = 0; y < r->height; y++ ) {
|
||||||
VipsPel *q =
|
VipsPel *q =
|
||||||
VIPS_REGION_ADDR( out_region, r->left, r->top + y );
|
VIPS_REGION_ADDR( out_region, r->left, r->top + y );
|
||||||
const double Y = (r->top + y) * reducev->vshrink +
|
const int py = (int) Y;
|
||||||
(reducev->centre ? 0.5 : 0.0);
|
VipsPel *p = VIPS_REGION_ADDR( ir, r->left, py );
|
||||||
VipsPel *p = VIPS_REGION_ADDR( ir, r->left, (int) Y );
|
const int sy = Y * VIPS_TRANSFORM_SCALE;
|
||||||
const int sy = Y * VIPS_TRANSFORM_SCALE * 2;
|
const int ty = sy & (VIPS_TRANSFORM_SCALE - 1);
|
||||||
const int siy = sy & (VIPS_TRANSFORM_SCALE * 2 - 1);
|
|
||||||
const int ty = (siy + 1) >> 1;
|
|
||||||
const int *cyi = reducev->matrixi[ty];
|
const int *cyi = reducev->matrixi[ty];
|
||||||
const double *cyf = reducev->matrixf[ty];
|
const double *cyf = reducev->matrixf[ty];
|
||||||
const int lskip = VIPS_REGION_LSKIP( ir );
|
const int lskip = VIPS_REGION_LSKIP( ir );
|
||||||
@ -606,13 +613,15 @@ vips_reducev_gen( VipsRegion *out_region, void *vseq,
|
|||||||
case VIPS_FORMAT_DPCOMPLEX:
|
case VIPS_FORMAT_DPCOMPLEX:
|
||||||
case VIPS_FORMAT_DOUBLE:
|
case VIPS_FORMAT_DOUBLE:
|
||||||
reducev_notab<double>( reducev,
|
reducev_notab<double>( reducev,
|
||||||
q, p, ne, lskip, Y - (int) Y );
|
q, p, ne, lskip, Y - py );
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
g_assert_not_reached();
|
g_assert_not_reached();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Y += reducev->vshrink;
|
||||||
}
|
}
|
||||||
|
|
||||||
VIPS_GATE_STOP( "vips_reducev_gen: work" );
|
VIPS_GATE_STOP( "vips_reducev_gen: work" );
|
||||||
@ -644,11 +653,9 @@ vips_reducev_vector_gen( VipsRegion *out_region, void *vseq,
|
|||||||
#endif /*DEBUG_PIXELS*/
|
#endif /*DEBUG_PIXELS*/
|
||||||
|
|
||||||
s.left = r->left;
|
s.left = r->left;
|
||||||
s.top = r->top * reducev->vshrink;
|
s.top = r->top * reducev->vshrink - reducev->voffset;
|
||||||
s.width = r->width;
|
s.width = r->width;
|
||||||
s.height = r->height * reducev->vshrink + reducev->n_point;
|
s.height = r->height * reducev->vshrink + reducev->n_point;
|
||||||
if( reducev->centre )
|
|
||||||
s.height += 1;
|
|
||||||
if( vips_region_prepare( ir, &s ) )
|
if( vips_region_prepare( ir, &s ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
@ -663,15 +670,15 @@ vips_reducev_vector_gen( VipsRegion *out_region, void *vseq,
|
|||||||
|
|
||||||
VIPS_GATE_START( "vips_reducev_vector_gen: work" );
|
VIPS_GATE_START( "vips_reducev_vector_gen: work" );
|
||||||
|
|
||||||
for( int y = 0; y < r->height; y ++ ) {
|
double Y = (r->top + 0.5) * reducev->vshrink - 0.5 -
|
||||||
|
reducev->voffset;
|
||||||
|
|
||||||
|
for( int y = 0; y < r->height; y++ ) {
|
||||||
VipsPel *q =
|
VipsPel *q =
|
||||||
VIPS_REGION_ADDR( out_region, r->left, r->top + y );
|
VIPS_REGION_ADDR( out_region, r->left, r->top + y );
|
||||||
const double Y = (r->top + y) * reducev->vshrink +
|
|
||||||
(reducev->centre ? 0.5 : 0.0);
|
|
||||||
const int py = (int) Y;
|
const int py = (int) Y;
|
||||||
const int sy = Y * VIPS_TRANSFORM_SCALE * 2;
|
const int sy = Y * VIPS_TRANSFORM_SCALE;
|
||||||
const int siy = sy & (VIPS_TRANSFORM_SCALE * 2 - 1);
|
const int ty = sy & (VIPS_TRANSFORM_SCALE - 1);
|
||||||
const int ty = (siy + 1) >> 1;
|
|
||||||
const int *cyo = reducev->matrixo[ty];
|
const int *cyo = reducev->matrixo[ty];
|
||||||
|
|
||||||
#ifdef DEBUG_PIXELS
|
#ifdef DEBUG_PIXELS
|
||||||
@ -682,7 +689,7 @@ vips_reducev_vector_gen( VipsRegion *out_region, void *vseq,
|
|||||||
printf( "first column of pixel values:\n" );
|
printf( "first column of pixel values:\n" );
|
||||||
for( int i = 0; i < reducev->n_point; i++ )
|
for( int i = 0; i < reducev->n_point; i++ )
|
||||||
printf( "\t%d - %d\n", i,
|
printf( "\t%d - %d\n", i,
|
||||||
*VIPS_REGION_ADDR( ir, r->left, r->top + y + i ) );
|
*VIPS_REGION_ADDR( ir, r->left, py ) );
|
||||||
#endif /*DEBUG_PIXELS*/
|
#endif /*DEBUG_PIXELS*/
|
||||||
|
|
||||||
/* We run our n passes to generate this scanline.
|
/* We run our n passes to generate this scanline.
|
||||||
@ -709,6 +716,8 @@ vips_reducev_vector_gen( VipsRegion *out_region, void *vseq,
|
|||||||
printf( "pixel result:\n" );
|
printf( "pixel result:\n" );
|
||||||
printf( "\t%d\n", *q );
|
printf( "\t%d\n", *q );
|
||||||
#endif /*DEBUG_PIXELS*/
|
#endif /*DEBUG_PIXELS*/
|
||||||
|
|
||||||
|
Y += reducev->vshrink;
|
||||||
}
|
}
|
||||||
|
|
||||||
VIPS_GATE_STOP( "vips_reducev_vector_gen: work" );
|
VIPS_GATE_STOP( "vips_reducev_vector_gen: work" );
|
||||||
@ -726,45 +735,11 @@ vips_reducev_raw( VipsReducev *reducev, VipsImage *in, VipsImage **out )
|
|||||||
|
|
||||||
VipsGenerateFn generate;
|
VipsGenerateFn generate;
|
||||||
|
|
||||||
/* Build masks.
|
/* We need an 2.6 version if we will use the vector path.
|
||||||
*/
|
|
||||||
for( int y = 0; y < VIPS_TRANSFORM_SCALE + 1; y++ ) {
|
|
||||||
reducev->matrixf[y] =
|
|
||||||
VIPS_ARRAY( NULL, reducev->n_point, double );
|
|
||||||
if( !reducev->matrixf[y] )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
vips_reduce_make_mask( reducev->matrixf[y],
|
|
||||||
reducev->kernel, reducev->vshrink,
|
|
||||||
(float) y / VIPS_TRANSFORM_SCALE );
|
|
||||||
|
|
||||||
#ifdef DEBUG
|
|
||||||
printf( "%6.2g", (double) y / VIPS_TRANSFORM_SCALE );
|
|
||||||
for( int i = 0; i < reducev->n_point; i++ )
|
|
||||||
printf( ", %6.2g", reducev->matrixf[y][i] );
|
|
||||||
printf( "\n" );
|
|
||||||
#endif /*DEBUG*/
|
|
||||||
}
|
|
||||||
|
|
||||||
/* uchar and ushort need an int version of the masks.
|
|
||||||
*/
|
|
||||||
if( VIPS_IMAGE_SIZEOF_ELEMENT( in ) <= 2 )
|
|
||||||
for( int y = 0; y < VIPS_TRANSFORM_SCALE + 1; y++ ) {
|
|
||||||
reducev->matrixi[y] =
|
|
||||||
VIPS_ARRAY( NULL, reducev->n_point, int );
|
|
||||||
if( !reducev->matrixi[y] )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
vips_vector_to_fixed_point(
|
|
||||||
reducev->matrixf[y], reducev->matrixi[y],
|
|
||||||
reducev->n_point, VIPS_INTERPOLATE_SCALE );
|
|
||||||
}
|
|
||||||
|
|
||||||
/* And we need an 2.6 version if we will use the vector path.
|
|
||||||
*/
|
*/
|
||||||
if( in->BandFmt == VIPS_FORMAT_UCHAR &&
|
if( in->BandFmt == VIPS_FORMAT_UCHAR &&
|
||||||
vips_vector_isenabled() )
|
vips_vector_isenabled() )
|
||||||
for( int y = 0; y < VIPS_TRANSFORM_SCALE + 1; y++ ) {
|
for( int y = 0; y < VIPS_TRANSFORM_SCALE; y++ ) {
|
||||||
reducev->matrixo[y] =
|
reducev->matrixo[y] =
|
||||||
VIPS_ARRAY( NULL, reducev->n_point, int );
|
VIPS_ARRAY( NULL, reducev->n_point, int );
|
||||||
if( !reducev->matrixo[y] )
|
if( !reducev->matrixo[y] )
|
||||||
@ -787,7 +762,7 @@ vips_reducev_raw( VipsReducev *reducev, VipsImage *in, VipsImage **out )
|
|||||||
|
|
||||||
*out = vips_image_new();
|
*out = vips_image_new();
|
||||||
if( vips_image_pipelinev( *out,
|
if( vips_image_pipelinev( *out,
|
||||||
VIPS_DEMAND_STYLE_FATSTRIP, in, (void *) NULL ) )
|
VIPS_DEMAND_STYLE_THINSTRIP, in, (void *) NULL ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
/* Size output. We need to always round to nearest, so round(), not
|
/* Size output. We need to always round to nearest, so round(), not
|
||||||
@ -830,7 +805,7 @@ vips_reducev_build( VipsObject *object )
|
|||||||
VipsImage **t = (VipsImage **) vips_object_local_array( object, 4 );
|
VipsImage **t = (VipsImage **) vips_object_local_array( object, 4 );
|
||||||
|
|
||||||
VipsImage *in;
|
VipsImage *in;
|
||||||
int height;
|
double height, extra_pixels;
|
||||||
|
|
||||||
if( VIPS_OBJECT_CLASS( vips_reducev_parent_class )->build( object ) )
|
if( VIPS_OBJECT_CLASS( vips_reducev_parent_class )->build( object ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -855,6 +830,52 @@ vips_reducev_build( VipsObject *object )
|
|||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Output size. We need to always round to nearest, so round(), not
|
||||||
|
* rint().
|
||||||
|
*/
|
||||||
|
height = VIPS_ROUND_UINT(
|
||||||
|
(double) resample->in->Ysize / reducev->vshrink );
|
||||||
|
|
||||||
|
/* How many pixels we are inventing in the input, -ve for
|
||||||
|
* discarding.
|
||||||
|
*/
|
||||||
|
extra_pixels =
|
||||||
|
height * reducev->vshrink - resample->in->Ysize;
|
||||||
|
|
||||||
|
/* If we are rounding down, we are not using some input
|
||||||
|
* pixels. We need to move the origin *inside* the input image
|
||||||
|
* by half that distance so that we discard pixels equally
|
||||||
|
* from left and right.
|
||||||
|
*/
|
||||||
|
reducev->voffset = (1 + extra_pixels) / 2.0;
|
||||||
|
|
||||||
|
/* Build the tables of pre-computed coefficients.
|
||||||
|
*/
|
||||||
|
for( int y = 0; y < VIPS_TRANSFORM_SCALE; y++ ) {
|
||||||
|
reducev->matrixf[y] =
|
||||||
|
VIPS_ARRAY( NULL, reducev->n_point, double );
|
||||||
|
reducev->matrixi[y] =
|
||||||
|
VIPS_ARRAY( NULL, reducev->n_point, int );
|
||||||
|
if( !reducev->matrixf[y] ||
|
||||||
|
!reducev->matrixi[y] )
|
||||||
|
return( -1 );
|
||||||
|
|
||||||
|
vips_reduce_make_mask( reducev->matrixf[y],
|
||||||
|
reducev->kernel, reducev->vshrink,
|
||||||
|
(float) y / VIPS_TRANSFORM_SCALE );
|
||||||
|
|
||||||
|
for( int i = 0; i < reducev->n_point; i++ )
|
||||||
|
reducev->matrixi[y][i] = reducev->matrixf[y][i] *
|
||||||
|
VIPS_INTERPOLATE_SCALE;
|
||||||
|
|
||||||
|
#ifdef DEBUG
|
||||||
|
printf( "vips_reducev_build: mask %d\n ", y );
|
||||||
|
for( int i = 0; i < reducev->n_point; i++ )
|
||||||
|
printf( "%d ", reducev->matrixi[y][i] );
|
||||||
|
printf( "\n" );
|
||||||
|
#endif /*DEBUG*/
|
||||||
|
}
|
||||||
|
|
||||||
/* Unpack for processing.
|
/* Unpack for processing.
|
||||||
*/
|
*/
|
||||||
if( vips_image_decode( in, &t[0] ) )
|
if( vips_image_decode( in, &t[0] ) )
|
||||||
@ -863,12 +884,9 @@ vips_reducev_build( VipsObject *object )
|
|||||||
|
|
||||||
/* Add new pixels around the input so we can interpolate at the edges.
|
/* Add new pixels around the input so we can interpolate at the edges.
|
||||||
*/
|
*/
|
||||||
height = in->Ysize + reducev->n_point - 1;
|
|
||||||
if( reducev->centre )
|
|
||||||
height += 1;
|
|
||||||
if( vips_embed( in, &t[1],
|
if( vips_embed( in, &t[1],
|
||||||
0, reducev->n_point / 2 - 1,
|
0, reducev->n_point / 2 - 1,
|
||||||
in->Xsize, height,
|
in->Xsize, in->Ysize + reducev->n_point,
|
||||||
"extend", VIPS_EXTEND_COPY,
|
"extend", VIPS_EXTEND_COPY,
|
||||||
(void *) NULL ) )
|
(void *) NULL ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -939,13 +957,6 @@ vips_reducev_class_init( VipsReducevClass *reducev_class )
|
|||||||
G_STRUCT_OFFSET( VipsReducev, kernel ),
|
G_STRUCT_OFFSET( VipsReducev, kernel ),
|
||||||
VIPS_TYPE_KERNEL, VIPS_KERNEL_LANCZOS3 );
|
VIPS_TYPE_KERNEL, VIPS_KERNEL_LANCZOS3 );
|
||||||
|
|
||||||
VIPS_ARG_BOOL( reducev_class, "centre", 7,
|
|
||||||
_( "Centre" ),
|
|
||||||
_( "Use centre sampling convention" ),
|
|
||||||
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
|
||||||
G_STRUCT_OFFSET( VipsReducev, centre ),
|
|
||||||
FALSE );
|
|
||||||
|
|
||||||
/* Old name.
|
/* Old name.
|
||||||
*/
|
*/
|
||||||
VIPS_ARG_DOUBLE( reducev_class, "yshrink", 3,
|
VIPS_ARG_DOUBLE( reducev_class, "yshrink", 3,
|
||||||
@ -955,6 +966,15 @@ vips_reducev_class_init( VipsReducevClass *reducev_class )
|
|||||||
G_STRUCT_OFFSET( VipsReducev, vshrink ),
|
G_STRUCT_OFFSET( VipsReducev, vshrink ),
|
||||||
1, 1000000, 1 );
|
1, 1000000, 1 );
|
||||||
|
|
||||||
|
/* We used to let people pick centre or corner, but it's automatic now.
|
||||||
|
*/
|
||||||
|
VIPS_ARG_BOOL( reducev_class, "centre", 7,
|
||||||
|
_( "Centre" ),
|
||||||
|
_( "Use centre sampling convention" ),
|
||||||
|
VIPS_ARGUMENT_OPTIONAL_INPUT | VIPS_ARGUMENT_DEPRECATED,
|
||||||
|
G_STRUCT_OFFSET( VipsReducev, centre ),
|
||||||
|
FALSE );
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -211,7 +211,6 @@ vips_resize_build( VipsObject *object )
|
|||||||
g_info( "residual reducev by %g", vscale );
|
g_info( "residual reducev by %g", vscale );
|
||||||
if( vips_reducev( in, &t[2], 1.0 / vscale,
|
if( vips_reducev( in, &t[2], 1.0 / vscale,
|
||||||
"kernel", resize->kernel,
|
"kernel", resize->kernel,
|
||||||
"centre", TRUE,
|
|
||||||
NULL ) )
|
NULL ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
in = t[2];
|
in = t[2];
|
||||||
@ -222,7 +221,6 @@ vips_resize_build( VipsObject *object )
|
|||||||
hscale );
|
hscale );
|
||||||
if( vips_reduceh( in, &t[3], 1.0 / hscale,
|
if( vips_reduceh( in, &t[3], 1.0 / hscale,
|
||||||
"kernel", resize->kernel,
|
"kernel", resize->kernel,
|
||||||
"centre", TRUE,
|
|
||||||
NULL ) )
|
NULL ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
in = t[3];
|
in = t[3];
|
||||||
|
@ -416,7 +416,7 @@ calculate_coefficients_lanczos( double *c,
|
|||||||
|
|
||||||
sum = 0;
|
sum = 0;
|
||||||
for( i = 0; i < n_points; i++ ) {
|
for( i = 0; i < n_points; i++ ) {
|
||||||
double xp = (i - half) / shrink;
|
const double xp = (i - half) / shrink;
|
||||||
|
|
||||||
double l;
|
double l;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user