close jpeg read early

The current behaviour (close input handles on unref) works for languages
like C / C++ / Python / Rust / etc. where things get unreffed automatically
when they go out of scope.

On languages like Ruby / C# / node / etc. where things are unreffed on GC,
files can stay open for a long time after you've finished with them. This
interacts in an unfortunate way with the Windows default of refusing to
remove open files.

This change closes file handles as soon as the scan of the input file
finishes, and therefore produces something closer to expected behaviour
for GCd languages on Windows.

see https://github.com/kleisauke/net-vips/issues/12
This commit is contained in:
John Cupitt 2018-08-16 15:47:48 +01:00
parent f3ef6e52ea
commit b836749b75
2 changed files with 28 additions and 22 deletions

View File

@ -188,15 +188,30 @@ typedef struct _ReadJpeg {
int output_height; int output_height;
} ReadJpeg; } ReadJpeg;
/* This can be called many times. It's called directly at the end of image
* read.
*/
static void
readjpeg_close_input( ReadJpeg *jpeg )
{
VIPS_FREEF( fclose, jpeg->eman.fp );
/* Don't call jpeg_finish_decompress(). It just checks the tail of the
* file and who cares about that. All mem is freed in
* jpeg_destroy_decompress().
*/
/* I don't think this can fail. It's harmless to call many times.
*/
jpeg_destroy_decompress( &jpeg->cinfo );
}
/* This can be called many times. /* This can be called many times.
*/ */
static int static int
readjpeg_free( ReadJpeg *jpeg ) readjpeg_free( ReadJpeg *jpeg )
{ {
int result;
result = 0;
if( jpeg->eman.pub.num_warnings != 0 ) { if( jpeg->eman.pub.num_warnings != 0 ) {
g_warning( _( "read gave %ld warnings" ), g_warning( _( "read gave %ld warnings" ),
jpeg->eman.pub.num_warnings ); jpeg->eman.pub.num_warnings );
@ -207,24 +222,15 @@ readjpeg_free( ReadJpeg *jpeg )
jpeg->eman.pub.num_warnings = 0; jpeg->eman.pub.num_warnings = 0;
} }
/* Don't call jpeg_finish_decompress(). It just checks the tail of the readjpeg_close_input( jpeg );
* file and who cares about that. All mem is freed in
* jpeg_destroy_decompress().
*/
VIPS_FREEF( fclose, jpeg->eman.fp );
VIPS_FREE( jpeg->filename ); VIPS_FREE( jpeg->filename );
jpeg->eman.fp = NULL;
/* I don't think this can fail. It's harmless to call many times. return( 0 );
*/
jpeg_destroy_decompress( &jpeg->cinfo );
return( result );
} }
static void static void
readjpeg_close( VipsObject *object, ReadJpeg *jpeg ) readjpeg_close_cb( VipsObject *object, ReadJpeg *jpeg )
{ {
(void) readjpeg_free( jpeg ); (void) readjpeg_free( jpeg );
} }
@ -261,7 +267,7 @@ readjpeg_new( VipsImage *out, int shrink, gboolean fail, gboolean autorotate )
jpeg_create_decompress( &jpeg->cinfo ); jpeg_create_decompress( &jpeg->cinfo );
g_signal_connect( out, "close", g_signal_connect( out, "close",
G_CALLBACK( readjpeg_close ), jpeg ); G_CALLBACK( readjpeg_close_cb ), jpeg );
return( jpeg ); return( jpeg );
} }
@ -664,11 +670,10 @@ read_jpeg_generate( VipsRegion *or,
jpeg->y_pos += 1; jpeg->y_pos += 1;
} }
/* Progressive images can have a lot of memory in the decompress /* Shut down the input file as soon as we can.
* object, destroy as soon as we can. Safe to call many times.
*/ */
if( jpeg->y_pos >= or->im->Ysize ) if( jpeg->y_pos >= or->im->Ysize )
jpeg_destroy_decompress( &jpeg->cinfo ); readjpeg_close_input( jpeg );
VIPS_GATE_STOP( "read_jpeg_generate: work" ); VIPS_GATE_STOP( "read_jpeg_generate: work" );

View File

@ -385,9 +385,10 @@ class TestForeign:
self.file_loader("magickload", BMP_FILE, bmp_valid) self.file_loader("magickload", BMP_FILE, bmp_valid)
self.buffer_loader("magickload_buffer", BMP_FILE, bmp_valid) self.buffer_loader("magickload_buffer", BMP_FILE, bmp_valid)
# we should have rgba for svg files # we should have rgb or rgba for svg files ... different versions of
# IM handle this differently
im = pyvips.Image.magickload(SVG_FILE) im = pyvips.Image.magickload(SVG_FILE)
assert im.bands == 4 assert im.bands == 3 or im.bands == 4
# density should change size of generated svg # density should change size of generated svg
im = pyvips.Image.magickload(SVG_FILE, density='100') im = pyvips.Image.magickload(SVG_FILE, density='100')