sink_screen works

This commit is contained in:
John Cupitt 2010-04-16 13:44:24 +00:00
parent 41921942e6
commit 31b06dbc5a
2 changed files with 100 additions and 63 deletions

7
TODO
View File

@ -1,16 +1,11 @@
- load big.v, scroll from the bottom to the top in 1:1 mode, does not catch up
with repaints when we get to the top
segv on exit during leak check
- im_prepare_thread() can go? I think it was only used by im_render() - im_prepare_thread() can go? I think it was only used by im_render()
wtf, im_vips2tiff.c seems to be using it, what's wrong with im_wbuffer()? wtf, im_vips2tiff.c seems to be using it, what's wrong with im_wbuffer()?
- get rid of threadgroup too (deprecate?)
- expose more of the tone funcs in nip2 - expose more of the tone funcs in nip2

View File

@ -35,12 +35,12 @@
*/ */
/* Trace allocate/free. /* Trace allocate/free.
*/
#define VIPS_DEBUG_RED #define VIPS_DEBUG_RED
*/
/* Trace reschedule /* Trace reschedule
*/
#define VIPS_DEBUG_GREEN #define VIPS_DEBUG_GREEN
*/
#ifdef HAVE_CONFIG_H #ifdef HAVE_CONFIG_H
#include <config.h> #include <config.h>
@ -206,7 +206,11 @@ render_free( Render *render )
g_mutex_lock( render_dirty_lock ); g_mutex_lock( render_dirty_lock );
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 );
im_semaphore_upn( &render_dirty_sem, -1 );
/* We could im_semaphore_upn( &render_dirty_sem, -1 ), but
* what's the point. We'd just wake up the bg thread
* for no reason.
*/
} }
g_mutex_unlock( render_dirty_lock ); g_mutex_unlock( render_dirty_lock );
@ -309,6 +313,7 @@ render_allocate( VipsThreadState *state, void *a, gboolean *stop )
Tile *tile; Tile *tile;
tile = (Tile *) render->dirty->data; tile = (Tile *) render->dirty->data;
g_assert( !tile->painted );
render->dirty = g_slist_remove( render->dirty, tile ); render->dirty = g_slist_remove( render->dirty, tile );
rstate->tile = tile; rstate->tile = tile;
} }
@ -326,21 +331,20 @@ render_work( VipsThreadState *state, void *a )
Tile *tile = rstate->tile; Tile *tile = rstate->tile;
g_assert( tile ); g_assert( tile );
g_assert( !tile->painted );
if( !tile->painted ) { VIPS_DEBUG_MSG( "calculating tile %dx%d\n",
VIPS_DEBUG_MSG( "calculating tile %dx%d\n", tile->area.left, tile->area.top );
tile->area.left, tile->area.top );
if( im_prepare_to( state->reg, tile->region, if( im_prepare_to( state->reg, tile->region,
&tile->area, tile->area.left, tile->area.top ) ) &tile->area, tile->area.left, tile->area.top ) )
return( -1 ); return( -1 );
tile->painted = TRUE; tile->painted = TRUE;
/* Now clients can update. /* Now clients can update.
*/ */
if( render->notify ) if( render->notify )
render->notify( render->out, &tile->area, render->a ); render->notify( render->out, &tile->area, render->a );
}
return( 0 ); return( 0 );
} }
@ -568,8 +572,9 @@ tile_new( Render *render )
tile->region = NULL; tile->region = NULL;
tile->area.left = 0; tile->area.left = 0;
tile->area.top = 0; tile->area.top = 0;
tile->area.width = 0; tile->area.width = render->tile_width;
tile->area.height = 0; tile->area.height = render->tile_height;
tile->painted = FALSE;
tile->ticks = render->ticks; tile->ticks = render->ticks;
if( !(tile->region = im_region_create( render->in )) ) { if( !(tile->region = im_region_create( render->in )) ) {
@ -591,6 +596,43 @@ render_tile_lookup( Render *render, Rect *area )
return( (Tile *) g_hash_table_lookup( render->tiles, area ) ); return( (Tile *) g_hash_table_lookup( render->tiles, area ) );
} }
/* Add a new tile to the table.
*/
static void
render_tile_add( Tile *tile, Rect *area )
{
Render *render = tile->render;
g_assert( !render_tile_lookup( render, area ) );
tile->painted = FALSE;
tile->area = *area;
/* Ignore buffer allocate errors, not much we could do with them.
*/
if( im_region_buffer( tile->region, area ) )
VIPS_DEBUG_MSG( "render_tile_add: buffer allocate failed\n" );
g_hash_table_insert( render->tiles, &tile->area, tile );
}
/* Move a tile to a new position.
*/
static void
render_tile_move( Tile *tile, Rect *area )
{
Render *render = tile->render;
g_assert( render_tile_lookup( render, &tile->area ) );
if( tile->area.left != area->left ||
tile->area.top != area->top ) {
g_assert( !render_tile_lookup( render, area ) );
g_hash_table_remove( render->tiles, &tile->area );
render_tile_add( tile, area );
}
}
/* We've looked at a tile ... bump to end of LRU and front of dirty. /* We've looked at a tile ... bump to end of LRU and front of dirty.
*/ */
static void static void
@ -601,37 +643,33 @@ tile_touch( Tile *tile )
tile->ticks = render->ticks; tile->ticks = render->ticks;
render->ticks += 1; render->ticks += 1;
if( !tile->painted ) { if( g_slist_find( render->dirty, tile ) ) {
g_assert( !tile->painted );
VIPS_DEBUG_MSG( "tile_bump_dirty: bumping tile %dx%d\n", VIPS_DEBUG_MSG( "tile_bump_dirty: bumping tile %dx%d\n",
tile->area.left, tile->area.top ); tile->area.left, tile->area.top );
if( g_slist_find( render->dirty, tile ) ) { render->dirty = g_slist_remove( render->dirty, tile );
render->dirty = g_slist_remove( render->dirty, tile ); render->dirty = g_slist_prepend( render->dirty, tile );
render->dirty = g_slist_prepend( render->dirty, tile );
}
} }
} }
/* Queue a tile for calculation. It might need moving too. /* Queue a tile for calculation.
*/ */
static void static void
tile_queue( Tile *tile, Rect *area ) tile_queue( Tile *tile )
{ {
Render *render = tile->render; Render *render = tile->render;
VIPS_DEBUG_MSG( "tile_set_dirty: adding tile %dx%d to dirty\n", VIPS_DEBUG_MSG( "tile_queue: adding tile %dx%d to dirty\n",
area->left, area->top ); tile->area.left, tile->area.top );
tile->painted = FALSE;
tile->area = *area;
if( im_region_buffer( tile->region, area ) )
printf( "poop!\n" );
g_hash_table_insert( render->tiles, &tile->area, tile );
if( render->notify && have_threads ) { if( render->notify && have_threads ) {
/* Add to the list of renders with dirty tiles. The bg /* Add to the list of renders with dirty tiles. The bg
* thread will pick it up and paint it. * thread will pick it up and paint it.
*/ */
g_assert( !g_slist_find( render->dirty, tile ) );
tile->painted = FALSE;
render->dirty = g_slist_prepend( render->dirty, tile ); render->dirty = g_slist_prepend( render->dirty, tile );
render_dirty_put( render ); render_dirty_put( render );
} }
@ -640,9 +678,9 @@ tile_queue( Tile *tile, Rect *area )
* 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.
*/ */
VIPS_DEBUG_MSG( "tile_set_dirty: " VIPS_DEBUG_MSG( "tile_queue: "
"painting tile %dx%d synchronously\n", "painting tile %dx%d synchronously\n",
area->left, area->top ); tile->area.left, tile->area.top );
im_prepare( tile->region, &tile->area ); im_prepare( tile->region, &tile->area );
tile->painted = TRUE; tile->painted = TRUE;
@ -692,6 +730,9 @@ render_tile_get_dirty( Render *render )
else { else {
tile = (Tile *) g_slist_last( render->dirty )->data; tile = (Tile *) g_slist_last( render->dirty )->data;
render->dirty = g_slist_remove( render->dirty, tile ); render->dirty = g_slist_remove( render->dirty, tile );
VIPS_DEBUG_MSG( "render_tile_get_dirty: "
"reusing dirty %p\n", tile );
} }
return( tile ); return( tile );
@ -705,12 +746,15 @@ tile_request( Render *render, Rect *area )
{ {
Tile *tile; Tile *tile;
VIPS_DEBUG_MSG( "tile_request: asking for %dx%d\n",
area->left, area->top );
if( (tile = render_tile_lookup( render, area )) ) { if( (tile = render_tile_lookup( render, area )) ) {
/* We already have a tile at this position. If it's invalid, /* We already have a tile at this position. If it's invalid,
* ask for a repaint. * ask for a repaint.
*/ */
if( !tile->painted || tile->region->invalid ) if( tile->region->invalid )
tile_queue( tile, area ); tile_queue( tile );
} }
else if( render->ntiles < render->max_tiles || else if( render->ntiles < render->max_tiles ||
render->max_tiles == -1 ) { render->max_tiles == -1 ) {
@ -720,26 +764,23 @@ tile_request( Render *render, Rect *area )
if( !(tile = tile_new( render )) ) if( !(tile = tile_new( render )) )
return( NULL ); return( NULL );
tile_queue( tile, area ); render_tile_add( tile, area );
tile_queue( tile );
} }
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. * then if that fails, reuse a dirty tile.
*/ */
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 )) ) {
VIPS_DEBUG_MSG( "tile_request: no tiles to reuse\n" );
return( NULL ); return( NULL );
}
VIPS_DEBUG_MSG( "(render_tile_get: was at %dx%d, " render_tile_move( tile, area );
"moving to %dx%d)\n",
tile->area.left, tile->area.top,
area->left, area->top );
/* Need to remove from the old position. tile_queue( tile );
*/
g_hash_table_remove( render->tiles, &tile->area );
tile_queue( tile, area );
} }
tile_touch( tile ); tile_touch( tile );
@ -892,16 +933,16 @@ mask_fill( REGION *out, void *seq, void *a, void *b )
* This operation renders @in in the background, making pixels available on * This operation renders @in in the background, making pixels available on
* @out as they are calculated. The @notify callback is run every time a new * @out as they are calculated. The @notify callback is run every time a new
* set of pixels are available. Calculated pixels are kept in a cache with * set of pixels are available. Calculated pixels are kept in a cache with
* tiles sized @tile_width by @tile_height pixels and at most @max_tiles tiles. * tiles sized @tile_width by @tile_height pixels and with at most @max_tiles
* tiles.
* If @max_tiles is -1, the cache is of unlimited size (up to the maximum image * If @max_tiles is -1, the cache is of unlimited size (up to the maximum image
* size). * size).
* The @mask image s a one-band uchar image and has 255 for pixels which are * The @mask image is a one-band uchar image and has 255 for pixels which are
* currently in cache and 0 for uncalculated pixels. * currently in cache and 0 for uncalculated pixels.
* *
* The pixel rendering system has a single global #im_threadgroup_t which is * Only a single sink is calculated at any one time, though many may be
* used for all currently active instances of im_render_priority(). As * alive. Use @priority to indicate which renders are more important:
* renderers are added and removed from the system, the threadgroup switches * zero means normal
* between renderers based on their priority setting. Zero means normal
* priority, negative numbers are low priority, positive numbers high * priority, negative numbers are low priority, positive numbers high
* priority. * priority.
* *
@ -911,15 +952,16 @@ mask_fill( REGION *out, void *seq, void *a, void *b )
* to a queue, and the @notify callback will trigger when those pixels are * to a queue, and the @notify callback will trigger when those pixels are
* ready. * ready.
* *
* The @notify callback is run from the background thread. In the callback, * The @notify callback is run from one of the background threads. In the
* callback
* you need to somehow send a message to the main thread that the pixels are * you need to somehow send a message to the main thread that the pixels are
* ready. In a glib-based application, this is easily done with g_idle_add(). * ready. In a glib-based application, this is easily done with g_idle_add().
* *
* If @notify is %NULL, then im_render_priority() runs synchronously. * If @notify is %NULL then im_render_priority() runs synchronously.
* im_prepare() on @out will always block until the pixels have been * im_prepare() on @out will always block until the pixels have been
* calculated by the background #im_threadgroup_t. * calculated.
* *
* See also: im_cache(), im_prepare(). * See also: im_cache(), im_prepare(), vips_sink_disc(), vips_sink().
* *
* Returns: 0 on sucess, -1 on error. * Returns: 0 on sucess, -1 on error.
*/ */