fix map of custom seekable streams

We failed for seekable custom streams.

See https://github.com/libvips/libvips/issues/1494#issuecomment-567190830
This commit is contained in:
John Cupitt 2019-12-20 12:44:56 +00:00
parent 53dac98b55
commit 8030d7b926

View File

@ -672,7 +672,7 @@ vips_streami_read( VipsStreami *streami, void *buffer, size_t length )
bytes_read += available; bytes_read += available;
} }
else { else {
/* Some kind of filesystem source. /* Some kind of filesystem or custom source.
* *
* Get what we can from header_bytes. We may need to read * Get what we can from header_bytes. We may need to read
* some more after this. * some more after this.
@ -769,12 +769,12 @@ vips_streami_pipe_read_to_position( VipsStreami *streami, gint64 target )
while( target == -1 || while( target == -1 ||
streami->read_position < target ) { streami->read_position < target ) {
ssize_t read; ssize_t bytes_read;
read = vips_streami_read( streami, buffer, 4096 ); bytes_read = vips_streami_read( streami, buffer, 4096 );
if( read == -1 ) if( bytes_read == -1 )
return( -1 ); return( -1 );
if( read == 0 ) if( bytes_read == 0 )
break; break;
if( target == -1 && if( target == -1 &&
@ -790,6 +790,66 @@ vips_streami_pipe_read_to_position( VipsStreami *streami, gint64 target )
return( 0 ); return( 0 );
} }
/* Convert a seekable source that can't be mapped (eg. a custom input with a
* seek method) into a memory source.
*/
static int
vips_streami_read_to_memory( VipsStreami *streami )
{
GByteArray *byte_array;
gint64 read_position;
unsigned char *q;
VIPS_DEBUG_MSG( "vips_streami_read_to_memory:\n" );
g_assert( !streami->is_pipe );
g_assert( !streami->blob );
g_assert( !streami->header_bytes );
g_assert( streami->length >= 0 );
if( vips_streami_rewind( streami ) )
return( -1 );
/* We know the length, so we can size the buffer correctly and read
* directly to it.
*/
byte_array = g_byte_array_new();
g_byte_array_set_size( byte_array, streami->length );
/* Read in a series of chunks to reduce stress upstream.
*/
read_position = 0;
q = byte_array->data;
while( read_position < streami->length ) {
ssize_t bytes_read;
bytes_read = vips_streami_read( streami, q,
VIPS_MAX( 4096, streami->length - read_position ) );
if( bytes_read == -1 ) {
VIPS_FREEF( g_byte_array_unref, byte_array );
return( -1 );
}
if( bytes_read == 0 )
break;
read_position += bytes_read;
q += bytes_read;
}
/* Steal the byte_array pointer and turn into a memory source.
*
* We save byte_array in the header_bytes field to get it freed when
* we are freed.
*/
streami->data = byte_array->data;
streami->is_pipe = FALSE;
streami->header_bytes = byte_array;
vips_streami_minimise( streami );
return( 0 );
}
/* Read the entire pipe into memory and turn this into a memory source stream. /* Read the entire pipe into memory and turn this into a memory source stream.
*/ */
static int static int
@ -844,6 +904,10 @@ vips_streami_descriptor_to_memory( VipsStreami *streami )
* vips_streami_is_mappable: * vips_streami_is_mappable:
* @streami: input stream to operate on * @streami: input stream to operate on
* *
* Some streams can be efficiently mapped into memory.
* You can still use vips_streami_map() if this function returns %FALSE,
* but it will be slow.
*
* Returns: %TRUE if the stream can be efficiently mapped into memory. * Returns: %TRUE if the stream can be efficiently mapped into memory.
*/ */
gboolean gboolean
@ -859,7 +923,7 @@ vips_streami_is_mappable( VipsStreami *streami )
return( streami->data || return( streami->data ||
VIPS_STREAM( streami )->filename || VIPS_STREAM( streami )->filename ||
(!streami->is_pipe && (!streami->is_pipe &&
VIPS_STREAM( streami )->descriptor) ); VIPS_STREAM( streami )->descriptor != -1) );
} }
/** /**
@ -889,13 +953,17 @@ vips_streami_map( VipsStreami *streami, size_t *length_out )
return( NULL ); return( NULL );
if( !streami->data ) { if( !streami->data ) {
/* Seekable descriptors can simply be mapped. All other sources /* Seekable descriptors can simply be mapped. Seekable sources
* must be read into memory. * can be read. All other sources must be streamed into memory.
*/ */
if( vips_streami_is_mappable( streami ) ) { if( vips_streami_is_mappable( streami ) ) {
if( vips_streami_descriptor_to_memory( streami ) ) if( vips_streami_descriptor_to_memory( streami ) )
return( NULL ); return( NULL );
} }
else if( !streami->is_pipe ) {
if( vips_streami_read_to_memory( streami ) )
return( NULL );
}
else { else {
if( vips_streami_pipe_to_memory( streami ) ) if( vips_streami_pipe_to_memory( streami ) )
return( NULL ); return( NULL );
@ -1131,7 +1199,7 @@ vips_streami_sniff_at_most( VipsStreami *streami,
unsigned char **data, size_t length ) unsigned char **data, size_t length )
{ {
unsigned char *q; unsigned char *q;
size_t bytes_read; gint64 read_position;
VIPS_DEBUG_MSG( "vips_streami_sniff_at_most: %zd bytes\n", length ); VIPS_DEBUG_MSG( "vips_streami_sniff_at_most: %zd bytes\n", length );
@ -1143,26 +1211,27 @@ vips_streami_sniff_at_most( VipsStreami *streami,
g_byte_array_set_size( streami->sniff, length ); g_byte_array_set_size( streami->sniff, length );
read_position = 0;
q = streami->sniff->data; q = streami->sniff->data;
bytes_read = 0; while( read_position < length ) {
while( bytes_read < length ) { ssize_t bytes_read;
ssize_t n;
n = vips_streami_read( streami, q, length - bytes_read ); bytes_read = vips_streami_read( streami, q,
if( n == -1 ) length - read_position );
if( bytes_read == -1 )
return( -1 ); return( -1 );
if( n == 0 ) if( bytes_read == 0 )
break; break;
bytes_read += n; read_position += bytes_read;
q += n; q += bytes_read;
} }
SANITY( streami ); SANITY( streami );
*data = streami->sniff->data; *data = streami->sniff->data;
return( bytes_read ); return( read_position );
} }
/** /**