This commit is contained in:
John Cupitt 2009-03-05 13:20:43 +00:00
parent e2581e75a8
commit 88af68df64
3 changed files with 42 additions and 185 deletions

View File

@ -3,6 +3,7 @@
- isradiance was returning TRUE too often - isradiance was returning TRUE too often
- radiance loader now loads packed RGBE/XYZE instead of unpacking to float - radiance loader now loads packed RGBE/XYZE instead of unpacking to float
- added im_rad2float() - added im_rad2float()
- remove fading stuff from im_render() -- cleaner and simpler
11/10/08 started 7.17.0 11/10/08 started 7.17.0
- merge vips-7.16 brach back into trunk - merge vips-7.16 brach back into trunk

8
TODO
View File

@ -10,7 +10,13 @@
- im_render should use a hash for tile lookup ... or+shift x/y together - im_render should use a hash for tile lookup ... or+shift x/y together
get rid of the fading stuff, too complicated, a bit useless load wtc.jpg, rotate 42 degrees, zoom in and out quickly for a while, quit
tiles are leaked :(
does not leak if you wait for repaint to finish before zooming/unzooming
problem with waiting for paint to finish in render_kill?

View File

@ -33,6 +33,9 @@
* - more instrumenting * - more instrumenting
* 23/4/08 * 23/4/08
* - oop, broken for mask == NULL * - oop, broken for mask == NULL
* 5/3/09
* - remove all the fading stuff, a bit useless and it adds a lot of
* complexity
*/ */
/* /*
@ -67,7 +70,6 @@
#define DEBUG_MAKE #define DEBUG_MAKE
#define DEBUG_PAINT #define DEBUG_PAINT
#define DEBUG_TG #define DEBUG_TG
#define DEBUG_FADE
*/ */
#ifdef HAVE_CONFIG_H #ifdef HAVE_CONFIG_H
@ -84,12 +86,6 @@
#include <unistd.h> #include <unistd.h>
#endif /*HAVE_UNISTD_H*/ #endif /*HAVE_UNISTD_H*/
/* Need Sleep() on win32.
*/
#ifdef OS_WIN32
#include <windows.h>
#endif /*OS_WIN32*/
#include <vips/vips.h> #include <vips/vips.h>
#include <vips/thread.h> #include <vips/thread.h>
@ -108,11 +104,6 @@ static int threadgroup_count = 0;
static int threadgroup_active = 0; static int threadgroup_active = 0;
#endif /*DEBUG_TG*/ #endif /*DEBUG_TG*/
#ifdef DEBUG_FADE
static int fade_count = 0;
static int fade_active = 0;
#endif /*DEBUG_FADE*/
/* A manager thread. We have a fixed number of these taking jobs off the list /* A manager thread. We have a fixed number of these taking jobs off the list
* of current renders with dirty tiles, doing a tile, and putting the render * of current renders with dirty tiles, doing a tile, and putting the render
* back. * back.
@ -131,7 +122,6 @@ typedef void (*notify_fn)( IMAGE *, Rect *, void * );
typedef enum { typedef enum {
TILE_DIRTY, /* On the dirty list .. contains no pixels */ TILE_DIRTY, /* On the dirty list .. contains no pixels */
TILE_WORKING, /* Currently being worked on */ TILE_WORKING, /* Currently being worked on */
TILE_PAINTED_FADING, /* Painted, on fade list */
TILE_PAINTED /* Painted, ready for reuse */ TILE_PAINTED /* Painted, ready for reuse */
} TileState; } TileState;
@ -161,7 +151,7 @@ typedef struct {
*/ */
typedef struct _Render { typedef struct _Render {
/* Reference count this, since we use these things from several /* Reference count this, since we use these things from several
* threads. Can't easily use the gobject ref count system, since we * threads. Can't easily use the gobject ref count system since we
* need a lock around operations. * need a lock around operations.
*/ */
int ref_count; int ref_count;
@ -174,8 +164,6 @@ typedef struct _Render {
IMAGE *mask; /* Set valid pixels here */ IMAGE *mask; /* Set valid pixels here */
int width, height; /* Tile size */ int width, height; /* Tile size */
int max; /* Maximum number of tiles */ int max; /* Maximum number of tiles */
int fps; /* FPS for fade in */
int steps; /* Steps fade occurs in */
int priority; /* Larger numbers done sooner */ int priority; /* Larger numbers done sooner */
notify_fn notify; /* Tell caller about paints here */ notify_fn notify; /* Tell caller about paints here */
void *client; void *client;
@ -196,14 +184,6 @@ typedef struct _Render {
GMutex *dirty_lock; /* Lock before we read/write the dirty list */ GMutex *dirty_lock; /* Lock before we read/write the dirty list */
GSList *dirty; /* Tiles which need painting */ GSList *dirty; /* Tiles which need painting */
/* List of painted tiles being faded in. Plus a fade-in thread.
*/
GMutex *fade_lock; /* Lock before we read/write the fade list */
GSList *fade; /* Tiles which are being faded */
GThread *fade_gthread; /* Fade tiles in thread */
int time; /* Increment each frame */
int fade_kill; /* Set to ask fade thread to exit */
/* Render thread stuff. /* Render thread stuff.
*/ */
im_threadgroup_t *tg; /* Render with this threadgroup */ im_threadgroup_t *tg; /* Render with this threadgroup */
@ -233,9 +213,9 @@ render_dirty_remove( Render *render )
if( g_slist_find( render_dirty_all, render ) ) { if( g_slist_find( render_dirty_all, render ) ) {
render_dirty_all = g_slist_remove( render_dirty_all, render ); render_dirty_all = g_slist_remove( render_dirty_all, render );
/* We know this can't block, since there is at least 1 item in /* We know this can't block since there is at least 1 item in
* the render_dirty_all list (us). Except that * the render_dirty_all list (us). Except that
* render_dirty_get() does a _down() before it locks, so this * render_dirty_get() does a _down() before it locks so this
* could block if we run inbetween :-( possible deadlock. * could block if we run inbetween :-( possible deadlock.
*/ */
im_semaphore_down( &render_dirty_sem ); im_semaphore_down( &render_dirty_sem );
@ -272,20 +252,6 @@ render_free( Render *render )
render_dirty_remove( render ); render_dirty_remove( render );
/* Kill fade thread.
*/
if( render->fade_gthread ) {
render->fade_kill = 1;
(void) g_thread_join( render->fade_gthread );
render->fade_gthread = NULL;
render->fade_kill = 0;
#ifdef DEBUG_FADE
fade_active -= 1;
printf( "render_free: %d fade active\n", fade_active );
#endif /*DEBUG_FADE*/
}
IM_FREEF( im_threadgroup_free, render->tg ); IM_FREEF( im_threadgroup_free, render->tg );
/* Free cache. /* Free cache.
@ -295,12 +261,10 @@ render_free( Render *render )
IM_FREEF( g_slist_free, render->cache ); IM_FREEF( g_slist_free, render->cache );
render->ntiles = 0; render->ntiles = 0;
IM_FREEF( g_slist_free, render->dirty ); IM_FREEF( g_slist_free, render->dirty );
IM_FREEF( g_slist_free, render->fade );
g_mutex_free( render->ref_count_lock ); g_mutex_free( render->ref_count_lock );
g_mutex_free( render->dirty_lock ); g_mutex_free( render->dirty_lock );
g_mutex_free( render->read_lock ); g_mutex_free( render->read_lock );
g_mutex_free( render->fade_lock );
im_free( render ); im_free( render );
@ -373,8 +337,8 @@ render_dirty_get( void )
return( render ); return( render );
} }
/* Do a single tile. Take a dirty tile from the dirty list, fill with pixels /* Do a single tile. Take a dirty tile from the dirty list and fill with
* and add to the fade list. * pixels.
*/ */
static void static void
render_dirty_process( Render *render ) render_dirty_process( Render *render )
@ -442,32 +406,14 @@ render_dirty_process( Render *render )
im_error_buffer() ); im_error_buffer() );
#endif /*DEBUG_PAINT*/ #endif /*DEBUG_PAINT*/
/* Are we fading? Hand the tile over to the fade thread. /* All done.
*/ */
if( render->fade_gthread ) {
g_mutex_lock( render->fade_lock );
render->fade = g_slist_prepend( render->fade, tile );
tile->time = render->time;
tile->state = TILE_PAINTED_FADING;
/* Hand tile over to another thread.
*/
im__region_no_ownership( tile->region );
g_mutex_unlock( render->fade_lock );
}
else {
/* Not fading. Tell the user we're done ourselves.
*/
tile->time = render->time;
tile->state = TILE_PAINTED; tile->state = TILE_PAINTED;
im__region_no_ownership( tile->region ); im__region_no_ownership( tile->region );
if( render->notify ) if( render->notify )
render->notify( render->out, render->notify( render->out,
&tile->area, render->client ); &tile->area, render->client );
} }
}
} }
static int static int
@ -519,7 +465,15 @@ render_thread_main( void *client )
Render *render; Render *render;
if( (render = render_dirty_get()) ) { if( (render = render_dirty_get()) ) {
/* Loop here if this is the only dirty render, rather
* than bouncing back to _put()/_get(). We don't
* lock before testing since this is just a trivial
* optimisation and does not affect integrity.
*/
do {
render_dirty_process( render ); render_dirty_process( render );
} while( !render_dirty_all && render->dirty );
render_dirty_put( render ); render_dirty_put( render );
assert( render->ref_count == 1 || assert( render->ref_count == 1 ||
@ -603,68 +557,9 @@ tile_print( Tile *tile )
} }
#endif /*DEBUG*/ #endif /*DEBUG*/
/* Come here for the fade work thread.
*/
static void *
render_fade( void *client )
{
Render *render = (Render *) client;
for(;;) {
GSList *p;
GSList *next;
if( render->fade_kill )
break;
/* Increment frame counter. Used to set bg/fg ratio for fade.
*/
render->time += 1;
g_mutex_lock( render->fade_lock );
for( p = render->fade; p; p = next ) {
Tile *tile = (Tile *) p->data;
assert( tile->state == TILE_PAINTED_FADING );
/* We remove the current element as we scan :-( so get
* next beforehand.
*/
next = p->next;
if( render->time - tile->time > render->steps ) {
/* Tile is solid ... comes off the fade list.
*/
render->fade = g_slist_remove( render->fade,
tile );
tile->state = TILE_PAINTED;
}
/* Ask client to update.
*/
if( render->notify )
render->notify( render->out,
&tile->area, render->client );
}
g_mutex_unlock( render->fade_lock );
/* Sleep at the end of the loop, so if we have fading turned
* off there's no wait.
*/
#ifdef OS_WIN32
Sleep( 1000 / render->fps );
#else /*!OS_WIN32*/
usleep( 1000000 / render->fps );
#endif /*!OS_WIN32*/
}
return( NULL );
}
static Render * static Render *
render_new( IMAGE *in, IMAGE *out, IMAGE *mask, render_new( IMAGE *in, IMAGE *out, IMAGE *mask,
int width, int height, int max, int width, int height, int max,
int fps, int steps,
int priority, int priority,
notify_fn notify, void *client ) notify_fn notify, void *client )
{ {
@ -685,8 +580,6 @@ render_new( IMAGE *in, IMAGE *out, IMAGE *mask,
render->width = width; render->width = width;
render->height = height; render->height = height;
render->max = max; render->max = max;
render->fps = fps;
render->steps = steps;
render->priority = priority; render->priority = priority;
render->notify = notify; render->notify = notify;
render->client = client; render->client = client;
@ -700,12 +593,6 @@ render_new( IMAGE *in, IMAGE *out, IMAGE *mask,
render->dirty_lock = g_mutex_new(); render->dirty_lock = g_mutex_new();
render->dirty = NULL; render->dirty = NULL;
render->fade_lock = g_mutex_new();
render->fade = NULL;
render->fade_gthread = NULL;
render->time = 0;
render->fade_kill = 0;
render->tg = NULL; render->tg = NULL;
render->render_kill = FALSE; render->render_kill = FALSE;
@ -715,26 +602,6 @@ render_new( IMAGE *in, IMAGE *out, IMAGE *mask,
return( NULL ); return( NULL );
} }
/* Only need the fade thread if we're going to fade: ie. steps > 1.
*/
if( have_threads && steps > 1 ) {
if( !(render->fade_gthread = g_thread_create_full(
render_fade, render,
IM__DEFAULT_STACK_SIZE, TRUE, FALSE,
G_THREAD_PRIORITY_NORMAL, NULL )) ) {
im_error( "im_render",
"%s", _( "unable to create thread" ) );
return( NULL );
}
#ifdef DEBUG_FADE
fade_count += 1;
fade_active += 1;
printf( "render_new: creating fade thread %d\n", fade_count );
printf( "render_new: %d active\n", fade_active );
#endif /*DEBUG_FADE*/
}
return( render ); return( render );
} }
@ -859,7 +726,7 @@ tile_set_dirty( Tile *tile, Rect *area )
render_dirty_put( render ); render_dirty_put( render );
else { else {
/* No threads, or no notify ... paint the tile ourselves, /* No threads, or no notify ... paint the tile ourselves,
* sychronously. No need to notify the client, since they'll * sychronously. No need to notify the client since they'll
* never see black tiles. * never see black tiles.
*/ */
#ifdef DEBUG_PAINT #ifdef DEBUG_PAINT
@ -878,8 +745,6 @@ tile_set_dirty( Tile *tile, Rect *area )
im_prepare_thread( render->tg, im_prepare_thread( render->tg,
tile->region, &tile->area ); tile->region, &tile->area );
/* Can't fade in synchronous mode .. straight to painted.
*/
tile->state = TILE_PAINTED; tile->state = TILE_PAINTED;
render->dirty = g_slist_remove( render->dirty, tile ); render->dirty = g_slist_remove( render->dirty, tile );
} }
@ -978,8 +843,7 @@ render_tile_get( Render *render, Rect *area )
} }
else { else {
/* Need to reuse a tile. Try for an old painted tile first, /* Need to reuse a tile. Try for an old painted tile first,
* then if that fails, reuse a dirty tile. Don't search the * then if that fails, reuse a dirty tile.
* fading list, though we could maybe scavenge there.
*/ */
if( !(tile = render_tile_get_painted( render )) && if( !(tile = render_tile_get_painted( render )) &&
!(tile = render_tile_get_dirty( render )) ) !(tile = render_tile_get_dirty( render )) )
@ -1020,8 +884,7 @@ tile_copy( Tile *tile, REGION *to )
/* If the tile is painted, copy over the pixels. Otherwise, fill with /* If the tile is painted, copy over the pixels. Otherwise, fill with
* zero. * zero.
*/ */
if( tile->state == TILE_PAINTED || if( tile->state == TILE_PAINTED ) {
tile->state == TILE_PAINTED_FADING ) {
#ifdef DEBUG_PAINT #ifdef DEBUG_PAINT
printf( "tile_copy: copying calculated pixels for %dx%d\n", printf( "tile_copy: copying calculated pixels for %dx%d\n",
tile->area.left, tile->area.top ); tile->area.left, tile->area.top );
@ -1095,12 +958,12 @@ region_fill( REGION *out, void *seq, void *a, void *b )
return( 0 ); return( 0 );
} }
/* Paint the state of the 'painted' flag for a tile. Fade too. /* Paint the state of the 'painted' flag for a tile.
*/ */
static void static void
tile_paint_mask( Tile *tile, REGION *to ) tile_paint_mask( Tile *tile, REGION *to )
{ {
int mask; int mask = tile->state == TILE_PAINTED ? 255 : 0;
Rect ovlap; Rect ovlap;
int y; int y;
@ -1112,18 +975,6 @@ tile_paint_mask( Tile *tile, REGION *to )
assert( !im_rect_isempty( &ovlap ) ); assert( !im_rect_isempty( &ovlap ) );
len = IM_IMAGE_SIZEOF_PEL( to->im ) * ovlap.width; len = IM_IMAGE_SIZEOF_PEL( to->im ) * ovlap.width;
if( tile->state == TILE_PAINTED_FADING ) {
Render *render = tile->render;
int age = render->time - tile->time;
double opacity = IM_CLIP( 0, (double) age / render->steps, 1 );
mask = 255 * pow( opacity, 4 );
}
else if( tile->state == TILE_PAINTED )
mask = 255;
else
mask = 0;
for( y = ovlap.top; y < IM_RECT_BOTTOM( &ovlap ); y++ ) { for( y = ovlap.top; y < IM_RECT_BOTTOM( &ovlap ); y++ ) {
PEL *q = (PEL *) IM_REGION_ADDR( to, ovlap.left, y ); PEL *q = (PEL *) IM_REGION_ADDR( to, ovlap.left, y );
@ -1166,6 +1017,9 @@ mask_fill( REGION *out, void *seq, void *a, void *b )
return( 0 ); return( 0 );
} }
/* fps and steps are there for backwards compat only and no longer do
* anything.
*/
int int
im_render_fade( IMAGE *in, IMAGE *out, IMAGE *mask, im_render_fade( IMAGE *in, IMAGE *out, IMAGE *mask,
int width, int height, int max, int width, int height, int max,
@ -1180,12 +1034,8 @@ im_render_fade( IMAGE *in, IMAGE *out, IMAGE *mask,
if( render_thread_create() ) if( render_thread_create() )
return( -1 ); return( -1 );
/* Don't allow fps == 0, since we divide by this value to get wait if( width <= 0 || height <= 0 || max < -1 ) {
* time. im_error( "im_render", "%s", _( "bad parameters" ) );
*/
if( width <= 0 || height <= 0 || max < -1 || fps <= 0 || steps < 0 ) {
im_error( "im_render",
"%s", _( "bad parameters" ) );
return( -1 ); return( -1 );
} }
if( im_pincheck( in ) || im_poutcheck( out ) ) if( im_pincheck( in ) || im_poutcheck( out ) )
@ -1210,7 +1060,7 @@ im_render_fade( IMAGE *in, IMAGE *out, IMAGE *mask,
return( -1 ); return( -1 );
if( !(render = render_new( in, out, mask, if( !(render = render_new( in, out, mask,
width, height, max, fps, steps, priority, notify, client )) ) width, height, max, priority, notify, client )) )
return( -1 ); return( -1 );
#ifdef DEBUG_MAKE #ifdef DEBUG_MAKE