revise cgifsave to save mem

a bit simpler too

copying dloebl's idea from https://github.com/libvips/libvips/pull/3018
This commit is contained in:
John Cupitt 2022-09-08 10:16:50 +01:00
parent 5d1e26255d
commit c0e91d139d

View File

@ -33,8 +33,8 @@
*/ */
/* /*
#define DEBUG_VERBOSE
*/ */
#define DEBUG_VERBOSE
#ifdef HAVE_CONFIG_H #ifdef HAVE_CONFIG_H
#include <config.h> #include <config.h>
@ -100,16 +100,13 @@ typedef struct _VipsForeignSaveCgif {
int *palette; int *palette;
int n_colours; int n_colours;
/* The current frame coming from libvips, and the y position /* The frame we are building, the y position in the frame.
* in the input image.
*/
VipsRegion *frame;
int write_y;
/* VipsRegion is not always contiguous, but we need contiguous RGBA
* forthe quantizer. We need to copy each frame to a local buffer.
*/ */
int frame_width;
int frame_height;
VipsPel *frame_bytes; VipsPel *frame_bytes;
int write_y;
int page_number;
/* The current frame as seen by libimagequant. /* The current frame as seen by libimagequant.
*/ */
@ -152,12 +149,8 @@ vips_foreign_save_cgif_dispose( GObject *gobject )
{ {
VipsForeignSaveCgif *cgif = (VipsForeignSaveCgif *) gobject; VipsForeignSaveCgif *cgif = (VipsForeignSaveCgif *) gobject;
if( cgif->frame ) { g_info( "cgifsave: %d frames", cgif->page_number );
g_info( "cgifsave: %d frames", g_info( "cgifsave: %d unique palettes", cgif->n_palettes_generated );
cgif->frame->im->Ysize / cgif->frame->valid.height );
g_info( "cgifsave: %d unique palettes",
cgif->n_palettes_generated );
}
VIPS_FREEF( cgif_close, cgif->cgif_context ); VIPS_FREEF( cgif_close, cgif->cgif_context );
@ -166,8 +159,6 @@ vips_foreign_save_cgif_dispose( GObject *gobject )
free_quantisation_result ); free_quantisation_result );
VIPS_FREEF( vips__quantise_attr_destroy, cgif->attr ); VIPS_FREEF( vips__quantise_attr_destroy, cgif->attr );
VIPS_UNREF( cgif->frame );
VIPS_UNREF( cgif->target ); VIPS_UNREF( cgif->target );
VIPS_FREE( cgif->index ); VIPS_FREE( cgif->index );
@ -427,15 +418,12 @@ static int
vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif ) vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif )
{ {
VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( cgif ); VipsObjectClass *class = VIPS_OBJECT_GET_CLASS( cgif );
VipsRect *frame_rect = &cgif->frame->valid; int n_pels = cgif->frame_height * cgif->frame_width;
int page_index = frame_rect->top / frame_rect->height;
int n_pels = frame_rect->height * frame_rect->width;
gboolean has_transparency; gboolean has_transparency;
gboolean has_alpha_constraint; gboolean has_alpha_constraint;
VipsPel * restrict p; VipsPel * restrict p;
int i; int i;
int y;
VipsQuantiseImage *image; VipsQuantiseImage *image;
gboolean use_local; gboolean use_local;
VipsQuantiseResult *quantisation_result; VipsQuantiseResult *quantisation_result;
@ -445,16 +433,9 @@ vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif )
VipsPel palette_rgb[256 * 3]; VipsPel palette_rgb[256 * 3];
#ifdef DEBUG_VERBOSE #ifdef DEBUG_VERBOSE
printf( "vips_foreign_save_cgif_write_frame: %d\n", page_index ); printf( "vips_foreign_save_cgif_write_frame: %d\n", cgif->page_number );
#endif/*DEBUG_VERBOSE*/ #endif/*DEBUG_VERBOSE*/
/* We need the frame as a contiguous RGBA buffer for the quantiser.
*/
for( y = 0; y < frame_rect->height; y++ )
memcpy( cgif->frame_bytes + y * 4 * frame_rect->width,
VIPS_REGION_ADDR( cgif->frame, 0, frame_rect->top + y ),
4 * frame_rect->width );
/* Threshold the alpha channel. /* Threshold the alpha channel.
* *
* Also, check if the alpha channel of the current frame matches the * Also, check if the alpha channel of the current frame matches the
@ -478,7 +459,7 @@ vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif )
p[2] = 0; p[2] = 0;
p[3] = 0; p[3] = 0;
if( page_index > 0 && if( cgif->page_number > 0 &&
cgif->previous_frame[i * 4 + 3] ) cgif->previous_frame[i * 4 + 3] )
has_alpha_constraint = TRUE; has_alpha_constraint = TRUE;
} }
@ -489,7 +470,7 @@ vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif )
/* Set up new frame for libimagequant. /* Set up new frame for libimagequant.
*/ */
image = vips__quantise_image_create_rgba( cgif->attr, image = vips__quantise_image_create_rgba( cgif->attr,
cgif->frame_bytes, frame_rect->width, frame_rect->height, 0 ); cgif->frame_bytes, cgif->frame_width, cgif->frame_height, 0 );
/* Quantise. /* Quantise.
*/ */
@ -541,8 +522,8 @@ vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif )
cgif->cgif_config.numLoops = cgif->loop; cgif->cgif_config.numLoops = cgif->loop;
#endif/*HAVE_CGIF_ATTR_NO_LOOP*/ #endif/*HAVE_CGIF_ATTR_NO_LOOP*/
cgif->cgif_config.width = frame_rect->width; cgif->cgif_config.width = cgif->frame_width;
cgif->cgif_config.height = frame_rect->height; cgif->cgif_config.height = cgif->frame_height;
cgif->cgif_config.pGlobalPalette = palette_rgb; cgif->cgif_config.pGlobalPalette = palette_rgb;
cgif->cgif_config.numGlobalPaletteEntries = n_colours; cgif->cgif_config.numGlobalPaletteEntries = n_colours;
cgif->cgif_config.pWriteFn = vips__cgif_write; cgif->cgif_config.pWriteFn = vips__cgif_write;
@ -570,7 +551,7 @@ vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif )
/* Pixels which are equal to pixels in the previous frame can be made /* Pixels which are equal to pixels in the previous frame can be made
* transparent, provided no alpha channel constraint is present. * transparent, provided no alpha channel constraint is present.
*/ */
if( page_index > 0 && if( cgif->page_number > 0 &&
!has_alpha_constraint ) { !has_alpha_constraint ) {
int trans = has_transparency ? 0 : n_colours; int trans = has_transparency ? 0 : n_colours;
@ -590,9 +571,9 @@ vips_foreign_save_cgif_write_frame( VipsForeignSaveCgif *cgif )
} }
if( cgif->delay && if( cgif->delay &&
page_index < cgif->delay_length ) cgif->page_number < cgif->delay_length )
frame_config.delay = frame_config.delay =
VIPS_RINT( cgif->delay[page_index] / 10.0 ); VIPS_RINT( cgif->delay[cgif->page_number] / 10.0 );
/* Attach a local palette, if we need one. /* Attach a local palette, if we need one.
*/ */
@ -627,57 +608,27 @@ static int
vips_foreign_save_cgif_sink_disc( VipsRegion *region, VipsRect *area, void *a ) vips_foreign_save_cgif_sink_disc( VipsRegion *region, VipsRect *area, void *a )
{ {
VipsForeignSaveCgif *cgif = (VipsForeignSaveCgif *) a; VipsForeignSaveCgif *cgif = (VipsForeignSaveCgif *) a;
int line_size = cgif->frame_width * 4;
int y;
#ifdef DEBUG_VERBOSE #ifdef DEBUG_VERBOSE
printf( "vips_foreign_save_cgif_sink_disc: strip at %d, height %d\n", printf( "vips_foreign_save_cgif_sink_disc: strip at %d, height %d\n",
area->top, area->height ); area->top, area->height );
#endif/*DEBUG_VERBOSE*/ #endif/*DEBUG_VERBOSE*/
/* Write the new pixels into frame. for( y = 0; y < area->height; y++ ) {
*/ memcpy( cgif->frame_bytes + cgif->write_y * line_size,
while( cgif->write_y < VIPS_RECT_BOTTOM( area ) ) { VIPS_REGION_ADDR( region, 0, area->top + y ),
VipsRect *to = &cgif->frame->valid; line_size );
cgif->write_y += 1;
VipsRect hit;
/* The bit of the frame that we can fill.
*/
vips_rect_intersectrect( area, to, &hit );
/* Write the new pixels into the frame.
*/
vips_region_copy( region, cgif->frame,
&hit, hit.left, hit.top );
cgif->write_y += hit.height;
/* If we've filled the frame, write and move it down.
*/
if( VIPS_RECT_BOTTOM( &hit ) == VIPS_RECT_BOTTOM( to ) ) {
VipsRect new_frame;
VipsRect image;
if( cgif->write_y >= cgif->frame_height ) {
if( vips_foreign_save_cgif_write_frame( cgif ) ) if( vips_foreign_save_cgif_write_frame( cgif ) )
return( -1 ); return( -1 );
new_frame.left = 0; cgif->write_y = 0;
new_frame.top = cgif->write_y; cgif->page_number += 1;
new_frame.width = to->width;
new_frame.height = to->height;
image.left = 0;
image.top = 0;
image.width = cgif->in->Xsize;
image.height = cgif->in->Ysize;
vips_rect_intersectrect( &new_frame, &image,
&new_frame );
/* End of image?
*/
if( vips_rect_isempty( &new_frame ) )
break;
if( vips_region_buffer( cgif->frame, &new_frame ) )
return( -1 );
} }
} }
@ -693,9 +644,6 @@ vips_foreign_save_cgif_build( VipsObject *object )
VipsImage **t = (VipsImage **) VipsImage **t = (VipsImage **)
vips_object_local_array( VIPS_OBJECT( cgif ), 2 ); vips_object_local_array( VIPS_OBJECT( cgif ), 2 );
int page_height;
VipsRect frame_rect;
if( VIPS_OBJECT_CLASS( vips_foreign_save_cgif_parent_class )-> if( VIPS_OBJECT_CLASS( vips_foreign_save_cgif_parent_class )->
build( object ) ) build( object ) )
return( -1 ); return( -1 );
@ -712,16 +660,14 @@ vips_foreign_save_cgif_build( VipsObject *object )
/* Animation properties. /* Animation properties.
*/ */
page_height = vips_image_get_page_height( cgif->in );
if( vips_image_get_typeof( cgif->in, "delay" ) ) if( vips_image_get_typeof( cgif->in, "delay" ) )
vips_image_get_array_int( cgif->in, "delay", vips_image_get_array_int( cgif->in, "delay",
&cgif->delay, &cgif->delay_length ); &cgif->delay, &cgif->delay_length );
if( vips_image_get_typeof( cgif->in, "loop" ) ) if( vips_image_get_typeof( cgif->in, "loop" ) )
vips_image_get_int( cgif->in, "loop", &cgif->loop ); vips_image_get_int( cgif->in, "loop", &cgif->loop );
frame_rect.left = 0;
frame_rect.top = 0; cgif->frame_height = vips_image_get_page_height( cgif->in );
frame_rect.width = cgif->in->Xsize; cgif->frame_width = cgif->in->Xsize;
frame_rect.height = page_height;
/* Reject images that exceed the pixel limit of libimagequant, /* Reject images that exceed the pixel limit of libimagequant,
* or that exceed the GIF limit of 64k per axis. * or that exceed the GIF limit of 64k per axis.
@ -729,38 +675,27 @@ vips_foreign_save_cgif_build( VipsObject *object )
* Frame width * height will fit in an int, though frame size will * Frame width * height will fit in an int, though frame size will
* need at least a uint. * need at least a uint.
*/ */
if( (guint64) frame_rect.width * frame_rect.height > INT_MAX / 4 || if( (guint64) cgif->frame_width * cgif->frame_height > INT_MAX / 4 ||
frame_rect.width > 65535 || cgif->frame_width > 65535 ||
frame_rect.height > 65535 ) { cgif->frame_height > 65535 ) {
vips_error( class->nickname, "%s", _( "frame too large" ) ); vips_error( class->nickname, "%s", _( "frame too large" ) );
return( -1 ); return( -1 );
} }
/* Assemble frames here.
*/
cgif->frame = vips_region_new( cgif->in );
if( vips_region_buffer( cgif->frame, &frame_rect ) )
return( -1 );
/* The regions will get used in the bg thread callback,
* so make sure we don't own them.
*/
vips__region_no_ownership( cgif->frame );
/* This RGBA frame as a contiguous buffer. /* This RGBA frame as a contiguous buffer.
*/ */
cgif->frame_bytes = g_malloc0( (size_t) 4 * cgif->frame_bytes = g_malloc0( (size_t) 4 *
frame_rect.width * frame_rect.height ); cgif->frame_width * cgif->frame_height );
/* The previous RGBA frame (for spotting pixels which haven't changed). /* The previous RGBA frame (for spotting pixels which haven't changed).
*/ */
cgif->previous_frame = g_malloc0( (size_t) 4 * cgif->previous_frame = g_malloc0( (size_t) 4 *
frame_rect.width * frame_rect.height ); cgif->frame_width * cgif->frame_height );
/* The frame index buffer. /* The frame index buffer.
*/ */
cgif->index = g_malloc0( (size_t) frame_rect.width * cgif->index = g_malloc0( (size_t) cgif->frame_width *
frame_rect.height ); cgif->frame_height );
/* Set up libimagequant. /* Set up libimagequant.
*/ */