works!
This commit is contained in:
parent
f5f0dda551
commit
fa5677abd7
@ -41,8 +41,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
*/
|
|
||||||
#define VIPS_DEBUG
|
#define VIPS_DEBUG
|
||||||
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
@ -59,6 +59,10 @@
|
|||||||
|
|
||||||
#include "conversion.h"
|
#include "conversion.h"
|
||||||
|
|
||||||
|
/* The number of scanlines we cache behind the read point.
|
||||||
|
*/
|
||||||
|
#define NLINES (500)
|
||||||
|
|
||||||
typedef struct _VipsSequential {
|
typedef struct _VipsSequential {
|
||||||
VipsConversion parent_instance;
|
VipsConversion parent_instance;
|
||||||
|
|
||||||
@ -72,6 +76,9 @@ typedef struct _VipsSequential {
|
|||||||
GMutex *lock;
|
GMutex *lock;
|
||||||
GCond *ready;
|
GCond *ready;
|
||||||
|
|
||||||
|
/* The next read from our source will fetch this scanline, ie. it's 0
|
||||||
|
* when we start.
|
||||||
|
*/
|
||||||
int y_pos;
|
int y_pos;
|
||||||
} VipsSequential;
|
} VipsSequential;
|
||||||
|
|
||||||
@ -98,10 +105,14 @@ vips_sequential_generate( VipsRegion *or,
|
|||||||
VipsRect *r = &or->valid;
|
VipsRect *r = &or->valid;
|
||||||
VipsRegion *ir = (VipsRegion *) seq;
|
VipsRegion *ir = (VipsRegion *) seq;
|
||||||
|
|
||||||
printf( "vips_sequential_generate: thread %p "
|
VIPS_DEBUG_MSG( "thread %p request for %d lines from at line %d\n",
|
||||||
"request for %d lines, starting at line %d\n",
|
g_thread_self(), r->height, r->top );
|
||||||
g_thread_self(),
|
|
||||||
r->height, r->top );
|
if( r->height > NLINES ) {
|
||||||
|
vips_error( "VipsSequential", _( "request for %d scanlines "
|
||||||
|
"will break caching" ), r->height );
|
||||||
|
return( -1 );
|
||||||
|
}
|
||||||
|
|
||||||
if( sequential->trace )
|
if( sequential->trace )
|
||||||
vips_diag( "VipsSequential",
|
vips_diag( "VipsSequential",
|
||||||
@ -111,39 +122,46 @@ retry:
|
|||||||
|
|
||||||
g_mutex_lock( sequential->lock );
|
g_mutex_lock( sequential->lock );
|
||||||
|
|
||||||
|
VIPS_DEBUG_MSG( "thread %p has lock ...\n", g_thread_self() );
|
||||||
|
|
||||||
if( r->top > sequential->y_pos ) {
|
if( r->top > sequential->y_pos ) {
|
||||||
/* This is for stuff in the future, stall.
|
/* This is for stuff in the future, stall.
|
||||||
*/
|
*/
|
||||||
printf( "thread %p stalling ...\n", g_thread_self() );
|
VIPS_DEBUG_MSG( "thread %p stalling ...\n", g_thread_self() );
|
||||||
g_cond_wait( sequential->ready, sequential->lock );
|
g_cond_wait( sequential->ready, sequential->lock );
|
||||||
printf( "thread %p awake again, retrying ...\n",
|
VIPS_DEBUG_MSG( "thread %p awake again, retrying ...\n",
|
||||||
g_thread_self() );
|
g_thread_self() );
|
||||||
|
g_mutex_unlock( sequential->lock );
|
||||||
goto retry;
|
goto retry;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is a request for old or present pixels -- serve from cache.
|
/* This is a request for old or present pixels -- serve from cache.
|
||||||
* This may trigger further, sequential reads.
|
* This may trigger further, sequential reads.
|
||||||
*/
|
*/
|
||||||
printf( "thread %p reading ...\n", g_thread_self() );
|
VIPS_DEBUG_MSG( "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 ) ) {
|
||||||
|
VIPS_DEBUG_MSG( "thread %p unlocking ...\n", g_thread_self() );
|
||||||
g_mutex_unlock( sequential->lock );
|
g_mutex_unlock( sequential->lock );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
if( VIPS_RECT_BOTTOM( r ) >= sequential->y_pos ) {
|
if( VIPS_RECT_BOTTOM( r ) > sequential->y_pos ) {
|
||||||
/* This request has straddled or continued on from the read
|
/* This request has moved the read point. Update it, and wake
|
||||||
* point. Update it, and wake up all stalled threads for a
|
* up all stalled threads for a retry.
|
||||||
* retry.
|
|
||||||
*/
|
*/
|
||||||
sequential->y_pos = VIPS_RECT_BOTTOM( r );
|
sequential->y_pos = VIPS_RECT_BOTTOM( r );
|
||||||
|
|
||||||
printf( "thread %p updating y_pos and waking stalled\n",
|
VIPS_DEBUG_MSG( "thread %p updating y_pos to %d and "
|
||||||
g_thread_self() );
|
"waking stalled\n",
|
||||||
|
g_thread_self(),
|
||||||
|
sequential->y_pos );
|
||||||
|
|
||||||
g_cond_broadcast( sequential->ready );
|
g_cond_broadcast( sequential->ready );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VIPS_DEBUG_MSG( "thread %p unlocking ...\n", g_thread_self() );
|
||||||
|
|
||||||
g_mutex_unlock( sequential->lock );
|
g_mutex_unlock( sequential->lock );
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
@ -168,7 +186,7 @@ vips_sequential_build( VipsObject *object )
|
|||||||
if( vips_tilecache( sequential->in, &t,
|
if( vips_tilecache( sequential->in, &t,
|
||||||
"tile_width", sequential->in->Xsize,
|
"tile_width", sequential->in->Xsize,
|
||||||
"tile_height", 1,
|
"tile_height", 1,
|
||||||
"max_tiles", 500,
|
"max_tiles", NLINES,
|
||||||
"strategy", VIPS_CACHE_SEQUENTIAL,
|
"strategy", VIPS_CACHE_SEQUENTIAL,
|
||||||
NULL ) )
|
NULL ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
@ -75,8 +75,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
*/
|
|
||||||
#define DEBUG
|
#define DEBUG
|
||||||
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
@ -38,8 +38,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
*/
|
|
||||||
#define VIPS_DEBUG
|
#define VIPS_DEBUG
|
||||||
|
*/
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
@ -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