ppm load uses streams

This commit is contained in:
John Cupitt 2019-11-14 12:57:39 +00:00
parent 3bbadc0198
commit c1a027c8d7
6 changed files with 235 additions and 139 deletions

View File

@ -17,8 +17,8 @@
- nifti load/save uses double for all floating point metadata - nifti load/save uses double for all floating point metadata
- add vips_error_buffer_copy() - add vips_error_buffer_copy()
- add VipsStream: a universal IO class for loaders and savers - add VipsStream: a universal IO class for loaders and savers
- jpeg, png, tiff (though not tiffsave), rad, svg and webp use the new IO - jpeg, png, tiff (though not tiffsave), rad, svg, ppm and webp use the
class new IO class
- add @no_strip option to dzsave [kalozka1] - add @no_strip option to dzsave [kalozka1]
- add iiif layout to dzsave - add iiif layout to dzsave
- fix use of resolution-unit metadata on tiff save [kayarre] - fix use of resolution-unit metadata on tiff save [kayarre]

View File

@ -143,10 +143,10 @@ int vips__mat_load( const char *filename, VipsImage *out );
int vips__mat_header( const char *filename, VipsImage *out ); int vips__mat_header( const char *filename, VipsImage *out );
int vips__mat_ismat( const char *filename ); int vips__mat_ismat( const char *filename );
int vips__ppm_header( const char *name, VipsImage *out ); int vips__ppm_header_stream( VipsStreamib *streamib, VipsImage *out );
int vips__ppm_load( const char *name, VipsImage *out ); int vips__ppm_load_stream( VipsStreamib *streamib, VipsImage *out );
int vips__ppm_isppm( const char *filename ); VipsForeignFlags vips__ppm_flags_stream( VipsStreamib *streamib );
VipsForeignFlags vips__ppm_flags( const char *filename ); int vips__ppm_isppm_stream( VipsStreami *streami );
extern const char *vips__ppm_suffs[]; extern const char *vips__ppm_suffs[];
int vips__ppm_save_stream( VipsImage *in, VipsStreamo *streamo, int vips__ppm_save_stream( VipsImage *in, VipsStreamo *streamo,

View File

@ -37,7 +37,7 @@
* 29/7/19 Kyle-Kyle * 29/7/19 Kyle-Kyle
* - fix a loop with malformed ppm * - fix a loop with malformed ppm
* 13/11/19 * 13/11/19
* - ppm save redone with streams * - redone with streams
*/ */
/* /*
@ -90,36 +90,78 @@
*/ */
#define MAX_THING (80) #define MAX_THING (80)
/* After this, the next getc will be the first char of the next line (or EOF).
*/
static void static void
skip_line( FILE *fp ) skip_line( VipsStreamib *streamib )
{ {
int ch; int ch;
while( (ch = vips__fgetc( fp )) != '\n' && while( (ch = VIPS_STREAMIB_GETC( streamib )) != '\n' &&
ch != EOF ) ch != EOF )
; ;
} }
/* After this, the next getc will be the first char of the next block of
* non-whitespace (or EOF).
*/
static void static void
skip_white_space( FILE *fp ) skip_white_space( VipsStreamib *streamib )
{ {
int ch; int ch;
while( isspace( ch = vips__fgetc( fp ) ) ) while( isspace( ch = VIPS_STREAMIB_GETC( streamib ) ) )
; ;
ungetc( ch, fp ); VIPS_STREAMIB_UNGETC( streamib );
if( ch == '#' ) { if( ch == '#' ) {
skip_line( fp ); skip_line( streamib );
skip_white_space( fp ); skip_white_space( streamib );
} }
} }
/* After this, the next getc will be the first char of the next block of
* whitespace (or EOF). buf will be filled with up to length bytes, and
* null-terminated.
*
* If the first getc is whitespace, stop instantly and return nothing.
*/
static void
read_non_white_space( VipsStreamib *streamib, char *buf, int length )
{
int ch;
int i;
for( i = 0; i < length - 1 &&
!isspace( ch = VIPS_STREAMIB_GETC( streamib ) ) &&
ch != EOF; i++ )
buf[i] = ch;
buf[i] = '\0';
/* If we stopped before seeing any whitespace, skip to the end of the
* block of non-whitespace.
*/
if( !isspace( ch ) )
while( !isspace( ch = VIPS_STREAMIB_GETC( streamib ) ) &&
ch != EOF )
;
/* If we finally stopped on whitespace, step back one so the next get
* will be whitespace (or EOF).
*/
if( isspace( ch ) )
VIPS_STREAMIB_UNGETC( streamib );
}
static int static int
read_int( FILE *fp, int *i ) read_int( VipsStreamib *streamib, int *i )
{ {
skip_white_space( fp ); char buf[MAX_THING];
if( fscanf( fp, "%d", i ) != 1 ) {
skip_white_space( streamib );
read_non_white_space( streamib, buf, MAX_THING );
if( sscanf( buf, "%d", i ) != 1 ) {
vips_error( "ppm2vips", "%s", _( "bad int" ) ); vips_error( "ppm2vips", "%s", _( "bad int" ) );
return( -1 ); return( -1 );
} }
@ -128,10 +170,14 @@ read_int( FILE *fp, int *i )
} }
static int static int
read_float( FILE *fp, float *f ) read_float( VipsStreamib *streamib, float *f )
{ {
skip_white_space( fp ); char buf[MAX_THING];
if( fscanf( fp, "%f", f ) != 1 ) {
skip_white_space( streamib );
read_non_white_space( streamib, buf, MAX_THING );
if( sscanf( buf, "%f", f ) != 1 ) {
vips_error( "ppm2vips", "%s", _( "bad float" ) ); vips_error( "ppm2vips", "%s", _( "bad float" ) );
return( -1 ); return( -1 );
} }
@ -153,7 +199,8 @@ static char *magic_names[] = {
}; };
static int static int
read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first ) read_header( VipsStreamib *streamib, VipsImage *out,
int *bits, int *ascii, int *msb_first )
{ {
int width, height, bands; int width, height, bands;
VipsBandFormat format; VipsBandFormat format;
@ -175,8 +222,8 @@ read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first )
/* Read in the magic number. /* Read in the magic number.
*/ */
buf[0] = vips__fgetc( fp ); buf[0] = VIPS_STREAMIB_GETC( streamib );
buf[1] = vips__fgetc( fp ); buf[1] = VIPS_STREAMIB_GETC( streamib );
buf[2] = '\0'; buf[2] = '\0';
for( index = 0; index < VIPS_NUMBER( magic_names ); index++ ) for( index = 0; index < VIPS_NUMBER( magic_names ); index++ )
@ -196,8 +243,8 @@ read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first )
/* Read in size. /* Read in size.
*/ */
if( read_int( fp, &width ) || if( read_int( streamib, &width ) ||
read_int( fp, &height ) ) read_int( streamib, &height ) )
return( -1 ); return( -1 );
/* Read in max value / scale for >1 bit images. /* Read in max value / scale for >1 bit images.
@ -206,7 +253,7 @@ read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first )
if( index == 6 || index == 7 ) { if( index == 6 || index == 7 ) {
float scale; float scale;
if( read_float( fp, &scale ) ) if( read_float( streamib, &scale ) )
return( -1 ); return( -1 );
/* Scale > 0 means big-endian. /* Scale > 0 means big-endian.
@ -218,7 +265,7 @@ read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first )
else { else {
int max_value; int max_value;
if( read_int( fp, &max_value ) ) if( read_int( streamib, &max_value ) )
return( -1 ); return( -1 );
if( max_value > 255 ) if( max_value > 255 )
@ -232,7 +279,7 @@ read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first )
* character before the data starts. * character before the data starts.
*/ */
if( !*ascii && if( !*ascii &&
!isspace( vips__fgetc( fp ) ) ) { !isspace( VIPS_STREAMIB_GETC( streamib ) ) ) {
vips_error( "ppm2vips", "%s", vips_error( "ppm2vips", "%s",
_( "not whitespace before start of binary data" ) ); _( "not whitespace before start of binary data" ) );
return( -1 ); return( -1 );
@ -284,35 +331,53 @@ read_header( FILE *fp, VipsImage *out, int *bits, int *ascii, int *msb_first )
width, height, bands, format, width, height, bands, format,
VIPS_CODING_NONE, interpretation, 1.0, 1.0 ); VIPS_CODING_NONE, interpretation, 1.0, 1.0 );
VIPS_SETSTR( out->filename,
vips_stream_filename( VIPS_STREAM( streamib->streami ) ) );
return( 0 ); return( 0 );
} }
static void
read_mmap_close_cb( VipsObject *object, VipsStreamib *streamib )
{
VIPS_UNREF( streamib );
}
/* Read a ppm/pgm file using mmap(). /* Read a ppm/pgm file using mmap().
*/ */
static int static int
read_mmap( FILE *fp, const char *filename, int msb_first, VipsImage *out ) read_mmap( VipsStreamib *streamib, int msb_first, VipsImage *out )
{ {
const guint64 header_offset = ftell( fp );
VipsImage *x = vips_image_new();
VipsImage **t = (VipsImage **) VipsImage **t = (VipsImage **)
vips_object_local_array( VIPS_OBJECT( x ), 3 ); vips_object_local_array( VIPS_OBJECT( out ), 3 );
VipsStreami *streami = streamib->streami;
if( vips_rawload( filename, &t[0], gint64 header_offset;
out->Xsize, out->Ysize, VIPS_IMAGE_SIZEOF_PEL( out ), size_t length;
"offset", header_offset, const void *base_addr;
NULL ) ||
vips_streamib_unbuffer( streamib );
if( (header_offset = vips_streami_seek( streami, 0, SEEK_CUR )) < 0 ||
!(base_addr = vips_streami_map( streami, &length )) )
return( -1 );
if( !(t[0] = vips_image_new_from_memory(
base_addr + header_offset, length - header_offset,
out->Xsize, out->Ysize, out->Bands, out->BandFmt )) ||
vips_copy( t[0], &t[1], vips_copy( t[0], &t[1],
"bands", out->Bands, "interpretation", out->Type,
"format", out->BandFmt,
"coding", out->Coding,
NULL ) || NULL ) ||
vips__byteswap_bool( t[1], &t[2], vips__byteswap_bool( t[1], &t[2],
vips_amiMSBfirst() != msb_first ) || vips_amiMSBfirst() != msb_first ) ||
vips_image_write( t[2], out ) ) { vips_image_write( t[2], out ) )
g_object_unref( x );
return( -1 ); return( -1 );
}
g_object_unref( x ); /* out is using memory from the streami mmap. We need to keep a ref
* until out is closed.
*/
g_object_ref( streamib );
g_signal_connect( out, "close",
G_CALLBACK( read_mmap_close_cb ), streamib );
return( 0 ); return( 0 );
} }
@ -320,7 +385,7 @@ read_mmap( FILE *fp, const char *filename, int msb_first, VipsImage *out )
/* Read an ascii ppm/pgm file. /* Read an ascii ppm/pgm file.
*/ */
static int static int
read_ascii( FILE *fp, VipsImage *out ) read_ascii( VipsStreamib *streamib, VipsImage *out )
{ {
int x, y; int x, y;
VipsPel *buf; VipsPel *buf;
@ -332,7 +397,7 @@ read_ascii( FILE *fp, VipsImage *out )
for( x = 0; x < out->Xsize * out->Bands; x++ ) { for( x = 0; x < out->Xsize * out->Bands; x++ ) {
int val; int val;
if( read_int( fp, &val ) ) if( read_int( streamib, &val ) )
return( -1 ); return( -1 );
switch( out->BandFmt ) { switch( out->BandFmt ) {
@ -364,7 +429,7 @@ read_ascii( FILE *fp, VipsImage *out )
/* Read an ascii 1 bit file. /* Read an ascii 1 bit file.
*/ */
static int static int
read_1bit_ascii( FILE *fp, VipsImage *out ) read_1bit_ascii( VipsStreamib *streamib, VipsImage *out )
{ {
int x, y; int x, y;
VipsPel *buf; VipsPel *buf;
@ -376,7 +441,7 @@ read_1bit_ascii( FILE *fp, VipsImage *out )
for( x = 0; x < out->Xsize * out->Bands; x++ ) { for( x = 0; x < out->Xsize * out->Bands; x++ ) {
int val; int val;
if( read_int( fp, &val ) ) if( read_int( streamib, &val ) )
return( -1 ); return( -1 );
if( val ) if( val )
@ -395,7 +460,7 @@ read_1bit_ascii( FILE *fp, VipsImage *out )
/* Read a 1 bit binary file. /* Read a 1 bit binary file.
*/ */
static int static int
read_1bit_binary( FILE *fp, VipsImage *out ) read_1bit_binary( VipsStreamib *streamib, VipsImage *out )
{ {
int x, y; int x, y;
int bits; int bits;
@ -404,19 +469,19 @@ read_1bit_binary( FILE *fp, VipsImage *out )
if( !(buf = VIPS_ARRAY( out, VIPS_IMAGE_SIZEOF_LINE( out ), VipsPel )) ) if( !(buf = VIPS_ARRAY( out, VIPS_IMAGE_SIZEOF_LINE( out ), VipsPel )) )
return( -1 ); return( -1 );
bits = fgetc( fp ); bits = VIPS_STREAMIB_GETC( streamib );
for( y = 0; y < out->Ysize; y++ ) { for( y = 0; y < out->Ysize; y++ ) {
for( x = 0; x < out->Xsize * out->Bands; x++ ) { for( x = 0; x < out->Xsize * out->Bands; x++ ) {
buf[x] = (bits & 128) ? 0 : 255; buf[x] = (bits & 128) ? 0 : 255;
bits = VIPS_LSHIFT_INT( bits, 1 ); bits = VIPS_LSHIFT_INT( bits, 1 );
if( (x & 7) == 7 ) if( (x & 7) == 7 )
bits = fgetc( fp ); bits = VIPS_STREAMIB_GETC( streamib );
} }
/* Skip any unaligned bits at end of line. /* Skip any unaligned bits at end of line.
*/ */
if( (x & 7) != 0 ) if( (x & 7) != 0 )
bits = fgetc( fp ); bits = VIPS_STREAMIB_GETC( streamib );
if( vips_image_write_line( out, y, buf ) ) if( vips_image_write_line( out, y, buf ) )
return( -1 ); return( -1 );
@ -425,45 +490,39 @@ read_1bit_binary( FILE *fp, VipsImage *out )
return( 0 ); return( 0 );
} }
static int int
parse_ppm( FILE *fp, const char *filename, VipsImage *out ) vips__ppm_header_stream( VipsStreamib *streamib, VipsImage *out )
{ {
int bits; int bits;
int ascii; int ascii;
int msb_first; int msb_first;
if( read_header( fp, out, &bits, &ascii, &msb_first ) ) if( read_header( streamib, out, &bits, &ascii, &msb_first ) )
return( -1 );
return( 0 );
}
int
vips__ppm_load_stream( VipsStreamib *streamib, VipsImage *out )
{
int bits;
int ascii;
int msb_first;
if( read_header( streamib, out, &bits, &ascii, &msb_first ) )
return( -1 ); return( -1 );
/* What sort of read are we doing? /* What sort of read are we doing?
*/ */
if( !ascii && bits >= 8 ) if( !ascii && bits >= 8 )
return( read_mmap( fp, filename, msb_first, out ) ); return( read_mmap( streamib, msb_first, out ) );
else if( !ascii && bits == 1 ) else if( !ascii && bits == 1 )
return( read_1bit_binary( fp, out ) ); return( read_1bit_binary( streamib, out ) );
else if( ascii && bits == 1 ) else if( ascii && bits == 1 )
return( read_1bit_ascii( fp, out ) ); return( read_1bit_ascii( streamib, out ) );
else else
return( read_ascii( fp, out ) ); return( read_ascii( streamib, out ) );
}
int
vips__ppm_header( const char *filename, VipsImage *out )
{
FILE *fp;
int bits;
int ascii;
int msb_first;
if( !(fp = vips__file_open_read( filename, NULL, FALSE )) )
return( -1 );
if( read_header( fp, out, &bits, &ascii, &msb_first ) ) {
fclose( fp );
return( -1 );
}
fclose( fp );
return( 0 ); return( 0 );
} }
@ -471,82 +530,54 @@ vips__ppm_header( const char *filename, VipsImage *out )
/* Can this PPM file be read with a mmap? /* Can this PPM file be read with a mmap?
*/ */
static int static int
isppmmmap( const char *filename ) isppmmmap( VipsStreamib *streamib )
{ {
VipsImage *im; VipsImage *im;
FILE *fp;
int bits; int bits;
int ascii; int ascii;
int msb_first; int msb_first;
if( !(fp = vips__file_open_read( filename, NULL, FALSE )) )
return( -1 );
im = vips_image_new(); im = vips_image_new();
if( read_header( fp, im, &bits, &ascii, &msb_first ) ) { if( read_header( streamib, im, &bits, &ascii, &msb_first ) ) {
g_object_unref( im ); g_object_unref( im );
fclose( fp );
return( 0 ); return( 0 );
} }
g_object_unref( im ); g_object_unref( im );
fclose( fp );
return( !ascii && bits >= 8 ); return( !ascii && bits >= 8 );
} }
int
vips__ppm_load( const char *filename, VipsImage *out )
{
FILE *fp;
/* Note that we open in binary mode. If this is a binary PPM, we need
* to be able to mmap it.
*/
if( !(fp = vips__file_open_read( filename, NULL, FALSE )) )
return( -1 );
if( parse_ppm( fp, filename, out ) ) {
fclose( fp );
return( -1 );
}
fclose( fp );
return( 0 );
}
int
vips__ppm_isppm( const char *filename )
{
VipsPel buf[3];
if( vips__get_bytes( filename, buf, 2 ) == 2 ) {
int i;
buf[2] = '\0';
for( i = 0; i < VIPS_NUMBER( magic_names ); i++ )
if( strcmp( (char *) buf, magic_names[i] ) == 0 )
return( TRUE );
}
return( 0 );
}
/* ppm flags function. /* ppm flags function.
*/ */
VipsForeignFlags VipsForeignFlags
vips__ppm_flags( const char *filename ) vips__ppm_flags_stream( VipsStreamib *streamib )
{ {
VipsForeignFlags flags; VipsForeignFlags flags;
flags = 0; flags = 0;
if( isppmmmap( filename ) ) if( isppmmmap( streamib ) )
flags |= VIPS_FOREIGN_PARTIAL; flags |= VIPS_FOREIGN_PARTIAL;
return( flags ); return( flags );
} }
int
vips__ppm_isppm_stream( VipsStreami *streami )
{
const unsigned char *data;
if( (data = vips_streami_sniff( streami, 2 )) ) {
int i;
for( i = 0; i < VIPS_NUMBER( magic_names ); i++ )
if( vips_isprefix( magic_names[i], (char *) data ) )
return( TRUE );
}
return( 0 );
}
const char *vips__ppm_suffs[] = { ".ppm", ".pgm", ".pbm", ".pfm", NULL }; const char *vips__ppm_suffs[] = { ".ppm", ".pgm", ".pbm", ".pfm", NULL };
struct _Write; struct _Write;

View File

@ -66,10 +66,37 @@ typedef VipsForeignLoadClass VipsForeignLoadPpmClass;
G_DEFINE_TYPE( VipsForeignLoadPpm, vips_foreign_load_ppm, G_DEFINE_TYPE( VipsForeignLoadPpm, vips_foreign_load_ppm,
VIPS_TYPE_FOREIGN_LOAD ); VIPS_TYPE_FOREIGN_LOAD );
static gboolean
vips_foreign_load_ppm_is_a( const char *filename )
{
VipsStreami *streami;
gboolean result;
if( !(streami = vips_streami_new_from_filename( filename )) )
return( FALSE );
result = vips__ppm_isppm_stream( streami );
VIPS_UNREF( streami );
return( result );
}
static VipsForeignFlags static VipsForeignFlags
vips_foreign_load_ppm_get_flags_filename( const char *filename ) vips_foreign_load_ppm_get_flags_filename( const char *filename )
{ {
return( (VipsForeignFlags) vips__ppm_flags( filename ) ); VipsStreami *streami;
VipsStreamib *streamib;
VipsForeignFlags flags;
if( !(streami = vips_streami_new_from_filename( filename )) )
return( 0 );
streamib = vips_streamib_new( streami );
VIPS_UNREF( streami );
flags = vips__ppm_flags_stream( streamib );
VIPS_UNREF( streamib );
return( flags );
} }
static VipsForeignFlags static VipsForeignFlags
@ -85,10 +112,20 @@ vips_foreign_load_ppm_header( VipsForeignLoad *load )
{ {
VipsForeignLoadPpm *ppm = (VipsForeignLoadPpm *) load; VipsForeignLoadPpm *ppm = (VipsForeignLoadPpm *) load;
if( vips__ppm_header( ppm->filename, load->out ) ) VipsStreami *streami;
return( -1 ); VipsStreamib *streamib;
VIPS_SETSTR( load->out->filename, ppm->filename ); if( !(streami = vips_streami_new_from_filename( ppm->filename )) )
return( -1 );
streamib = vips_streamib_new( streami );
VIPS_UNREF( streami );
if( vips__ppm_header_stream( streamib, load->out ) ) {
VIPS_UNREF( streamib );
return( -1 );
}
VIPS_UNREF( streamib );
return( 0 ); return( 0 );
} }
@ -98,8 +135,20 @@ vips_foreign_load_ppm_load( VipsForeignLoad *load )
{ {
VipsForeignLoadPpm *ppm = (VipsForeignLoadPpm *) load; VipsForeignLoadPpm *ppm = (VipsForeignLoadPpm *) load;
if( vips__ppm_load( ppm->filename, load->real ) ) VipsStreami *streami;
VipsStreamib *streamib;
if( !(streami = vips_streami_new_from_filename( ppm->filename )) )
return( -1 ); return( -1 );
streamib = vips_streamib_new( streami );
VIPS_UNREF( streami );
if( vips__ppm_load_stream( streamib, load->real ) ) {
VIPS_UNREF( streamib );
return( -1 );
}
VIPS_UNREF( streamib );
return( 0 ); return( 0 );
} }
@ -124,7 +173,7 @@ vips_foreign_load_ppm_class_init( VipsForeignLoadPpmClass *class )
*/ */
foreign_class->priority = 200; foreign_class->priority = 200;
load_class->is_a = vips__ppm_isppm; load_class->is_a = vips_foreign_load_ppm_is_a;
load_class->get_flags_filename = load_class->get_flags_filename =
vips_foreign_load_ppm_get_flags_filename; vips_foreign_load_ppm_get_flags_filename;
load_class->get_flags = vips_foreign_load_ppm_get_flags; load_class->get_flags = vips_foreign_load_ppm_get_flags;

View File

@ -283,6 +283,10 @@ int vips_streamib_getc( VipsStreamib *streamib );
vips_streamib_getc( S ) \ vips_streamib_getc( S ) \
) )
void vips_streamib_ungetc( VipsStreamib *streamib ); void vips_streamib_ungetc( VipsStreamib *streamib );
#define VIPS_STREAMIB_UNGETC( S ) { \
if( (S)->read_point > 0 ) \
(S)->read_point -= 1; \
}
int vips_streamib_require( VipsStreamib *streamib, int require ); int vips_streamib_require( VipsStreamib *streamib, int require );
#define VIPS_STREAMIB_REQUIRE( S, R ) ( \ #define VIPS_STREAMIB_REQUIRE( S, R ) ( \

View File

@ -131,7 +131,7 @@ vips_streamib_unbuffer( VipsStreamib *streamib )
/* We'd read ahead a little way -- seek backwards by that amount. /* We'd read ahead a little way -- seek backwards by that amount.
*/ */
vips_streami_seek( streamib->streami, vips_streami_seek( streamib->streami,
streamib->read_point - streamib->chars_in_buffer, SEEK_SET ); streamib->read_point - streamib->chars_in_buffer, SEEK_CUR );
streamib->read_point = 0; streamib->read_point = 0;
streamib->chars_in_buffer = 0; streamib->chars_in_buffer = 0;
} }
@ -171,7 +171,7 @@ vips_streamib_refill( VipsStreamib *streamib )
* *
* Fetch the next character from the stream. * Fetch the next character from the stream.
* *
* Use the macro VIPS_STREAMIB_GETC() instead for speed. * If you can, use the macro VIPS_STREAMIB_GETC() instead for speed.
* *
* Returns: the next char from @streamib, -1 on read error or EOF. * Returns: the next char from @streamib, -1 on read error or EOF.
*/ */
@ -204,6 +204,8 @@ vips_streamib_getc( VipsStreamib *streamib )
* *
* unget more than one character is undefined. Unget at the start of the file * unget more than one character is undefined. Unget at the start of the file
* does nothing. * does nothing.
*
* If you can, use the macro VIPS_STREAMIB_UNGETC() instead for speed.
*/ */
void void
vips_streamib_ungetc( VipsStreamib *streamib ) vips_streamib_ungetc( VipsStreamib *streamib )
@ -212,6 +214,16 @@ vips_streamib_ungetc( VipsStreamib *streamib )
streamib->read_point -= 1; streamib->read_point -= 1;
} }
/**
* VIPS_STREAMIB_UNGETC:
* @streamib: stream to operate on
*
* The opposite of vips_streamib_getc(): undo the previous getc.
*
* unget more than one character is undefined. Unget at the start of the file
* does nothing.
*/
/** /**
* vips_streamib_require: * vips_streamib_require:
* @streamib: stream to operate on * @streamib: stream to operate on