fix up byteswapping path (again)

This commit is contained in:
John Cupitt 2011-12-01 18:03:41 +00:00
parent 48763493e5
commit 261af58e00
6 changed files with 55 additions and 45 deletions

10
TODO
View File

@ -1,3 +1,7 @@
- get copy swap to use the glib byte order macros as well, faster?
- test vips_foreign_load_vips_get_flags(), sense inverted? - test vips_foreign_load_vips_get_flags(), sense inverted?
test load/save jpeg buffer test load/save jpeg buffer
@ -347,12 +351,6 @@ g++ -shared -nostdlib /usr/lib/gcc/x86_64-linux-gnu/4.4.3/../../../../lib/crti.o
- look into G_GNUC_DEPRECATED for back compat in vips8 - look into G_GNUC_DEPRECATED for back compat in vips8
- use
http://library.gnome.org/devel/glib/stable/glib-Byte-Order-Macros.html
for swapping ... they are asm macros so we should see a speedup
- can we use conv_sep to speed up the memuse benchmarks? - can we use conv_sep to speed up the memuse benchmarks?
- check mosaic1, global_balance, similarity etc. use of im__affine - check mosaic1, global_balance, similarity etc. use of im__affine

View File

@ -332,14 +332,14 @@ read_header( const char *header )
case SHORT: case SHORT:
p = &G_STRUCT_MEMBER( unsigned char, d, p = &G_STRUCT_MEMBER( unsigned char, d,
dsr_header[i].offset ); dsr_header[i].offset );
im__read_2byte( 1, p, &p ); vips__copy_2byte( TRUE, p, p );
break; break;
case INT: case INT:
case FLOAT: case FLOAT:
p = &G_STRUCT_MEMBER( unsigned char, d, p = &G_STRUCT_MEMBER( unsigned char, d,
dsr_header[i].offset ); dsr_header[i].offset );
im__read_4byte( 1, p, &p ); vips__copy_4byte( TRUE, p, p );
break; break;
case BYTE: case BYTE:

View File

@ -103,10 +103,9 @@ int vips_remapfilerw( VipsImage * );
void vips__buffer_init( void ); void vips__buffer_init( void );
void vips__read_4byte( int msb_first, unsigned char *to, unsigned char **from ); void vips__copy_4byte( int swap, unsigned char *to, unsigned char *from );
void vips__read_2byte( int msb_first, unsigned char *to, unsigned char **from ); void vips__copy_2byte( gboolean swap, unsigned char *to, unsigned char *from );
void vips__write_4byte( unsigned char **to, unsigned char *from );
void vips__write_2byte( unsigned char **to, unsigned char *from );
guint32 vips__file_magic( const char *filename ); guint32 vips__file_magic( const char *filename );
int vips__has_extension_block( VipsImage *im ); int vips__has_extension_block( VipsImage *im );
void *vips__read_extension_block( VipsImage *im, int *size ); void *vips__read_extension_block( VipsImage *im, int *size );

View File

@ -450,10 +450,6 @@ size_t im_ref_string_get_length( const GValue *value );
#define im__open_image_read vips__open_image_read #define im__open_image_read vips__open_image_read
#define im_image_open_input vips_image_open_input #define im_image_open_input vips_image_open_input
#define im_image_open_output vips_image_open_output #define im_image_open_output vips_image_open_output
#define im__read_4byte vips__read_4byte
#define im__read_2byte vips__read_2byte
#define im__write_4byte vips__write_4byte
#define im__write_2byte vips__write_2byte
#define im__has_extension_block vips__has_extension_block #define im__has_extension_block vips__has_extension_block
#define im__read_extension_block vips__read_extension_block #define im__read_extension_block vips__read_extension_block
#define im__write_extension_block vips__write_extension_block #define im__write_extension_block vips__write_extension_block

View File

@ -852,7 +852,7 @@ vips__file_write( void *data, size_t size, size_t nmemb, FILE *stream )
if( (n = fwrite( data, size, nmemb, stream )) != nmemb ) { if( (n = fwrite( data, size, nmemb, stream )) != nmemb ) {
vips_error( "vips__file_write", vips_error( "vips__file_write",
_( "writing error (%zd out of %zd blocks written) " _( "write error (%zd out of %zd blocks written) "
"... disc full?" ), n, nmemb ); "... disc full?" ), n, nmemb );
return( -1 ); return( -1 );
} }

View File

