reorganise sink base class

the thread state made by sinkmemory was missing a member needed by the
base class ... move ->stop into vips thread state
This commit is contained in:
John Cupitt 2011-11-02 10:27:33 +00:00
parent f2d9001cce
commit b34c07a88f
7 changed files with 32 additions and 84 deletions

52
TODO
View File

@ -1,55 +1,3 @@
- strange
$ valgrind vips embed Gugg_coloured.jpg x.jpg 100 100 5000 5000 --extend
mirror
==32146== Memcheck, a memory error detector
==32146== Copyright (C) 2002-2010, and GNU GPL'd, by Julian Seward et al.
==32146== Using Valgrind-3.6.1-Debian and LibVEX; rerun with -h for copyright
info
==32146== Command: vips embed Gugg_coloured.jpg x.jpg 100 100 5000 5000
--extend mirror
==32146==
==32146== Thread 2:
==32146== Conditional jump or move depends on uninitialised value(s)
==32146== at 0x4F8ED7D: vips_sink_base_allocate (sink.c:257)
==32146== by 0x4F99BC2: vips_thread_allocate (threadpool.c:450)
==32146== by 0x4F99CB5: vips_thread_work_unit (threadpool.c:508)
==32146== by 0x4F99DF0: vips_thread_main_loop (threadpool.c:547)
==32146== by 0x55852B5: ??? (in
/lib/x86_64-linux-gnu/libglib-2.0.so.0.3000.0)
==32146== by 0x5818EFB: start_thread (pthread_create.c:304)
==32146== by 0x5B0F89C: clone (clone.S:112)
==32146==
it's the 'stop' member of SinkThreadState
/* Has work requested early termination?
*/
if( sstate->stop ) {
*stop = TRUE;
return( 0 );
}
argh we have ->stop (a bool) and ->stop (a function)
rename the function as stop_fn
sstate is the "state" member of VipsThread
which is created by the ->start member of the thread
which is set by vips_threadpool_run()
which is vips_thread_state_new(), see vips_sink_memory()
which creates a VipsThreadState
so .. move ->stop from SinkThreadState into the base class, VipsThreadState
- don't do vips_image_write() at the end of a pipeline, instead try: - don't do vips_image_write() at the end of a pipeline, instead try:

View File

@ -57,8 +57,8 @@
*/ */
/* /*
*/
#define VIPS_DEBUG #define VIPS_DEBUG
*/
#ifdef HAVE_CONFIG_H #ifdef HAVE_CONFIG_H
#include <config.h> #include <config.h>

View File

