lock for metadata changes

Another attempt at fixing crashes on metadata chenage in highly threaded
applications.

Global lock around set, remove and copy metadata. This is crude, but
simple, the performance impact should be small, and ought to resolve the
problem.

We'll do something better for the next version.

see https://github.com/lovell/sharp/issues/1986
This commit is contained in:
John Cupitt 2020-01-21 08:29:59 +00:00
parent eebc6e5636
commit fda5e5c402
2 changed files with 37 additions and 6 deletions

View File

@ -5,6 +5,7 @@
- fix a warning with magicksave with no delay array [chregu] - fix a warning with magicksave with no delay array [chregu]
- fix a race in tiled tiff load [kleisauke] - fix a race in tiled tiff load [kleisauke]
- better imagemagick init [LebronCurry] - better imagemagick init [LebronCurry]
- lock for metadata changes [jcupitt]
20/6/19 started 8.9.0 20/6/19 started 8.9.0
- add vips_image_get/set_array_int() - add vips_image_get/set_array_int()

View File

@ -930,8 +930,14 @@ meta_cp( VipsImage *dst, const VipsImage *src )
/* Loop, copying fields. /* Loop, copying fields.
*/ */
meta_init( dst ); meta_init( dst );
/* We lock with vips_image_set() to stop races in highly-
* threaded applications.
*/
g_mutex_lock( vips__global_lock );
vips_slist_map2( src->meta_traverse, vips_slist_map2( src->meta_traverse,
(VipsSListMap2Fn) meta_cp_field, dst, NULL ); (VipsSListMap2Fn) meta_cp_field, dst, NULL );
g_mutex_unlock( vips__global_lock );
} }
return( 0 ); return( 0 );
@ -1027,7 +1033,17 @@ vips_image_set( VipsImage *image, const char *name, GValue *value )
} }
meta_init( image ); meta_init( image );
/* We lock between modifying metadata and copying metadata between
* images, see meta_cp().
*
* This prevents modification of metadata by one thread racing with
* metadata copy on another -- this can lead to crashes in
* highly-threaded applications.
*/
g_mutex_lock( vips__global_lock );
(void) meta_new( image, name, value ); (void) meta_new( image, name, value );
g_mutex_unlock( vips__global_lock );
/* If we're setting an EXIF data block, we need to automatically expand /* If we're setting an EXIF data block, we need to automatically expand
* out all the tags. This will set things like xres/yres too. * out all the tags. This will set things like xres/yres too.
@ -1042,6 +1058,7 @@ vips_image_set( VipsImage *image, const char *name, GValue *value )
#ifdef DEBUG #ifdef DEBUG
meta_sanity( image ); meta_sanity( image );
#endif /*DEBUG*/ #endif /*DEBUG*/
} }
/* Unforunately gvalue seems to have no way of doing this. Just handle the vips /* Unforunately gvalue seems to have no way of doing this. Just handle the vips
@ -1219,19 +1236,32 @@ vips_image_get_typeof( const VipsImage *image, const char *name )
gboolean gboolean
vips_image_remove( VipsImage *image, const char *name ) vips_image_remove( VipsImage *image, const char *name )
{ {
gboolean result;
result = FALSE;
/* If this image is shared, block metadata changes. /* If this image is shared, block metadata changes.
*/ */
if( G_OBJECT( image )->ref_count > 1 ) { if( G_OBJECT( image )->ref_count > 1 ) {
g_warning( "can't remove metadata \"%s\" on shared image", g_warning( "can't remove metadata \"%s\" on shared image",
name ); name );
return( FALSE ); return( result );
} }
if( image->meta && if( image->meta ) {
g_hash_table_remove( image->meta, name ) ) /* We lock between modifying metadata and copying metadata
return( TRUE ); * between images, see meta_cp().
*
* This prevents modification of metadata by one thread
* racing with metadata copy on another -- this can lead to
* crashes in highly-threaded applications.
*/
g_mutex_lock( vips__global_lock );
result = g_hash_table_remove( image->meta, name );
g_mutex_unlock( vips__global_lock );
}
return( FALSE ); return( result );
} }
/* Deprecated header fields we hide from _map. /* Deprecated header fields we hide from _map.