vips_conva() done and working
remove im_conva() nest, add tests
This commit is contained in:
parent
cedb904773
commit
c658332215
@ -27,10 +27,10 @@
|
|||||||
- better quality for vips_resize() with linear/cubic kernels
|
- better quality for vips_resize() with linear/cubic kernels
|
||||||
- pyvips8 can create new metadata
|
- pyvips8 can create new metadata
|
||||||
- better upsizing with vips_resize()
|
- better upsizing with vips_resize()
|
||||||
- added vips_convf(), vips_convi(), vips_convasep() ... im_conv*() functions
|
- added vips_convf(), vips_convi(), vips_convasep(), vips_conva() ...
|
||||||
rewritten as classes
|
im_conv*() functions rewritten as classes
|
||||||
- vips_convsep() calls vips_convasep() for the approximate case
|
- vips_convsep() calls vips_convasep() for the approximate case
|
||||||
- new fixed-point vector path for convi is up to 2x faster
|
- new fixed-point vector path for convi is up to about 2x faster
|
||||||
- added vips_worley(), generate Worley noise
|
- added vips_worley(), generate Worley noise
|
||||||
- added vips_perlin(), generate Perlin noise
|
- added vips_perlin(), generate Perlin noise
|
||||||
- gif loader can write 1, 2, 3, or 4 bands depending on file contents
|
- gif loader can write 1, 2, 3, or 4 bands depending on file contents
|
||||||
|
2
TODO
2
TODO
@ -1,5 +1,7 @@
|
|||||||
- tests for convasep ... just check it matches convsep +/- some small amount
|
- tests for convasep ... just check it matches convsep +/- some small amount
|
||||||
|
|
||||||
|
- tests for conva
|
||||||
|
|
||||||
- try this blur.mat
|
- try this blur.mat
|
||||||
|
|
||||||
6 1 3896
|
6 1 3896
|
||||||
|
@ -6,6 +6,7 @@ libconvolution_la_SOURCES = \
|
|||||||
correlation.c \
|
correlation.c \
|
||||||
correlation.h \
|
correlation.h \
|
||||||
conv.c \
|
conv.c \
|
||||||
|
conva.c \
|
||||||
convf.c \
|
convf.c \
|
||||||
convi.c \
|
convi.c \
|
||||||
convasep.c \
|
convasep.c \
|
||||||
|
@ -72,12 +72,16 @@ $ vips im_max abs.v
|
|||||||
|
|
||||||
- are we handling mask offset correctly?
|
- are we handling mask offset correctly?
|
||||||
|
|
||||||
|
- could we do better with an h and a v cumulativization image? we might
|
||||||
|
not need the huge intermediate we have now, since any line sum an be
|
||||||
|
found with simple indexing
|
||||||
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
*/
|
|
||||||
#define DEBUG
|
#define DEBUG
|
||||||
#define VIPS_DEBUG
|
#define VIPS_DEBUG
|
||||||
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
@ -93,10 +97,14 @@ $ vips im_max abs.v
|
|||||||
#include <vips/vips.h>
|
#include <vips/vips.h>
|
||||||
#include <vips/vector.h>
|
#include <vips/vector.h>
|
||||||
#include <vips/debug.h>
|
#include <vips/debug.h>
|
||||||
|
#include <vips/internal.h>
|
||||||
|
|
||||||
/* Maximum number of boxes we can break the mask into.
|
#include "pconvolution.h"
|
||||||
|
|
||||||
|
/* Maximum number of boxes we can break the mask into. Don't have this too
|
||||||
|
* high, it'll make the instance huge, and gobject has a 64kb limit.
|
||||||
*/
|
*/
|
||||||
#define MAX_LINES (10000)
|
#define MAX_LINES (1000)
|
||||||
|
|
||||||
/* The number of edges we consider at once in clustering. Higher values are
|
/* The number of edges we consider at once in clustering. Higher values are
|
||||||
* faster, but risk pushing up average error in the result.
|
* faster, but risk pushing up average error in the result.
|
||||||
@ -167,8 +175,6 @@ typedef struct {
|
|||||||
int area;
|
int area;
|
||||||
int rounding;
|
int rounding;
|
||||||
int offset;
|
int offset;
|
||||||
int width;
|
|
||||||
int height;
|
|
||||||
|
|
||||||
/* The horizontal lines we gather. hline[3] writes to band 3 in the
|
/* The horizontal lines we gather. hline[3] writes to band 3 in the
|
||||||
* intermediate image. max_line is the length of the longest hline:
|
* intermediate image. max_line is the length of the longest hline:
|
||||||
@ -259,7 +265,7 @@ vips_conva_hprint( VipsConva *conva )
|
|||||||
conva->velement[y].factor,
|
conva->velement[y].factor,
|
||||||
conva->hline[b].weight );
|
conva->hline[b].weight );
|
||||||
for( x = 0; x < 45; x++ ) {
|
for( x = 0; x < 45; x++ ) {
|
||||||
int rx = x * (conva->width + 1) / 45;
|
int rx = x * (conva->iM->Xsize + 1) / 45;
|
||||||
|
|
||||||
if( rx >= conva->hline[b].start &&
|
if( rx >= conva->hline[b].start &&
|
||||||
rx < conva->hline[b].end )
|
rx < conva->hline[b].end )
|
||||||
@ -300,6 +306,7 @@ vips_conva_decompose_lines( VipsConva *conva )
|
|||||||
{
|
{
|
||||||
VipsImage *iM = conva->iM;
|
VipsImage *iM = conva->iM;
|
||||||
const int size = iM->Xsize * iM->Ysize;
|
const int size = iM->Xsize * iM->Ysize;
|
||||||
|
double *coeff = VIPS_MATRIX( iM, 0, 0 );
|
||||||
|
|
||||||
double max;
|
double max;
|
||||||
double min;
|
double min;
|
||||||
@ -355,14 +362,14 @@ vips_conva_decompose_lines( VipsConva *conva )
|
|||||||
inside = 0;
|
inside = 0;
|
||||||
|
|
||||||
for( x = 0; x < iM->Xsize; x++ ) {
|
for( x = 0; x < iM->Xsize; x++ ) {
|
||||||
double coeff = *VIPS_MATRIX( iM, x, y );
|
double c = coeff[x + y * iM->Xsize];
|
||||||
|
|
||||||
/* The vertical line from mask[x, y] to 0 is
|
/* The vertical line from mask[x, y] to 0 is
|
||||||
* inside. Is our current square (x, y) part
|
* inside. Is our current square (x, y) part
|
||||||
* of that line?
|
* of that line?
|
||||||
*/
|
*/
|
||||||
if( (z_positive && coeff >= z_ph) ||
|
if( (z_positive && c >= z_ph) ||
|
||||||
(!z_positive && coeff <= z_ph) ) {
|
(!z_positive && c <= z_ph) ) {
|
||||||
if( !inside ) {
|
if( !inside ) {
|
||||||
vips_conva_hline_start( conva,
|
vips_conva_hline_start( conva,
|
||||||
x );
|
x );
|
||||||
@ -382,7 +389,7 @@ vips_conva_decompose_lines( VipsConva *conva )
|
|||||||
|
|
||||||
if( inside &&
|
if( inside &&
|
||||||
vips_conva_hline_end( conva,
|
vips_conva_hline_end( conva,
|
||||||
mask->xsize, y, z_positive ? 1 : -1 ) )
|
iM->Xsize, y, z_positive ? 1 : -1 ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -480,7 +487,7 @@ vips_conva_cluster2( VipsConva *conva )
|
|||||||
if( conva->hline[j].weight == 0 )
|
if( conva->hline[j].weight == 0 )
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
distance = vips_conva_distance( boxes, i, j );
|
distance = vips_conva_distance( conva, i, j );
|
||||||
if( distance < worst ) {
|
if( distance < worst ) {
|
||||||
conva->edge[worst_i].a = i;
|
conva->edge[worst_i].a = i;
|
||||||
conva->edge[worst_i].b = j;
|
conva->edge[worst_i].b = j;
|
||||||
@ -672,7 +679,7 @@ vips_conva_decompose_boxes( VipsConva *conva )
|
|||||||
{
|
{
|
||||||
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( conva );
|
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( conva );
|
||||||
VipsImage *iM = conva->iM;
|
VipsImage *iM = conva->iM;
|
||||||
double *coeff = VIPS_MATRIX( im, 0, 0 );
|
double *coeff = VIPS_MATRIX( iM, 0, 0 );
|
||||||
const int size = iM->Xsize * iM->Ysize;
|
const int size = iM->Xsize * iM->Ysize;
|
||||||
double scale = vips_image_get_scale( iM );
|
double scale = vips_image_get_scale( iM );
|
||||||
double offset = vips_image_get_offset( iM );
|
double offset = vips_image_get_offset( iM );
|
||||||
@ -680,18 +687,15 @@ vips_conva_decompose_boxes( VipsConva *conva )
|
|||||||
double sum;
|
double sum;
|
||||||
int x, y, z;
|
int x, y, z;
|
||||||
|
|
||||||
boxes->n_hline = 0;
|
|
||||||
boxes->n_velement = 0;
|
|
||||||
boxes->n_vline = 0;
|
|
||||||
|
|
||||||
/* Break into a set of hlines.
|
/* Break into a set of hlines.
|
||||||
*/
|
*/
|
||||||
if( vips_conva_decompose_lines( boxes ) )
|
if( vips_conva_decompose_lines( conva ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
/* Cluster to find groups of lines.
|
/* Cluster to find groups of lines.
|
||||||
*/
|
*/
|
||||||
VIPS_DEBUG_MSG( "boxes_new: clustering with thresh %d ...\n", cluster );
|
VIPS_DEBUG_MSG( "vips_conva_decompose_boxes: "
|
||||||
|
"clustering with thresh %d ...\n", conva->cluster );
|
||||||
while( vips_conva_cluster2( conva ) )
|
while( vips_conva_cluster2( conva ) )
|
||||||
;
|
;
|
||||||
|
|
||||||
@ -734,7 +738,7 @@ vips_conva_decompose_boxes( VipsConva *conva )
|
|||||||
sum += coeff[z];
|
sum += coeff[z];
|
||||||
|
|
||||||
conva->area = VIPS_RINT( sum * conva->area / scale );
|
conva->area = VIPS_RINT( sum * conva->area / scale );
|
||||||
conva->rounding = (conva->area + 1) / 2 + offset * conva->area;
|
conva->rounding = (conva->area + 1) / 2;
|
||||||
conva->offset = offset;
|
conva->offset = offset;
|
||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
@ -795,9 +799,7 @@ vips_conva_start( VipsImage *out, void *a, void *b )
|
|||||||
|
|
||||||
VipsConvaSeq *seq;
|
VipsConvaSeq *seq;
|
||||||
|
|
||||||
if( !(seq = VIPS_NEW( out, VipsConvaSeq )) )
|
seq = VIPS_NEW( out, VipsConvaSeq );
|
||||||
return( NULL );
|
|
||||||
|
|
||||||
seq->conva = conva;
|
seq->conva = conva;
|
||||||
seq->ir = vips_region_new( in );
|
seq->ir = vips_region_new( in );
|
||||||
|
|
||||||
@ -815,11 +817,6 @@ vips_conva_start( VipsImage *out, void *a, void *b )
|
|||||||
seq->sum = VIPS_ARRAY( out, conva->n_velement, double );
|
seq->sum = VIPS_ARRAY( out, conva->n_velement, double );
|
||||||
seq->last_stride = -1;
|
seq->last_stride = -1;
|
||||||
|
|
||||||
if( !seq->ir || !seq->start || !seq->end || !seq->sum ) {
|
|
||||||
vips_conva_stop( seq, in, boxes );
|
|
||||||
return( NULL );
|
|
||||||
}
|
|
||||||
|
|
||||||
return( seq );
|
return( seq );
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -863,7 +860,8 @@ G_STMT_START { \
|
|||||||
/* Do horizontal masks ... we scan the mask along scanlines.
|
/* Do horizontal masks ... we scan the mask along scanlines.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
vips_conva_hgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
vips_conva_hgenerate( VipsRegion *or, void *vseq,
|
||||||
|
void *a, void *b, gboolean *stop )
|
||||||
{
|
{
|
||||||
VipsConvaSeq *seq = (VipsConvaSeq *) vseq;
|
VipsConvaSeq *seq = (VipsConvaSeq *) vseq;
|
||||||
VipsImage *in = (VipsImage *) a;
|
VipsImage *in = (VipsImage *) a;
|
||||||
@ -888,7 +886,7 @@ vips_conva_hgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
|||||||
* than the section of the output image we are producing.
|
* than the section of the output image we are producing.
|
||||||
*/
|
*/
|
||||||
s = *r;
|
s = *r;
|
||||||
s.width += mask->xsize - 1;
|
s.width += iM->Xsize - 1;
|
||||||
if( vips_region_prepare( ir, &s ) )
|
if( vips_region_prepare( ir, &s ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
@ -967,7 +965,7 @@ vips_conva_hgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
|||||||
static int
|
static int
|
||||||
vips_conva_horizontal( VipsConva *conva, VipsImage *in, VipsImage **out )
|
vips_conva_horizontal( VipsConva *conva, VipsImage *in, VipsImage **out )
|
||||||
{
|
{
|
||||||
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( convasep );
|
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( conva );
|
||||||
|
|
||||||
/* Prepare output. Consider a 7x7 mask and a 7x7 image --- the output
|
/* Prepare output. Consider a 7x7 mask and a 7x7 image --- the output
|
||||||
* would be 1x1.
|
* would be 1x1.
|
||||||
@ -1083,11 +1081,13 @@ G_STMT_START { \
|
|||||||
/* Do vertical masks ... we scan the mask down columns of pixels.
|
/* Do vertical masks ... we scan the mask down columns of pixels.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
vips_conva_vgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
vips_conva_vgenerate( VipsRegion *or, void *vseq,
|
||||||
|
void *a, void *b, gboolean *stop )
|
||||||
{
|
{
|
||||||
VipsConvaSeq *seq = (VipsConvaSeq *) vseq;
|
VipsConvaSeq *seq = (VipsConvaSeq *) vseq;
|
||||||
VipsImage *in = (VipsImage *) a;
|
VipsImage *in = (VipsImage *) a;
|
||||||
VipsConva *conva = (VipsConva *) b;
|
VipsConva *conva = (VipsConva *) b;
|
||||||
|
VipsConvolution *convolution = (VipsConvolution *) conva;
|
||||||
|
|
||||||
VipsRegion *ir = seq->ir;
|
VipsRegion *ir = seq->ir;
|
||||||
const int n_vline = conva->n_vline;
|
const int n_vline = conva->n_vline;
|
||||||
@ -1108,14 +1108,14 @@ vips_conva_vgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
|||||||
* than the section of the output image we are producing.
|
* than the section of the output image we are producing.
|
||||||
*/
|
*/
|
||||||
s = *r;
|
s = *r;
|
||||||
s.height += mask->ysize - 1;
|
s.height += iM->Ysize - 1;
|
||||||
if( vips_region_prepare( ir, &s ) )
|
if( vips_region_prepare( ir, &s ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
istride = VIPS_REGION_LSKIP( ir ) /
|
istride = VIPS_REGION_LSKIP( ir ) /
|
||||||
VIPS_IMAGE_SIZEOF_ELEMENT( in );
|
VIPS_IMAGE_SIZEOF_ELEMENT( in );
|
||||||
ostride = VIPS_REGION_LSKIP( or ) /
|
ostride = VIPS_REGION_LSKIP( or ) /
|
||||||
VIPS_IMAGE_SIZEOF_ELEMENT( boxes->out );
|
VIPS_IMAGE_SIZEOF_ELEMENT( convolution->out );
|
||||||
|
|
||||||
/* Init offset array.
|
/* Init offset array.
|
||||||
*/
|
*/
|
||||||
@ -1130,7 +1130,7 @@ vips_conva_vgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
switch( conva->in->BandFmt ) {
|
switch( convolution->in->BandFmt ) {
|
||||||
case VIPS_FORMAT_UCHAR:
|
case VIPS_FORMAT_UCHAR:
|
||||||
if( conva->max_line > 256 )
|
if( conva->max_line > 256 )
|
||||||
VCONV( unsigned int, \
|
VCONV( unsigned int, \
|
||||||
@ -1143,10 +1143,10 @@ vips_conva_vgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
|||||||
case VIPS_FORMAT_CHAR:
|
case VIPS_FORMAT_CHAR:
|
||||||
if( conva->max_line > 256 )
|
if( conva->max_line > 256 )
|
||||||
VCONV( signed int, \
|
VCONV( signed int, \
|
||||||
signed int, signed char, CLIP_UCHAR );
|
signed int, signed char, CLIP_CHAR );
|
||||||
else
|
else
|
||||||
VCONV( signed int, \
|
VCONV( signed int, \
|
||||||
signed short, signed char, CLIP_UCHAR );
|
signed short, signed char, CLIP_CHAR );
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case VIPS_FORMAT_USHORT:
|
case VIPS_FORMAT_USHORT:
|
||||||
@ -1192,7 +1192,8 @@ vips_conva_vgenerate( VipsRegion *or, void *vseq, void *a, void *b )
|
|||||||
static int
|
static int
|
||||||
vips_conva_vertical( VipsConva *conva, VipsImage *in, VipsImage **out )
|
vips_conva_vertical( VipsConva *conva, VipsImage *in, VipsImage **out )
|
||||||
{
|
{
|
||||||
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( convasep );
|
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( conva );
|
||||||
|
VipsConvolution *convolution = (VipsConvolution *) conva;
|
||||||
|
|
||||||
/* Prepare output. Consider a 7x7 mask and a 7x7 image --- the output
|
/* Prepare output. Consider a 7x7 mask and a 7x7 image --- the output
|
||||||
* would be 1x1.
|
* would be 1x1.
|
||||||
@ -1208,12 +1209,12 @@ vips_conva_vertical( VipsConva *conva, VipsImage *in, VipsImage **out )
|
|||||||
"%s", _( "image too small for mask" ) );
|
"%s", _( "image too small for mask" ) );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
out->Bands = boxes->in->Bands;
|
(*out)->Bands = convolution->in->Bands;
|
||||||
out->BandFmt = boxes->in->BandFmt;
|
(*out)->BandFmt = convolution->in->BandFmt;
|
||||||
|
|
||||||
if( vips_image_generate( out,
|
if( vips_image_generate( *out,
|
||||||
vips_conva_start, vips_conva_vgenerate, vips_conva_stop,
|
vips_conva_start, vips_conva_vgenerate, vips_conva_stop,
|
||||||
in, boxes ) )
|
in, conva ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
@ -1222,7 +1223,6 @@ vips_conva_vertical( VipsConva *conva, VipsImage *in, VipsImage **out )
|
|||||||
static int
|
static int
|
||||||
vips_conva_build( VipsObject *object )
|
vips_conva_build( VipsObject *object )
|
||||||
{
|
{
|
||||||
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( object );
|
|
||||||
VipsConvolution *convolution = (VipsConvolution *) object;
|
VipsConvolution *convolution = (VipsConvolution *) object;
|
||||||
VipsConva *conva = (VipsConva *) object;
|
VipsConva *conva = (VipsConva *) object;
|
||||||
VipsImage **t = (VipsImage **) vips_object_local_array( object, 4 );
|
VipsImage **t = (VipsImage **) vips_object_local_array( object, 4 );
|
||||||
@ -1237,8 +1237,6 @@ vips_conva_build( VipsObject *object )
|
|||||||
if( vips__image_intize( convolution->M, &t[0] ) )
|
if( vips__image_intize( convolution->M, &t[0] ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
conva->iM = t[0];
|
conva->iM = t[0];
|
||||||
conva->width = conva->iM->Xsize;
|
|
||||||
conva->height = conva->iM->Ysize;
|
|
||||||
|
|
||||||
in = convolution->in;
|
in = convolution->in;
|
||||||
|
|
||||||
@ -1248,14 +1246,14 @@ vips_conva_build( VipsObject *object )
|
|||||||
g_object_set( conva, "out", vips_image_new(), NULL );
|
g_object_set( conva, "out", vips_image_new(), NULL );
|
||||||
if(
|
if(
|
||||||
vips_embed( in, &t[1],
|
vips_embed( in, &t[1],
|
||||||
conva->width / 2,
|
t[0]->Xsize / 2,
|
||||||
conva->height / 2,
|
t[0]->Ysize / 2,
|
||||||
in->Xsize + convasep->width - 1,
|
in->Xsize + t[0]->Xsize - 1,
|
||||||
in->Ysize + convasep->height - 1,
|
in->Ysize + t[0]->Ysize - 1,
|
||||||
"extend", VIPS_EXTEND_COPY,
|
"extend", VIPS_EXTEND_COPY,
|
||||||
NULL ) ||
|
NULL ) ||
|
||||||
vips_conva_horizontal( convasep, t[1], &t[2] ) ||
|
vips_conva_horizontal( conva, t[1], &t[2] ) ||
|
||||||
vips_conva_vertical( convasep, t[2], &t[3] ) ||
|
vips_conva_vertical( conva, t[2], &t[3] ) ||
|
||||||
vips_image_write( t[3], convolution->out ) )
|
vips_image_write( t[3], convolution->out ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
@ -1275,7 +1273,7 @@ vips_conva_class_init( VipsConvaClass *class )
|
|||||||
gobject_class->get_property = vips_object_get_property;
|
gobject_class->get_property = vips_object_get_property;
|
||||||
|
|
||||||
object_class->nickname = "conva";
|
object_class->nickname = "conva";
|
||||||
object_class->description = _( "approximate convolution" );
|
object_class->description = _( "approximate integer convolution" );
|
||||||
object_class->build = vips_conva_build;
|
object_class->build = vips_conva_build;
|
||||||
|
|
||||||
VIPS_ARG_INT( class, "layers", 104,
|
VIPS_ARG_INT( class, "layers", 104,
|
||||||
@ -1289,7 +1287,7 @@ vips_conva_class_init( VipsConvaClass *class )
|
|||||||
_( "Cluster" ),
|
_( "Cluster" ),
|
||||||
_( "Cluster lines closer than this in approximation" ),
|
_( "Cluster lines closer than this in approximation" ),
|
||||||
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
VIPS_ARGUMENT_OPTIONAL_INPUT,
|
||||||
G_STRUCT_OFFSET( VipsConv, cluster ),
|
G_STRUCT_OFFSET( VipsConva, cluster ),
|
||||||
1, 100, 1 );
|
1, 100, 1 );
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -1313,12 +1311,14 @@ vips_conva_init( VipsConva *conva )
|
|||||||
* * @layers: %gint, number of layers for approximation
|
* * @layers: %gint, number of layers for approximation
|
||||||
* * @cluster: %gint, cluster lines closer than this distance
|
* * @cluster: %gint, cluster lines closer than this distance
|
||||||
*
|
*
|
||||||
* Perform an approximate convolution of @in with @mask.
|
* Perform an approximate integer convolution of @in with @mask.
|
||||||
* This is a low-level operation, see
|
* This is a low-level operation, see
|
||||||
* vips_conv() for something more convenient.
|
* vips_conv() for something more convenient.
|
||||||
*
|
*
|
||||||
* The output image
|
* The output image
|
||||||
* always has the same #VipsBandFormat as the input image.
|
* always has the same #VipsBandFormat as the input image.
|
||||||
|
* Elements of @mask are converted to
|
||||||
|
* integers before convolution.
|
||||||
*
|
*
|
||||||
* Larger values for @layers give more accurate
|
* Larger values for @layers give more accurate
|
||||||
* results, but are slower. As @layers approaches the mask radius, the
|
* results, but are slower. As @layers approaches the mask radius, the
|
||||||
|
@ -61,12 +61,16 @@
|
|||||||
we could then use orc to write a bit of code to implement this set
|
we could then use orc to write a bit of code to implement this set
|
||||||
of lines
|
of lines
|
||||||
|
|
||||||
|
stackoverflow has an algorithm for cumulativization using SIMD and
|
||||||
|
threads, see that font rasterization with rust piece on medium by
|
||||||
|
ralph levien
|
||||||
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
*/
|
|
||||||
#define DEBUG
|
#define DEBUG
|
||||||
#define VIPS_DEBUG
|
#define VIPS_DEBUG
|
||||||
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
@ -883,7 +887,8 @@ vips_convasep_class_init( VipsConvasepClass *class )
|
|||||||
gobject_class->get_property = vips_object_get_property;
|
gobject_class->get_property = vips_object_get_property;
|
||||||
|
|
||||||
object_class->nickname = "convasep";
|
object_class->nickname = "convasep";
|
||||||
object_class->description = _( "approximate separable convolution" );
|
object_class->description =
|
||||||
|
_( "approximate separable integer convolution" );
|
||||||
object_class->build = vips_convasep_build;
|
object_class->build = vips_convasep_build;
|
||||||
|
|
||||||
VIPS_ARG_INT( class, "layers", 104,
|
VIPS_ARG_INT( class, "layers", 104,
|
||||||
@ -913,7 +918,7 @@ vips_convasep_init( VipsConvasep *convasep )
|
|||||||
*
|
*
|
||||||
* * @layers: %gint, number of layers for approximation
|
* * @layers: %gint, number of layers for approximation
|
||||||
*
|
*
|
||||||
* Approximate separable convolution. This is a low-level operation, see
|
* Approximate separable integer convolution. This is a low-level operation, see
|
||||||
* vips_convsep() for something more convenient.
|
* vips_convsep() for something more convenient.
|
||||||
*
|
*
|
||||||
* The image is convolved twice: once with @mask and then again with @mask
|
* The image is convolved twice: once with @mask and then again with @mask
|
||||||
|
@ -157,6 +157,7 @@ void
|
|||||||
vips_convolution_operation_init( void )
|
vips_convolution_operation_init( void )
|
||||||
{
|
{
|
||||||
extern int vips_conv_get_type( void );
|
extern int vips_conv_get_type( void );
|
||||||
|
extern int vips_conva_get_type( void );
|
||||||
extern int vips_convf_get_type( void );
|
extern int vips_convf_get_type( void );
|
||||||
extern int vips_convi_get_type( void );
|
extern int vips_convi_get_type( void );
|
||||||
extern int vips_convsep_get_type( void );
|
extern int vips_convsep_get_type( void );
|
||||||
@ -168,6 +169,7 @@ vips_convolution_operation_init( void )
|
|||||||
extern int vips_gaussblur_get_type( void );
|
extern int vips_gaussblur_get_type( void );
|
||||||
|
|
||||||
vips_conv_get_type();
|
vips_conv_get_type();
|
||||||
|
vips_conva_get_type();
|
||||||
vips_convf_get_type();
|
vips_convf_get_type();
|
||||||
vips_convi_get_type();
|
vips_convi_get_type();
|
||||||
vips_compass_get_type();
|
vips_compass_get_type();
|
||||||
|
@ -44,15 +44,17 @@ typedef enum {
|
|||||||
VIPS_COMBINE_LAST
|
VIPS_COMBINE_LAST
|
||||||
} VipsCombine;
|
} VipsCombine;
|
||||||
|
|
||||||
|
int vips_conv( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
||||||
|
__attribute__((sentinel));
|
||||||
int vips_convf( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
int vips_convf( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
||||||
__attribute__((sentinel));
|
__attribute__((sentinel));
|
||||||
int vips_convi( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
int vips_convi( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
||||||
__attribute__((sentinel));
|
__attribute__((sentinel));
|
||||||
int vips_convasep( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
int vips_conva( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
||||||
__attribute__((sentinel));
|
__attribute__((sentinel));
|
||||||
int vips_convsep( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
int vips_convsep( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
||||||
__attribute__((sentinel));
|
__attribute__((sentinel));
|
||||||
int vips_conv( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
int vips_convasep( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
||||||
__attribute__((sentinel));
|
__attribute__((sentinel));
|
||||||
|
|
||||||
int vips_compass( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
int vips_compass( VipsImage *in, VipsImage **out, VipsImage *mask, ... )
|
||||||
|
@ -109,7 +109,15 @@ static GMutex *vips_tracked_mutex = NULL;
|
|||||||
* @OBJ: allocate memory local to @OBJ, or %NULL for no auto-free
|
* @OBJ: allocate memory local to @OBJ, or %NULL for no auto-free
|
||||||
* @T: type of thing to allocate
|
* @T: type of thing to allocate
|
||||||
*
|
*
|
||||||
* Returns: A pointer of type @T *, or %NULL on error.
|
* Allocate memory for a thing of type @T. The memory is not
|
||||||
|
* cleared.
|
||||||
|
*
|
||||||
|
* This macro cannot fail. See vips_tracked_malloc() if you are
|
||||||
|
* allocating large amounts of memory.
|
||||||
|
*
|
||||||
|
* See also: vips_malloc().
|
||||||
|
*
|
||||||
|
* Returns: A pointer of type @T *.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -118,7 +126,15 @@ static GMutex *vips_tracked_mutex = NULL;
|
|||||||
* @N: number of @T 's to allocate
|
* @N: number of @T 's to allocate
|
||||||
* @T: type of thing to allocate
|
* @T: type of thing to allocate
|
||||||
*
|
*
|
||||||
* Returns: A pointer of type @T *, or %NULL on error.
|
* Allocate memory for an array of objects of type @T. The memory is not
|
||||||
|
* cleared.
|
||||||
|
*
|
||||||
|
* This macro cannot fail. See vips_tracked_malloc() if you are
|
||||||
|
* allocating large amounts of memory.
|
||||||
|
*
|
||||||
|
* See also: vips_malloc().
|
||||||
|
*
|
||||||
|
* Returns: A pointer of type @T *.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -141,7 +157,7 @@ vips_malloc_cb( VipsObject *object, char *buf )
|
|||||||
*
|
*
|
||||||
* See also: vips_tracked_malloc().
|
* See also: vips_tracked_malloc().
|
||||||
*
|
*
|
||||||
* Returns: (transfer full): a pointer to the allocated memory
|
* Returns: (transfer full): a pointer to the allocated memory.
|
||||||
*/
|
*/
|
||||||
void *
|
void *
|
||||||
vips_malloc( VipsObject *object, size_t size )
|
vips_malloc( VipsObject *object, size_t size )
|
||||||
@ -166,7 +182,7 @@ vips_malloc( VipsObject *object, size_t size )
|
|||||||
*
|
*
|
||||||
* g_strdup() a string. When @object is freed, the string will be freed for
|
* g_strdup() a string. When @object is freed, the string will be freed for
|
||||||
* you. If @object is %NULL, you need to
|
* you. If @object is %NULL, you need to
|
||||||
* free the memory explicitly with g_free().
|
* free the memory yourself with g_free().
|
||||||
*
|
*
|
||||||
* This function cannot fail.
|
* This function cannot fail.
|
||||||
*
|
*
|
||||||
|
Loading…
Reference in New Issue
Block a user