@ -79,6 +79,10 @@ typedef struct _VipsThreadState {
VipsRect pos; VipsRect pos;
int x, y; int x, y;
/* Set in work to get the allocate to signal stop.
*/
gboolean stop;
/* The client data passed to the enclosing vips_threadpool_run(). /* The client data passed to the enclosing vips_threadpool_run().
*/ */
void *a; void *a;

View File

@ -81,10 +81,6 @@ typedef struct _SinkThreadState {
* parent_object.reg, it's defined on the outer image. * parent_object.reg, it's defined on the outer image.
*/ */
VipsRegion *reg; VipsRegion *reg;
/* Set this in work to get the allocate to signal stop.
*/
gboolean stop;
} SinkThreadState; } SinkThreadState;
typedef struct _SinkThreadStateClass { typedef struct _SinkThreadStateClass {
@ -184,11 +180,10 @@ sink_thread_state_init( SinkThreadState *state )
{ {
state->seq = NULL; state->seq = NULL;
state->reg = NULL; state->reg = NULL;
state->stop = FALSE;
} }
static VipsThreadState * VipsThreadState *
sink_thread_state_new( VipsImage *im, void *a ) vips_sink_thread_state_new( VipsImage *im, void *a )
{ {
return( VIPS_THREAD_STATE( vips_object_new( return( VIPS_THREAD_STATE( vips_object_new(
sink_thread_state_get_type(), sink_thread_state_get_type(),
@ -247,14 +242,13 @@ sink_init( Sink *sink,
int int
vips_sink_base_allocate( VipsThreadState *state, void *a, gboolean *stop ) vips_sink_base_allocate( VipsThreadState *state, void *a, gboolean *stop )
{ {
SinkThreadState *sstate = (SinkThreadState *) state;
SinkBase *sink_base = (SinkBase *) a; SinkBase *sink_base = (SinkBase *) a;
VipsRect image, tile; VipsRect image, tile;
/* Has work requested early termination? /* Has work requested early termination?
*/ */
if( sstate->stop ) { if( state->stop ) {
*stop = TRUE; *stop = TRUE;
return( 0 ); return( 0 );
@ -300,7 +294,7 @@ sink_work( VipsThreadState *state, void *a )
if( vips_region_prepare( sstate->reg, &state->pos ) || if( vips_region_prepare( sstate->reg, &state->pos ) ||
sink->generate_fn( sstate->reg, sstate->seq, sink->generate_fn( sstate->reg, sstate->seq,
sink->a, sink->b, &sstate->stop ) ) sink->a, sink->b, &state->stop ) )
return( -1 ); return( -1 );
return( 0 ); return( 0 );
@ -379,7 +373,7 @@ vips_sink_tile( VipsImage *im,
vips_image_preeval( im ); vips_image_preeval( im );
result = vips_threadpool_run( im, result = vips_threadpool_run( im,
sink_thread_state_new, vips_sink_thread_state_new,
vips_sink_base_allocate, vips_sink_base_allocate,
sink_work, sink_work,
vips_sink_base_progress, vips_sink_base_progress,

View File

@ -58,6 +58,7 @@ typedef struct _SinkBase {
/* Some function we can share. /* Some function we can share.
*/ */
void vips_sink_base_init( SinkBase *sink_base, VipsImage *image ); void vips_sink_base_init( SinkBase *sink_base, VipsImage *image );
VipsThreadState *vips_sink_thread_state_new( VipsImage *im, void *a );
int vips_sink_base_allocate( VipsThreadState *state, void *a, gboolean *stop ); int vips_sink_base_allocate( VipsThreadState *state, void *a, gboolean *stop );
int vips_sink_base_progress( void *a ); int vips_sink_base_progress( void *a );

View File

@ -51,35 +51,35 @@
/* Per-call state. /* Per-call state.
*/ */
typedef struct _Sink { typedef struct _SinkMemory {
SinkBase sink_base; SinkBase sink_base;
/* A big region for the image memory. All the threads write to this. /* A big region for the image memory. All the threads write to this.
*/ */
VipsRegion *all; VipsRegion *all;
} Sink; } SinkMemory;
static void static void
sink_free( Sink *sink ) sink_memory_free( SinkMemory *memory )
{ {
VIPS_UNREF( sink->all ); VIPS_UNREF( memory->all );
} }
static int static int
sink_init( Sink *sink, VipsImage *im ) sink_memory_init( SinkMemory *memory, VipsImage *im )
{ {
VipsRect all; VipsRect all;
vips_sink_base_init( &sink->sink_base, im ); vips_sink_base_init( &memory->sink_base, im );
all.left = 0; all.left = 0;
all.top = 0; all.top = 0;
all.width = im->Xsize; all.width = im->Xsize;
all.height = im->Ysize; all.height = im->Ysize;
if( !(sink->all = vips_region_new( im )) || if( !(memory->all = vips_region_new( im )) ||
vips_region_image( sink->all, &all ) ) { vips_region_image( memory->all, &all ) ) {
sink_free( sink ); sink_memory_free( memory );
return( -1 ); return( -1 );
} }
@ -87,19 +87,19 @@ sink_init( Sink *sink, VipsImage *im )
} }
static int static int
sink_work( VipsThreadState *state, void *a ) sink_memory_work( VipsThreadState *state, void *a )
{ {
Sink *sink = (Sink *) a; SinkMemory *memory = (SinkMemory *) a;
VIPS_DEBUG_MSG( "sink_work: %p " VIPS_DEBUG_MSG( "sink_memory_work: %p "
"left = %d, top = %d, width = %d, height = %d\n", "left = %d, top = %d, width = %d, height = %d\n",
sink, memory,
state->pos.left, state->pos.left,
state->pos.top, state->pos.top,
state->pos.width, state->pos.width,
state->pos.height ); state->pos.height );
if( vips_region_prepare_to( state->reg, sink->all, if( vips_region_prepare_to( state->reg, memory->all,
&state->pos, state->pos.left, state->pos.top ) ) &state->pos, state->pos.left, state->pos.top ) )
return( -1 ); return( -1 );
@ -109,7 +109,7 @@ sink_work( VipsThreadState *state, void *a )
state->pos.left, state->pos.top ); state->pos.left, state->pos.top );
int i; int i;
VIPS_DEBUG_MSG( "sink_work: %p\n", sink ); VIPS_DEBUG_MSG( "sink_memory_work: %p\n", memory );
for( i = 0; i < VIPS_IMAGE_SIZEOF_PEL( state->reg->im ); i++ ) for( i = 0; i < VIPS_IMAGE_SIZEOF_PEL( state->reg->im ); i++ )
printf( "\t%d) %02x\n", i, p[i] ); printf( "\t%d) %02x\n", i, p[i] );
} }
@ -132,7 +132,7 @@ sink_work( VipsThreadState *state, void *a )
int int
vips_sink_memory( VipsImage *image ) vips_sink_memory( VipsImage *image )
{ {
Sink sink; SinkMemory memory;
int result; int result;
VIPS_DEBUG_MSG( "vips_sink_memory: %p\n", image ); VIPS_DEBUG_MSG( "vips_sink_memory: %p\n", image );
@ -144,7 +144,7 @@ vips_sink_memory( VipsImage *image )
*/ */
image->Bbits = vips_format_sizeof( image->BandFmt ) << 3; image->Bbits = vips_format_sizeof( image->BandFmt ) << 3;
if( sink_init( &sink, image ) ) if( sink_memory_init( &memory, image ) )
return( -1 ); return( -1 );
vips_image_preeval( image ); vips_image_preeval( image );
@ -152,13 +152,13 @@ vips_sink_memory( VipsImage *image )
result = vips_threadpool_run( image, result = vips_threadpool_run( image,
vips_thread_state_new, vips_thread_state_new,
vips_sink_base_allocate, vips_sink_base_allocate,
sink_work, sink_memory_work,
vips_sink_base_progress, vips_sink_base_progress,
&sink ); &memory );
vips_image_posteval( image ); vips_image_posteval( image );
sink_free( &sink ); sink_memory_free( &memory );
return( result ); return( result );
} }

View File

@ -287,6 +287,7 @@ vips_thread_state_init( VipsThreadState *state )
VIPS_DEBUG_MSG( "vips_thread_state_init:\n" ); VIPS_DEBUG_MSG( "vips_thread_state_init:\n" );
state->reg = NULL; state->reg = NULL;
state->stop = FALSE;
} }
void * void *