@ -47,6 +47,7 @@
/* /*
#define DEBUG #define DEBUG
#define SHOW_HEADER
*/ */
#ifdef HAVE_CONFIG_H #ifdef HAVE_CONFIG_H
@ -170,32 +171,30 @@ image_pixel_length( VipsImage *image )
return( psize + image->sizeof_header ); return( psize + image->sizeof_header );
} }
/* Copy 2 and 4 bytes, swapping to and from native. /* Copy 2 and 4 bytes, optionally swapping byte order.
*/ */
void void
vips__copy_4byte( int msb_first, unsigned char *to, unsigned char *from ) vips__copy_4byte( int swap, unsigned char *to, unsigned char *from )
{ {
guint32 out; guint32 *in = (guint32 *) from;
guint32 *out = (guint32 *) to;
if( msb_first ) if( swap )
out = from[0] << 24 | from[1] << 16 | from[2] << 8 | from[3]; *out = GUINT32_SWAP_LE_BE( *in );
else else
out = from[3] << 24 | from[2] << 16 | from[1] << 8 | from[0]; *out = *in;
*((guint32 *) to) = out;
} }
void void
vips__copy_2byte( int msb_first, unsigned char *to, unsigned char *from ) vips__copy_2byte( gboolean swap, unsigned char *to, unsigned char *from )
{ {
guint16 out; guint16 *in = (guint16 *) from;
guint16 *out = (guint16 *) to;
if( msb_first ) if( swap )
out = from[0] << 8 | from[1]; *out = GUINT16_SWAP_LE_BE( *in );
else else
out = from[1] << 8 | from[0]; *out = *in;
*((guint16 *) to) = out;
} }
guint32 guint32
@ -216,7 +215,7 @@ vips__file_magic( const char *filename )
typedef struct _FieldIO { typedef struct _FieldIO {
glong offset; glong offset;
int size; int size;
void (*copy)( int msb_first, unsigned char *to, unsigned char *from ); void (*copy)( gboolean swap, unsigned char *to, unsigned char *from );
} FieldIO; } FieldIO;
static FieldIO fields[] = { static FieldIO fields[] = {
@ -239,10 +238,19 @@ static FieldIO fields[] = {
int int
vips__read_header_bytes( VipsImage *im, unsigned char *from ) vips__read_header_bytes( VipsImage *im, unsigned char *from )
{ {
int msb_first; gboolean swap;
int i; int i;
vips__copy_4byte( 1, (unsigned char *) &im->magic, from ); #ifdef SHOW_HEADER
printf( "vips__read_header_bytes: file bytes:\n" );
for( i = 0; i < im->sizeof_header; i++ )
printf( "%2d - 0x%02x\n", i, from[i] );
#endif /*SHOW_HEADER*/
/* The magic number is always written MSB first, we may need to swap.
*/
vips__copy_4byte( !vips_amiMSBfirst(),
(unsigned char *) &im->magic, from );
from += 4; from += 4;
if( im->magic != VIPS_MAGIC_INTEL && if( im->magic != VIPS_MAGIC_INTEL &&
im->magic != VIPS_MAGIC_SPARC ) { im->magic != VIPS_MAGIC_SPARC ) {
@ -251,10 +259,13 @@ vips__read_header_bytes( VipsImage *im, unsigned char *from )
return( -1 ); return( -1 );
} }
msb_first = im->magic == VIPS_MAGIC_SPARC; /* We need to swap for other fields if the file byte order is
* different from ours.
*/
swap = vips_amiMSBfirst() != (im->magic == VIPS_MAGIC_SPARC);
for( i = 0; i < VIPS_NUMBER( fields ); i++ ) { for( i = 0; i < VIPS_NUMBER( fields ); i++ ) {
fields[i].copy( msb_first, fields[i].copy( swap,
&G_STRUCT_MEMBER( unsigned char, im, fields[i].offset ), &G_STRUCT_MEMBER( unsigned char, im, fields[i].offset ),
from ); from );
from += fields[i].size; from += fields[i].size;
@ -270,22 +281,22 @@ vips__read_header_bytes( VipsImage *im, unsigned char *from )
int int
vips__write_header_bytes( VipsImage *im, unsigned char *to ) vips__write_header_bytes( VipsImage *im, unsigned char *to )
{ {
int msb_first; /* Swap if the byte order we are asked to write the header in is
* different from ours.
*/
gboolean swap = vips_amiMSBfirst() != (im->magic == VIPS_MAGIC_SPARC);
int i; int i;
unsigned char *q; unsigned char *q;
/* Always write the magic number MSB first. /* Always write the magic number MSB first.
*/ */
to[0] = im->magic >> 24; vips__copy_4byte( !vips_amiMSBfirst(),
to[1] = im->magic >> 16; to, (unsigned char *) &im->magic );
to[2] = im->magic >> 8;
to[3] = im->magic;
q = to + 4; q = to + 4;
msb_first = im->magic == VIPS_MAGIC_SPARC;
for( i = 0; i < VIPS_NUMBER( fields ); i++ ) { for( i = 0; i < VIPS_NUMBER( fields ); i++ ) {
fields[i].copy( msb_first, fields[i].copy( swap,
q, q,
&G_STRUCT_MEMBER( unsigned char, im, &G_STRUCT_MEMBER( unsigned char, im,
fields[i].offset ) ); fields[i].offset ) );
@ -297,6 +308,12 @@ vips__write_header_bytes( VipsImage *im, unsigned char *to )
while( q - to < im->sizeof_header ) while( q - to < im->sizeof_header )
*q++ = 0; *q++ = 0;
#ifdef SHOW_HEADER
printf( "vips__write_header_bytes: file bytes:\n" );
for( i = 0; i < im->sizeof_header; i++ )
printf( "%2d - 0x%02x\n", i, to[i] );
#endif /*SHOW_HEADER*/
return( 0 ); return( 0 );
} }