tiff reader allows many more formats
eg. 32-bit ints, signed and unsigned, complex, extra alpha, etc.
This commit is contained in:
parent
326365ab6c
commit
a11e5352f2
@ -18,6 +18,7 @@
|
|||||||
- added vips_webpload(), vips_webpload_buffer(), vips_webpsave(),
|
- added vips_webpload(), vips_webpload_buffer(), vips_webpsave(),
|
||||||
vips_webpsave_buffer(), vips_webpsave_mime()
|
vips_webpsave_buffer(), vips_webpsave_mime()
|
||||||
- tiff reader allows separate planes for strip read
|
- tiff reader allows separate planes for strip read
|
||||||
|
- tiff reader allows many more formats, eg. 32-bit int, complex, etc.
|
||||||
- added vips_image_new_matrixv()
|
- added vips_image_new_matrixv()
|
||||||
|
|
||||||
3/7/13 started 7.34.2
|
3/7/13 started 7.34.2
|
||||||
|
@ -140,6 +140,7 @@
|
|||||||
* 17/9/13
|
* 17/9/13
|
||||||
* - support separate planes for strip read
|
* - support separate planes for strip read
|
||||||
* - big cleanup
|
* - big cleanup
|
||||||
|
* - support for many more formats, eg. 32-bit int etc.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -170,8 +171,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#define DEBUG
|
|
||||||
*/
|
*/
|
||||||
|
#define DEBUG
|
||||||
|
|
||||||
#ifdef HAVE_CONFIG_H
|
#ifdef HAVE_CONFIG_H
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
@ -233,6 +234,7 @@ typedef struct _ReadTiff {
|
|||||||
int samples_per_pixel;
|
int samples_per_pixel;
|
||||||
int bits_per_sample;
|
int bits_per_sample;
|
||||||
int photometric_interpretation;
|
int photometric_interpretation;
|
||||||
|
int sample_format;
|
||||||
|
|
||||||
/* Turn on separate plane reading.
|
/* Turn on separate plane reading.
|
||||||
*/
|
*/
|
||||||
@ -326,24 +328,6 @@ tfget16( TIFF *tif, ttag_t tag, int *out )
|
|||||||
return( 1 );
|
return( 1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Test a uint16 field. Field must be defined and equal to the value.
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
tfequals( TIFF *tif, ttag_t tag, uint16 val )
|
|
||||||
{
|
|
||||||
int v;
|
|
||||||
|
|
||||||
if( !tfget16( tif, tag, &v ) )
|
|
||||||
return( 0 );
|
|
||||||
if( v != val ) {
|
|
||||||
vips_error( "tiff2vips",
|
|
||||||
_( "required field %d = %d, not %d" ), tag, v, val );
|
|
||||||
return( 0 );
|
|
||||||
}
|
|
||||||
|
|
||||||
return( 1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
check_samples( ReadTiff *rtiff, int samples_per_pixel )
|
check_samples( ReadTiff *rtiff, int samples_per_pixel )
|
||||||
{
|
{
|
||||||
@ -359,13 +343,25 @@ check_samples( ReadTiff *rtiff, int samples_per_pixel )
|
|||||||
/* Check n and n+1 so we can have an alpha.
|
/* Check n and n+1 so we can have an alpha.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
check_samples_alpha( ReadTiff *rtiff, int samples_per_pixel )
|
check_min_samples( ReadTiff *rtiff, int samples_per_pixel )
|
||||||
{
|
{
|
||||||
if( rtiff->samples_per_pixel != samples_per_pixel &&
|
if( rtiff->samples_per_pixel < samples_per_pixel ) {
|
||||||
rtiff->samples_per_pixel != samples_per_pixel + 1 ) {
|
|
||||||
vips_error( "tiff2vips",
|
vips_error( "tiff2vips",
|
||||||
_( "not %d or %d bands" ),
|
_( "not at least %d samples per pixel" ),
|
||||||
samples_per_pixel, samples_per_pixel + 1 );
|
samples_per_pixel );
|
||||||
|
return( -1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
return( 0 );
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
check_interpretation( ReadTiff *rtiff, int photometric_interpretation )
|
||||||
|
{
|
||||||
|
if( rtiff->photometric_interpretation != photometric_interpretation ) {
|
||||||
|
vips_error( "tiff2vips",
|
||||||
|
_( "not photometric interpretation %d" ),
|
||||||
|
photometric_interpretation );
|
||||||
return( -1 );
|
return( -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -400,20 +396,55 @@ check_bits_palette( ReadTiff *rtiff )
|
|||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static VipsBandFormat
|
||||||
check_float( ReadTiff *rtiff )
|
guess_format( ReadTiff *rtiff )
|
||||||
{
|
{
|
||||||
int format;
|
switch( rtiff->bits_per_sample ) {
|
||||||
|
case 1:
|
||||||
|
case 2:
|
||||||
|
case 4:
|
||||||
|
case 8:
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_INT )
|
||||||
|
return( VIPS_FORMAT_CHAR );
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_UINT )
|
||||||
|
return( VIPS_FORMAT_UCHAR );
|
||||||
|
break;
|
||||||
|
|
||||||
if( !tfget16( rtiff->tiff, TIFFTAG_SAMPLEFORMAT, &format ) )
|
case 16:
|
||||||
return( -1 );
|
if( rtiff->sample_format == SAMPLEFORMAT_INT )
|
||||||
if( format != SAMPLEFORMAT_IEEEFP ) {
|
return( VIPS_FORMAT_SHORT );
|
||||||
vips_error( "tiff2vips",
|
if( rtiff->sample_format == SAMPLEFORMAT_UINT )
|
||||||
"%s", _( "not a floating-point image" ) );
|
return( VIPS_FORMAT_USHORT );
|
||||||
return( -1 );
|
break;
|
||||||
|
|
||||||
|
case 32:
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_INT )
|
||||||
|
return( VIPS_FORMAT_INT );
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_UINT )
|
||||||
|
return( VIPS_FORMAT_UINT );
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_IEEEFP )
|
||||||
|
return( VIPS_FORMAT_FLOAT );
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 64:
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_IEEEFP )
|
||||||
|
return( VIPS_FORMAT_DOUBLE );
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_COMPLEXIEEEFP )
|
||||||
|
return( VIPS_FORMAT_COMPLEX );
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 128:
|
||||||
|
if( rtiff->sample_format == SAMPLEFORMAT_COMPLEXIEEEFP )
|
||||||
|
return( VIPS_FORMAT_DPCOMPLEX );
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return( 0 );
|
vips_error( "tiff2vips", "%s", _( "unsupported image format\n" ) );
|
||||||
|
|
||||||
|
return( VIPS_FORMAT_NOTSET );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Per-scanline process function for VIPS_CODING_LABQ.
|
/* Per-scanline process function for VIPS_CODING_LABQ.
|
||||||
@ -439,8 +470,9 @@ labpack_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *dummy )
|
|||||||
static int
|
static int
|
||||||
parse_labpack( ReadTiff *rtiff, VipsImage *out )
|
parse_labpack( ReadTiff *rtiff, VipsImage *out )
|
||||||
{
|
{
|
||||||
if( check_samples_alpha( rtiff, 3 ) ||
|
if( check_min_samples( rtiff, 3 ) ||
|
||||||
check_bits( rtiff, 8 ) )
|
check_bits( rtiff, 8 ) ||
|
||||||
|
check_interpretation( rtiff, PHOTOMETRIC_CIELAB ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
out->Bands = 4;
|
out->Bands = 4;
|
||||||
@ -461,13 +493,17 @@ labs_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *dummy )
|
|||||||
int x;
|
int x;
|
||||||
unsigned short *p1 = (unsigned short *) p;
|
unsigned short *p1 = (unsigned short *) p;
|
||||||
short *q1 = (short *) q;
|
short *q1 = (short *) q;
|
||||||
|
int i;
|
||||||
|
|
||||||
for( x = 0; x < n; x++ ) {
|
for( x = 0; x < n; x++ ) {
|
||||||
|
/* We use a signed int16 for L.
|
||||||
|
*/
|
||||||
q1[0] = p1[0] >> 1;
|
q1[0] = p1[0] >> 1;
|
||||||
q1[1] = p1[1];
|
|
||||||
q1[2] = p1[2];
|
|
||||||
|
|
||||||
q1 += 3;
|
for( i = 1; i < rtiff->samples_per_pixel; i++ )
|
||||||
|
q1[i] = p1[i];
|
||||||
|
|
||||||
|
q1 += rtiff->samples_per_pixel;
|
||||||
p1 += rtiff->samples_per_pixel;
|
p1 += rtiff->samples_per_pixel;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -477,11 +513,12 @@ labs_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *dummy )
|
|||||||
static int
|
static int
|
||||||
parse_labs( ReadTiff *rtiff, VipsImage *out )
|
parse_labs( ReadTiff *rtiff, VipsImage *out )
|
||||||
{
|
{
|
||||||
if( check_samples_alpha( rtiff, 3 ) ||
|
if( check_min_samples( rtiff, 3 ) ||
|
||||||
check_bits( rtiff, 16 ) )
|
check_bits( rtiff, 16 ) ||
|
||||||
|
check_interpretation( rtiff, PHOTOMETRIC_CIELAB ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
out->Bands = 3;
|
out->Bands = rtiff->samples_per_pixel;
|
||||||
out->BandFmt = VIPS_FORMAT_SHORT;
|
out->BandFmt = VIPS_FORMAT_SHORT;
|
||||||
out->Coding = VIPS_CODING_NONE;
|
out->Coding = VIPS_CODING_NONE;
|
||||||
out->Type = VIPS_INTERPRETATION_LABS;
|
out->Type = VIPS_INTERPRETATION_LABS;
|
||||||
@ -526,7 +563,7 @@ onebit_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *flg )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Read a 1-bit TIFF image. Pass in pixel values to use for black and white.
|
/* Read a 1-bit TIFF image.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
parse_onebit( ReadTiff *rtiff, VipsImage *out )
|
parse_onebit( ReadTiff *rtiff, VipsImage *out )
|
||||||
@ -545,129 +582,98 @@ parse_onebit( ReadTiff *rtiff, VipsImage *out )
|
|||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Per-scanline process function for 8-bit greyscale images.
|
/* Swap the sense of the first channel, if necessary.
|
||||||
|
*/
|
||||||
|
#define GREY_LOOP( TYPE, MAX ) { \
|
||||||
|
TYPE *p1; \
|
||||||
|
TYPE *q1; \
|
||||||
|
\
|
||||||
|
p1 = (TYPE *) p; \
|
||||||
|
q1 = (TYPE *) q; \
|
||||||
|
for( x = 0; x < n; x++ ) { \
|
||||||
|
if( invert ) \
|
||||||
|
q1[0] = MAX - p1[0]; \
|
||||||
|
else \
|
||||||
|
q1[0] = p1[0]; \
|
||||||
|
\
|
||||||
|
for( i = 1; i < rtiff->samples_per_pixel; i++ ) \
|
||||||
|
q1[i] = p1[i]; \
|
||||||
|
\
|
||||||
|
q1 += rtiff->samples_per_pixel; \
|
||||||
|
p1 += rtiff->samples_per_pixel; \
|
||||||
|
} \
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Per-scanline process function for greyscale images.
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
greyscale8_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *client )
|
greyscale_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *client )
|
||||||
{
|
{
|
||||||
int mask =
|
gboolean invert =
|
||||||
rtiff->photometric_interpretation == PHOTOMETRIC_MINISBLACK ?
|
rtiff->photometric_interpretation == PHOTOMETRIC_MINISWHITE;
|
||||||
0 : -1;
|
VipsBandFormat format = guess_format( rtiff );
|
||||||
|
|
||||||
int x;
|
int x, i;
|
||||||
|
|
||||||
/* Read bytes, swapping sense if necessary.
|
switch( format ) {
|
||||||
*/
|
case VIPS_FORMAT_UCHAR:
|
||||||
for( x = 0; x < n; x++ ) {
|
case VIPS_FORMAT_CHAR:
|
||||||
q[0] = p[0] ^ mask;
|
GREY_LOOP( guchar, UCHAR_MAX );
|
||||||
|
break;
|
||||||
|
|
||||||
/* Process alpha, if any. Don't swap this.
|
case VIPS_FORMAT_SHORT:
|
||||||
*/
|
GREY_LOOP( gshort, SHRT_MAX );
|
||||||
if( rtiff->samples_per_pixel == 2 )
|
break;
|
||||||
q[1] = p[1];
|
|
||||||
|
|
||||||
q += rtiff->samples_per_pixel;
|
case VIPS_FORMAT_USHORT:
|
||||||
p += rtiff->samples_per_pixel;
|
GREY_LOOP( gushort, USHRT_MAX );
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VIPS_FORMAT_INT:
|
||||||
|
GREY_LOOP( gint, INT_MAX );
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VIPS_FORMAT_UINT:
|
||||||
|
GREY_LOOP( guint, UINT_MAX );
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VIPS_FORMAT_FLOAT:
|
||||||
|
GREY_LOOP( float, 1.0 );
|
||||||
|
break;
|
||||||
|
|
||||||
|
case VIPS_FORMAT_DOUBLE:
|
||||||
|
GREY_LOOP( double, 1.0 );
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
g_assert( 0 );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Read a 8-bit grey-scale TIFF image.
|
/* Read a grey-scale TIFF image. We have to invert the first band if
|
||||||
|
* PHOTOMETRIC_MINISBLACK is set.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
parse_greyscale8( ReadTiff *rtiff, VipsImage *out )
|
parse_greyscale( ReadTiff *rtiff, VipsImage *out )
|
||||||
{
|
{
|
||||||
if( check_samples_alpha( rtiff, 1 ) ||
|
if( check_min_samples( rtiff, 1 ) )
|
||||||
check_bits( rtiff, 8 ) )
|
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
out->Bands = rtiff->samples_per_pixel;
|
out->Bands = rtiff->samples_per_pixel;
|
||||||
out->BandFmt = VIPS_FORMAT_UCHAR;
|
out->BandFmt = guess_format( rtiff );
|
||||||
out->Coding = VIPS_CODING_NONE;
|
out->Coding = VIPS_CODING_NONE;
|
||||||
out->Type = VIPS_INTERPRETATION_B_W;
|
|
||||||
|
|
||||||
rtiff->sfn = greyscale8_line;
|
if( rtiff->bits_per_sample == 16 )
|
||||||
|
out->Type = VIPS_INTERPRETATION_GREY16;
|
||||||
|
else
|
||||||
|
out->Type = VIPS_INTERPRETATION_B_W;
|
||||||
|
|
||||||
return( 0 );
|
/* greyscale_line() doesn't do complex.
|
||||||
}
|
|
||||||
|
|
||||||
/* Per-scanline process function for 16-bit greyscale images.
|
|
||||||
*/
|
|
||||||
static void
|
|
||||||
greyscale16_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *client )
|
|
||||||
{
|
|
||||||
int mask =
|
|
||||||
rtiff->photometric_interpretation == PHOTOMETRIC_MINISBLACK ?
|
|
||||||
0 : -1;
|
|
||||||
|
|
||||||
unsigned short *p1;
|
|
||||||
unsigned short *q1;
|
|
||||||
int x;
|
|
||||||
|
|
||||||
/* Read bytes, swapping sense if necessary.
|
|
||||||
*/
|
*/
|
||||||
p1 = (unsigned short *) p;
|
if( vips_check_noncomplex( "tiff2vips", out ) )
|
||||||
q1 = (unsigned short *) q;
|
return( -1 );
|
||||||
for( x = 0; x < n; x++ ) {
|
|
||||||
q1[0] = p1[0] ^ mask;
|
|
||||||
|
|
||||||
/* Process alpha, if any. Don't swap this.
|
rtiff->sfn = greyscale_line;
|
||||||
*/
|
|
||||||
if( rtiff->samples_per_pixel == 2 )
|
|
||||||
q1[1] = p1[1];
|
|
||||||
|
|
||||||
q1 += rtiff->samples_per_pixel;
|
|
||||||
p1 += rtiff->samples_per_pixel;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Read a 16-bit grey-scale TIFF image.
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
parse_greyscale16( ReadTiff *rtiff, VipsImage *out )
|
|
||||||
{
|
|
||||||
if( check_samples_alpha( rtiff, 1 ) ||
|
|
||||||
check_bits( rtiff, 16 ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
out->Bands = rtiff->samples_per_pixel;
|
|
||||||
out->BandFmt = VIPS_FORMAT_USHORT;
|
|
||||||
out->Coding = VIPS_CODING_NONE;
|
|
||||||
out->Type = VIPS_INTERPRETATION_GREY16;
|
|
||||||
|
|
||||||
rtiff->sfn = greyscale16_line;
|
|
||||||
|
|
||||||
return( 0 );
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Per-scanline process function when we just need to copy.
|
|
||||||
*/
|
|
||||||
static void
|
|
||||||
memcpy_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *client )
|
|
||||||
{
|
|
||||||
VipsImage *im = (VipsImage *) client;
|
|
||||||
size_t len = n * VIPS_IMAGE_SIZEOF_PEL( im );
|
|
||||||
|
|
||||||
memcpy( q, p, len );
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Read a 32-bit floating point greyscale TIFF image. What do we do about
|
|
||||||
* MINISWHITE/MINISBLACK (pm)? Not sure ... just ignore it.
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
parse_greyscale32f( ReadTiff *rtiff, VipsImage *out )
|
|
||||||
{
|
|
||||||
if( check_samples_alpha( rtiff, 1 ) ||
|
|
||||||
check_bits( rtiff, 32 ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
out->Bands = rtiff->samples_per_pixel;
|
|
||||||
out->BandFmt = VIPS_FORMAT_FLOAT;
|
|
||||||
out->Coding = VIPS_CODING_NONE;
|
|
||||||
out->Type = VIPS_INTERPRETATION_B_W;
|
|
||||||
|
|
||||||
rtiff->sfn = memcpy_line;
|
|
||||||
rtiff->client = out;
|
|
||||||
rtiff->memcpy = TRUE;
|
|
||||||
|
|
||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
@ -787,96 +793,43 @@ parse_palette( ReadTiff *rtiff, VipsImage *out )
|
|||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Read an 8-bit RGB/RGBA image.
|
/* Per-scanline process function when we just need to copy.
|
||||||
*/
|
*/
|
||||||
static int
|
static void
|
||||||
parse_rgb8( ReadTiff *rtiff, VipsImage *out )
|
memcpy_line( ReadTiff *rtiff, VipsPel *q, VipsPel *p, int n, void *client )
|
||||||
{
|
{
|
||||||
if( check_samples_alpha( rtiff, 3 ) ||
|
VipsImage *im = (VipsImage *) client;
|
||||||
check_bits( rtiff, 8 ) )
|
size_t len = n * VIPS_IMAGE_SIZEOF_PEL( im );
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
out->Bands = rtiff->samples_per_pixel;
|
memcpy( q, p, len );
|
||||||
out->BandFmt = VIPS_FORMAT_UCHAR;
|
|
||||||
out->Coding = VIPS_CODING_NONE;
|
|
||||||
out->Type = VIPS_INTERPRETATION_sRGB;
|
|
||||||
|
|
||||||
rtiff->sfn = memcpy_line;
|
|
||||||
rtiff->client = out;
|
|
||||||
rtiff->memcpy = TRUE;
|
|
||||||
|
|
||||||
return( 0 );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Read a 16-bit RGB/RGBA image.
|
/* Read a regular multiband image where we can just copy pixels from the tiff
|
||||||
|
* buffer.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
parse_rgb16( ReadTiff *rtiff, VipsImage *out )
|
parse_copy( ReadTiff *rtiff, VipsImage *out )
|
||||||
{
|
{
|
||||||
if( check_samples_alpha( rtiff, 3 ) ||
|
|
||||||
check_bits( rtiff, 16 ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
out->Bands = rtiff->samples_per_pixel;
|
out->Bands = rtiff->samples_per_pixel;
|
||||||
out->BandFmt = VIPS_FORMAT_USHORT;
|
out->BandFmt = guess_format( rtiff );
|
||||||
out->Coding = VIPS_CODING_NONE;
|
|
||||||
out->Type = VIPS_INTERPRETATION_RGB16;
|
|
||||||
|
|
||||||
rtiff->sfn = memcpy_line;
|
|
||||||
rtiff->client = out;
|
|
||||||
rtiff->memcpy = TRUE;
|
|
||||||
|
|
||||||
return( 0 );
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Read a 32-bit float image. RGB or LAB, with or without alpha.
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
parse_32f( ReadTiff *rtiff, VipsImage *out )
|
|
||||||
{
|
|
||||||
if( check_samples_alpha( rtiff, 3 ) ||
|
|
||||||
check_bits( rtiff, 32 ) ||
|
|
||||||
check_float( rtiff ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
out->Bands = rtiff->samples_per_pixel;
|
|
||||||
out->BandFmt = VIPS_FORMAT_FLOAT;
|
|
||||||
out->Coding = VIPS_CODING_NONE;
|
out->Coding = VIPS_CODING_NONE;
|
||||||
|
|
||||||
switch( rtiff->photometric_interpretation ) {
|
if( rtiff->samples_per_pixel >= 3 &&
|
||||||
case PHOTOMETRIC_CIELAB:
|
(rtiff->photometric_interpretation == PHOTOMETRIC_RGB ||
|
||||||
out->Type = VIPS_INTERPRETATION_LAB;
|
rtiff->photometric_interpretation == PHOTOMETRIC_YCBCR) ) {
|
||||||
break;
|
if( rtiff->bits_per_sample == 16 )
|
||||||
|
out->Type = VIPS_INTERPRETATION_RGB16;
|
||||||
case PHOTOMETRIC_RGB:
|
else
|
||||||
out->Type = VIPS_INTERPRETATION_sRGB;
|
out->Type = VIPS_INTERPRETATION_sRGB;
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
g_assert( 0 );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
rtiff->sfn = memcpy_line;
|
if( rtiff->samples_per_pixel >= 3 &&
|
||||||
rtiff->client = out;
|
rtiff->photometric_interpretation == PHOTOMETRIC_CIELAB )
|
||||||
rtiff->memcpy = TRUE;
|
out->Type = VIPS_INTERPRETATION_LAB;
|
||||||
|
|
||||||
return( 0 );
|
if( rtiff->samples_per_pixel >= 4 &&
|
||||||
}
|
rtiff->photometric_interpretation == PHOTOMETRIC_SEPARATED )
|
||||||
|
out->Type = VIPS_INTERPRETATION_CMYK;
|
||||||
/* Read a CMYK image.
|
|
||||||
*/
|
|
||||||
static int
|
|
||||||
parse_cmyk( ReadTiff *rtiff, VipsImage *out )
|
|
||||||
{
|
|
||||||
if( check_samples_alpha( rtiff, 4 ) ||
|
|
||||||
check_bits( rtiff, 8 ) ||
|
|
||||||
!tfequals( rtiff->tiff, TIFFTAG_INKSET, INKSET_CMYK ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
out->Bands = rtiff->samples_per_pixel;
|
|
||||||
out->BandFmt = VIPS_FORMAT_UCHAR;
|
|
||||||
out->Coding = VIPS_CODING_NONE;
|
|
||||||
out->Type = VIPS_INTERPRETATION_CMYK;
|
|
||||||
|
|
||||||
rtiff->sfn = memcpy_line;
|
rtiff->sfn = memcpy_line;
|
||||||
rtiff->client = out;
|
rtiff->client = out;
|
||||||
@ -938,13 +891,49 @@ parse_resolution( TIFF *tiff, VipsImage *out )
|
|||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef int (*reader_fn)( ReadTiff *rtiff, VipsImage *out );
|
||||||
|
|
||||||
|
/* We have a range of output paths. Look at the tiff header and try to
|
||||||
|
* route the input image to the best output path.
|
||||||
|
*/
|
||||||
|
static reader_fn
|
||||||
|
pick_reader( ReadTiff *rtiff )
|
||||||
|
{
|
||||||
|
if( rtiff->photometric_interpretation == PHOTOMETRIC_CIELAB ) {
|
||||||
|
if( rtiff->bits_per_sample == 8 )
|
||||||
|
return( parse_labpack );
|
||||||
|
if( rtiff->bits_per_sample == 16 )
|
||||||
|
return( parse_labs );
|
||||||
|
}
|
||||||
|
|
||||||
|
if( rtiff->photometric_interpretation == PHOTOMETRIC_MINISWHITE ||
|
||||||
|
rtiff->photometric_interpretation == PHOTOMETRIC_MINISBLACK ) {
|
||||||
|
if( rtiff->bits_per_sample == 1 )
|
||||||
|
return( parse_onebit );
|
||||||
|
else
|
||||||
|
return( parse_greyscale );
|
||||||
|
}
|
||||||
|
|
||||||
|
if( rtiff->photometric_interpretation == PHOTOMETRIC_PALETTE )
|
||||||
|
return( parse_palette );
|
||||||
|
|
||||||
|
if( rtiff->photometric_interpretation == PHOTOMETRIC_YCBCR ) {
|
||||||
|
/* Sometimes JPEG in TIFF images are tagged as YCBCR. Ask
|
||||||
|
* libtiff to convert to RGB for us.
|
||||||
|
*/
|
||||||
|
TIFFSetField( rtiff->tiff,
|
||||||
|
TIFFTAG_JPEGCOLORMODE, JPEGCOLORMODE_RGB );
|
||||||
|
}
|
||||||
|
|
||||||
|
return( parse_copy );
|
||||||
|
}
|
||||||
|
|
||||||
/* Look at PhotometricInterpretation and BitsPerPixel and try to figure out
|
/* Look at PhotometricInterpretation and BitsPerPixel and try to figure out
|
||||||
* which of the image classes this is.
|
* which of the image classes this is.
|
||||||
*/
|
*/
|
||||||
static int
|
static int
|
||||||
parse_header( ReadTiff *rtiff, VipsImage *out )
|
parse_header( ReadTiff *rtiff, VipsImage *out )
|
||||||
{
|
{
|
||||||
int format;
|
|
||||||
uint32 data_length;
|
uint32 data_length;
|
||||||
uint32 width, height;
|
uint32 width, height;
|
||||||
void *data;
|
void *data;
|
||||||
@ -970,6 +959,15 @@ parse_header( ReadTiff *rtiff, VipsImage *out )
|
|||||||
&rtiff->photometric_interpretation ) )
|
&rtiff->photometric_interpretation ) )
|
||||||
return( -1 );
|
return( -1 );
|
||||||
|
|
||||||
|
/* Some optional fields.
|
||||||
|
*/
|
||||||
|
{
|
||||||
|
uint16 v;
|
||||||
|
|
||||||
|
TIFFGetFieldDefaulted( rtiff->tiff, TIFFTAG_SAMPLEFORMAT, &v );
|
||||||
|
rtiff->sample_format = v;
|
||||||
|
}
|
||||||
|
|
||||||
/* Arbitrary sanity-checking limits.
|
/* Arbitrary sanity-checking limits.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@ -999,140 +997,15 @@ parse_header( ReadTiff *rtiff, VipsImage *out )
|
|||||||
rtiff->samples_per_pixel );
|
rtiff->samples_per_pixel );
|
||||||
printf( "parse_header: bits_per_sample = %d\n",
|
printf( "parse_header: bits_per_sample = %d\n",
|
||||||
rtiff->bits_per_sample );
|
rtiff->bits_per_sample );
|
||||||
|
printf( "parse_header: sample_format = %d\n",
|
||||||
|
rtiff->sample_format );
|
||||||
#endif /*DEBUG*/
|
#endif /*DEBUG*/
|
||||||
|
|
||||||
switch( rtiff->photometric_interpretation ) {
|
/* We have a range of output paths. Look at the tiff header and try to
|
||||||
case PHOTOMETRIC_CIELAB:
|
* route the input image to the best output path.
|
||||||
switch( rtiff->bits_per_sample ) {
|
*/
|
||||||
case 8:
|
if( pick_reader( rtiff )( rtiff, out ) )
|
||||||
if( parse_labpack( rtiff, out ) )
|
return( -1 );
|
||||||
return( -1 );
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 16:
|
|
||||||
if( parse_labs( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 32:
|
|
||||||
if( parse_32f( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
vips_error( "tiff2vips",
|
|
||||||
_( "unsupported depth %d for LAB image" ),
|
|
||||||
rtiff->bits_per_sample );
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PHOTOMETRIC_MINISWHITE:
|
|
||||||
case PHOTOMETRIC_MINISBLACK:
|
|
||||||
switch( rtiff->bits_per_sample ) {
|
|
||||||
case 1:
|
|
||||||
if( parse_onebit( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 8:
|
|
||||||
if( parse_greyscale8( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 16:
|
|
||||||
if( parse_greyscale16( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 32:
|
|
||||||
if( !tfget16( rtiff->tiff,
|
|
||||||
TIFFTAG_SAMPLEFORMAT, &format ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
if( format == SAMPLEFORMAT_IEEEFP ) {
|
|
||||||
if( parse_greyscale32f( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
vips_error( "tiff2vips",
|
|
||||||
_( "unsupported sample format "
|
|
||||||
"%d for greyscale image" ),
|
|
||||||
format );
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
vips_error( "tiff2vips",
|
|
||||||
_( "unsupported depth %d for greyscale image" ),
|
|
||||||
rtiff->bits_per_sample );
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PHOTOMETRIC_PALETTE:
|
|
||||||
/* Full colour pallette.
|
|
||||||
*/
|
|
||||||
if( parse_palette( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PHOTOMETRIC_YCBCR:
|
|
||||||
/* Sometimes JPEG in TIFF images are tagged as YCBCR. Ask
|
|
||||||
* libtiff to convert to RGB for us.
|
|
||||||
*/
|
|
||||||
TIFFSetField( rtiff->tiff,
|
|
||||||
TIFFTAG_JPEGCOLORMODE, JPEGCOLORMODE_RGB );
|
|
||||||
if( parse_rgb8( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PHOTOMETRIC_RGB:
|
|
||||||
switch( rtiff->bits_per_sample ) {
|
|
||||||
case 8:
|
|
||||||
if( parse_rgb8( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 16:
|
|
||||||
if( parse_rgb16( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 32:
|
|
||||||
if( parse_32f( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
vips_error( "tiff2vips",
|
|
||||||
_( "unsupported depth %d for RGB image" ),
|
|
||||||
rtiff->bits_per_sample );
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PHOTOMETRIC_SEPARATED:
|
|
||||||
if( parse_cmyk( rtiff, out ) )
|
|
||||||
return( -1 );
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
vips_error( "tiff2vips",
|
|
||||||
_( "unknown photometric interpretation %d" ),
|
|
||||||
rtiff->photometric_interpretation );
|
|
||||||
return( -1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Read any ICC profile.
|
/* Read any ICC profile.
|
||||||
*/
|
*/
|
||||||
|
Loading…
Reference in New Issue
Block a user