diff --git a/libsrc/iofuncs/im_open_vips.c b/libsrc/iofuncs/im_open_vips.c new file mode 100644 index 00000000..9351302a --- /dev/null +++ b/libsrc/iofuncs/im_open_vips.c @@ -0,0 +1,806 @@ +/* Read a VIPS file into a IMAGE * + * + * 22/5/08 + * - from im_open.c, im_openin.c, im_desc_hd.c, im_readhist.c + */ + +/* + + This file is part of VIPS. + + VIPS is free software; you can redistribute it and/or modify + it under the terms of the GNU Lesser General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU Lesser General Public License for more details. + + You should have received a copy of the GNU Lesser General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + + */ + +/* + + These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk + + */ + +/* +#define DEBUG + */ + +#ifdef HAVE_CONFIG_H +#include +#endif /*HAVE_CONFIG_H*/ +#include + +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#ifdef HAVE_SYS_FILE_H +#include +#endif /*HAVE_SYS_FILE_H*/ +#include +#ifdef HAVE_UNISTD_H +#include +#endif /*HAVE_UNISTD_H*/ +#ifdef HAVE_IO_H +#include +#endif /*HAVE_IO_H*/ +#include +#include + +#ifdef OS_WIN32 +#include +#endif /*OS_WIN32*/ + +#include +#include +#include + +#ifdef WITH_DMALLOC +#include +#endif /*WITH_DMALLOC*/ + +/* Try to make an O_BINARY ... sometimes need the leading '_'. + */ +#ifdef BINARY_OPEN +#ifndef O_BINARY +#ifdef _O_BINARY +#define O_BINARY _O_BINARY +#endif /*_O_BINARY*/ +#endif /*!O_BINARY*/ +#endif /*BINARY_OPEN*/ + +/* Our XML namespace. + */ +#define NAMESPACE "http://www.vips.ecs.soton.ac.uk/vips" + +/* mmap() whole vs. window threshold ... an int, so we can tune easily from a + * debugger. + */ +#ifdef DEBUG +int im__mmap_limit = 1; +#else +int im__mmap_limit = IM__MMAP_LIMIT; +#endif /*DEBUG*/ + +/* Sort of open for read for image files. + */ +static int +im__open_image_file( const char *filename ) +{ + int fd; + + /* Try to open read-write, so that calls to im_makerw() will work. + * When we later mmap this file, we set read-only, so there + * is little danger of scrubbing over files we own. + */ +#ifdef BINARY_OPEN + if( (fd = open( filename, O_RDWR | O_BINARY )) == -1 ) { +#else /*BINARY_OPEN*/ + if( (fd = open( filename, O_RDWR )) == -1 ) { +#endif /*BINARY_OPEN*/ + /* Open read-write failed. Fall back to open read-only. + */ +#ifdef BINARY_OPEN + if( (fd = open( filename, O_RDONLY | O_BINARY )) == -1 ) { +#else /*BINARY_OPEN*/ + if( (fd = open( filename, O_RDONLY )) == -1 ) { +#endif /*BINARY_OPEN*/ + im_error( "im__open_image_file", + _( "unable to open \"%s\", %s" ), + filename, strerror( errno ) ); + return( -1 ); + } + } + + return( fd ); +} + +/* Predict the size of the header plus pixel data. Don't use off_t, + * it's sometimes only 32 bits (eg. on many windows build environments) and we + * want to always be 64 bit. + */ +static gint64 +im__image_pixel_length( IMAGE *im ) +{ + gint64 psize; + + switch( im->Coding ) { + case IM_CODING_LABQ: + case IM_CODING_NONE: + psize = (gint64) IM_IMAGE_SIZEOF_LINE( im ) * im->Ysize; + break; + + default: + psize = im->Length; + break; + } + + return( psize + im->sizeof_header ); +} + +/* Read short/int/float LSB and MSB first. + */ +void +im__read_4byte( int msb_first, unsigned char *to, unsigned char **from ) +{ + unsigned char *p = *from; + int out; + + if( msb_first ) + out = p[0] << 24 | p[1] << 16 | p[2] << 8 | p[3]; + else + out = p[3] << 24 | p[2] << 16 | p[1] << 8 | p[0]; + + *from += 4; + *((guint32 *) to) = out; +} + +void +im__read_2byte( int msb_first, unsigned char *to, unsigned char **from ) +{ + int out; + unsigned char *p = *from; + + if( msb_first ) + out = p[0] << 8 | p[1]; + else + out = p[1] << 8 | p[0]; + + *from += 2; + *((guint16 *) to) = out; +} + +/* We always write in native byte order. + */ +void +im__write_4byte( unsigned char **to, unsigned char *from ) +{ + *((guint32 *) *to) = *((guint32 *) from); + *to += 4; +} + +void +im__write_2byte( unsigned char **to, unsigned char *from ) +{ + *((guint16 *) *to) = *((guint16 *) from); + *to += 2; +} + +/* offset, read, write functions. + */ +typedef struct _FieldIO { + glong offset; + void (*read)( int msb_first, unsigned char *to, unsigned char **from ); + void (*write)( unsigned char **to, unsigned char *from ); +} FieldIO; + +static FieldIO fields[] = { + { G_STRUCT_OFFSET( IMAGE, Xsize ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Ysize ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Bands ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Bbits ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, BandFmt ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Coding ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Type ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Xres ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Yres ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Length ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Compression ), + im__read_2byte, im__write_2byte }, + { G_STRUCT_OFFSET( IMAGE, Level ), + im__read_2byte, im__write_2byte }, + { G_STRUCT_OFFSET( IMAGE, Xoffset ), + im__read_4byte, im__write_4byte }, + { G_STRUCT_OFFSET( IMAGE, Yoffset ), + im__read_4byte, im__write_4byte } +}; + +static int +im__read_header_bytes( IMAGE *im, unsigned char *from ) +{ + int msb_first; + int i; + + im__read_4byte( 1, (unsigned char *) &im->magic, &from ); + if( im->magic != IM_MAGIC_INTEL && im->magic != IM_MAGIC_SPARC ) { + im_error( "im_open", _( "\"%s\" is not a VIPS image" ), + im->filename ); + return( -1 ); + } + msb_first = im->magic == IM_MAGIC_SPARC; + + for( i = 0; i < IM_NUMBER( fields ); i++ ) + fields[i].read( msb_first, + &G_STRUCT_MEMBER( unsigned char, im, fields[i].offset ), + &from ); + + /* Set this ourselves ... bbits is deprecated in the file format. + */ + im->Bbits = im_bits_of_fmt( im->BandFmt ); + + return( 0 ); +} + +int +im__write_header_bytes( IMAGE *im, unsigned char *to ) +{ + guint32 magic; + int i; + unsigned char *q; + + /* Always write the magic number MSB first. + */ + magic = im_amiMSBfirst() ? IM_MAGIC_SPARC : IM_MAGIC_INTEL; + to[0] = magic >> 24; + to[1] = magic >> 16; + to[2] = magic >> 8; + to[3] = magic; + q = to + 4; + + for( i = 0; i < IM_NUMBER( fields ); i++ ) + fields[i].write( &q, + &G_STRUCT_MEMBER( unsigned char, im, + fields[i].offset ) ); + + /* Pad spares with zeros. + */ + while( q - to < im->sizeof_header ) + *q++ = 0; + + return( 0 ); +} + +/* Read a chunk of an fd into memory. Add a '\0' at the end. + */ +static char * +read_chunk( int fd, gint64 offset, size_t length ) +{ + char *buf; + + if( im__seek( fd, offset ) ) + return( NULL ); + if( !(buf = im_malloc( NULL, length + 1 )) ) + return( NULL ); + if( read( fd, buf, length ) != length ) { + im_free( buf ); + im_error( "im_readhist", _( "unable to read history" ) ); + return( NULL ); + } + buf[length] = '\0'; + + return( buf ); +} + +/* Does it look like an image has an extension block? + */ +static int +im__has_extension_block( IMAGE *im ) +{ + gint64 length; + gint64 psize; + + psize = im__image_pixel_length( im ); + if( (length = im_file_length( im->fd )) == -1 ) + return( 0 ); + + return( length - psize > 0 ); +} + +/* Read everything after the pixels into memory. + */ +static void * +im__read_extension_block( IMAGE *im, int *size ) +{ + gint64 length; + gint64 psize; + void *buf; + + psize = im__image_pixel_length( im ); + if( (length = im_file_length( im->fd )) == -1 ) + return( NULL ); + if( length - psize > 10 * 1024 * 1024 ) { + im_error( "im_readhist", + _( "more than a 10 megabytes of XML? " + "sufferin' succotash!" ) ); + return( NULL ); + } + if( length - psize == 0 ) + return( NULL ); + if( !(buf = read_chunk( im->fd, psize, length - psize )) ) + return( NULL ); + if( size ) + *size = length - psize; + +#ifdef DEBUG + printf( "im__read_extension_block: read %d bytes from %s\n", + (int) (length - psize), im->filename ); + printf( "data: \"%s\"\n", (char *) buf ); +#endif /*DEBUG*/ + + return( buf ); +} + +/* Read everything after the pixels into memory. + + FIXME ... why can't we use xmlParserInputBufferCreateFd and parse + directly from the fd rather than having to read the stupid thing into + memory + + the libxml API docs are impossible to decipher + + */ +static xmlDoc * +read_xml( IMAGE *im ) +{ + void *buf; + int size; + xmlDoc *doc; + xmlNode *node; + + if( !(buf = im__read_extension_block( im, &size )) ) + return( NULL ); + if( !(doc = xmlParseMemory( buf, size )) ) { + im_free( buf ); + return( NULL ); + } + im_free( buf ); + if( !(node = xmlDocGetRootElement( doc )) || + !node->nsDef || + !im_isprefix( NAMESPACE, (char *) node->nsDef->href ) ) { + im_error( "im__readhist", _( "incorrect namespace in XML" ) ); + xmlFreeDoc( doc ); + return( NULL ); + } + +#ifdef DEBUG + printf( "read_xml: namespace == %s\n", node->nsDef->href ); +#endif /*DEBUG*/ + + return( doc ); +} + +/* Find the first child node with a name. + */ +static xmlNode * +get_node( xmlNode *base, const char *name ) +{ + xmlNode *i; + + for( i = base->children; i; i = i->next ) + if( strcmp( (char *) i->name, name ) == 0 ) + return( i ); + + return( NULL ); +} + +/* Read a string property to a buffer. TRUE for success. + */ +static int +get_sprop( xmlNode *xnode, const char *name, char *buf, int sz ) +{ + char *value = (char *) xmlGetProp( xnode, (xmlChar *) name ); + + if( !value ) + return( 0 ); + + im_strncpy( buf, value, sz ); + IM_FREEF( xmlFree, value ); + + return( 1 ); +} + +/* Chop history into lines, add each one as a refstring. + */ +static void +set_history( IMAGE *im, char *history ) +{ + GSList *history_list; + char *p, *q; + + /* There can be history there already if we're rewinding. + */ + IM_FREEF( im__gslist_gvalue_free, im->history_list ); + + history_list = NULL; + + for( p = history; *p; p = q ) { + if( (q = strchr( p, '\n' )) ) + *q = '\0'; + else + q = p + strlen( p ); + + history_list = g_slist_prepend( history_list, + im__gvalue_ref_string_new( p ) ); + } + + im->history_list = g_slist_reverse( history_list ); +} + +/* Load header fields. + */ +static int +rebuild_header_builtin( IMAGE *im, xmlNode *i ) +{ + char name[256]; + + if( get_sprop( i, "name", name, 256 ) ) { + if( strcmp( name, "Hist" ) == 0 ) { + char *history; + + /* Have to take (another) copy, since we need to free + * with xmlFree(). + */ + history = (char *) xmlNodeGetContent( i ); + set_history( im, history ); + xmlFree( history ); + } + } + + return( 0 ); +} + +/* Load meta fields. + */ +static int +rebuild_header_meta( IMAGE *im, xmlNode *i ) +{ + char name[256]; + char type[256]; + + if( get_sprop( i, "name", name, 256 ) && + get_sprop( i, "type", type, 256 ) ) { + GType gtype = g_type_from_name( type ); + + /* Can we convert from IM_SAVE_STRING to type? + */ + if( gtype && + g_value_type_transformable( + IM_TYPE_SAVE_STRING, gtype ) ) { + char *content; + GValue save_value = { 0 }; + GValue value = { 0 }; + + content = (char *) xmlNodeGetContent( i ); + g_value_init( &save_value, IM_TYPE_SAVE_STRING ); + im_save_string_set( &save_value, content ); + xmlFree( content ); + + g_value_init( &value, gtype ); + if( !g_value_transform( &save_value, &value ) ) { + g_value_unset( &save_value ); + im_error( "im__readhist", _( "error " + "transforming from save format" ) ); + return( -1 ); + } + if( im_meta_set( im, name, &value ) ) { + g_value_unset( &save_value ); + g_value_unset( &value ); + return( -1 ); + } + g_value_unset( &save_value ); + g_value_unset( &value ); + } + } + + return( 0 ); +} + +static xmlDoc * +get_xml( IMAGE *im ) +{ + if( im_header_get_type( im, IM_META_XML ) ) { + xmlDoc *doc; + + if( im_meta_get_area( im, IM_META_XML, (void *) &doc ) ) + return( NULL ); + + return( doc ); + } + + return( NULL ); +} + +/* Rebuild header fields that depend on stuff saved in xml. + */ +static int +rebuild_header( IMAGE *im ) +{ + xmlDoc *doc; + + if( (doc = get_xml( im )) ) { + xmlNode *root; + xmlNode *block; + + if( !(root = xmlDocGetRootElement( doc )) ) + return( -1 ); + if( (block = get_node( root, "header" )) ) { + xmlNode *i; + + for( i = block->children; i; i = i->next ) + if( strcmp( (char *) i->name, "field" ) == 0 ) + if( rebuild_header_builtin( im, i ) ) + return( -1 ); + } + if( (block = get_node( root, "meta" )) ) { + xmlNode *i; + + for( i = block->children; i; i = i->next ) + if( strcmp( (char *) i->name, "field" ) == 0 ) + if( rebuild_header_meta( im, i ) ) + return( -1 ); + } + } + + return( 0 ); +} + +/* Called at the end of im__read_header ... get any XML after the pixel data + * and read it in. + */ +static int +im__readhist( IMAGE *im ) +{ + /* Junk any old xml meta. + */ + if( im_header_get_type( im, IM_META_XML ) ) + im_meta_set_area( im, IM_META_XML, NULL, NULL ); + + if( im__has_extension_block( im ) ) { + xmlDoc *doc; + + if( !(doc = read_xml( im )) ) + return( -1 ); + if( im_meta_set_area( im, IM_META_XML, + (im_callback_fn) xmlFreeDoc, doc ) ) { + xmlFreeDoc( doc ); + return( -1 ); + } + } + + if( rebuild_header( im ) ) + return( -1 ); + + return( 0 ); +} + +/* Open the filename, read the header, some sanity checking. + */ +static int +im__read_header( IMAGE *image ) +{ + /* We don't use im->sizeof_header here, but we know we're reading a + * VIPS image anyway. + */ + unsigned char header[IM_SIZEOF_HEADER]; + + gint64 length; + gint64 psize; + + image->dtype = IM_OPENIN; + if( (image->fd = im__open_image_file( image->filename )) == -1 ) + return( -1 ); + if( read( image->fd, header, IM_SIZEOF_HEADER ) != IM_SIZEOF_HEADER || + im__read_header_bytes( image, header ) ) { + im_error( "im_openin", + _( "unable to read header for \"%s\", %s" ), + image->filename, strerror( errno ) ); + return( -1 ); + } + + /* Predict and check the file size. + */ + psize = im__image_pixel_length( image ); + if( (length = im_file_length( image->fd )) == -1 ) + return( -1 ); + if( psize > length ) { + im_error( "im_openin", _( "unable to open \"%s\", %s" ), + image->filename, _( "file has been truncated" ) ); + return( -1 ); + } + + /* Set demand style. Allow the most permissive sort. + */ + image->dhint = IM_THINSTRIP; + + /* Set the history part of im descriptor. Don't return an error if this + * fails (due to eg. corrupted XML) because it's probably mostly + * harmless. + */ + if( im__readhist( image ) ) { + im_warn( "im_openin", _( "error reading XML: %s" ), + im_error_buffer() ); + im_error_clear(); + } + + return( 0 ); +} + +/* Open, then mmap() small images, leave large images to have a rolling mmap() + * window for each region. + */ +static int +im_openin( IMAGE *image ) +{ + gint64 size; + +#ifdef DEBUG + char *str; + + if( (str = g_getenv( "IM_MMAP_LIMIT" )) ) { + im__mmap_limit = atoi( str ); + printf( "im_openin: setting maplimit to %d from environment\n", + im__mmap_limit ); + } +#endif /*DEBUG*/ + + if( im__read_header( image ) ) + return( -1 ); + + size = (gint64) IM_IMAGE_SIZEOF_LINE( image ) * image->Ysize + + image->sizeof_header; + if( size < im__mmap_limit ) { + if( im_mapfile( image ) ) + return( -1 ); + image->data = image->baseaddr + image->sizeof_header; + image->dtype = IM_MMAPIN; + +#ifdef DEBUG + printf( "im_openin: completely mmap()ing \"%s\": it's small\n", + image->filename ); +#endif /*DEBUG*/ + } + else { +#ifdef DEBUG + printf( "im_openin: delaying mmap() of \"%s\": it's big!\n", + image->filename ); +#endif /*DEBUG*/ + } + + return( 0 ); +} + +/* Open, then mmap() read/write. + */ +static int +im_openinrw( IMAGE *image ) +{ + if( im__read_header( image ) ) + return( -1 ); + if( im_mapfilerw( image ) ) + return( -1 ); + image->data = image->baseaddr + image->sizeof_header; + image->dtype = IM_MMAPINRW; + +#ifdef DEBUG + printf( "im_openin: completely mmap()ing \"%s\" read-write\n", + image->filename ); +#endif /*DEBUG*/ + + return( 0 ); +} + +/* Open a VIPS image for reading and byte-swap the image data if necessary. A + * ":w" at the end of the filename means we open read-write. + */ +IMAGE * +im_open_vips( const char *filename ) +{ + char name[FILENAME_MAX]; + char mode[FILENAME_MAX]; + IMAGE *im; + + im_filename_split( filename, name, mode ); + + if( !(im = im_init( name )) ) + return( NULL ); + if( mode[0] == 'w' ) { + if( im_openinrw( im ) ) { + im_close( im ); + return( NULL ); + } + if( im->Bbits != IM_BBITS_BYTE && + im_isMSBfirst( im ) != im_amiMSBfirst() ) { + im_close( im ); + im_error( "im_open_vips", _( "open for read-write for " + "native format images only" ) ); + return( NULL ); + } + } + else { + if( im_openin( im ) ) { + im_close( im ); + return( NULL ); + } + } + + /* Not in native format? + */ + if( im_isMSBfirst( im ) != im_amiMSBfirst() ) { + /* Does it need swapping? + */ + switch( im->Coding ) { + case IM_CODING_LABQ: + break; + + case IM_CODING_NONE: + if( im->BandFmt != IM_BANDFMT_CHAR && + im->BandFmt != IM_BANDFMT_UCHAR ) { + IMAGE *im2; + + /* Needs swapping :( make a little pipeline up + * to do this for us. + */ + if( !(im2 = im_open( filename, "p" )) ) { + im_close( im ); + return( NULL ); + } + if( im_add_close_callback( im2, + (im_callback_fn)im_close, im, NULL ) ) { + im_close( im ); + im_close( im2 ); + return( NULL ); + } + if( im_copy_swap( im, im2 ) ) { + im_close( im2 ); + return( NULL ); + } + im = im2; + } + break; + + default: + im_close( im ); + im_error( "im_open", _( "unknown coding type" ) ); + return( NULL ); + } + } + + return( im ); +} +