2011-08-03 09:51:40 +02:00
|
|
|
/* $Id: tif_dirread.c,v 1.92.2.15 2010-12-31 16:12:40 olivier Exp $ */
|
2010-05-11 19:44:00 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 1988-1997 Sam Leffler
|
|
|
|
* Copyright (c) 1991-1997 Silicon Graphics, Inc.
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software and
|
|
|
|
* its documentation for any purpose is hereby granted without fee, provided
|
|
|
|
* that (i) the above copyright notices and this permission notice appear in
|
|
|
|
* all copies of the software and related documentation, and (ii) the names of
|
|
|
|
* Sam Leffler and Silicon Graphics may not be used in any advertising or
|
|
|
|
* publicity relating to the software without the specific, prior written
|
|
|
|
* permission of Sam Leffler and Silicon Graphics.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
* EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
|
|
|
|
* WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
*
|
|
|
|
* IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
|
|
|
|
* OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
|
|
|
|
* WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF
|
|
|
|
* LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
|
|
|
|
* OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TIFF Library.
|
|
|
|
*
|
|
|
|
* Directory Read Support Routines.
|
|
|
|
*/
|
|
|
|
#include "tiffiop.h"
|
|
|
|
|
|
|
|
#define IGNORE 0 /* tag placeholder used below */
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
#ifdef HAVE_IEEEFP
|
|
|
|
# define TIFFCvtIEEEFloatToNative(tif, n, fp)
|
|
|
|
# define TIFFCvtIEEEDoubleToNative(tif, n, dp)
|
2010-05-11 19:44:00 +02:00
|
|
|
#else
|
|
|
|
extern void TIFFCvtIEEEFloatToNative(TIFF*, uint32, float*);
|
|
|
|
extern void TIFFCvtIEEEDoubleToNative(TIFF*, uint32, double*);
|
|
|
|
#endif
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
static TIFFDirEntry* TIFFReadDirectoryFind(TIFFDirEntry* dir,
|
|
|
|
uint16 dircount, uint16 tagid);
|
2010-05-11 19:44:00 +02:00
|
|
|
static int EstimateStripByteCounts(TIFF*, TIFFDirEntry*, uint16);
|
|
|
|
static void MissingRequired(TIFF*, const char*);
|
2010-07-16 14:54:53 +02:00
|
|
|
static int TIFFCheckDirOffset(TIFF*, toff_t);
|
2010-05-11 19:44:00 +02:00
|
|
|
static int CheckDirCount(TIFF*, TIFFDirEntry*, uint32);
|
2010-07-16 14:54:53 +02:00
|
|
|
static uint16 TIFFFetchDirectory(TIFF*, toff_t, TIFFDirEntry**, toff_t *);
|
2010-05-11 19:44:00 +02:00
|
|
|
static tsize_t TIFFFetchData(TIFF*, TIFFDirEntry*, char*);
|
|
|
|
static tsize_t TIFFFetchString(TIFF*, TIFFDirEntry*, char*);
|
|
|
|
static float TIFFFetchRational(TIFF*, TIFFDirEntry*);
|
|
|
|
static int TIFFFetchNormalTag(TIFF*, TIFFDirEntry*);
|
|
|
|
static int TIFFFetchPerSampleShorts(TIFF*, TIFFDirEntry*, uint16*);
|
|
|
|
static int TIFFFetchPerSampleLongs(TIFF*, TIFFDirEntry*, uint32*);
|
2011-08-03 09:51:40 +02:00
|
|
|
static int TIFFFetchPerSampleAnys(TIFF*, TIFFDirEntry*, double*, double*);
|
2010-05-11 19:44:00 +02:00
|
|
|
static int TIFFFetchShortArray(TIFF*, TIFFDirEntry*, uint16*);
|
|
|
|
static int TIFFFetchStripThing(TIFF*, TIFFDirEntry*, long, uint32**);
|
|
|
|
static int TIFFFetchRefBlackWhite(TIFF*, TIFFDirEntry*);
|
2010-07-16 14:54:53 +02:00
|
|
|
static int TIFFFetchSubjectDistance(TIFF*, TIFFDirEntry*);
|
2010-05-11 19:44:00 +02:00
|
|
|
static float TIFFFetchFloat(TIFF*, TIFFDirEntry*);
|
|
|
|
static int TIFFFetchFloatArray(TIFF*, TIFFDirEntry*, float*);
|
|
|
|
static int TIFFFetchDoubleArray(TIFF*, TIFFDirEntry*, double*);
|
|
|
|
static int TIFFFetchAnyArray(TIFF*, TIFFDirEntry*, double*);
|
|
|
|
static int TIFFFetchShortPair(TIFF*, TIFFDirEntry*);
|
|
|
|
static void ChopUpSingleUncompressedStrip(TIFF*);
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Read the next TIFF directory from a file and convert it to the internal
|
|
|
|
* format. We read directories sequentially.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
int
|
|
|
|
TIFFReadDirectory(TIFF* tif)
|
|
|
|
{
|
|
|
|
static const char module[] = "TIFFReadDirectory";
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
int n;
|
|
|
|
TIFFDirectory* td;
|
|
|
|
TIFFDirEntry *dp, *dir = NULL;
|
2010-05-11 19:44:00 +02:00
|
|
|
uint16 iv;
|
|
|
|
uint32 v;
|
|
|
|
const TIFFFieldInfo* fip;
|
2010-07-16 14:54:53 +02:00
|
|
|
size_t fix;
|
2010-05-11 19:44:00 +02:00
|
|
|
uint16 dircount;
|
2011-08-03 09:51:40 +02:00
|
|
|
uint16 previous_tag = 0;
|
2010-07-16 14:54:53 +02:00
|
|
|
int diroutoforderwarning = 0, compressionknown = 0;
|
|
|
|
int haveunknowntags = 0;
|
2010-05-11 19:44:00 +02:00
|
|
|
|
|
|
|
tif->tif_diroff = tif->tif_nextdiroff;
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Check whether we have the last offset or bad offset (IFD looping).
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
if (!TIFFCheckDirOffset(tif, tif->tif_nextdiroff))
|
|
|
|
return 0;
|
2010-05-11 19:44:00 +02:00
|
|
|
/*
|
|
|
|
* Cleanup any previous compression state.
|
|
|
|
*/
|
|
|
|
(*tif->tif_cleanup)(tif);
|
|
|
|
tif->tif_curdir++;
|
2010-07-16 14:54:53 +02:00
|
|
|
dircount = TIFFFetchDirectory(tif, tif->tif_nextdiroff,
|
|
|
|
&dir, &tif->tif_nextdiroff);
|
|
|
|
if (!dircount) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: Failed to read directory at offset %u",
|
|
|
|
tif->tif_name, tif->tif_nextdiroff);
|
|
|
|
return 0;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
2011-08-03 09:51:40 +02:00
|
|
|
{
|
|
|
|
TIFFDirEntry* ma;
|
|
|
|
uint16 mb;
|
|
|
|
for (ma=dir, mb=0; mb<dircount; ma++, mb++)
|
|
|
|
{
|
|
|
|
TIFFDirEntry* na;
|
|
|
|
uint16 nb;
|
|
|
|
for (na=ma+1, nb=mb+1; nb<dircount; na++, nb++)
|
|
|
|
{
|
|
|
|
if (ma->tdir_tag==na->tdir_tag)
|
|
|
|
na->tdir_tag=IGNORE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
tif->tif_flags &= ~TIFF_BEENWRITING; /* reset before new dir */
|
|
|
|
/*
|
|
|
|
* Setup default value and then make a pass over
|
|
|
|
* the fields to check type and tag information,
|
|
|
|
* and to extract info required to size data
|
|
|
|
* structures. A second pass is made afterwards
|
|
|
|
* to read in everthing not taken in the first pass.
|
|
|
|
*/
|
|
|
|
td = &tif->tif_dir;
|
|
|
|
/* free any old stuff and reinit */
|
|
|
|
TIFFFreeDirectory(tif);
|
|
|
|
TIFFDefaultDirectory(tif);
|
|
|
|
/*
|
|
|
|
* Electronic Arts writes gray-scale TIFF files
|
|
|
|
* without a PlanarConfiguration directory entry.
|
|
|
|
* Thus we setup a default value here, even though
|
|
|
|
* the TIFF spec says there is no default value.
|
|
|
|
*/
|
|
|
|
TIFFSetField(tif, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Sigh, we must make a separate pass through the
|
|
|
|
* directory for the following reason:
|
|
|
|
*
|
|
|
|
* We must process the Compression tag in the first pass
|
|
|
|
* in order to merge in codec-private tag definitions (otherwise
|
|
|
|
* we may get complaints about unknown tags). However, the
|
|
|
|
* Compression tag may be dependent on the SamplesPerPixel
|
|
|
|
* tag value because older TIFF specs permited Compression
|
|
|
|
* to be written as a SamplesPerPixel-count tag entry.
|
|
|
|
* Thus if we don't first figure out the correct SamplesPerPixel
|
|
|
|
* tag value then we may end up ignoring the Compression tag
|
|
|
|
* value because it has an incorrect count value (if the
|
|
|
|
* true value of SamplesPerPixel is not 1).
|
|
|
|
*
|
|
|
|
* It sure would have been nice if Aldus had really thought
|
|
|
|
* this stuff through carefully.
|
2010-07-16 14:54:53 +02:00
|
|
|
*/
|
2010-05-11 19:44:00 +02:00
|
|
|
for (dp = dir, n = dircount; n > 0; n--, dp++) {
|
|
|
|
if (tif->tif_flags & TIFF_SWAB) {
|
|
|
|
TIFFSwabArrayOfShort(&dp->tdir_tag, 2);
|
|
|
|
TIFFSwabArrayOfLong(&dp->tdir_count, 2);
|
|
|
|
}
|
|
|
|
if (dp->tdir_tag == TIFFTAG_SAMPLESPERPIXEL) {
|
|
|
|
if (!TIFFFetchNormalTag(tif, dp))
|
|
|
|
goto bad;
|
|
|
|
dp->tdir_tag = IGNORE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* First real pass over the directory.
|
|
|
|
*/
|
|
|
|
fix = 0;
|
|
|
|
for (dp = dir, n = dircount; n > 0; n--, dp++) {
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
if (dp->tdir_tag == IGNORE)
|
2010-05-11 19:44:00 +02:00
|
|
|
continue;
|
2010-07-16 14:54:53 +02:00
|
|
|
|
2010-05-11 19:44:00 +02:00
|
|
|
/*
|
|
|
|
* Silicon Beach (at least) writes unordered
|
|
|
|
* directory tags (violating the spec). Handle
|
|
|
|
* it here, but be obnoxious (maybe they'll fix it?).
|
|
|
|
*/
|
2011-08-03 09:51:40 +02:00
|
|
|
if (dp->tdir_tag < previous_tag) {
|
2010-05-11 19:44:00 +02:00
|
|
|
if (!diroutoforderwarning) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
|
|
|
"%s: invalid TIFF directory; tags are not sorted in ascending order",
|
|
|
|
tif->tif_name);
|
2010-05-11 19:44:00 +02:00
|
|
|
diroutoforderwarning = 1;
|
|
|
|
}
|
|
|
|
}
|
2011-08-03 09:51:40 +02:00
|
|
|
previous_tag = dp->tdir_tag;
|
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
dp->tdir_tag < tif->tif_fieldinfo[fix]->field_tag)
|
|
|
|
fix = 0; /* O(n^2) */
|
2010-05-11 19:44:00 +02:00
|
|
|
while (fix < tif->tif_nfields &&
|
2010-07-16 14:54:53 +02:00
|
|
|
tif->tif_fieldinfo[fix]->field_tag < dp->tdir_tag)
|
2010-05-11 19:44:00 +02:00
|
|
|
fix++;
|
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
tif->tif_fieldinfo[fix]->field_tag != dp->tdir_tag) {
|
2010-07-16 14:54:53 +02:00
|
|
|
/* Unknown tag ... we'll deal with it below */
|
|
|
|
haveunknowntags = 1;
|
|
|
|
continue;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Null out old tags that we ignore.
|
|
|
|
*/
|
|
|
|
if (tif->tif_fieldinfo[fix]->field_bit == FIELD_IGNORE) {
|
|
|
|
ignore:
|
|
|
|
dp->tdir_tag = IGNORE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Check data type.
|
|
|
|
*/
|
|
|
|
fip = tif->tif_fieldinfo[fix];
|
|
|
|
while (dp->tdir_type != (unsigned short) fip->field_type
|
2010-07-16 14:54:53 +02:00
|
|
|
&& fix < tif->tif_nfields) {
|
2010-05-11 19:44:00 +02:00
|
|
|
if (fip->field_type == TIFF_ANY) /* wildcard */
|
|
|
|
break;
|
2010-07-16 14:54:53 +02:00
|
|
|
fip = tif->tif_fieldinfo[++fix];
|
2010-05-11 19:44:00 +02:00
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
fip->field_tag != dp->tdir_tag) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
2010-05-11 19:44:00 +02:00
|
|
|
"%s: wrong data type %d for \"%s\"; tag ignored",
|
|
|
|
tif->tif_name, dp->tdir_type,
|
|
|
|
tif->tif_fieldinfo[fix-1]->field_name);
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Check count if known in advance.
|
|
|
|
*/
|
|
|
|
if (fip->field_readcount != TIFF_VARIABLE
|
|
|
|
&& fip->field_readcount != TIFF_VARIABLE2) {
|
|
|
|
uint32 expected = (fip->field_readcount == TIFF_SPP) ?
|
|
|
|
(uint32) td->td_samplesperpixel :
|
|
|
|
(uint32) fip->field_readcount;
|
|
|
|
if (!CheckDirCount(tif, dp, expected))
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (dp->tdir_tag) {
|
|
|
|
case TIFFTAG_COMPRESSION:
|
|
|
|
/*
|
|
|
|
* The 5.0 spec says the Compression tag has
|
|
|
|
* one value, while earlier specs say it has
|
|
|
|
* one value per sample. Because of this, we
|
|
|
|
* accept the tag if one value is supplied.
|
|
|
|
*/
|
|
|
|
if (dp->tdir_count == 1) {
|
|
|
|
v = TIFFExtractData(tif,
|
|
|
|
dp->tdir_type, dp->tdir_offset);
|
|
|
|
if (!TIFFSetField(tif, dp->tdir_tag, (uint16)v))
|
|
|
|
goto bad;
|
2010-07-16 14:54:53 +02:00
|
|
|
else
|
|
|
|
compressionknown = 1;
|
2010-05-11 19:44:00 +02:00
|
|
|
break;
|
|
|
|
/* XXX: workaround for broken TIFFs */
|
|
|
|
} else if (dp->tdir_type == TIFF_LONG) {
|
|
|
|
if (!TIFFFetchPerSampleLongs(tif, dp, &v) ||
|
|
|
|
!TIFFSetField(tif, dp->tdir_tag, (uint16)v))
|
|
|
|
goto bad;
|
|
|
|
} else {
|
|
|
|
if (!TIFFFetchPerSampleShorts(tif, dp, &iv)
|
|
|
|
|| !TIFFSetField(tif, dp->tdir_tag, iv))
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
dp->tdir_tag = IGNORE;
|
|
|
|
break;
|
|
|
|
case TIFFTAG_STRIPOFFSETS:
|
|
|
|
case TIFFTAG_STRIPBYTECOUNTS:
|
|
|
|
case TIFFTAG_TILEOFFSETS:
|
|
|
|
case TIFFTAG_TILEBYTECOUNTS:
|
|
|
|
TIFFSetFieldBit(tif, fip->field_bit);
|
|
|
|
break;
|
|
|
|
case TIFFTAG_IMAGEWIDTH:
|
|
|
|
case TIFFTAG_IMAGELENGTH:
|
|
|
|
case TIFFTAG_IMAGEDEPTH:
|
|
|
|
case TIFFTAG_TILELENGTH:
|
|
|
|
case TIFFTAG_TILEWIDTH:
|
|
|
|
case TIFFTAG_TILEDEPTH:
|
|
|
|
case TIFFTAG_PLANARCONFIG:
|
|
|
|
case TIFFTAG_ROWSPERSTRIP:
|
2010-07-16 14:54:53 +02:00
|
|
|
case TIFFTAG_EXTRASAMPLES:
|
2010-05-11 19:44:00 +02:00
|
|
|
if (!TIFFFetchNormalTag(tif, dp))
|
|
|
|
goto bad;
|
|
|
|
dp->tdir_tag = IGNORE;
|
|
|
|
break;
|
2010-07-16 14:54:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we saw any unknown tags, make an extra pass over the directory
|
|
|
|
* to deal with them. This must be done separately because the tags
|
|
|
|
* could have become known when we registered a codec after finding
|
|
|
|
* the Compression tag. In a correctly-sorted directory there's
|
|
|
|
* no problem because Compression will come before any codec-private
|
|
|
|
* tags, but if the sorting is wrong that might not hold.
|
|
|
|
*/
|
|
|
|
if (haveunknowntags) {
|
|
|
|
fix = 0;
|
|
|
|
for (dp = dir, n = dircount; n > 0; n--, dp++) {
|
|
|
|
if (dp->tdir_tag == IGNORE)
|
|
|
|
continue;
|
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
dp->tdir_tag < tif->tif_fieldinfo[fix]->field_tag)
|
|
|
|
fix = 0; /* O(n^2) */
|
|
|
|
while (fix < tif->tif_nfields &&
|
|
|
|
tif->tif_fieldinfo[fix]->field_tag < dp->tdir_tag)
|
|
|
|
fix++;
|
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
tif->tif_fieldinfo[fix]->field_tag != dp->tdir_tag) {
|
|
|
|
|
|
|
|
TIFFWarningExt(tif->tif_clientdata,
|
|
|
|
module,
|
|
|
|
"%s: unknown field with tag %d (0x%x) encountered",
|
|
|
|
tif->tif_name,
|
|
|
|
dp->tdir_tag,
|
|
|
|
dp->tdir_tag);
|
|
|
|
|
|
|
|
if (!_TIFFMergeFieldInfo(tif,
|
|
|
|
_TIFFCreateAnonFieldInfo(tif,
|
|
|
|
dp->tdir_tag,
|
|
|
|
(TIFFDataType) dp->tdir_type),
|
|
|
|
1))
|
|
|
|
{
|
|
|
|
TIFFWarningExt(tif->tif_clientdata,
|
|
|
|
module,
|
|
|
|
"Registering anonymous field with tag %d (0x%x) failed",
|
|
|
|
dp->tdir_tag,
|
|
|
|
dp->tdir_tag);
|
|
|
|
dp->tdir_tag = IGNORE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
fix = 0;
|
|
|
|
while (fix < tif->tif_nfields &&
|
|
|
|
tif->tif_fieldinfo[fix]->field_tag < dp->tdir_tag)
|
|
|
|
fix++;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Check data type.
|
|
|
|
*/
|
|
|
|
fip = tif->tif_fieldinfo[fix];
|
|
|
|
while (dp->tdir_type != (unsigned short) fip->field_type
|
|
|
|
&& fix < tif->tif_nfields) {
|
|
|
|
if (fip->field_type == TIFF_ANY) /* wildcard */
|
|
|
|
break;
|
|
|
|
fip = tif->tif_fieldinfo[++fix];
|
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
fip->field_tag != dp->tdir_tag) {
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
|
|
|
"%s: wrong data type %d for \"%s\"; tag ignored",
|
|
|
|
tif->tif_name, dp->tdir_type,
|
|
|
|
tif->tif_fieldinfo[fix-1]->field_name);
|
|
|
|
dp->tdir_tag = IGNORE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX: OJPEG hack.
|
|
|
|
* If a) compression is OJPEG, b) planarconfig tag says it's separate,
|
|
|
|
* c) strip offsets/bytecounts tag are both present and
|
|
|
|
* d) both contain exactly one value, then we consistently find
|
|
|
|
* that the buggy implementation of the buggy compression scheme
|
|
|
|
* matches contig planarconfig best. So we 'fix-up' the tag here
|
|
|
|
*/
|
|
|
|
if ((td->td_compression==COMPRESSION_OJPEG) &&
|
|
|
|
(td->td_planarconfig==PLANARCONFIG_SEPARATE)) {
|
|
|
|
dp = TIFFReadDirectoryFind(dir,dircount,TIFFTAG_STRIPOFFSETS);
|
|
|
|
if ((dp!=0) && (dp->tdir_count==1)) {
|
|
|
|
dp = TIFFReadDirectoryFind(dir, dircount,
|
|
|
|
TIFFTAG_STRIPBYTECOUNTS);
|
|
|
|
if ((dp!=0) && (dp->tdir_count==1)) {
|
|
|
|
td->td_planarconfig=PLANARCONFIG_CONTIG;
|
|
|
|
TIFFWarningExt(tif->tif_clientdata,
|
|
|
|
"TIFFReadDirectory",
|
|
|
|
"Planarconfig tag value assumed incorrect, "
|
|
|
|
"assuming data is contig instead of chunky");
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate directory structure and setup defaults.
|
|
|
|
*/
|
|
|
|
if (!TIFFFieldSet(tif, FIELD_IMAGEDIMENSIONS)) {
|
|
|
|
MissingRequired(tif, "ImageLength");
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Setup appropriate structures (by strip or by tile)
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
if (!TIFFFieldSet(tif, FIELD_TILEDIMENSIONS)) {
|
|
|
|
td->td_nstrips = TIFFNumberOfStrips(tif);
|
|
|
|
td->td_tilewidth = td->td_imagewidth;
|
|
|
|
td->td_tilelength = td->td_rowsperstrip;
|
|
|
|
td->td_tiledepth = td->td_imagedepth;
|
|
|
|
tif->tif_flags &= ~TIFF_ISTILED;
|
|
|
|
} else {
|
|
|
|
td->td_nstrips = TIFFNumberOfTiles(tif);
|
|
|
|
tif->tif_flags |= TIFF_ISTILED;
|
|
|
|
}
|
|
|
|
if (!td->td_nstrips) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: cannot handle zero number of %s",
|
|
|
|
tif->tif_name, isTiled(tif) ? "tiles" : "strips");
|
2010-05-11 19:44:00 +02:00
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
td->td_stripsperimage = td->td_nstrips;
|
|
|
|
if (td->td_planarconfig == PLANARCONFIG_SEPARATE)
|
|
|
|
td->td_stripsperimage /= td->td_samplesperpixel;
|
|
|
|
if (!TIFFFieldSet(tif, FIELD_STRIPOFFSETS)) {
|
2010-07-16 14:54:53 +02:00
|
|
|
if ((td->td_compression==COMPRESSION_OJPEG) &&
|
|
|
|
(isTiled(tif)==0) &&
|
|
|
|
(td->td_nstrips==1)) {
|
|
|
|
/*
|
|
|
|
* XXX: OJPEG hack.
|
|
|
|
* If a) compression is OJPEG, b) it's not a tiled TIFF,
|
|
|
|
* and c) the number of strips is 1,
|
|
|
|
* then we tolerate the absence of stripoffsets tag,
|
|
|
|
* because, presumably, all required data is in the
|
|
|
|
* JpegInterchangeFormat stream.
|
|
|
|
*/
|
|
|
|
TIFFSetFieldBit(tif, FIELD_STRIPOFFSETS);
|
|
|
|
} else {
|
|
|
|
MissingRequired(tif,
|
|
|
|
isTiled(tif) ? "TileOffsets" : "StripOffsets");
|
|
|
|
goto bad;
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Second pass: extract other information.
|
|
|
|
*/
|
|
|
|
for (dp = dir, n = dircount; n > 0; n--, dp++) {
|
|
|
|
if (dp->tdir_tag == IGNORE)
|
|
|
|
continue;
|
|
|
|
switch (dp->tdir_tag) {
|
|
|
|
case TIFFTAG_MINSAMPLEVALUE:
|
|
|
|
case TIFFTAG_MAXSAMPLEVALUE:
|
|
|
|
case TIFFTAG_BITSPERSAMPLE:
|
|
|
|
case TIFFTAG_DATATYPE:
|
|
|
|
case TIFFTAG_SAMPLEFORMAT:
|
|
|
|
/*
|
|
|
|
* The 5.0 spec says the Compression tag has
|
|
|
|
* one value, while earlier specs say it has
|
|
|
|
* one value per sample. Because of this, we
|
|
|
|
* accept the tag if one value is supplied.
|
|
|
|
*
|
2010-07-16 14:54:53 +02:00
|
|
|
* The MinSampleValue, MaxSampleValue, BitsPerSample
|
|
|
|
* DataType and SampleFormat tags are supposed to be
|
|
|
|
* written as one value/sample, but some vendors
|
|
|
|
* incorrectly write one value only -- so we accept
|
|
|
|
* that as well (yech). Other vendors write correct
|
2010-05-11 19:44:00 +02:00
|
|
|
* value for NumberOfSamples, but incorrect one for
|
|
|
|
* BitsPerSample and friends, and we will read this
|
|
|
|
* too.
|
|
|
|
*/
|
|
|
|
if (dp->tdir_count == 1) {
|
|
|
|
v = TIFFExtractData(tif,
|
|
|
|
dp->tdir_type, dp->tdir_offset);
|
|
|
|
if (!TIFFSetField(tif, dp->tdir_tag, (uint16)v))
|
|
|
|
goto bad;
|
|
|
|
/* XXX: workaround for broken TIFFs */
|
|
|
|
} else if (dp->tdir_tag == TIFFTAG_BITSPERSAMPLE
|
|
|
|
&& dp->tdir_type == TIFF_LONG) {
|
|
|
|
if (!TIFFFetchPerSampleLongs(tif, dp, &v) ||
|
|
|
|
!TIFFSetField(tif, dp->tdir_tag, (uint16)v))
|
|
|
|
goto bad;
|
|
|
|
} else {
|
|
|
|
if (!TIFFFetchPerSampleShorts(tif, dp, &iv) ||
|
|
|
|
!TIFFSetField(tif, dp->tdir_tag, iv))
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFFTAG_SMINSAMPLEVALUE:
|
2011-08-03 09:51:40 +02:00
|
|
|
{
|
|
|
|
double minv = 0.0, maxv = 0.0;
|
|
|
|
if (!TIFFFetchPerSampleAnys(tif, dp, &minv, &maxv) ||
|
|
|
|
!TIFFSetField(tif, dp->tdir_tag, minv))
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
break;
|
2010-05-11 19:44:00 +02:00
|
|
|
case TIFFTAG_SMAXSAMPLEVALUE:
|
2010-07-16 14:54:53 +02:00
|
|
|
{
|
2011-08-03 09:51:40 +02:00
|
|
|
double minv = 0.0, maxv = 0.0;
|
|
|
|
if (!TIFFFetchPerSampleAnys(tif, dp, &minv, &maxv) ||
|
|
|
|
!TIFFSetField(tif, dp->tdir_tag, maxv))
|
2010-07-16 14:54:53 +02:00
|
|
|
goto bad;
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
break;
|
|
|
|
case TIFFTAG_STRIPOFFSETS:
|
|
|
|
case TIFFTAG_TILEOFFSETS:
|
|
|
|
if (!TIFFFetchStripThing(tif, dp,
|
|
|
|
td->td_nstrips, &td->td_stripoffset))
|
|
|
|
goto bad;
|
|
|
|
break;
|
|
|
|
case TIFFTAG_STRIPBYTECOUNTS:
|
|
|
|
case TIFFTAG_TILEBYTECOUNTS:
|
|
|
|
if (!TIFFFetchStripThing(tif, dp,
|
|
|
|
td->td_nstrips, &td->td_stripbytecount))
|
|
|
|
goto bad;
|
|
|
|
break;
|
|
|
|
case TIFFTAG_COLORMAP:
|
|
|
|
case TIFFTAG_TRANSFERFUNCTION:
|
2010-07-16 14:54:53 +02:00
|
|
|
{
|
|
|
|
char* cp;
|
|
|
|
/*
|
|
|
|
* TransferFunction can have either 1x or 3x
|
|
|
|
* data values; Colormap can have only 3x
|
|
|
|
* items.
|
|
|
|
*/
|
|
|
|
v = 1L<<td->td_bitspersample;
|
|
|
|
if (dp->tdir_tag == TIFFTAG_COLORMAP ||
|
|
|
|
dp->tdir_count != v) {
|
|
|
|
if (!CheckDirCount(tif, dp, 3 * v))
|
|
|
|
break;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
2010-07-16 14:54:53 +02:00
|
|
|
v *= sizeof(uint16);
|
|
|
|
cp = (char *)_TIFFCheckMalloc(tif,
|
|
|
|
dp->tdir_count,
|
|
|
|
sizeof (uint16),
|
|
|
|
"to read \"TransferFunction\" tag");
|
|
|
|
if (cp != NULL) {
|
|
|
|
if (TIFFFetchData(tif, dp, cp)) {
|
|
|
|
/*
|
|
|
|
* This deals with there being
|
|
|
|
* only one array to apply to
|
|
|
|
* all samples.
|
|
|
|
*/
|
|
|
|
uint32 c = 1L << td->td_bitspersample;
|
|
|
|
if (dp->tdir_count == c)
|
|
|
|
v = 0L;
|
|
|
|
TIFFSetField(tif, dp->tdir_tag,
|
|
|
|
cp, cp+v, cp+2*v);
|
|
|
|
}
|
|
|
|
_TIFFfree(cp);
|
|
|
|
}
|
|
|
|
break;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
case TIFFTAG_PAGENUMBER:
|
|
|
|
case TIFFTAG_HALFTONEHINTS:
|
|
|
|
case TIFFTAG_YCBCRSUBSAMPLING:
|
|
|
|
case TIFFTAG_DOTRANGE:
|
|
|
|
(void) TIFFFetchShortPair(tif, dp);
|
|
|
|
break;
|
|
|
|
case TIFFTAG_REFERENCEBLACKWHITE:
|
|
|
|
(void) TIFFFetchRefBlackWhite(tif, dp);
|
|
|
|
break;
|
|
|
|
/* BEGIN REV 4.0 COMPATIBILITY */
|
|
|
|
case TIFFTAG_OSUBFILETYPE:
|
|
|
|
v = 0L;
|
|
|
|
switch (TIFFExtractData(tif, dp->tdir_type,
|
|
|
|
dp->tdir_offset)) {
|
|
|
|
case OFILETYPE_REDUCEDIMAGE:
|
|
|
|
v = FILETYPE_REDUCEDIMAGE;
|
|
|
|
break;
|
|
|
|
case OFILETYPE_PAGE:
|
|
|
|
v = FILETYPE_PAGE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (v)
|
|
|
|
TIFFSetField(tif, TIFFTAG_SUBFILETYPE, v);
|
|
|
|
break;
|
|
|
|
/* END REV 4.0 COMPATIBILITY */
|
|
|
|
default:
|
|
|
|
(void) TIFFFetchNormalTag(tif, dp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-07-16 14:54:53 +02:00
|
|
|
/*
|
|
|
|
* OJPEG hack:
|
|
|
|
* - If a) compression is OJPEG, and b) photometric tag is missing,
|
|
|
|
* then we consistently find that photometric should be YCbCr
|
|
|
|
* - If a) compression is OJPEG, and b) photometric tag says it's RGB,
|
|
|
|
* then we consistently find that the buggy implementation of the
|
|
|
|
* buggy compression scheme matches photometric YCbCr instead.
|
|
|
|
* - If a) compression is OJPEG, and b) bitspersample tag is missing,
|
|
|
|
* then we consistently find bitspersample should be 8.
|
|
|
|
* - If a) compression is OJPEG, b) samplesperpixel tag is missing,
|
|
|
|
* and c) photometric is RGB or YCbCr, then we consistently find
|
|
|
|
* samplesperpixel should be 3
|
|
|
|
* - If a) compression is OJPEG, b) samplesperpixel tag is missing,
|
|
|
|
* and c) photometric is MINISWHITE or MINISBLACK, then we consistently
|
|
|
|
* find samplesperpixel should be 3
|
|
|
|
*/
|
|
|
|
if (td->td_compression==COMPRESSION_OJPEG)
|
|
|
|
{
|
|
|
|
if (!TIFFFieldSet(tif,FIELD_PHOTOMETRIC))
|
|
|
|
{
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, "TIFFReadDirectory",
|
|
|
|
"Photometric tag is missing, assuming data is YCbCr");
|
|
|
|
if (!TIFFSetField(tif,TIFFTAG_PHOTOMETRIC,PHOTOMETRIC_YCBCR))
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
else if (td->td_photometric==PHOTOMETRIC_RGB)
|
|
|
|
{
|
|
|
|
td->td_photometric=PHOTOMETRIC_YCBCR;
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, "TIFFReadDirectory",
|
|
|
|
"Photometric tag value assumed incorrect, "
|
|
|
|
"assuming data is YCbCr instead of RGB");
|
|
|
|
}
|
|
|
|
if (!TIFFFieldSet(tif,FIELD_BITSPERSAMPLE))
|
|
|
|
{
|
|
|
|
TIFFWarningExt(tif->tif_clientdata,"TIFFReadDirectory",
|
|
|
|
"BitsPerSample tag is missing, assuming 8 bits per sample");
|
|
|
|
if (!TIFFSetField(tif,TIFFTAG_BITSPERSAMPLE,8))
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
if (!TIFFFieldSet(tif,FIELD_SAMPLESPERPIXEL))
|
|
|
|
{
|
2011-08-03 09:51:40 +02:00
|
|
|
if (td->td_photometric==PHOTOMETRIC_RGB)
|
2010-07-16 14:54:53 +02:00
|
|
|
{
|
|
|
|
TIFFWarningExt(tif->tif_clientdata,
|
|
|
|
"TIFFReadDirectory",
|
|
|
|
"SamplesPerPixel tag is missing, "
|
|
|
|
"assuming correct SamplesPerPixel value is 3");
|
|
|
|
if (!TIFFSetField(tif,TIFFTAG_SAMPLESPERPIXEL,3))
|
|
|
|
goto bad;
|
|
|
|
}
|
2011-08-03 09:51:40 +02:00
|
|
|
if (td->td_photometric==PHOTOMETRIC_YCBCR)
|
2010-07-16 14:54:53 +02:00
|
|
|
{
|
|
|
|
TIFFWarningExt(tif->tif_clientdata,
|
|
|
|
"TIFFReadDirectory",
|
|
|
|
"SamplesPerPixel tag is missing, "
|
2011-08-03 09:51:40 +02:00
|
|
|
"applying correct SamplesPerPixel value of 3");
|
|
|
|
if (!TIFFSetField(tif,TIFFTAG_SAMPLESPERPIXEL,3))
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
else if ((td->td_photometric==PHOTOMETRIC_MINISWHITE)
|
|
|
|
|| (td->td_photometric==PHOTOMETRIC_MINISBLACK))
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* SamplesPerPixel tag is missing, but is not required
|
|
|
|
* by spec. Assume correct SamplesPerPixel value of 1.
|
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
if (!TIFFSetField(tif,TIFFTAG_SAMPLESPERPIXEL,1))
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
/*
|
|
|
|
* Verify Palette image has a Colormap.
|
|
|
|
*/
|
|
|
|
if (td->td_photometric == PHOTOMETRIC_PALETTE &&
|
|
|
|
!TIFFFieldSet(tif, FIELD_COLORMAP)) {
|
2011-08-03 09:51:40 +02:00
|
|
|
if ( tif->tif_dir.td_bitspersample>=8 && tif->tif_dir.td_samplesperpixel==3)
|
|
|
|
tif->tif_dir.td_photometric = PHOTOMETRIC_RGB;
|
|
|
|
else if (tif->tif_dir.td_bitspersample>=8)
|
|
|
|
tif->tif_dir.td_photometric = PHOTOMETRIC_MINISBLACK;
|
|
|
|
else {
|
|
|
|
MissingRequired(tif, "Colormap");
|
|
|
|
goto bad;
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* OJPEG hack:
|
|
|
|
* We do no further messing with strip/tile offsets/bytecounts in OJPEG
|
|
|
|
* TIFFs
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
if (td->td_compression!=COMPRESSION_OJPEG)
|
|
|
|
{
|
2010-05-11 19:44:00 +02:00
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Attempt to deal with a missing StripByteCounts tag.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
if (!TIFFFieldSet(tif, FIELD_STRIPBYTECOUNTS)) {
|
|
|
|
/*
|
|
|
|
* Some manufacturers violate the spec by not giving
|
|
|
|
* the size of the strips. In this case, assume there
|
|
|
|
* is one uncompressed strip of data.
|
|
|
|
*/
|
|
|
|
if ((td->td_planarconfig == PLANARCONFIG_CONTIG &&
|
|
|
|
td->td_nstrips > 1) ||
|
|
|
|
(td->td_planarconfig == PLANARCONFIG_SEPARATE &&
|
|
|
|
td->td_nstrips != td->td_samplesperpixel)) {
|
|
|
|
MissingRequired(tif, "StripByteCounts");
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
|
|
|
"%s: TIFF directory is missing required "
|
|
|
|
"\"%s\" field, calculating from imagelength",
|
|
|
|
tif->tif_name,
|
|
|
|
_TIFFFieldWithTag(tif,TIFFTAG_STRIPBYTECOUNTS)->field_name);
|
|
|
|
if (EstimateStripByteCounts(tif, dir, dircount) < 0)
|
|
|
|
goto bad;
|
2010-05-11 19:44:00 +02:00
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Assume we have wrong StripByteCount value (in case
|
|
|
|
* of single strip) in following cases:
|
|
|
|
* - it is equal to zero along with StripOffset;
|
|
|
|
* - it is larger than file itself (in case of uncompressed
|
|
|
|
* image);
|
|
|
|
* - it is smaller than the size of the bytes per row
|
|
|
|
* multiplied on the number of rows. The last case should
|
|
|
|
* not be checked in the case of writing new image,
|
|
|
|
* because we may do not know the exact strip size
|
|
|
|
* until the whole image will be written and directory
|
|
|
|
* dumped out.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
#define BYTECOUNTLOOKSBAD \
|
|
|
|
( (td->td_stripbytecount[0] == 0 && td->td_stripoffset[0] != 0) || \
|
|
|
|
(td->td_compression == COMPRESSION_NONE && \
|
|
|
|
td->td_stripbytecount[0] > TIFFGetFileSize(tif) - td->td_stripoffset[0]) || \
|
|
|
|
(tif->tif_mode == O_RDONLY && \
|
|
|
|
td->td_compression == COMPRESSION_NONE && \
|
|
|
|
td->td_stripbytecount[0] < TIFFScanlineSize(tif) * td->td_imagelength) )
|
|
|
|
|
|
|
|
} else if (td->td_nstrips == 1
|
|
|
|
&& td->td_stripoffset[0] != 0
|
|
|
|
&& BYTECOUNTLOOKSBAD) {
|
|
|
|
/*
|
|
|
|
* XXX: Plexus (and others) sometimes give a value of
|
|
|
|
* zero for a tag when they don't know what the
|
|
|
|
* correct value is! Try and handle the simple case
|
|
|
|
* of estimating the size of a one strip image.
|
|
|
|
*/
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
2010-05-11 19:44:00 +02:00
|
|
|
"%s: Bogus \"%s\" field, ignoring and calculating from imagelength",
|
2010-07-16 14:54:53 +02:00
|
|
|
tif->tif_name,
|
|
|
|
_TIFFFieldWithTag(tif,TIFFTAG_STRIPBYTECOUNTS)->field_name);
|
|
|
|
if(EstimateStripByteCounts(tif, dir, dircount) < 0)
|
|
|
|
goto bad;
|
|
|
|
} else if (td->td_planarconfig == PLANARCONFIG_CONTIG
|
|
|
|
&& td->td_nstrips > 2
|
|
|
|
&& td->td_compression == COMPRESSION_NONE
|
|
|
|
&& td->td_stripbytecount[0] != td->td_stripbytecount[1]
|
|
|
|
&& td->td_stripbytecount[0] != 0
|
|
|
|
&& td->td_stripbytecount[1] != 0 ) {
|
|
|
|
/*
|
|
|
|
* XXX: Some vendors fill StripByteCount array with
|
|
|
|
* absolutely wrong values (it can be equal to
|
|
|
|
* StripOffset array, for example). Catch this case
|
|
|
|
* here.
|
|
|
|
*/
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
|
|
|
"%s: Wrong \"%s\" field, ignoring and calculating from imagelength",
|
|
|
|
tif->tif_name,
|
|
|
|
_TIFFFieldWithTag(tif,TIFFTAG_STRIPBYTECOUNTS)->field_name);
|
|
|
|
if (EstimateStripByteCounts(tif, dir, dircount) < 0)
|
|
|
|
goto bad;
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
if (dir) {
|
|
|
|
_TIFFfree((char *)dir);
|
|
|
|
dir = NULL;
|
|
|
|
}
|
|
|
|
if (!TIFFFieldSet(tif, FIELD_MAXSAMPLEVALUE))
|
|
|
|
td->td_maxsamplevalue = (uint16)((1L<<td->td_bitspersample)-1);
|
|
|
|
/*
|
|
|
|
* Setup default compression scheme.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX: We can optimize checking for the strip bounds using the sorted
|
|
|
|
* bytecounts array. See also comments for TIFFAppendToStrip()
|
|
|
|
* function in tif_write.c.
|
|
|
|
*/
|
|
|
|
if (td->td_nstrips > 1) {
|
|
|
|
tstrip_t strip;
|
|
|
|
|
|
|
|
td->td_stripbytecountsorted = 1;
|
|
|
|
for (strip = 1; strip < td->td_nstrips; strip++) {
|
|
|
|
if (td->td_stripoffset[strip - 1] >
|
|
|
|
td->td_stripoffset[strip]) {
|
|
|
|
td->td_stripbytecountsorted = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!TIFFFieldSet(tif, FIELD_COMPRESSION))
|
|
|
|
TIFFSetField(tif, TIFFTAG_COMPRESSION, COMPRESSION_NONE);
|
2010-07-16 14:54:53 +02:00
|
|
|
/*
|
|
|
|
* Some manufacturers make life difficult by writing
|
2010-05-11 19:44:00 +02:00
|
|
|
* large amounts of uncompressed data as a single strip.
|
|
|
|
* This is contrary to the recommendations of the spec.
|
2010-07-16 14:54:53 +02:00
|
|
|
* The following makes an attempt at breaking such images
|
2010-05-11 19:44:00 +02:00
|
|
|
* into strips closer to the recommended 8k bytes. A
|
|
|
|
* side effect, however, is that the RowsPerStrip tag
|
|
|
|
* value may be changed.
|
2010-07-16 14:54:53 +02:00
|
|
|
*/
|
2010-05-11 19:44:00 +02:00
|
|
|
if (td->td_nstrips == 1 && td->td_compression == COMPRESSION_NONE &&
|
|
|
|
(tif->tif_flags & (TIFF_STRIPCHOP|TIFF_ISTILED)) == TIFF_STRIPCHOP)
|
|
|
|
ChopUpSingleUncompressedStrip(tif);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Reinitialize i/o since we are starting on a new directory.
|
|
|
|
*/
|
|
|
|
tif->tif_row = (uint32) -1;
|
|
|
|
tif->tif_curstrip = (tstrip_t) -1;
|
|
|
|
tif->tif_col = (uint32) -1;
|
|
|
|
tif->tif_curtile = (ttile_t) -1;
|
|
|
|
tif->tif_tilesize = (tsize_t) -1;
|
|
|
|
|
|
|
|
tif->tif_scanlinesize = TIFFScanlineSize(tif);
|
|
|
|
if (!tif->tif_scanlinesize) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: cannot handle zero scanline size",
|
|
|
|
tif->tif_name);
|
2010-05-11 19:44:00 +02:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isTiled(tif)) {
|
|
|
|
tif->tif_tilesize = TIFFTileSize(tif);
|
|
|
|
if (!tif->tif_tilesize) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: cannot handle zero tile size",
|
|
|
|
tif->tif_name);
|
2010-05-11 19:44:00 +02:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!TIFFStripSize(tif)) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: cannot handle zero strip size",
|
|
|
|
tif->tif_name);
|
2010-05-11 19:44:00 +02:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (1);
|
|
|
|
bad:
|
|
|
|
if (dir)
|
|
|
|
_TIFFfree(dir);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
static TIFFDirEntry*
|
|
|
|
TIFFReadDirectoryFind(TIFFDirEntry* dir, uint16 dircount, uint16 tagid)
|
|
|
|
{
|
|
|
|
TIFFDirEntry* m;
|
|
|
|
uint16 n;
|
|
|
|
for (m=dir, n=0; n<dircount; m++, n++)
|
|
|
|
{
|
|
|
|
if (m->tdir_tag==tagid)
|
|
|
|
return(m);
|
|
|
|
}
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read custom directory from the arbitarry offset.
|
|
|
|
* The code is very similar to TIFFReadDirectory().
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
TIFFReadCustomDirectory(TIFF* tif, toff_t diroff,
|
|
|
|
const TIFFFieldInfo info[], size_t n)
|
|
|
|
{
|
|
|
|
static const char module[] = "TIFFReadCustomDirectory";
|
|
|
|
|
|
|
|
TIFFDirectory* td = &tif->tif_dir;
|
|
|
|
TIFFDirEntry *dp, *dir = NULL;
|
|
|
|
const TIFFFieldInfo* fip;
|
|
|
|
size_t fix;
|
|
|
|
uint16 i, dircount;
|
|
|
|
|
|
|
|
_TIFFSetupFieldInfo(tif, info, n);
|
|
|
|
|
|
|
|
dircount = TIFFFetchDirectory(tif, diroff, &dir, NULL);
|
|
|
|
if (!dircount) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: Failed to read custom directory at offset %u",
|
|
|
|
tif->tif_name, diroff);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
TIFFFreeDirectory(tif);
|
|
|
|
_TIFFmemset(&tif->tif_dir, 0, sizeof(TIFFDirectory));
|
|
|
|
|
|
|
|
fix = 0;
|
|
|
|
for (dp = dir, i = dircount; i > 0; i--, dp++) {
|
|
|
|
if (tif->tif_flags & TIFF_SWAB) {
|
|
|
|
TIFFSwabArrayOfShort(&dp->tdir_tag, 2);
|
|
|
|
TIFFSwabArrayOfLong(&dp->tdir_count, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fix >= tif->tif_nfields || dp->tdir_tag == IGNORE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
while (fix < tif->tif_nfields &&
|
|
|
|
tif->tif_fieldinfo[fix]->field_tag < dp->tdir_tag)
|
|
|
|
fix++;
|
|
|
|
|
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
tif->tif_fieldinfo[fix]->field_tag != dp->tdir_tag) {
|
|
|
|
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
|
|
|
"%s: unknown field with tag %d (0x%x) encountered",
|
|
|
|
tif->tif_name, dp->tdir_tag, dp->tdir_tag);
|
|
|
|
if (!_TIFFMergeFieldInfo(tif,
|
|
|
|
_TIFFCreateAnonFieldInfo(tif,
|
|
|
|
dp->tdir_tag,
|
|
|
|
(TIFFDataType) dp->tdir_type),
|
|
|
|
1))
|
|
|
|
{
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
|
|
|
"Registering anonymous field with tag %d (0x%x) failed",
|
|
|
|
dp->tdir_tag, dp->tdir_tag);
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
fix = 0;
|
|
|
|
while (fix < tif->tif_nfields &&
|
|
|
|
tif->tif_fieldinfo[fix]->field_tag < dp->tdir_tag)
|
|
|
|
fix++;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Null out old tags that we ignore.
|
|
|
|
*/
|
|
|
|
if (tif->tif_fieldinfo[fix]->field_bit == FIELD_IGNORE) {
|
|
|
|
ignore:
|
|
|
|
dp->tdir_tag = IGNORE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Check data type.
|
|
|
|
*/
|
|
|
|
fip = tif->tif_fieldinfo[fix];
|
|
|
|
while (dp->tdir_type != (unsigned short) fip->field_type
|
|
|
|
&& fix < tif->tif_nfields) {
|
|
|
|
if (fip->field_type == TIFF_ANY) /* wildcard */
|
|
|
|
break;
|
|
|
|
fip = tif->tif_fieldinfo[++fix];
|
|
|
|
if (fix >= tif->tif_nfields ||
|
|
|
|
fip->field_tag != dp->tdir_tag) {
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, module,
|
|
|
|
"%s: wrong data type %d for \"%s\"; tag ignored",
|
|
|
|
tif->tif_name, dp->tdir_type,
|
|
|
|
tif->tif_fieldinfo[fix-1]->field_name);
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Check count if known in advance.
|
|
|
|
*/
|
|
|
|
if (fip->field_readcount != TIFF_VARIABLE
|
|
|
|
&& fip->field_readcount != TIFF_VARIABLE2) {
|
|
|
|
uint32 expected = (fip->field_readcount == TIFF_SPP) ?
|
|
|
|
(uint32) td->td_samplesperpixel :
|
|
|
|
(uint32) fip->field_readcount;
|
|
|
|
if (!CheckDirCount(tif, dp, expected))
|
|
|
|
goto ignore;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* EXIF tags which need to be specifically processed.
|
|
|
|
*/
|
|
|
|
switch (dp->tdir_tag) {
|
|
|
|
case EXIFTAG_SUBJECTDISTANCE:
|
|
|
|
(void) TIFFFetchSubjectDistance(tif, dp);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
(void) TIFFFetchNormalTag(tif, dp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dir)
|
|
|
|
_TIFFfree(dir);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* EXIF is important special case of custom IFD, so we have a special
|
|
|
|
* function to read it.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
TIFFReadEXIFDirectory(TIFF* tif, toff_t diroff)
|
|
|
|
{
|
|
|
|
size_t exifFieldInfoCount;
|
|
|
|
const TIFFFieldInfo *exifFieldInfo =
|
|
|
|
_TIFFGetExifFieldInfo(&exifFieldInfoCount);
|
|
|
|
return TIFFReadCustomDirectory(tif, diroff, exifFieldInfo,
|
|
|
|
exifFieldInfoCount);
|
|
|
|
}
|
|
|
|
|
2010-05-11 19:44:00 +02:00
|
|
|
static int
|
|
|
|
EstimateStripByteCounts(TIFF* tif, TIFFDirEntry* dir, uint16 dircount)
|
|
|
|
{
|
|
|
|
static const char module[] = "EstimateStripByteCounts";
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFDirEntry *dp;
|
|
|
|
TIFFDirectory *td = &tif->tif_dir;
|
|
|
|
uint32 strip;
|
2010-05-11 19:44:00 +02:00
|
|
|
|
|
|
|
if (td->td_stripbytecount)
|
|
|
|
_TIFFfree(td->td_stripbytecount);
|
|
|
|
td->td_stripbytecount = (uint32*)
|
2010-07-16 14:54:53 +02:00
|
|
|
_TIFFCheckMalloc(tif, td->td_nstrips, sizeof (uint32),
|
2010-05-11 19:44:00 +02:00
|
|
|
"for \"StripByteCounts\" array");
|
2010-07-16 14:54:53 +02:00
|
|
|
if( td->td_stripbytecount == NULL )
|
|
|
|
return -1;
|
|
|
|
|
2010-05-11 19:44:00 +02:00
|
|
|
if (td->td_compression != COMPRESSION_NONE) {
|
|
|
|
uint32 space = (uint32)(sizeof (TIFFHeader)
|
|
|
|
+ sizeof (uint16)
|
|
|
|
+ (dircount * sizeof (TIFFDirEntry))
|
|
|
|
+ sizeof (uint32));
|
|
|
|
toff_t filesize = TIFFGetFileSize(tif);
|
|
|
|
uint16 n;
|
|
|
|
|
|
|
|
/* calculate amount of space used by indirect values */
|
|
|
|
for (dp = dir, n = dircount; n > 0; n--, dp++)
|
|
|
|
{
|
|
|
|
uint32 cc = TIFFDataWidth((TIFFDataType) dp->tdir_type);
|
|
|
|
if (cc == 0) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
2010-05-11 19:44:00 +02:00
|
|
|
"%s: Cannot determine size of unknown tag type %d",
|
|
|
|
tif->tif_name, dp->tdir_type);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
cc = cc * dp->tdir_count;
|
|
|
|
if (cc > sizeof (uint32))
|
|
|
|
space += cc;
|
|
|
|
}
|
|
|
|
space = filesize - space;
|
|
|
|
if (td->td_planarconfig == PLANARCONFIG_SEPARATE)
|
|
|
|
space /= td->td_samplesperpixel;
|
2010-07-16 14:54:53 +02:00
|
|
|
for (strip = 0; strip < td->td_nstrips; strip++)
|
|
|
|
td->td_stripbytecount[strip] = space;
|
2010-05-11 19:44:00 +02:00
|
|
|
/*
|
|
|
|
* This gross hack handles the case were the offset to
|
|
|
|
* the last strip is past the place where we think the strip
|
|
|
|
* should begin. Since a strip of data must be contiguous,
|
|
|
|
* it's safe to assume that we've overestimated the amount
|
|
|
|
* of data in the strip and trim this number back accordingly.
|
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
strip--;
|
|
|
|
if (((toff_t)(td->td_stripoffset[strip]+
|
|
|
|
td->td_stripbytecount[strip])) > filesize)
|
|
|
|
td->td_stripbytecount[strip] =
|
|
|
|
filesize - td->td_stripoffset[strip];
|
|
|
|
} else if (isTiled(tif)) {
|
|
|
|
uint32 bytespertile = TIFFTileSize(tif);
|
|
|
|
|
|
|
|
for (strip = 0; strip < td->td_nstrips; strip++)
|
|
|
|
td->td_stripbytecount[strip] = bytespertile;
|
2010-05-11 19:44:00 +02:00
|
|
|
} else {
|
|
|
|
uint32 rowbytes = TIFFScanlineSize(tif);
|
|
|
|
uint32 rowsperstrip = td->td_imagelength/td->td_stripsperimage;
|
2010-07-16 14:54:53 +02:00
|
|
|
for (strip = 0; strip < td->td_nstrips; strip++)
|
|
|
|
td->td_stripbytecount[strip] = rowbytes * rowsperstrip;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
TIFFSetFieldBit(tif, FIELD_STRIPBYTECOUNTS);
|
|
|
|
if (!TIFFFieldSet(tif, FIELD_ROWSPERSTRIP))
|
|
|
|
td->td_rowsperstrip = td->td_imagelength;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
MissingRequired(TIFF* tif, const char* tagname)
|
|
|
|
{
|
|
|
|
static const char module[] = "MissingRequired";
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
2010-05-11 19:44:00 +02:00
|
|
|
"%s: TIFF directory is missing required \"%s\" field",
|
|
|
|
tif->tif_name, tagname);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Check the directory offset against the list of already seen directory
|
|
|
|
* offsets. This is a trick to prevent IFD looping. The one can create TIFF
|
|
|
|
* file with looped directory pointers. We will maintain a list of already
|
|
|
|
* seen directories and check every IFD offset against that list.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFCheckDirOffset(TIFF* tif, toff_t diroff)
|
|
|
|
{
|
|
|
|
uint16 n;
|
|
|
|
|
|
|
|
if (diroff == 0) /* no more directories */
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (n = 0; n < tif->tif_dirnumber && tif->tif_dirlist; n++) {
|
|
|
|
if (tif->tif_dirlist[n] == diroff)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
tif->tif_dirnumber++;
|
|
|
|
|
|
|
|
if (tif->tif_dirnumber > tif->tif_dirlistsize) {
|
|
|
|
toff_t* new_dirlist;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX: Reduce memory allocation granularity of the dirlist
|
|
|
|
* array.
|
|
|
|
*/
|
|
|
|
new_dirlist = (toff_t *)_TIFFCheckRealloc(tif,
|
|
|
|
tif->tif_dirlist,
|
|
|
|
tif->tif_dirnumber,
|
|
|
|
2 * sizeof(toff_t),
|
|
|
|
"for IFD list");
|
|
|
|
if (!new_dirlist)
|
|
|
|
return 0;
|
|
|
|
tif->tif_dirlistsize = 2 * tif->tif_dirnumber;
|
|
|
|
tif->tif_dirlist = new_dirlist;
|
|
|
|
}
|
|
|
|
|
|
|
|
tif->tif_dirlist[tif->tif_dirnumber - 1] = diroff;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check the count field of a directory entry against a known value. The
|
|
|
|
* caller is expected to skip/ignore the tag if there is a mismatch.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
CheckDirCount(TIFF* tif, TIFFDirEntry* dir, uint32 count)
|
|
|
|
{
|
|
|
|
if (count > dir->tdir_count) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFWarningExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"incorrect count for field \"%s\" (%u, expecting %u); tag ignored",
|
2010-05-11 19:44:00 +02:00
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name,
|
|
|
|
dir->tdir_count, count);
|
|
|
|
return (0);
|
|
|
|
} else if (count < dir->tdir_count) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFWarningExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"incorrect count for field \"%s\" (%u, expecting %u); tag trimmed",
|
2010-05-11 19:44:00 +02:00
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name,
|
|
|
|
dir->tdir_count, count);
|
2011-08-03 09:51:40 +02:00
|
|
|
dir->tdir_count = count;
|
2010-05-11 19:44:00 +02:00
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
/*
|
|
|
|
* Read IFD structure from the specified offset. If the pointer to
|
|
|
|
* nextdiroff variable has been specified, read it too. Function returns a
|
|
|
|
* number of fields in the directory or 0 if failed.
|
|
|
|
*/
|
|
|
|
static uint16
|
|
|
|
TIFFFetchDirectory(TIFF* tif, toff_t diroff, TIFFDirEntry **pdir,
|
|
|
|
toff_t *nextdiroff)
|
|
|
|
{
|
|
|
|
static const char module[] = "TIFFFetchDirectory";
|
|
|
|
|
|
|
|
TIFFDirEntry *dir;
|
|
|
|
uint16 dircount;
|
|
|
|
|
|
|
|
assert(pdir);
|
|
|
|
|
|
|
|
tif->tif_diroff = diroff;
|
|
|
|
if (nextdiroff)
|
|
|
|
*nextdiroff = 0;
|
|
|
|
if (!isMapped(tif)) {
|
|
|
|
if (!SeekOK(tif, tif->tif_diroff)) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: Seek error accessing TIFF directory",
|
|
|
|
tif->tif_name);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (!ReadOK(tif, &dircount, sizeof (uint16))) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: Can not read TIFF directory count",
|
|
|
|
tif->tif_name);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (tif->tif_flags & TIFF_SWAB)
|
|
|
|
TIFFSwabShort(&dircount);
|
|
|
|
dir = (TIFFDirEntry *)_TIFFCheckMalloc(tif, dircount,
|
|
|
|
sizeof (TIFFDirEntry),
|
|
|
|
"to read TIFF directory");
|
|
|
|
if (dir == NULL)
|
|
|
|
return 0;
|
|
|
|
if (!ReadOK(tif, dir, dircount*sizeof (TIFFDirEntry))) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%.100s: Can not read TIFF directory",
|
|
|
|
tif->tif_name);
|
|
|
|
_TIFFfree(dir);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Read offset to next directory for sequential scans if
|
|
|
|
* needed.
|
|
|
|
*/
|
|
|
|
if (nextdiroff)
|
|
|
|
(void) ReadOK(tif, nextdiroff, sizeof(uint32));
|
|
|
|
} else {
|
|
|
|
toff_t off = tif->tif_diroff;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check for integer overflow when validating the dir_off,
|
|
|
|
* otherwise a very high offset may cause an OOB read and
|
|
|
|
* crash the client. Make two comparisons instead of
|
|
|
|
*
|
|
|
|
* off + sizeof(uint16) > tif->tif_size
|
|
|
|
*
|
|
|
|
* to avoid overflow.
|
|
|
|
*/
|
|
|
|
if (tif->tif_size < sizeof (uint16) ||
|
|
|
|
off > tif->tif_size - sizeof(uint16)) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: Can not read TIFF directory count",
|
|
|
|
tif->tif_name);
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
_TIFFmemcpy(&dircount, tif->tif_base + off,
|
|
|
|
sizeof(uint16));
|
|
|
|
}
|
|
|
|
off += sizeof (uint16);
|
|
|
|
if (tif->tif_flags & TIFF_SWAB)
|
|
|
|
TIFFSwabShort(&dircount);
|
|
|
|
dir = (TIFFDirEntry *)_TIFFCheckMalloc(tif, dircount,
|
|
|
|
sizeof(TIFFDirEntry),
|
|
|
|
"to read TIFF directory");
|
|
|
|
if (dir == NULL)
|
|
|
|
return 0;
|
|
|
|
if (off + dircount * sizeof (TIFFDirEntry) > tif->tif_size) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
|
|
"%s: Can not read TIFF directory",
|
|
|
|
tif->tif_name);
|
|
|
|
_TIFFfree(dir);
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
_TIFFmemcpy(dir, tif->tif_base + off,
|
|
|
|
dircount * sizeof(TIFFDirEntry));
|
|
|
|
}
|
|
|
|
if (nextdiroff) {
|
|
|
|
off += dircount * sizeof (TIFFDirEntry);
|
|
|
|
if (off + sizeof (uint32) <= tif->tif_size) {
|
|
|
|
_TIFFmemcpy(nextdiroff, tif->tif_base + off,
|
|
|
|
sizeof (uint32));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nextdiroff && tif->tif_flags & TIFF_SWAB)
|
|
|
|
TIFFSwabLong(nextdiroff);
|
|
|
|
*pdir = dir;
|
|
|
|
return dircount;
|
|
|
|
}
|
|
|
|
|
2010-05-11 19:44:00 +02:00
|
|
|
/*
|
|
|
|
* Fetch a contiguous directory item.
|
|
|
|
*/
|
|
|
|
static tsize_t
|
|
|
|
TIFFFetchData(TIFF* tif, TIFFDirEntry* dir, char* cp)
|
|
|
|
{
|
2010-07-16 14:54:53 +02:00
|
|
|
uint32 w = TIFFDataWidth((TIFFDataType) dir->tdir_type);
|
|
|
|
/*
|
|
|
|
* FIXME: butecount should have tsize_t type, but for now libtiff
|
|
|
|
* defines tsize_t as a signed 32-bit integer and we are losing
|
|
|
|
* ability to read arrays larger than 2^31 bytes. So we are using
|
|
|
|
* uint32 instead of tsize_t here.
|
|
|
|
*/
|
|
|
|
uint32 cc = dir->tdir_count * w;
|
|
|
|
|
|
|
|
/* Check for overflow. */
|
|
|
|
if (!dir->tdir_count || !w || cc / w != dir->tdir_count)
|
|
|
|
goto bad;
|
2010-05-11 19:44:00 +02:00
|
|
|
|
|
|
|
if (!isMapped(tif)) {
|
|
|
|
if (!SeekOK(tif, dir->tdir_offset))
|
|
|
|
goto bad;
|
|
|
|
if (!ReadOK(tif, cp, cc))
|
|
|
|
goto bad;
|
|
|
|
} else {
|
2010-07-16 14:54:53 +02:00
|
|
|
/* Check for overflow. */
|
|
|
|
if (dir->tdir_offset + cc < dir->tdir_offset
|
|
|
|
|| dir->tdir_offset + cc < cc
|
|
|
|
|| dir->tdir_offset + cc > tif->tif_size)
|
2010-05-11 19:44:00 +02:00
|
|
|
goto bad;
|
|
|
|
_TIFFmemcpy(cp, tif->tif_base + dir->tdir_offset, cc);
|
|
|
|
}
|
|
|
|
if (tif->tif_flags & TIFF_SWAB) {
|
|
|
|
switch (dir->tdir_type) {
|
|
|
|
case TIFF_SHORT:
|
|
|
|
case TIFF_SSHORT:
|
|
|
|
TIFFSwabArrayOfShort((uint16*) cp, dir->tdir_count);
|
|
|
|
break;
|
|
|
|
case TIFF_LONG:
|
|
|
|
case TIFF_SLONG:
|
|
|
|
case TIFF_FLOAT:
|
|
|
|
TIFFSwabArrayOfLong((uint32*) cp, dir->tdir_count);
|
|
|
|
break;
|
|
|
|
case TIFF_RATIONAL:
|
|
|
|
case TIFF_SRATIONAL:
|
|
|
|
TIFFSwabArrayOfLong((uint32*) cp, 2*dir->tdir_count);
|
|
|
|
break;
|
|
|
|
case TIFF_DOUBLE:
|
|
|
|
TIFFSwabArrayOfDouble((double*) cp, dir->tdir_count);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (cc);
|
|
|
|
bad:
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"Error fetching data for field \"%s\"",
|
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name);
|
|
|
|
return (tsize_t) 0;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch an ASCII item from the file.
|
|
|
|
*/
|
|
|
|
static tsize_t
|
|
|
|
TIFFFetchString(TIFF* tif, TIFFDirEntry* dir, char* cp)
|
|
|
|
{
|
|
|
|
if (dir->tdir_count <= 4) {
|
|
|
|
uint32 l = dir->tdir_offset;
|
|
|
|
if (tif->tif_flags & TIFF_SWAB)
|
|
|
|
TIFFSwabLong(&l);
|
|
|
|
_TIFFmemcpy(cp, &l, dir->tdir_count);
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
return (TIFFFetchData(tif, dir, cp));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert numerator+denominator to float.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
cvtRational(TIFF* tif, TIFFDirEntry* dir, uint32 num, uint32 denom, float* rv)
|
|
|
|
{
|
|
|
|
if (denom == 0) {
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"%s: Rational with zero denominator (num = %u)",
|
2010-05-11 19:44:00 +02:00
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name, num);
|
|
|
|
return (0);
|
|
|
|
} else {
|
|
|
|
if (dir->tdir_type == TIFF_RATIONAL)
|
|
|
|
*rv = ((float)num / (float)denom);
|
|
|
|
else
|
|
|
|
*rv = ((float)(int32)num / (float)(int32)denom);
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Fetch a rational item from the file at offset off and return the value as a
|
|
|
|
* floating point number.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
static float
|
|
|
|
TIFFFetchRational(TIFF* tif, TIFFDirEntry* dir)
|
|
|
|
{
|
|
|
|
uint32 l[2];
|
|
|
|
float v;
|
|
|
|
|
|
|
|
return (!TIFFFetchData(tif, dir, (char *)l) ||
|
|
|
|
!cvtRational(tif, dir, l[0], l[1], &v) ? 1.0f : v);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Fetch a single floating point value from the offset field and return it as
|
|
|
|
* a native float.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
static float
|
|
|
|
TIFFFetchFloat(TIFF* tif, TIFFDirEntry* dir)
|
|
|
|
{
|
|
|
|
float v;
|
|
|
|
int32 l = TIFFExtractData(tif, dir->tdir_type, dir->tdir_offset);
|
|
|
|
_TIFFmemcpy(&v, &l, sizeof(float));
|
|
|
|
TIFFCvtIEEEFloatToNative(tif, 1, &v);
|
|
|
|
return (v);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch an array of BYTE or SBYTE values.
|
|
|
|
*/
|
|
|
|
static int
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFFetchByteArray(TIFF* tif, TIFFDirEntry* dir, uint8* v)
|
2010-05-11 19:44:00 +02:00
|
|
|
{
|
|
|
|
if (dir->tdir_count <= 4) {
|
|
|
|
/*
|
|
|
|
* Extract data from offset field.
|
|
|
|
*/
|
|
|
|
if (tif->tif_header.tiff_magic == TIFF_BIGENDIAN) {
|
|
|
|
if (dir->tdir_type == TIFF_SBYTE)
|
|
|
|
switch (dir->tdir_count) {
|
2010-07-16 14:54:53 +02:00
|
|
|
case 4: v[3] = dir->tdir_offset & 0xff;
|
|
|
|
case 3: v[2] = (dir->tdir_offset >> 8) & 0xff;
|
|
|
|
case 2: v[1] = (dir->tdir_offset >> 16) & 0xff;
|
|
|
|
case 1: v[0] = dir->tdir_offset >> 24;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
switch (dir->tdir_count) {
|
2010-07-16 14:54:53 +02:00
|
|
|
case 4: v[3] = dir->tdir_offset & 0xff;
|
|
|
|
case 3: v[2] = (dir->tdir_offset >> 8) & 0xff;
|
|
|
|
case 2: v[1] = (dir->tdir_offset >> 16) & 0xff;
|
|
|
|
case 1: v[0] = dir->tdir_offset >> 24;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (dir->tdir_type == TIFF_SBYTE)
|
|
|
|
switch (dir->tdir_count) {
|
2010-07-16 14:54:53 +02:00
|
|
|
case 4: v[3] = dir->tdir_offset >> 24;
|
|
|
|
case 3: v[2] = (dir->tdir_offset >> 16) & 0xff;
|
|
|
|
case 2: v[1] = (dir->tdir_offset >> 8) & 0xff;
|
|
|
|
case 1: v[0] = dir->tdir_offset & 0xff;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
switch (dir->tdir_count) {
|
2010-07-16 14:54:53 +02:00
|
|
|
case 4: v[3] = dir->tdir_offset >> 24;
|
|
|
|
case 3: v[2] = (dir->tdir_offset >> 16) & 0xff;
|
|
|
|
case 2: v[1] = (dir->tdir_offset >> 8) & 0xff;
|
|
|
|
case 1: v[0] = dir->tdir_offset & 0xff;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return (1);
|
|
|
|
} else
|
|
|
|
return (TIFFFetchData(tif, dir, (char*) v) != 0); /* XXX */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch an array of SHORT or SSHORT values.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchShortArray(TIFF* tif, TIFFDirEntry* dir, uint16* v)
|
|
|
|
{
|
|
|
|
if (dir->tdir_count <= 2) {
|
|
|
|
if (tif->tif_header.tiff_magic == TIFF_BIGENDIAN) {
|
|
|
|
switch (dir->tdir_count) {
|
|
|
|
case 2: v[1] = (uint16) (dir->tdir_offset & 0xffff);
|
|
|
|
case 1: v[0] = (uint16) (dir->tdir_offset >> 16);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch (dir->tdir_count) {
|
|
|
|
case 2: v[1] = (uint16) (dir->tdir_offset >> 16);
|
|
|
|
case 1: v[0] = (uint16) (dir->tdir_offset & 0xffff);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (1);
|
|
|
|
} else
|
|
|
|
return (TIFFFetchData(tif, dir, (char *)v) != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Fetch a pair of SHORT or BYTE values. Some tags may have either BYTE
|
|
|
|
* or SHORT type and this function works with both ones.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchShortPair(TIFF* tif, TIFFDirEntry* dir)
|
|
|
|
{
|
2010-07-16 14:54:53 +02:00
|
|
|
/*
|
|
|
|
* Prevent overflowing the v stack arrays below by performing a sanity
|
|
|
|
* check on tdir_count, this should never be greater than two.
|
|
|
|
*/
|
|
|
|
if (dir->tdir_count > 2) {
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"unexpected count for field \"%s\", %u, expected 2; ignored",
|
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name,
|
|
|
|
dir->tdir_count);
|
|
|
|
return 0;
|
|
|
|
}
|
2010-05-11 19:44:00 +02:00
|
|
|
|
|
|
|
switch (dir->tdir_type) {
|
2010-07-16 14:54:53 +02:00
|
|
|
case TIFF_BYTE:
|
|
|
|
case TIFF_SBYTE:
|
|
|
|
{
|
2011-08-12 00:27:18 +02:00
|
|
|
uint8 v[4]={0,0,0,0};
|
2010-07-16 14:54:53 +02:00
|
|
|
return TIFFFetchByteArray(tif, dir, v)
|
|
|
|
&& TIFFSetField(tif, dir->tdir_tag, v[0], v[1]);
|
|
|
|
}
|
|
|
|
case TIFF_SHORT:
|
|
|
|
case TIFF_SSHORT:
|
|
|
|
{
|
2011-08-12 00:27:18 +02:00
|
|
|
uint16 v[2]={0,0};
|
2010-07-16 14:54:53 +02:00
|
|
|
return TIFFFetchShortArray(tif, dir, v)
|
|
|
|
&& TIFFSetField(tif, dir->tdir_tag, v[0], v[1]);
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return 0;
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch an array of LONG or SLONG values.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchLongArray(TIFF* tif, TIFFDirEntry* dir, uint32* v)
|
|
|
|
{
|
|
|
|
if (dir->tdir_count == 1) {
|
|
|
|
v[0] = dir->tdir_offset;
|
|
|
|
return (1);
|
|
|
|
} else
|
|
|
|
return (TIFFFetchData(tif, dir, (char*) v) != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch an array of RATIONAL or SRATIONAL values.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchRationalArray(TIFF* tif, TIFFDirEntry* dir, float* v)
|
|
|
|
{
|
|
|
|
int ok = 0;
|
|
|
|
uint32* l;
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
l = (uint32*)_TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dir->tdir_count, TIFFDataWidth((TIFFDataType) dir->tdir_type),
|
|
|
|
"to fetch array of rationals");
|
|
|
|
if (l) {
|
|
|
|
if (TIFFFetchData(tif, dir, (char *)l)) {
|
|
|
|
uint32 i;
|
|
|
|
for (i = 0; i < dir->tdir_count; i++) {
|
|
|
|
ok = cvtRational(tif, dir,
|
|
|
|
l[2*i+0], l[2*i+1], &v[i]);
|
|
|
|
if (!ok)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_TIFFfree((char *)l);
|
|
|
|
}
|
|
|
|
return (ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch an array of FLOAT values.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchFloatArray(TIFF* tif, TIFFDirEntry* dir, float* v)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (dir->tdir_count == 1) {
|
2010-07-16 14:54:53 +02:00
|
|
|
union
|
|
|
|
{
|
|
|
|
float f;
|
|
|
|
uint32 i;
|
|
|
|
} float_union;
|
|
|
|
|
|
|
|
float_union.i=dir->tdir_offset;
|
|
|
|
v[0]=float_union.f;
|
2010-05-11 19:44:00 +02:00
|
|
|
TIFFCvtIEEEFloatToNative(tif, dir->tdir_count, v);
|
|
|
|
return (1);
|
|
|
|
} else if (TIFFFetchData(tif, dir, (char*) v)) {
|
|
|
|
TIFFCvtIEEEFloatToNative(tif, dir->tdir_count, v);
|
|
|
|
return (1);
|
|
|
|
} else
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch an array of DOUBLE values.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchDoubleArray(TIFF* tif, TIFFDirEntry* dir, double* v)
|
|
|
|
{
|
|
|
|
if (TIFFFetchData(tif, dir, (char*) v)) {
|
|
|
|
TIFFCvtIEEEDoubleToNative(tif, dir->tdir_count, v);
|
|
|
|
return (1);
|
|
|
|
} else
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Fetch an array of ANY values. The actual values are returned as doubles
|
|
|
|
* which should be able hold all the types. Yes, there really should be an
|
|
|
|
* tany_t to avoid this potential non-portability ... Note in particular that
|
|
|
|
* we assume that the double return value vector is large enough to read in
|
|
|
|
* any fundamental type. We use that vector as a buffer to read in the base
|
|
|
|
* type vector and then convert it in place to double (from end to front of
|
|
|
|
* course).
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchAnyArray(TIFF* tif, TIFFDirEntry* dir, double* v)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
switch (dir->tdir_type) {
|
|
|
|
case TIFF_BYTE:
|
|
|
|
case TIFF_SBYTE:
|
2010-07-16 14:54:53 +02:00
|
|
|
if (!TIFFFetchByteArray(tif, dir, (uint8*) v))
|
2010-05-11 19:44:00 +02:00
|
|
|
return (0);
|
|
|
|
if (dir->tdir_type == TIFF_BYTE) {
|
2010-07-16 14:54:53 +02:00
|
|
|
uint8* vp = (uint8*) v;
|
2010-05-11 19:44:00 +02:00
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
} else {
|
2010-07-16 14:54:53 +02:00
|
|
|
int8* vp = (int8*) v;
|
2010-05-11 19:44:00 +02:00
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_SHORT:
|
|
|
|
case TIFF_SSHORT:
|
|
|
|
if (!TIFFFetchShortArray(tif, dir, (uint16*) v))
|
|
|
|
return (0);
|
|
|
|
if (dir->tdir_type == TIFF_SHORT) {
|
|
|
|
uint16* vp = (uint16*) v;
|
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
} else {
|
|
|
|
int16* vp = (int16*) v;
|
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_LONG:
|
|
|
|
case TIFF_SLONG:
|
|
|
|
if (!TIFFFetchLongArray(tif, dir, (uint32*) v))
|
|
|
|
return (0);
|
|
|
|
if (dir->tdir_type == TIFF_LONG) {
|
|
|
|
uint32* vp = (uint32*) v;
|
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
} else {
|
|
|
|
int32* vp = (int32*) v;
|
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_RATIONAL:
|
|
|
|
case TIFF_SRATIONAL:
|
|
|
|
if (!TIFFFetchRationalArray(tif, dir, (float*) v))
|
|
|
|
return (0);
|
|
|
|
{ float* vp = (float*) v;
|
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_FLOAT:
|
|
|
|
if (!TIFFFetchFloatArray(tif, dir, (float*) v))
|
|
|
|
return (0);
|
|
|
|
{ float* vp = (float*) v;
|
|
|
|
for (i = dir->tdir_count-1; i >= 0; i--)
|
|
|
|
v[i] = vp[i];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_DOUBLE:
|
|
|
|
return (TIFFFetchDoubleArray(tif, dir, (double*) v));
|
|
|
|
default:
|
|
|
|
/* TIFF_NOTYPE */
|
|
|
|
/* TIFF_ASCII */
|
|
|
|
/* TIFF_UNDEFINED */
|
2010-07-16 14:54:53 +02:00
|
|
|
TIFFErrorExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"cannot read TIFF_ANY type %d for field \"%s\"",
|
|
|
|
dir->tdir_type,
|
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name);
|
2010-05-11 19:44:00 +02:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch a tag that is not handled by special case code.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchNormalTag(TIFF* tif, TIFFDirEntry* dp)
|
|
|
|
{
|
|
|
|
static const char mesg[] = "to fetch tag value";
|
|
|
|
int ok = 0;
|
|
|
|
const TIFFFieldInfo* fip = _TIFFFieldWithTag(tif, dp->tdir_tag);
|
|
|
|
|
|
|
|
if (dp->tdir_count > 1) { /* array of values */
|
|
|
|
char* cp = NULL;
|
|
|
|
|
|
|
|
switch (dp->tdir_type) {
|
|
|
|
case TIFF_BYTE:
|
|
|
|
case TIFF_SBYTE:
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif,
|
|
|
|
dp->tdir_count, sizeof (uint8), mesg);
|
|
|
|
ok = cp && TIFFFetchByteArray(tif, dp, (uint8*) cp);
|
2010-05-11 19:44:00 +02:00
|
|
|
break;
|
|
|
|
case TIFF_SHORT:
|
|
|
|
case TIFF_SSHORT:
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dp->tdir_count, sizeof (uint16), mesg);
|
|
|
|
ok = cp && TIFFFetchShortArray(tif, dp, (uint16*) cp);
|
|
|
|
break;
|
|
|
|
case TIFF_LONG:
|
|
|
|
case TIFF_SLONG:
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dp->tdir_count, sizeof (uint32), mesg);
|
|
|
|
ok = cp && TIFFFetchLongArray(tif, dp, (uint32*) cp);
|
|
|
|
break;
|
|
|
|
case TIFF_RATIONAL:
|
|
|
|
case TIFF_SRATIONAL:
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dp->tdir_count, sizeof (float), mesg);
|
|
|
|
ok = cp && TIFFFetchRationalArray(tif, dp, (float*) cp);
|
|
|
|
break;
|
|
|
|
case TIFF_FLOAT:
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dp->tdir_count, sizeof (float), mesg);
|
|
|
|
ok = cp && TIFFFetchFloatArray(tif, dp, (float*) cp);
|
|
|
|
break;
|
|
|
|
case TIFF_DOUBLE:
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dp->tdir_count, sizeof (double), mesg);
|
|
|
|
ok = cp && TIFFFetchDoubleArray(tif, dp, (double*) cp);
|
|
|
|
break;
|
|
|
|
case TIFF_ASCII:
|
|
|
|
case TIFF_UNDEFINED: /* bit of a cheat... */
|
|
|
|
/*
|
|
|
|
* Some vendors write strings w/o the trailing
|
|
|
|
* NULL byte, so always append one just in case.
|
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif, dp->tdir_count + 1,
|
|
|
|
1, mesg);
|
2010-05-11 19:44:00 +02:00
|
|
|
if( (ok = (cp && TIFFFetchString(tif, dp, cp))) != 0 )
|
|
|
|
cp[dp->tdir_count] = '\0'; /* XXX */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (ok) {
|
|
|
|
ok = (fip->field_passcount ?
|
|
|
|
TIFFSetField(tif, dp->tdir_tag, dp->tdir_count, cp)
|
|
|
|
: TIFFSetField(tif, dp->tdir_tag, cp));
|
|
|
|
}
|
|
|
|
if (cp != NULL)
|
|
|
|
_TIFFfree(cp);
|
|
|
|
} else if (CheckDirCount(tif, dp, 1)) { /* singleton value */
|
|
|
|
switch (dp->tdir_type) {
|
|
|
|
case TIFF_BYTE:
|
|
|
|
case TIFF_SBYTE:
|
|
|
|
case TIFF_SHORT:
|
|
|
|
case TIFF_SSHORT:
|
|
|
|
/*
|
|
|
|
* If the tag is also acceptable as a LONG or SLONG
|
|
|
|
* then TIFFSetField will expect an uint32 parameter
|
|
|
|
* passed to it (through varargs). Thus, for machines
|
|
|
|
* where sizeof (int) != sizeof (uint32) we must do
|
|
|
|
* a careful check here. It's hard to say if this
|
|
|
|
* is worth optimizing.
|
|
|
|
*
|
|
|
|
* NB: We use TIFFFieldWithTag here knowing that
|
|
|
|
* it returns us the first entry in the table
|
|
|
|
* for the tag and that that entry is for the
|
|
|
|
* widest potential data type the tag may have.
|
|
|
|
*/
|
|
|
|
{ TIFFDataType type = fip->field_type;
|
|
|
|
if (type != TIFF_LONG && type != TIFF_SLONG) {
|
|
|
|
uint16 v = (uint16)
|
|
|
|
TIFFExtractData(tif, dp->tdir_type, dp->tdir_offset);
|
|
|
|
ok = (fip->field_passcount ?
|
|
|
|
TIFFSetField(tif, dp->tdir_tag, 1, &v)
|
|
|
|
: TIFFSetField(tif, dp->tdir_tag, v));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* fall thru... */
|
|
|
|
case TIFF_LONG:
|
|
|
|
case TIFF_SLONG:
|
|
|
|
{ uint32 v32 =
|
|
|
|
TIFFExtractData(tif, dp->tdir_type, dp->tdir_offset);
|
|
|
|
ok = (fip->field_passcount ?
|
|
|
|
TIFFSetField(tif, dp->tdir_tag, 1, &v32)
|
|
|
|
: TIFFSetField(tif, dp->tdir_tag, v32));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_RATIONAL:
|
|
|
|
case TIFF_SRATIONAL:
|
|
|
|
case TIFF_FLOAT:
|
|
|
|
{ float v = (dp->tdir_type == TIFF_FLOAT ?
|
|
|
|
TIFFFetchFloat(tif, dp)
|
|
|
|
: TIFFFetchRational(tif, dp));
|
|
|
|
ok = (fip->field_passcount ?
|
|
|
|
TIFFSetField(tif, dp->tdir_tag, 1, &v)
|
|
|
|
: TIFFSetField(tif, dp->tdir_tag, v));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_DOUBLE:
|
|
|
|
{ double v;
|
|
|
|
ok = (TIFFFetchDoubleArray(tif, dp, &v) &&
|
|
|
|
(fip->field_passcount ?
|
|
|
|
TIFFSetField(tif, dp->tdir_tag, 1, &v)
|
|
|
|
: TIFFSetField(tif, dp->tdir_tag, v))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TIFF_ASCII:
|
|
|
|
case TIFF_UNDEFINED: /* bit of a cheat... */
|
|
|
|
{ char c[2];
|
|
|
|
if( (ok = (TIFFFetchString(tif, dp, c) != 0)) != 0 ) {
|
|
|
|
c[1] = '\0'; /* XXX paranoid */
|
|
|
|
ok = (fip->field_passcount ?
|
|
|
|
TIFFSetField(tif, dp->tdir_tag, 1, c)
|
|
|
|
: TIFFSetField(tif, dp->tdir_tag, c));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define NITEMS(x) (sizeof (x) / sizeof (x[0]))
|
|
|
|
/*
|
|
|
|
* Fetch samples/pixel short values for
|
|
|
|
* the specified tag and verify that
|
|
|
|
* all values are the same.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchPerSampleShorts(TIFF* tif, TIFFDirEntry* dir, uint16* pl)
|
|
|
|
{
|
2010-07-16 14:54:53 +02:00
|
|
|
uint16 samples = tif->tif_dir.td_samplesperpixel;
|
|
|
|
int status = 0;
|
|
|
|
|
|
|
|
if (CheckDirCount(tif, dir, (uint32) samples)) {
|
|
|
|
uint16 buf[10];
|
|
|
|
uint16* v = buf;
|
|
|
|
|
|
|
|
if (dir->tdir_count > NITEMS(buf))
|
|
|
|
v = (uint16*) _TIFFCheckMalloc(tif, dir->tdir_count, sizeof(uint16),
|
|
|
|
"to fetch per-sample values");
|
|
|
|
if (v && TIFFFetchShortArray(tif, dir, v)) {
|
|
|
|
uint16 i;
|
|
|
|
int check_count = dir->tdir_count;
|
|
|
|
if( samples < check_count )
|
|
|
|
check_count = samples;
|
|
|
|
|
|
|
|
for (i = 1; i < check_count; i++)
|
|
|
|
if (v[i] != v[0]) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"Cannot handle different per-sample values for field \"%s\"",
|
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name);
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
*pl = v[0];
|
|
|
|
status = 1;
|
|
|
|
}
|
|
|
|
bad:
|
|
|
|
if (v && v != buf)
|
|
|
|
_TIFFfree(v);
|
|
|
|
}
|
|
|
|
return (status);
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch samples/pixel long values for
|
|
|
|
* the specified tag and verify that
|
|
|
|
* all values are the same.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchPerSampleLongs(TIFF* tif, TIFFDirEntry* dir, uint32* pl)
|
|
|
|
{
|
2010-07-16 14:54:53 +02:00
|
|
|
uint16 samples = tif->tif_dir.td_samplesperpixel;
|
|
|
|
int status = 0;
|
|
|
|
|
|
|
|
if (CheckDirCount(tif, dir, (uint32) samples)) {
|
|
|
|
uint32 buf[10];
|
|
|
|
uint32* v = buf;
|
|
|
|
|
|
|
|
if (dir->tdir_count > NITEMS(buf))
|
|
|
|
v = (uint32*) _TIFFCheckMalloc(tif, dir->tdir_count, sizeof(uint32),
|
|
|
|
"to fetch per-sample values");
|
|
|
|
if (v && TIFFFetchLongArray(tif, dir, v)) {
|
|
|
|
uint16 i;
|
|
|
|
int check_count = dir->tdir_count;
|
|
|
|
|
|
|
|
if( samples < check_count )
|
|
|
|
check_count = samples;
|
|
|
|
for (i = 1; i < check_count; i++)
|
|
|
|
if (v[i] != v[0]) {
|
|
|
|
TIFFErrorExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"Cannot handle different per-sample values for field \"%s\"",
|
|
|
|
_TIFFFieldWithTag(tif, dir->tdir_tag)->field_name);
|
|
|
|
goto bad;
|
|
|
|
}
|
|
|
|
*pl = v[0];
|
|
|
|
status = 1;
|
|
|
|
}
|
|
|
|
bad:
|
|
|
|
if (v && v != buf)
|
|
|
|
_TIFFfree(v);
|
|
|
|
}
|
|
|
|
return (status);
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2011-08-03 09:51:40 +02:00
|
|
|
* Fetch samples/pixel ANY values for the specified tag and returns their min
|
|
|
|
* and max.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
static int
|
2011-08-03 09:51:40 +02:00
|
|
|
TIFFFetchPerSampleAnys(TIFF* tif, TIFFDirEntry* dir, double* minv, double* maxv)
|
2010-05-11 19:44:00 +02:00
|
|
|
{
|
2010-07-16 14:54:53 +02:00
|
|
|
uint16 samples = tif->tif_dir.td_samplesperpixel;
|
|
|
|
int status = 0;
|
|
|
|
|
|
|
|
if (CheckDirCount(tif, dir, (uint32) samples)) {
|
|
|
|
double buf[10];
|
|
|
|
double* v = buf;
|
|
|
|
|
|
|
|
if (dir->tdir_count > NITEMS(buf))
|
|
|
|
v = (double*) _TIFFCheckMalloc(tif, dir->tdir_count, sizeof (double),
|
|
|
|
"to fetch per-sample values");
|
|
|
|
if (v && TIFFFetchAnyArray(tif, dir, v)) {
|
|
|
|
uint16 i;
|
|
|
|
int check_count = dir->tdir_count;
|
|
|
|
if( samples < check_count )
|
|
|
|
check_count = samples;
|
|
|
|
|
2011-08-03 09:51:40 +02:00
|
|
|
*minv = *maxv = v[0];
|
2010-07-16 14:54:53 +02:00
|
|
|
for (i = 1; i < check_count; i++)
|
2011-08-03 09:51:40 +02:00
|
|
|
{
|
|
|
|
if (v[i] < *minv)
|
|
|
|
*minv = v[i];
|
|
|
|
if (v[i] > *maxv)
|
|
|
|
*maxv = v[i];
|
|
|
|
}
|
2010-07-16 14:54:53 +02:00
|
|
|
status = 1;
|
|
|
|
}
|
|
|
|
if (v && v != buf)
|
|
|
|
_TIFFfree(v);
|
|
|
|
}
|
|
|
|
return (status);
|
2010-05-11 19:44:00 +02:00
|
|
|
}
|
|
|
|
#undef NITEMS
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch a set of offsets or lengths.
|
|
|
|
* While this routine says "strips", in fact it's also used for tiles.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchStripThing(TIFF* tif, TIFFDirEntry* dir, long nstrips, uint32** lpp)
|
|
|
|
{
|
|
|
|
register uint32* lp;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
CheckDirCount(tif, dir, (uint32) nstrips);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate space for strip information.
|
|
|
|
*/
|
|
|
|
if (*lpp == NULL &&
|
2010-07-16 14:54:53 +02:00
|
|
|
(*lpp = (uint32 *)_TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
nstrips, sizeof (uint32), "for strip array")) == NULL)
|
|
|
|
return (0);
|
|
|
|
lp = *lpp;
|
|
|
|
_TIFFmemset( lp, 0, sizeof(uint32) * nstrips );
|
|
|
|
|
|
|
|
if (dir->tdir_type == (int)TIFF_SHORT) {
|
|
|
|
/*
|
|
|
|
* Handle uint16->uint32 expansion.
|
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
uint16* dp = (uint16*) _TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dir->tdir_count, sizeof (uint16), "to fetch strip tag");
|
|
|
|
if (dp == NULL)
|
|
|
|
return (0);
|
|
|
|
if( (status = TIFFFetchShortArray(tif, dir, dp)) != 0 ) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for( i = 0; i < nstrips && i < (int) dir->tdir_count; i++ )
|
|
|
|
{
|
|
|
|
lp[i] = dp[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_TIFFfree((char*) dp);
|
|
|
|
|
|
|
|
} else if( nstrips != (int) dir->tdir_count ) {
|
|
|
|
/* Special case to correct length */
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
uint32* dp = (uint32*) _TIFFCheckMalloc(tif,
|
2010-05-11 19:44:00 +02:00
|
|
|
dir->tdir_count, sizeof (uint32), "to fetch strip tag");
|
|
|
|
if (dp == NULL)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
status = TIFFFetchLongArray(tif, dir, dp);
|
|
|
|
if( status != 0 ) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for( i = 0; i < nstrips && i < (int) dir->tdir_count; i++ )
|
|
|
|
{
|
|
|
|
lp[i] = dp[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_TIFFfree( (char *) dp );
|
|
|
|
} else
|
|
|
|
status = TIFFFetchLongArray(tif, dir, lp);
|
|
|
|
|
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fetch and set the RefBlackWhite tag.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchRefBlackWhite(TIFF* tif, TIFFDirEntry* dir)
|
|
|
|
{
|
|
|
|
static const char mesg[] = "for \"ReferenceBlackWhite\" array";
|
|
|
|
char* cp;
|
|
|
|
int ok;
|
|
|
|
|
|
|
|
if (dir->tdir_type == TIFF_RATIONAL)
|
|
|
|
return (TIFFFetchNormalTag(tif, dir));
|
|
|
|
/*
|
|
|
|
* Handle LONG's for backward compatibility.
|
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
cp = (char *)_TIFFCheckMalloc(tif, dir->tdir_count,
|
|
|
|
sizeof (uint32), mesg);
|
2010-05-11 19:44:00 +02:00
|
|
|
if( (ok = (cp && TIFFFetchLongArray(tif, dir, (uint32*) cp))) != 0) {
|
|
|
|
float* fp = (float*)
|
2010-07-16 14:54:53 +02:00
|
|
|
_TIFFCheckMalloc(tif, dir->tdir_count, sizeof (float), mesg);
|
2010-05-11 19:44:00 +02:00
|
|
|
if( (ok = (fp != NULL)) != 0 ) {
|
|
|
|
uint32 i;
|
|
|
|
for (i = 0; i < dir->tdir_count; i++)
|
|
|
|
fp[i] = (float)((uint32*) cp)[i];
|
|
|
|
ok = TIFFSetField(tif, dir->tdir_tag, fp);
|
|
|
|
_TIFFfree((char*) fp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cp)
|
|
|
|
_TIFFfree(cp);
|
|
|
|
return (ok);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Fetch and set the SubjectDistance EXIF tag.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
TIFFFetchSubjectDistance(TIFF* tif, TIFFDirEntry* dir)
|
|
|
|
{
|
|
|
|
uint32 l[2];
|
|
|
|
float v;
|
|
|
|
int ok = 0;
|
|
|
|
|
|
|
|
if( dir->tdir_count != 1 || dir->tdir_type != TIFF_RATIONAL )
|
|
|
|
{
|
|
|
|
TIFFWarningExt(tif->tif_clientdata, tif->tif_name,
|
|
|
|
"incorrect count or type for SubjectDistance, tag ignored" );
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (TIFFFetchData(tif, dir, (char *)l)
|
|
|
|
&& cvtRational(tif, dir, l[0], l[1], &v)) {
|
|
|
|
/*
|
|
|
|
* XXX: Numerator 0xFFFFFFFF means that we have infinite
|
|
|
|
* distance. Indicate that with a negative floating point
|
|
|
|
* SubjectDistance value.
|
|
|
|
*/
|
|
|
|
ok = TIFFSetField(tif, dir->tdir_tag,
|
|
|
|
(l[0] != 0xFFFFFFFF) ? v : -v);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Replace a single strip (tile) of uncompressed data by multiple strips
|
|
|
|
* (tiles), each approximately STRIP_SIZE_DEFAULT bytes. This is useful for
|
|
|
|
* dealing with large images or for dealing with machines with a limited
|
|
|
|
* amount memory.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
static void
|
|
|
|
ChopUpSingleUncompressedStrip(TIFF* tif)
|
|
|
|
{
|
|
|
|
register TIFFDirectory *td = &tif->tif_dir;
|
|
|
|
uint32 bytecount = td->td_stripbytecount[0];
|
|
|
|
uint32 offset = td->td_stripoffset[0];
|
|
|
|
tsize_t rowbytes = TIFFVTileSize(tif, 1), stripbytes;
|
|
|
|
tstrip_t strip, nstrips, rowsperstrip;
|
|
|
|
uint32* newcounts;
|
|
|
|
uint32* newoffsets;
|
|
|
|
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Make the rows hold at least one scanline, but fill specified amount
|
|
|
|
* of data if possible.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
2010-07-16 14:54:53 +02:00
|
|
|
if (rowbytes > STRIP_SIZE_DEFAULT) {
|
2010-05-11 19:44:00 +02:00
|
|
|
stripbytes = rowbytes;
|
|
|
|
rowsperstrip = 1;
|
|
|
|
} else if (rowbytes > 0 ) {
|
2010-07-16 14:54:53 +02:00
|
|
|
rowsperstrip = STRIP_SIZE_DEFAULT / rowbytes;
|
2010-05-11 19:44:00 +02:00
|
|
|
stripbytes = rowbytes * rowsperstrip;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return;
|
|
|
|
|
2010-07-16 14:54:53 +02:00
|
|
|
/*
|
|
|
|
* never increase the number of strips in an image
|
|
|
|
*/
|
2010-05-11 19:44:00 +02:00
|
|
|
if (rowsperstrip >= td->td_rowsperstrip)
|
|
|
|
return;
|
|
|
|
nstrips = (tstrip_t) TIFFhowmany(bytecount, stripbytes);
|
2010-07-16 14:54:53 +02:00
|
|
|
if( nstrips == 0 ) /* something is wonky, do nothing. */
|
|
|
|
return;
|
|
|
|
|
|
|
|
newcounts = (uint32*) _TIFFCheckMalloc(tif, nstrips, sizeof (uint32),
|
2010-05-11 19:44:00 +02:00
|
|
|
"for chopped \"StripByteCounts\" array");
|
2010-07-16 14:54:53 +02:00
|
|
|
newoffsets = (uint32*) _TIFFCheckMalloc(tif, nstrips, sizeof (uint32),
|
2010-05-11 19:44:00 +02:00
|
|
|
"for chopped \"StripOffsets\" array");
|
|
|
|
if (newcounts == NULL || newoffsets == NULL) {
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Unable to allocate new strip information, give up and use
|
|
|
|
* the original one strip information.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
if (newcounts != NULL)
|
|
|
|
_TIFFfree(newcounts);
|
|
|
|
if (newoffsets != NULL)
|
|
|
|
_TIFFfree(newoffsets);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/*
|
2010-07-16 14:54:53 +02:00
|
|
|
* Fill the strip information arrays with new bytecounts and offsets
|
|
|
|
* that reflect the broken-up format.
|
2010-05-11 19:44:00 +02:00
|
|
|
*/
|
|
|
|
for (strip = 0; strip < nstrips; strip++) {
|
2010-07-16 14:54:53 +02:00
|
|
|
if ((uint32)stripbytes > bytecount)
|
2010-05-11 19:44:00 +02:00
|
|
|
stripbytes = bytecount;
|
|
|
|
newcounts[strip] = stripbytes;
|
|
|
|
newoffsets[strip] = offset;
|
|
|
|
offset += stripbytes;
|
|
|
|
bytecount -= stripbytes;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Replace old single strip info with multi-strip info.
|
|
|
|
*/
|
|
|
|
td->td_stripsperimage = td->td_nstrips = nstrips;
|
|
|
|
TIFFSetField(tif, TIFFTAG_ROWSPERSTRIP, rowsperstrip);
|
|
|
|
|
|
|
|
_TIFFfree(td->td_stripbytecount);
|
|
|
|
_TIFFfree(td->td_stripoffset);
|
|
|
|
td->td_stripbytecount = newcounts;
|
|
|
|
td->td_stripoffset = newoffsets;
|
|
|
|
td->td_stripbytecountsorted = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vim: set ts=8 sts=8 sw=8 noet: */
|
2010-07-16 14:54:53 +02:00
|
|
|
/*
|
|
|
|
* Local Variables:
|
|
|
|
* mode: c
|
|
|
|
* c-basic-offset: 8
|
|
|
|
* fill-column: 78
|
|
|
|
* End:
|
|
|
|
*/
|