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:
parent
f3ef6e52ea
commit
b836749b75
@ -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" );
|
||||||
|
|
||||||
|
@ -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')
|
||||||
|
Loading…
Reference in New Issue
Block a user