experiment with thread stalling
This commit is contained in:
parent
45e3c0bd39
commit
f5f0dda551
@ -10,6 +10,7 @@
|
|||||||
- better error msgs for enum args
|
- better error msgs for enum args
|
||||||
- fix compiler warnings in production build (thanks Dmitry)
|
- fix compiler warnings in production build (thanks Dmitry)
|
||||||
- fix spurious warnings about exif updates
|
- fix spurious warnings about exif updates
|
||||||
|
- VipsSequential has an integrated cache and stalls out of order threads
|
||||||
|
|
||||||
20/7/12 started 7.30.0
|
20/7/12 started 7.30.0
|
||||||
- support "rs" mode in vips7
|
- support "rs" mode in vips7
|
||||||
|
@ -9,6 +9,9 @@
|
|||||||
* - support skip forwards as well, so we can do extract/insert
|
* - support skip forwards as well, so we can do extract/insert
|
||||||
* 10/8/12
|
* 10/8/12
|
||||||
* - add @trace option
|
* - add @trace option
|
||||||
|
* 21/8/12
|
||||||
|
* - remove skip forward, instead do thread stalling and have an
|
||||||
|
* integrated cache
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -38,8 +41,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define VIPS_DEBUG
|
|
||||||
*/
|
*/
|
||||||
|
#define VIPS_DEBUG
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
@ -61,14 +64,32 @@ typedef struct _VipsSequential {
|
|||||||
|
|
||||||
VipsImage *in;
|
VipsImage *in;
|
||||||
|
|
||||||
int y_pos;
|
|
||||||
gboolean trace;
|
gboolean trace;
|
||||||
|
|
||||||
|
/* Lock access to y_pos with this, use the cond to wake up stalled
|
||||||
|
* threads.
|
||||||
|
*/
|
||||||
|
GMutex *lock;
|
||||||
|
GCond *ready;
|
||||||
|
|
||||||
|
int y_pos;
|
||||||
} VipsSequential;
|
} VipsSequential;
|
||||||
|
|
||||||
typedef VipsConversionClass VipsSequentialClass;
|
typedef VipsConversionClass VipsSequentialClass;
|
||||||
|
|
||||||
G_DEFINE_TYPE( VipsSequential, vips_sequential, VIPS_TYPE_CONVERSION );
|
G_DEFINE_TYPE( VipsSequential, vips_sequential, VIPS_TYPE_CONVERSION );
|
||||||
|
|
||||||
|
static void
|
||||||
|
vips_sequential_dispose( GObject *gobject )
|
||||||
|
{
|
||||||
|
VipsSequential *sequential = (VipsSequential *) gobject;
|
||||||
|
|
||||||
|
VIPS_FREEF( g_mutex_free, sequential->lock );
|
||||||
|
VIPS_FREEF( g_cond_free, sequential->ready );
|
||||||
|
|
||||||
|
G_OBJECT_CLASS( vips_sequential_parent_class )->dispose( gobject );
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
vips_sequential_generate( VipsRegion *or,
|
vips_sequential_generate( VipsRegion *or,
|
||||||
void *seq, void *a, void *b, gboolean *stop )
|
void *seq, void *a, void *b, gboolean *stop )
|
||||||
@ -77,55 +98,53 @@ vips_sequential_generate( VipsRegion *or,
|
|||||||
VipsRect *r = &or->valid;
|
VipsRect *r = &or->valid;
|
||||||
VipsRegion *ir = (VipsRegion *) seq;
|
VipsRegion *ir = (VipsRegion *) seq;
|
||||||
|
|
||||||
if( sequential->trace )
|
printf( "vips_sequential_generate: thread %p "
|
||||||
vips_diag( "VipsSequential",
|
"request for %d lines, starting at line %d\n",
|
||||||
"%d lines, starting at line %d", r->height, r->top );
|
g_thread_self(),
|
||||||
|
r->height, r->top );
|
||||||
/* We can't go backwards, but we can skip forwards.
|
|
||||||
*/
|
|
||||||
if( r->top < sequential->y_pos ) {
|
|
||||||
vips_error( "VipsSequential",
|
|
||||||
_( "at line %d in file, but line %d requested" ),
|
|
||||||
sequential->y_pos, r->top );
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
/* We're inside a tilecache where tiles are the full image width, so
|
|
||||||
* this should always be true.
|
|
||||||
*/
|
|
||||||
g_assert( r->left == 0 );
|
|
||||||
g_assert( r->width == or->im->Xsize );
|
|
||||||
g_assert( VIPS_RECT_BOTTOM( r ) <= or->im->Ysize );
|
|
||||||
|
|
||||||
/* Skip forwards, if necessary.
|
|
||||||
*/
|
|
||||||
while( sequential->y_pos < r->top ) {
|
|
||||||
VipsRect rect;
|
|
||||||
|
|
||||||
rect.top = sequential->y_pos;
|
|
||||||
rect.left = 0;
|
|
||||||
rect.width = or->im->Xsize;
|
|
||||||
rect.height = VIPS_MIN( r->top - sequential->y_pos,
|
|
||||||
VIPS__FATSTRIP_HEIGHT );
|
|
||||||
if( vips_region_prepare( ir, &rect ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
sequential->y_pos += rect.height;
|
|
||||||
|
|
||||||
if( sequential->trace )
|
if( sequential->trace )
|
||||||
vips_diag( "VipsSequential",
|
vips_diag( "VipsSequential",
|
||||||
"skipping %d lines", rect.height );
|
"request for %d lines, starting at line %d",
|
||||||
|
r->height, r->top );
|
||||||
|
retry:
|
||||||
|
|
||||||
|
g_mutex_lock( sequential->lock );
|
||||||
|
|
||||||
|
if( r->top > sequential->y_pos ) {
|
||||||
|
/* This is for stuff in the future, stall.
|
||||||
|
*/
|
||||||
|
printf( "thread %p stalling ...\n", g_thread_self() );
|
||||||
|
g_cond_wait( sequential->ready, sequential->lock );
|
||||||
|
printf( "thread %p awake again, retrying ...\n",
|
||||||
|
g_thread_self() );
|
||||||
|
goto retry;
|
||||||
}
|
}
|
||||||
|
|
||||||
g_assert( sequential->y_pos == r->top );
|
/* This is a request for old or present pixels -- serve from cache.
|
||||||
|
* This may trigger further, sequential reads.
|
||||||
/* Pointer copy.
|
|
||||||
*/
|
*/
|
||||||
|
printf( "thread %p reading ...\n", g_thread_self() );
|
||||||
if( vips_region_prepare( ir, r ) ||
|
if( vips_region_prepare( ir, r ) ||
|
||||||
vips_region_region( or, ir, r, r->left, r->top ) )
|
vips_region_region( or, ir, r, r->left, r->top ) ) {
|
||||||
|
g_mutex_unlock( sequential->lock );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
}
|
||||||
|
|
||||||
sequential->y_pos += r->height;
|
if( VIPS_RECT_BOTTOM( r ) >= sequential->y_pos ) {
|
||||||
|
/* This request has straddled or continued on from the read
|
||||||
|
* point. Update it, and wake up all stalled threads for a
|
||||||
|
* retry.
|
||||||
|
*/
|
||||||
|
sequential->y_pos = VIPS_RECT_BOTTOM( r );
|
||||||
|
|
||||||
|
printf( "thread %p updating y_pos and waking stalled\n",
|
||||||
|
g_thread_self() );
|
||||||
|
|
||||||
|
g_cond_broadcast( sequential->ready );
|
||||||
|
}
|
||||||
|
|
||||||
|
g_mutex_unlock( sequential->lock );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
@ -136,6 +155,8 @@ vips_sequential_build( VipsObject *object )
|
|||||||
VipsConversion *conversion = VIPS_CONVERSION( object );
|
VipsConversion *conversion = VIPS_CONVERSION( object );
|
||||||
VipsSequential *sequential = (VipsSequential *) object;
|
VipsSequential *sequential = (VipsSequential *) object;
|
||||||
|
|
||||||
|
VipsImage *t;
|
||||||
|
|
||||||
VIPS_DEBUG_MSG( "vips_sequential_build\n" );
|
VIPS_DEBUG_MSG( "vips_sequential_build\n" );
|
||||||
|
|
||||||
if( VIPS_OBJECT_CLASS( vips_sequential_parent_class )->build( object ) )
|
if( VIPS_OBJECT_CLASS( vips_sequential_parent_class )->build( object ) )
|
||||||
@ -144,14 +165,22 @@ vips_sequential_build( VipsObject *object )
|
|||||||
if( vips_image_pio_input( sequential->in ) )
|
if( vips_image_pio_input( sequential->in ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
if( vips_image_copy_fields( conversion->out, sequential->in ) )
|
if( vips_tilecache( sequential->in, &t,
|
||||||
|
"tile_width", sequential->in->Xsize,
|
||||||
|
"tile_height", 1,
|
||||||
|
"max_tiles", 500,
|
||||||
|
"strategy", VIPS_CACHE_SEQUENTIAL,
|
||||||
|
NULL ) )
|
||||||
|
return( -1 );
|
||||||
|
vips_object_local( object, t );
|
||||||
|
|
||||||
|
if( vips_image_copy_fields( conversion->out, t ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
vips_demand_hint( conversion->out,
|
vips_demand_hint( conversion->out,
|
||||||
VIPS_DEMAND_STYLE_FATSTRIP, sequential->in, NULL );
|
VIPS_DEMAND_STYLE_FATSTRIP, t, NULL );
|
||||||
|
|
||||||
if( vips_image_generate( conversion->out,
|
if( vips_image_generate( conversion->out,
|
||||||
vips_start_one, vips_sequential_generate, vips_stop_one,
|
vips_start_one, vips_sequential_generate, vips_stop_one,
|
||||||
sequential->in, sequential ) )
|
t, sequential ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
@ -165,6 +194,7 @@ vips_sequential_class_init( VipsSequentialClass *class )
|
|||||||
|
|
||||||
VIPS_DEBUG_MSG( "vips_sequential_class_init\n" );
|
VIPS_DEBUG_MSG( "vips_sequential_class_init\n" );
|
||||||
|
|
||||||
|
gobject_class->dispose = vips_sequential_dispose;
|
||||||
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;
|
||||||
|
|
||||||
@ -190,6 +220,8 @@ static void
|
|||||||
vips_sequential_init( VipsSequential *sequential )
|
vips_sequential_init( VipsSequential *sequential )
|
||||||
{
|
{
|
||||||
sequential->trace = FALSE;
|
sequential->trace = FALSE;
|
||||||
|
sequential->lock = g_mutex_new();
|
||||||
|
sequential->ready = g_cond_new();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -79,7 +79,7 @@ typedef struct {
|
|||||||
int time; /* Time of last use for flush */
|
int time; /* Time of last use for flush */
|
||||||
int x; /* xy pos in VIPS image cods */
|
int x; /* xy pos in VIPS image cods */
|
||||||
int y;
|
int y;
|
||||||
} Tile;
|
} VipsTile;
|
||||||
|
|
||||||
typedef struct _VipsTileCache {
|
typedef struct _VipsTileCache {
|
||||||
VipsConversion parent_instance;
|
VipsConversion parent_instance;
|
||||||
@ -103,7 +103,7 @@ typedef VipsConversionClass VipsTileCacheClass;
|
|||||||
G_DEFINE_TYPE( VipsTileCache, vips_tile_cache, VIPS_TYPE_CONVERSION );
|
G_DEFINE_TYPE( VipsTileCache, vips_tile_cache, VIPS_TYPE_CONVERSION );
|
||||||
|
|
||||||
static void
|
static void
|
||||||
tile_destroy( Tile *tile )
|
vips_tile_destroy( VipsTile *tile )
|
||||||
{
|
{
|
||||||
VipsTileCache *cache = tile->cache;
|
VipsTileCache *cache = tile->cache;
|
||||||
|
|
||||||
@ -121,9 +121,9 @@ static void
|
|||||||
vips_tile_cache_drop_all( VipsTileCache *cache )
|
vips_tile_cache_drop_all( VipsTileCache *cache )
|
||||||
{
|
{
|
||||||
while( cache->tiles ) {
|
while( cache->tiles ) {
|
||||||
Tile *tile = (Tile *) cache->tiles->data;
|
VipsTile *tile = (VipsTile *) cache->tiles->data;
|
||||||
|
|
||||||
tile_destroy( tile );
|
vips_tile_destroy( tile );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,12 +138,12 @@ vips_tile_cache_dispose( GObject *gobject )
|
|||||||
G_OBJECT_CLASS( vips_tile_cache_parent_class )->dispose( gobject );
|
G_OBJECT_CLASS( vips_tile_cache_parent_class )->dispose( gobject );
|
||||||
}
|
}
|
||||||
|
|
||||||
static Tile *
|
static VipsTile *
|
||||||
tile_new( VipsTileCache *cache )
|
vips_tile_new( VipsTileCache *cache )
|
||||||
{
|
{
|
||||||
Tile *tile;
|
VipsTile *tile;
|
||||||
|
|
||||||
if( !(tile = VIPS_NEW( NULL, Tile )) )
|
if( !(tile = VIPS_NEW( NULL, VipsTile )) )
|
||||||
return( NULL );
|
return( NULL );
|
||||||
|
|
||||||
tile->cache = cache;
|
tile->cache = cache;
|
||||||
@ -156,7 +156,7 @@ tile_new( VipsTileCache *cache )
|
|||||||
cache->ntiles += 1;
|
cache->ntiles += 1;
|
||||||
|
|
||||||
if( !(tile->region = vips_region_new( cache->in )) ) {
|
if( !(tile->region = vips_region_new( cache->in )) ) {
|
||||||
tile_destroy( tile );
|
vips_tile_destroy( tile );
|
||||||
return( NULL );
|
return( NULL );
|
||||||
}
|
}
|
||||||
vips__region_no_ownership( tile->region );
|
vips__region_no_ownership( tile->region );
|
||||||
@ -165,7 +165,7 @@ tile_new( VipsTileCache *cache )
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
tile_move( Tile *tile, int x, int y )
|
vips_tile_move( VipsTile *tile, int x, int y )
|
||||||
{
|
{
|
||||||
VipsRect area;
|
VipsRect area;
|
||||||
|
|
||||||
@ -184,14 +184,16 @@ tile_move( Tile *tile, int x, int y )
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Do we have a tile in the cache?
|
/* Do we have a tile in the cache?
|
||||||
|
*
|
||||||
|
* FIXME .. use a hash?
|
||||||
*/
|
*/
|
||||||
static Tile *
|
static VipsTile *
|
||||||
tile_search( VipsTileCache *cache, int x, int y )
|
vips_tile_search( VipsTileCache *cache, int x, int y )
|
||||||
{
|
{
|
||||||
GSList *p;
|
GSList *p;
|
||||||
|
|
||||||
for( p = cache->tiles; p; p = p->next ) {
|
for( p = cache->tiles; p; p = p->next ) {
|
||||||
Tile *tile = (Tile *) p->data;
|
VipsTile *tile = (VipsTile *) p->data;
|
||||||
|
|
||||||
if( tile->x == x && tile->y == y )
|
if( tile->x == x && tile->y == y )
|
||||||
return( tile );
|
return( tile );
|
||||||
@ -201,7 +203,7 @@ tile_search( VipsTileCache *cache, int x, int y )
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
tile_touch( Tile *tile )
|
vips_tile_touch( VipsTile *tile )
|
||||||
{
|
{
|
||||||
g_assert( tile->cache->ntiles >= 0 );
|
g_assert( tile->cache->ntiles >= 0 );
|
||||||
|
|
||||||
@ -211,7 +213,7 @@ tile_touch( Tile *tile )
|
|||||||
/* Fill a tile with pixels.
|
/* Fill a tile with pixels.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
tile_fill( Tile *tile, VipsRegion *in )
|
vips_tile_fill( VipsTile *tile, VipsRegion *in )
|
||||||
{
|
{
|
||||||
VipsRect area;
|
VipsRect area;
|
||||||
|
|
||||||
@ -226,7 +228,7 @@ tile_fill( Tile *tile, VipsRegion *in )
|
|||||||
&area, area.left, area.top ) )
|
&area, area.left, area.top ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
tile_touch( tile );
|
vips_tile_touch( tile );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
@ -234,17 +236,17 @@ tile_fill( Tile *tile, VipsRegion *in )
|
|||||||
/* Find existing tile, make a new tile, or if we have a full set of tiles,
|
/* Find existing tile, make a new tile, or if we have a full set of tiles,
|
||||||
* reuse LRU.
|
* reuse LRU.
|
||||||
*/
|
*/
|
||||||
static Tile *
|
static VipsTile *
|
||||||
tile_find( VipsTileCache *cache, VipsRegion *in, int x, int y )
|
vips_tile_find( VipsTileCache *cache, VipsRegion *in, int x, int y )
|
||||||
{
|
{
|
||||||
Tile *tile;
|
VipsTile *tile;
|
||||||
int oldest, topmost;
|
int oldest, topmost;
|
||||||
GSList *p;
|
GSList *p;
|
||||||
|
|
||||||
/* In cache already?
|
/* In cache already?
|
||||||
*/
|
*/
|
||||||
if( (tile = tile_search( cache, x, y )) ) {
|
if( (tile = vips_tile_search( cache, x, y )) ) {
|
||||||
tile_touch( tile );
|
vips_tile_touch( tile );
|
||||||
|
|
||||||
return( tile );
|
return( tile );
|
||||||
}
|
}
|
||||||
@ -253,9 +255,9 @@ tile_find( VipsTileCache *cache, VipsRegion *in, int x, int y )
|
|||||||
*/
|
*/
|
||||||
if( cache->max_tiles == -1 ||
|
if( cache->max_tiles == -1 ||
|
||||||
cache->ntiles < cache->max_tiles ) {
|
cache->ntiles < cache->max_tiles ) {
|
||||||
if( !(tile = tile_new( cache )) ||
|
if( !(tile = vips_tile_new( cache )) ||
|
||||||
tile_move( tile, x, y ) ||
|
vips_tile_move( tile, x, y ) ||
|
||||||
tile_fill( tile, in ) )
|
vips_tile_fill( tile, in ) )
|
||||||
return( NULL );
|
return( NULL );
|
||||||
|
|
||||||
return( tile );
|
return( tile );
|
||||||
@ -268,7 +270,7 @@ tile_find( VipsTileCache *cache, VipsRegion *in, int x, int y )
|
|||||||
oldest = cache->time;
|
oldest = cache->time;
|
||||||
tile = NULL;
|
tile = NULL;
|
||||||
for( p = cache->tiles; p; p = p->next ) {
|
for( p = cache->tiles; p; p = p->next ) {
|
||||||
Tile *t = (Tile *) p->data;
|
VipsTile *t = (VipsTile *) p->data;
|
||||||
|
|
||||||
if( t->time < oldest ) {
|
if( t->time < oldest ) {
|
||||||
oldest = t->time;
|
oldest = t->time;
|
||||||
@ -281,7 +283,7 @@ tile_find( VipsTileCache *cache, VipsRegion *in, int x, int y )
|
|||||||
topmost = cache->in->Ysize;
|
topmost = cache->in->Ysize;
|
||||||
tile = NULL;
|
tile = NULL;
|
||||||
for( p = cache->tiles; p; p = p->next ) {
|
for( p = cache->tiles; p; p = p->next ) {
|
||||||
Tile *t = (Tile *) p->data;
|
VipsTile *t = (VipsTile *) p->data;
|
||||||
|
|
||||||
if( t->y < topmost ) {
|
if( t->y < topmost ) {
|
||||||
topmost = t->y;
|
topmost = t->y;
|
||||||
@ -298,35 +300,13 @@ tile_find( VipsTileCache *cache, VipsRegion *in, int x, int y )
|
|||||||
|
|
||||||
VIPS_DEBUG_MSG( "tilecache: reusing tile %d x %d\n", tile->x, tile->y );
|
VIPS_DEBUG_MSG( "tilecache: reusing tile %d x %d\n", tile->x, tile->y );
|
||||||
|
|
||||||
if( tile_move( tile, x, y ) ||
|
if( vips_tile_move( tile, x, y ) ||
|
||||||
tile_fill( tile, in ) )
|
vips_tile_fill( tile, in ) )
|
||||||
return( NULL );
|
return( NULL );
|
||||||
|
|
||||||
return( tile );
|
return( tile );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Copy rect from @from to @to.
|
|
||||||
*/
|
|
||||||
static void
|
|
||||||
copy_region( VipsRegion *from, VipsRegion *to, VipsRect *area )
|
|
||||||
{
|
|
||||||
int y;
|
|
||||||
|
|
||||||
/* Area should be inside both from and to.
|
|
||||||
*/
|
|
||||||
g_assert( vips_rect_includesrect( &from->valid, area ) );
|
|
||||||
g_assert( vips_rect_includesrect( &to->valid, area ) );
|
|
||||||
|
|
||||||
/* Loop down common area, copying.
|
|
||||||
*/
|
|
||||||
for( y = area->top; y < VIPS_RECT_BOTTOM( area ); y++ ) {
|
|
||||||
VipsPel *p = VIPS_REGION_ADDR( from, area->left, y );
|
|
||||||
VipsPel *q = VIPS_REGION_ADDR( to, area->left, y );
|
|
||||||
|
|
||||||
memcpy( q, p, VIPS_IMAGE_SIZEOF_PEL( from->im ) * area->width );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Generate func.
|
/* Generate func.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
@ -353,7 +333,7 @@ vips_tile_cache_gen( VipsRegion *or,
|
|||||||
* the output region directly to the tile.
|
* the output region directly to the tile.
|
||||||
*
|
*
|
||||||
* However this would mean that tile drop on minimise could then leave
|
* However this would mean that tile drop on minimise could then leave
|
||||||
* dangling pointers, if minimuse was called on an active pipeline.
|
* dangling pointers, if minimise were called on an active pipeline.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
VIPS_DEBUG_MSG( "vips_tile_cache_gen: "
|
VIPS_DEBUG_MSG( "vips_tile_cache_gen: "
|
||||||
@ -362,11 +342,11 @@ vips_tile_cache_gen( VipsRegion *or,
|
|||||||
|
|
||||||
for( y = ys; y < VIPS_RECT_BOTTOM( r ); y += th )
|
for( y = ys; y < VIPS_RECT_BOTTOM( r ); y += th )
|
||||||
for( x = xs; x < VIPS_RECT_RIGHT( r ); x += tw ) {
|
for( x = xs; x < VIPS_RECT_RIGHT( r ); x += tw ) {
|
||||||
Tile *tile;
|
VipsTile *tile;
|
||||||
VipsRect tarea;
|
VipsRect tarea;
|
||||||
VipsRect hit;
|
VipsRect hit;
|
||||||
|
|
||||||
if( !(tile = tile_find( cache, in, x, y )) ) {
|
if( !(tile = vips_tile_find( cache, in, x, y )) ) {
|
||||||
g_mutex_unlock( cache->lock );
|
g_mutex_unlock( cache->lock );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
@ -381,8 +361,8 @@ vips_tile_cache_gen( VipsRegion *or,
|
|||||||
/* The part of the tile that we need.
|
/* The part of the tile that we need.
|
||||||
*/
|
*/
|
||||||
vips_rect_intersectrect( &tarea, r, &hit );
|
vips_rect_intersectrect( &tarea, r, &hit );
|
||||||
|
vips_region_copy( tile->region, or, &hit,
|
||||||
copy_region( tile->region, or, &hit );
|
hit.left, hit.top );
|
||||||
}
|
}
|
||||||
|
|
||||||
g_mutex_unlock( cache->lock );
|
g_mutex_unlock( cache->lock );
|
||||||
|
@ -956,35 +956,6 @@ vips_foreign_load_init( VipsForeignLoad *load )
|
|||||||
load->disc = TRUE;
|
load->disc = TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Make a sequential cache for a file reader.
|
|
||||||
*/
|
|
||||||
int
|
|
||||||
vips_foreign_tilecache( VipsImage *in, VipsImage **out, int strip_height )
|
|
||||||
{
|
|
||||||
int tile_width;
|
|
||||||
int tile_height;
|
|
||||||
int nlines;
|
|
||||||
int nstrips;
|
|
||||||
|
|
||||||
vips_get_tile_size( in, &tile_width, &tile_height, &nlines );
|
|
||||||
|
|
||||||
/* We need two buffers, each with enough strips to make a complete
|
|
||||||
* buffer. And double to be safe, since input buffers must be larger
|
|
||||||
* than output, and our buffers may not align exactly.
|
|
||||||
*/
|
|
||||||
nstrips = 2 * (1 + nlines / strip_height);
|
|
||||||
|
|
||||||
if( vips_tilecache( in, out,
|
|
||||||
"tile_width", in->Xsize,
|
|
||||||
"tile_height", strip_height * nstrips,
|
|
||||||
"max_tiles", 2,
|
|
||||||
"strategy", VIPS_CACHE_SEQUENTIAL,
|
|
||||||
NULL ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
return( 0 );
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Abstract base class for image savers.
|
/* Abstract base class for image savers.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@ -918,8 +918,7 @@ read_jpeg_image( ReadJpeg *jpeg, VipsImage *out )
|
|||||||
NULL, read_jpeg_generate, NULL,
|
NULL, read_jpeg_generate, NULL,
|
||||||
jpeg, NULL ) ||
|
jpeg, NULL ) ||
|
||||||
vips_sequential( t[0], &t[1], NULL ) ||
|
vips_sequential( t[0], &t[1], NULL ) ||
|
||||||
vips_foreign_tilecache( t[1], &t[2], 8 ) ||
|
vips_image_write( t[1], out ) )
|
||||||
vips_image_write( t[2], out ) )
|
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
|
@ -1429,8 +1429,7 @@ read_stripwise( ReadTiff *rtiff, VipsImage *out )
|
|||||||
NULL, tiff2vips_stripwise_generate, NULL,
|
NULL, tiff2vips_stripwise_generate, NULL,
|
||||||
rtiff, tbuf ) ||
|
rtiff, tbuf ) ||
|
||||||
vips_sequential( t[0], &t[1], NULL ) ||
|
vips_sequential( t[0], &t[1], NULL ) ||
|
||||||
vips_foreign_tilecache( t[1], &t[2], rtiff->rows_per_strip ) ||
|
vips_image_write( t[1], out ) )
|
||||||
vips_image_write( t[2], out ) )
|
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
|
@ -75,8 +75,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define DEBUG
|
|
||||||
*/
|
*/
|
||||||
|
#define DEBUG
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
@ -490,8 +490,7 @@ vips__png_read( const char *name, VipsImage *out )
|
|||||||
NULL, png2vips_generate, NULL,
|
NULL, png2vips_generate, NULL,
|
||||||
read, NULL ) ||
|
read, NULL ) ||
|
||||||
vips_sequential( t[0], &t[1], NULL ) ||
|
vips_sequential( t[0], &t[1], NULL ) ||
|
||||||
vips_foreign_tilecache( t[1], &t[2], 16 ) ||
|
vips_image_write( t[1], out ) )
|
||||||
vips_image_write( t[2], out ) )
|
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,10 +139,6 @@ int vips__sizealike( VipsImage *in1, VipsImage *in2,
|
|||||||
int vips__bandalike( const char *domain,
|
int vips__bandalike( const char *domain,
|
||||||
VipsImage *in1, VipsImage *in2, VipsImage **out1, VipsImage **out2 );
|
VipsImage *in1, VipsImage *in2, VipsImage **out1, VipsImage **out2 );
|
||||||
|
|
||||||
int vips_foreign_tilecache( VipsImage *in, VipsImage **out, int strip_height );
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void im__format_init( void );
|
void im__format_init( void );
|
||||||
|
|
||||||
|
@ -281,8 +281,6 @@ vips_operation_class_init( VipsOperationClass *class )
|
|||||||
static void
|
static void
|
||||||
vips_operation_init( VipsOperation *operation )
|
vips_operation_init( VipsOperation *operation )
|
||||||
{
|
{
|
||||||
/* Init our instance fields.
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -87,8 +87,14 @@ vips_semaphore_upn( VipsSemaphore *s, int n )
|
|||||||
s->v += n;
|
s->v += n;
|
||||||
value_after_op = s->v;
|
value_after_op = s->v;
|
||||||
#ifdef HAVE_THREADS
|
#ifdef HAVE_THREADS
|
||||||
g_mutex_unlock( s->mutex );
|
/* If we are only incrementing by one, we only need to wake a single
|
||||||
|
* thread. If we are incrementing by a lot, we must wake all threads.
|
||||||
|
*/
|
||||||
|
if( n == 1 )
|
||||||
g_cond_signal( s->cond );
|
g_cond_signal( s->cond );
|
||||||
|
else
|
||||||
|
g_cond_broadcast( s->cond );
|
||||||
|
g_mutex_unlock( s->mutex );
|
||||||
#endif /*HAVE_THREADS*/
|
#endif /*HAVE_THREADS*/
|
||||||
|
|
||||||
#ifdef DEBUG_IO
|
#ifdef DEBUG_IO
|
||||||
|
@ -38,8 +38,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define VIPS_DEBUG
|
|
||||||
*/
|
*/
|
||||||
|
#define VIPS_DEBUG
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
@ -337,8 +337,8 @@ wbuffer_allocate_fn( VipsThreadState *state, void *a, gboolean *stop )
|
|||||||
sink_base->y += sink_base->tile_height;
|
sink_base->y += sink_base->tile_height;
|
||||||
|
|
||||||
if( sink_base->y >= VIPS_RECT_BOTTOM( &write->buf->area ) ) {
|
if( sink_base->y >= VIPS_RECT_BOTTOM( &write->buf->area ) ) {
|
||||||
/* Block until the last write is done, then set write
|
/* Block until the write of the previous buffer
|
||||||
* of the front buffer going.
|
* is done, then set write of this buffer going.
|
||||||
*/
|
*/
|
||||||
if( wbuffer_flush( write ) )
|
if( wbuffer_flush( write ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
@ -365,7 +365,8 @@ wbuffer_allocate_fn( VipsThreadState *state, void *a, gboolean *stop )
|
|||||||
/* Position buf at the new y.
|
/* Position buf at the new y.
|
||||||
*/
|
*/
|
||||||
if( wbuffer_position( write->buf,
|
if( wbuffer_position( write->buf,
|
||||||
sink_base->y, sink_base->nlines ) )
|
//sink_base->y, sink_base->nlines ) )
|
||||||
|
sink_base->y, sink_base->tile_height ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -495,7 +496,8 @@ vips_sink_disc( VipsImage *im, VipsRegionWrite write_fn, void *a )
|
|||||||
result = 0;
|
result = 0;
|
||||||
if( !write.buf ||
|
if( !write.buf ||
|
||||||
!write.buf_back ||
|
!write.buf_back ||
|
||||||
wbuffer_position( write.buf, 0, write.sink_base.nlines ) ||
|
//wbuffer_position( write.buf, 0, write.sink_base.nlines ) ||
|
||||||
|
wbuffer_position( write.buf, 0, write.sink_base.tile_height ) ||
|
||||||
vips_threadpool_run( im,
|
vips_threadpool_run( im,
|
||||||
write_thread_state_new,
|
write_thread_state_new,
|
||||||
wbuffer_allocate_fn,
|
wbuffer_allocate_fn,
|
||||||
|
@ -63,8 +63,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define DEBUG
|
|
||||||
*/
|
*/
|
||||||
|
#define DEBUG
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
Loading…
Reference in New Issue
Block a user