home *** CD-ROM | disk | FTP | other *** search
- #ifndef lint
- static char rcsid[] = "$Header: /usr/people/sam/tiff/libtiff/RCS/tif_dirread.c,v 1.20 92/03/25 09:58:09 sam Exp $";
- #endif
-
- /*
- * Copyright (c) 1988, 1989, 1990, 1991, 1992 Sam Leffler
- * Copyright (c) 1991, 1992 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.
- *
- * NB: Beware of the varargs declarations for routines in
- * this file. The names and types of variables has been
- * carefully chosen to make things work with compilers that
- * are busted in one way or another (e.g. SGI/MIPS).
- */
- #include "tiffioP.h"
-
- #define IGNORE 0 /* tag placeholder used below */
-
- #if HAVE_IEEEFP
- #define TIFFCvtIEEEFloatToNative(tif, n, fp)
- #endif
-
- #include "prototypes.h"
- #if USE_PROTOTYPES
- static EstimateStripByteCounts(TIFF *, TIFFDirEntry *, u_int);
- static MissingRequired(TIFF *, char *);
- static CheckDirCount(TIFF *, TIFFDirEntry *, u_long);
- static TIFFFetchData(TIFF *, TIFFDirEntry *, char *);
- static TIFFFetchString(TIFF *, TIFFDirEntry *, char *);
- static float TIFFFetchRational(TIFF *, TIFFDirEntry *);
- static TIFFFetchNormalTag(TIFF *, TIFFDirEntry *);
- static TIFFFetchPerSampleShorts(TIFF *, TIFFDirEntry *, long *);
- static TIFFFetchShortArray(TIFF *, TIFFDirEntry *, u_short []);
- static TIFFFetchStripThing(TIFF *, TIFFDirEntry *, long, u_long **);
- static TIFFFetchRefBlackWhite(TIFF *, TIFFDirEntry *);
- static TIFFFetchJPEGQTables(TIFF *, TIFFDirEntry *);
- static TIFFFetchJPEGCTables(TIFF *, TIFFDirEntry *, u_char ***);
- static TIFFFetchExtraSamples(TIFF *, TIFFDirEntry *);
- static float TIFFFetchFloat(TIFF *, TIFFDirEntry *);
- static int TIFFFetchFloatArray(TIFF *, TIFFDirEntry *, float *);
- extern int TIFFSetCompressionScheme(TIFF *, int);
- extern int TIFFDefaultDirectory(TIFF*);
- extern int TIFFFreeDirectory(TIFF*);
- #else
- static EstimateStripByteCounts();
- static MissingRequired();
- static CheckDirCount();
- static TIFFFetchData();
- static TIFFFetchString();
- static float TIFFFetchRational();
- static TIFFFetchNormalTag();
- static TIFFFetchPerSampleShorts();
- static TIFFFetchShortArray();
- static TIFFFetchStripThing();
- static TIFFFetchRefBlackWhite();
- static TIFFFetchJPEGQTables();
- static TIFFFetchJPEGCTables();
- static TIFFFetchExtraSamples();
- static float TIFFFetchFloat();
- static int TIFFFetchFloatArray();
- extern int TIFFSetCompressionScheme();
- extern int TIFFDefaultDirectory();
- extern int TIFFFreeDirectory();
- #endif
-
- static char *
- CheckMalloc(tif, n, what)
- TIFF *tif;
- int n;
- char *what;
- {
- char *cp = malloc(n);
- if (cp == NULL)
- TIFFError(tif->tif_name, "No space %s", what);
- return (cp);
- }
-
- /*
- * Read the next TIFF directory from a file
- * and convert it to the internal format.
- * We read directories sequentially.
- */
- TIFFReadDirectory(tif)
- TIFF *tif;
- {
- register TIFFDirEntry *dp;
- register int n;
- register TIFFDirectory *td;
- TIFFDirEntry *dir;
- long v;
- TIFFFieldInfo *fip;
- u_short dircount;
- char *cp;
- int diroutoforderwarning = 0;
-
- tif->tif_diroff = tif->tif_nextdiroff;
- if (tif->tif_diroff == 0) /* no more directories */
- return (0);
- tif->tif_curdir++;
- if (!isMapped(tif)) {
- if (!SeekOK(tif->tif_fd, tif->tif_diroff)) {
- TIFFError(tif->tif_name,
- "Seek error accessing TIFF directory");
- return (0);
- }
- if (!ReadOK(tif->tif_fd, &dircount, sizeof (short))) {
- TIFFError(tif->tif_name,
- "Can not read TIFF directory count");
- return (0);
- }
- if (tif->tif_flags & TIFF_SWAB)
- TIFFSwabShort(&dircount);
- dir = (TIFFDirEntry *)CheckMalloc(tif,
- dircount * sizeof (TIFFDirEntry), "to read TIFF directory");
- if (dir == NULL)
- return (0);
- if (!ReadOK(tif->tif_fd, dir, dircount*sizeof (TIFFDirEntry))) {
- TIFFError(tif->tif_name, "Can not read TIFF directory");
- goto bad;
- }
- /*
- * Read offset to next directory for sequential scans.
- */
- if (!ReadOK(tif->tif_fd, &tif->tif_nextdiroff, sizeof (long)))
- tif->tif_nextdiroff = 0;
- #ifdef MMAP_SUPPORT
- } else {
- off_t off = tif->tif_diroff;
-
- if (off + sizeof (short) > tif->tif_size) {
- TIFFError(tif->tif_name,
- "Can not read TIFF directory count");
- return (0);
- } else
- bcopy(tif->tif_base + off, &dircount, sizeof (short));
- off += sizeof (short);
- if (tif->tif_flags & TIFF_SWAB)
- TIFFSwabShort(&dircount);
- dir = (TIFFDirEntry *)CheckMalloc(tif,
- dircount * sizeof (TIFFDirEntry), "to read TIFF directory");
- if (dir == NULL)
- return (0);
- if (off + dircount*sizeof (TIFFDirEntry) > tif->tif_size) {
- TIFFError(tif->tif_name, "Can not read TIFF directory");
- goto bad;
- } else
- bcopy(tif->tif_base + off, dir,
- dircount*sizeof (TIFFDirEntry));
- off += dircount* sizeof (TIFFDirEntry);
- if (off + sizeof (long) < tif->tif_size)
- bcopy(tif->tif_base + off, &tif->tif_nextdiroff,
- sizeof (long));
- else
- tif->tif_nextdiroff = 0;
- #endif
- }
- if (tif->tif_flags & TIFF_SWAB)
- TIFFSwabLong((u_long *)&tif->tif_nextdiroff);
-
- 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);
- for (fip = tiffFieldInfo, dp = dir, n = dircount; n > 0; n--, dp++) {
- if (tif->tif_flags & TIFF_SWAB) {
- TIFFSwabArrayOfShort(&dp->tdir_tag, 2);
- TIFFSwabArrayOfLong(&dp->tdir_count, 2);
- }
- /*
- * Find the field information entry for this tag.
- */
- /*
- * Silicon Beach (at least) writes unordered
- * directory tags (violating the spec). Handle
- * it here, but be obnoxious (maybe they'll fix it?).
- */
- if (dp->tdir_tag < fip->field_tag) {
- if (!diroutoforderwarning) {
- TIFFWarning(tif->tif_name,
- "invalid TIFF directory; tags are not sorted in ascending order");
- diroutoforderwarning = 1;
- }
- fip = tiffFieldInfo; /* O(n^2) */
- }
- while (fip->field_tag && fip->field_tag < dp->tdir_tag)
- fip++;
- if (!fip->field_tag || fip->field_tag != dp->tdir_tag) {
- TIFFWarning(tif->tif_name,
- "unknown field with tag %d (0x%x) ignored",
- dp->tdir_tag, dp->tdir_tag);
- dp->tdir_tag = IGNORE;
- fip = tiffFieldInfo; /* restart search */
- continue;
- }
- /*
- * Null out old tags that we ignore.
- */
- if (fip->field_bit == FIELD_IGNORE) {
- ignore:
- dp->tdir_tag = IGNORE;
- continue;
- }
- /*
- * Check data type.
- */
- while (dp->tdir_type != (u_short)fip->field_type) {
- if (fip->field_type == TIFF_ANY) /* wildcard */
- break;
- fip++;
- if (!fip->field_tag || fip->field_tag != dp->tdir_tag) {
- TIFFWarning(tif->tif_name,
- "wrong data type %d for \"%s\"; tag ignored",
- dp->tdir_type, fip[-1].field_name);
- goto ignore;
- }
- }
- /*
- * Check count if known in advance.
- */
- if (fip->field_readcount != TIFF_VARIABLE) {
- u_long expected = (fip->field_readcount == TIFF_SPP) ?
- (u_long) td->td_samplesperpixel :
- (u_long) fip->field_readcount;
- if (!CheckDirCount(tif, dp, expected))
- goto ignore;
- }
-
- switch (dp->tdir_tag) {
- 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_SAMPLESPERPIXEL:
- case TIFFTAG_ROWSPERSTRIP:
- if (!TIFFFetchNormalTag(tif, dp))
- goto bad;
- break;
- }
- }
-
- /*
- * Allocate directory structure and setup defaults.
- */
- if (!TIFFFieldSet(tif, FIELD_IMAGEDIMENSIONS)) {
- MissingRequired(tif, "ImageLength");
- goto bad;
- }
- if (!TIFFFieldSet(tif, FIELD_PLANARCONFIG)) {
- MissingRequired(tif, "PlanarConfiguration");
- goto bad;
- }
- /*
- * Setup appropriate structures (by strip or by tile)
- */
- if (!TIFFFieldSet(tif, FIELD_TILEDIMENSIONS)) {
- td->td_stripsperimage = (td->td_rowsperstrip == 0xffffffff ?
- (td->td_imagelength != 0 ? 1 : 0) :
- howmany(td->td_imagelength, td->td_rowsperstrip));
- 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_stripsperimage = TIFFNumberOfTiles(tif);
- tif->tif_flags |= TIFF_ISTILED;
- }
- td->td_nstrips = td->td_stripsperimage;
- if (td->td_planarconfig == PLANARCONFIG_SEPARATE)
- td->td_nstrips *= td->td_samplesperpixel;
- if (td->td_nstrips > 0 && !TIFFFieldSet(tif, FIELD_STRIPOFFSETS)) {
- MissingRequired(tif,
- isTiled(tif) ? "TileOffsets" : "StripOffsets");
- goto bad;
- }
-
- /*
- * 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_COMPRESSION:
- case TIFFTAG_MINSAMPLEVALUE:
- case TIFFTAG_MAXSAMPLEVALUE:
- case TIFFTAG_BITSPERSAMPLE:
- /*
- * 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.
- *
- * The MinSampleValue, MaxSampleValue and
- * BitsPerSample 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).
- */
- if (dp->tdir_count == 1) {
- v = TIFFExtractData(tif,
- dp->tdir_type, dp->tdir_offset);
- if (!TIFFSetField(tif, dp->tdir_tag, (int)v))
- goto bad;
- break;
- }
- /* fall thru... */
- case TIFFTAG_DATATYPE:
- case TIFFTAG_SAMPLEFORMAT:
- if (!TIFFFetchPerSampleShorts(tif, dp, &v) ||
- !TIFFSetField(tif, dp->tdir_tag, (int)v))
- goto bad;
- 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_IMAGELENGTH:
- case TIFFTAG_ROWSPERSTRIP:
- case TIFFTAG_TILELENGTH:
- case TIFFTAG_TILEWIDTH:
- case TIFFTAG_TILEDEPTH:
- case TIFFTAG_SAMPLESPERPIXEL:
- case TIFFTAG_PLANARCONFIG:
- /* handled in first pass above */
- break;
- case TIFFTAG_COLORMAP:
- if (!CheckDirCount(tif,dp,3*(1L<<td->td_bitspersample)))
- break;
- /* fall thru... */
- case TIFFTAG_TRANSFERFUNCTION:
- v = (1L<<td->td_bitspersample) * sizeof (u_short);
- cp = CheckMalloc(tif,
- dp->tdir_count * sizeof (u_short),
- "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.
- */
- if (dp->tdir_count == 1L<<td->td_bitspersample)
- v = 0;
- /* NB: we assume samples/pixel <= 4 */
- TIFFSetField(tif, dp->tdir_tag,
- cp, cp+v, cp+2*v, cp+3*v);
- }
- free(cp);
- }
- break;
- case TIFFTAG_PAGENUMBER:
- if (TIFFFetchShortArray(tif, dp, td->td_pagenumber))
- TIFFSetFieldBit(tif, FIELD_PAGENUMBER);
- break;
- case TIFFTAG_HALFTONEHINTS:
- if (TIFFFetchShortArray(tif, dp, td->td_halftonehints))
- TIFFSetFieldBit(tif, FIELD_HALFTONEHINTS);
- break;
- #ifdef COLORIMETRY_SUPPORT
- case TIFFTAG_REFERENCEBLACKWHITE:
- (void) TIFFFetchRefBlackWhite(tif, dp);
- break;
- #endif
- #ifdef YCBCR_SUPPORT
- case TIFFTAG_YCBCRSUBSAMPLING:
- if (TIFFFetchShortArray(tif, dp, td->td_ycbcrsubsampling))
- TIFFSetFieldBit(tif, FIELD_YCBCRSUBSAMPLING);
- break;
- #endif
- #ifdef CMYK_SUPPORT
- case TIFFTAG_DOTRANGE:
- if (TIFFFetchShortArray(tif, dp, td->td_dotrange))
- TIFFSetFieldBit(tif, FIELD_DOTRANGE);
- break;
- #endif
- #ifdef JPEG_SUPPORT
- case TIFFTAG_JPEGQTABLES:
- if (TIFFFetchJPEGQTables(tif, dp))
- TIFFSetFieldBit(tif, FIELD_JPEGQTABLES);
- break;
- case TIFFTAG_JPEGDCTABLES:
- if (TIFFFetchJPEGCTables(tif, dp, &td->td_dctab))
- TIFFSetFieldBit(tif, FIELD_JPEGDCTABLES);
- break;
- case TIFFTAG_JPEGACTABLES:
- if (TIFFFetchJPEGCTables(tif, dp, &td->td_actab))
- TIFFSetFieldBit(tif, FIELD_JPEGACTABLES);
- break;
- #endif
- case TIFFTAG_EXTRASAMPLES:
- (void) TIFFFetchExtraSamples(tif, dp);
- break;
- /* BEGIN REV 4.0 COMPATIBILITY */
- case TIFFTAG_OSUBFILETYPE:
- v = 0;
- 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)
- (void) TIFFSetField(tif,
- TIFFTAG_SUBFILETYPE, (int)v);
- break;
- /* END REV 4.0 COMPATIBILITY */
- default:
- (void) TIFFFetchNormalTag(tif, dp);
- break;
- }
- }
- /*
- * Verify Palette image has a Colormap.
- */
- if (td->td_photometric == PHOTOMETRIC_PALETTE &&
- !TIFFFieldSet(tif, FIELD_COLORMAP)) {
- MissingRequired(tif, "Colormap");
- goto bad;
- }
- /*
- * Attempt to deal with a missing StripByteCounts tag.
- */
- 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_nstrips > 1) {
- MissingRequired(tif, "StripByteCounts");
- goto bad;
- }
- TIFFWarning(tif->tif_name,
- "TIFF directory is missing required \"%s\" field, calculating from imagelength",
- TIFFFieldWithTag(TIFFTAG_STRIPBYTECOUNTS)->field_name);
- EstimateStripByteCounts(tif, dir, dircount);
- } else if (td->td_nstrips == 1 && td->td_stripbytecount[0] == 0) {
- /*
- * 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.
- */
- TIFFWarning(tif->tif_name,
- "Bogus \"%s\" field, ignoring and calculating from imagelength",
- TIFFFieldWithTag(TIFFTAG_STRIPBYTECOUNTS)->field_name);
- EstimateStripByteCounts(tif, dir, dircount);
- }
- if (dir)
- free((char *)dir);
- if (!TIFFFieldSet(tif, FIELD_MAXSAMPLEVALUE))
- td->td_maxsamplevalue = (1L<<td->td_bitspersample)-1;
- /*
- * Setup default compression scheme.
- */
- if (!TIFFFieldSet(tif, FIELD_COMPRESSION))
- TIFFSetField(tif, TIFFTAG_COMPRESSION, COMPRESSION_NONE);
- /*
- * Reinitialize i/o since we are starting on a new directory.
- */
- tif->tif_row = -1;
- tif->tif_curstrip = -1;
- tif->tif_col = -1;
- tif->tif_curtile = -1;
- tif->tif_tilesize = TIFFTileSize(tif);
- tif->tif_scanlinesize = TIFFScanlineSize(tif);
- return (1);
- bad:
- if (dir)
- free((char *)dir);
- return (0);
- }
-
- static
- EstimateStripByteCounts(tif, dir, dircount)
- TIFF *tif;
- TIFFDirEntry *dir;
- u_int dircount;
- {
- register TIFFDirEntry *dp;
- register TIFFDirectory *td = &tif->tif_dir;
- register int n;
-
- td->td_stripbytecount = (u_long *)
- CheckMalloc(tif, sizeof (u_long), "for \"StripByteCounts\" array");
- if (td->td_compression != COMPRESSION_NONE) {
- u_long space = sizeof (TIFFHeader)
- + sizeof (short)
- + (dircount * sizeof (TIFFDirEntry))
- + sizeof (long);
- long filesize = TIFFGetFileSize(tif->tif_fd);
- /* calculate amount of space used by indirect values */
- for (dp = dir, n = dircount; n > 0; n--, dp++) {
- int cc = dp->tdir_count * tiffDataWidth[dp->tdir_type];
- if (cc > sizeof (long))
- space += cc;
- }
- td->td_stripbytecount[0] = filesize - space;
- /*
- * This gross hack handles the case were the offset to
- * the 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.
- */
- if (td->td_stripoffset[0] + td->td_stripbytecount[0] > filesize)
- td->td_stripbytecount[0] =
- filesize - td->td_stripoffset[0];
- } else {
- u_long rowbytes = howmany(td->td_bitspersample *
- td->td_samplesperpixel * td->td_imagewidth, 8);
- td->td_stripbytecount[0] = td->td_imagelength * rowbytes;
- }
- TIFFSetFieldBit(tif, FIELD_STRIPBYTECOUNTS);
- if (!TIFFFieldSet(tif, FIELD_ROWSPERSTRIP))
- td->td_rowsperstrip = td->td_imagelength;
- }
-
- static
- MissingRequired(tif, tagname)
- TIFF *tif;
- char *tagname;
- {
- TIFFError(tif->tif_name,
- "TIFF directory is missing required \"%s\" field", tagname);
- }
-
- /*
- * 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.
- */
- static
- CheckDirCount(tif, dir, count)
- TIFF *tif;
- TIFFDirEntry *dir;
- u_long count;
- {
- if (count != dir->tdir_count) {
- TIFFWarning(tif->tif_name,
- "incorrect count for field \"%s\" (%lu, expecting %lu); tag ignored",
- TIFFFieldWithTag(dir->tdir_tag)->field_name,
- dir->tdir_count, count);
- return (0);
- }
- return (1);
- }
-
- /*
- * Fetch a contiguous directory item.
- */
- static
- TIFFFetchData(tif, dir, cp)
- TIFF *tif;
- TIFFDirEntry *dir;
- char *cp;
- {
- int cc, w;
-
- w = tiffDataWidth[dir->tdir_type];
- cc = dir->tdir_count * w;
- if (!isMapped(tif)) {
- if (!SeekOK(tif->tif_fd, dir->tdir_offset))
- goto bad;
- if (!ReadOK(tif->tif_fd, cp, cc))
- goto bad;
- #ifdef MMAP_SUPPORT
- } else {
- if (dir->tdir_offset + cc > tif->tif_size)
- goto bad;
- bcopy(tif->tif_base + dir->tdir_offset, cp, cc);
- #endif
- }
- if (tif->tif_flags & TIFF_SWAB) {
- switch (dir->tdir_type) {
- case TIFF_SHORT:
- case TIFF_SSHORT:
- TIFFSwabArrayOfShort((u_short *)cp, dir->tdir_count);
- break;
- case TIFF_LONG:
- case TIFF_SLONG:
- case TIFF_FLOAT:
- TIFFSwabArrayOfLong((u_long *)cp, dir->tdir_count);
- break;
- case TIFF_RATIONAL:
- case TIFF_SRATIONAL:
- TIFFSwabArrayOfLong((u_long *)cp, 2*dir->tdir_count);
- break;
- }
- }
- return (cc);
- bad:
- TIFFError(tif->tif_name, "Error fetching data for field \"%s\"",
- TIFFFieldWithTag(dir->tdir_tag)->field_name);
- return (0);
- }
-
- /*
- * Fetch an ASCII item from the file.
- */
- static
- TIFFFetchString(tif, dir, cp)
- TIFF *tif;
- TIFFDirEntry *dir;
- char *cp;
- {
- if (dir->tdir_count <= 4) {
- u_long l = dir->tdir_offset;
- if (tif->tif_flags & TIFF_SWAB)
- TIFFSwabLong(&l);
- bcopy(&l, cp, dir->tdir_count);
- return (1);
- }
- return (TIFFFetchData(tif, dir, cp));
- }
-
- /*
- * Convert numerator+denominator to float.
- */
- static int
- cvtRational(tif, dir, num, denom, rv)
- TIFF *tif;
- TIFFDirEntry *dir;
- u_long num, denom;
- float *rv;
- {
- if (denom == 0) {
- TIFFError(tif->tif_name,
- "%s: Rational with zero denominator (num = %lu)",
- TIFFFieldWithTag(dir->tdir_tag)->field_name, num);
- return (0);
- } else {
- if (dir->tdir_type == TIFF_RATIONAL)
- *rv = ((float)num / (float)denom);
- else
- *rv = ((float)(long)num / (float)(long)denom);
- return (1);
- }
- }
-
- /*
- * Fetch a rational item from the file
- * at offset off and return the value
- * as a floating point number.
- */
- static float
- TIFFFetchRational(tif, dir)
- TIFF *tif;
- TIFFDirEntry *dir;
- {
- u_long l[2];
- float v;
-
- return (!TIFFFetchData(tif, dir, (char *)l) ||
- !cvtRational(tif, dir, l[0], l[1], &v) ? 1. : v);
- }
-
- /*
- * Fetch a single floating point value
- * from the offset field and return it
- * as a native float.
- */
- static float
- TIFFFetchFloat(tif, dir)
- TIFF *tif;
- TIFFDirEntry *dir;
- {
- float v = (float)
- TIFFExtractData(tif, dir->tdir_type, dir->tdir_offset);
- TIFFCvtIEEEFloatToNative(tif, 1, &v);
- return (v);
- }
-
- /*
- * Fetch an array of BYTE or SBYTE values.
- */
- static
- TIFFFetchByteArray(tif, dir, v)
- TIFF *tif;
- TIFFDirEntry *dir;
- u_short v[];
- {
-
- if (dir->tdir_count <= 4) {
- /*
- * Extract data from offset field.
- */
- if (tif->tif_header.tiff_magic == TIFF_BIGENDIAN) {
- switch (dir->tdir_count) {
- 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;
- }
- } else {
- switch (dir->tdir_count) {
- 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;
- }
- }
- return (1);
- } else
- return (TIFFFetchData(tif, dir, (char *)v)); /* XXX */
- }
-
- /*
- * Fetch an array of SHORT or SSHORT values.
- */
- static
- TIFFFetchShortArray(tif, dir, v)
- TIFF *tif;
- TIFFDirEntry *dir;
- u_short v[];
- {
- if (dir->tdir_count <= 2) {
- if (tif->tif_header.tiff_magic == TIFF_BIGENDIAN) {
- switch (dir->tdir_count) {
- case 2: v[1] = dir->tdir_offset & 0xffff;
- case 1: v[0] = dir->tdir_offset >> 16;
- }
- } else {
- switch (dir->tdir_count) {
- case 2: v[1] = dir->tdir_offset >> 16;
- case 1: v[0] = dir->tdir_offset & 0xffff;
- }
- }
- return (1);
- } else
- return (TIFFFetchData(tif, dir, (char *)v));
- }
-
- /*
- * Fetch an array of LONG or SLONG values.
- */
- static
- TIFFFetchLongArray(tif, dir, v)
- TIFF *tif;
- TIFFDirEntry *dir;
- u_long v[];
- {
- if (dir->tdir_count == 1) {
- v[0] = dir->tdir_offset;
- return (1);
- } else
- return (TIFFFetchData(tif, dir, (char *)v));
- }
-
- /*
- * Fetch an array of RATIONAL or SRATIONAL values.
- */
- static
- TIFFFetchRationalArray(tif, dir, v)
- TIFF *tif;
- TIFFDirEntry *dir;
- float v[];
- {
- int ok = 0;
- u_long *l;
-
- l = (u_long *)CheckMalloc(tif,
- dir->tdir_count*tiffDataWidth[dir->tdir_type],
- "to fetch array of rationals");
- if (l) {
- if (TIFFFetchData(tif, dir, (char *)l)) {
- u_long 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;
- }
- }
- free((char *)l);
- }
- return (ok);
- }
-
- /*
- * Fetch an array of FLOAT values.
- */
- static
- TIFFFetchFloatArray(tif, dir, v)
- TIFF *tif;
- TIFFDirEntry *dir;
- float v[];
- {
- if (TIFFFetchData(tif, dir, (char *)v)) {
- TIFFCvtIEEEFloatToNative(tif, dir->tdir_count, v);
- return (1);
- } else
- return (0);
- }
-
- /*
- * Fetch a tag that is not handled by special case code.
- *
- * NB: DOUBLE and UNDEFINED types are not handled.
- */
- static
- TIFFFetchNormalTag(tif, dp)
- TIFF *tif;
- TIFFDirEntry *dp;
- {
- static char mesg[] = "to fetch tag value";
- int ok = 0;
-
- if (dp->tdir_count > 1) { /* array of values */
- char *cp = NULL;
-
- switch (dp->tdir_type) {
- case TIFF_BYTE:
- case TIFF_SBYTE:
- /* NB: always expand BYTE values to shorts */
- cp = CheckMalloc(tif,
- dp->tdir_count * sizeof (u_short), mesg);
- ok = cp && TIFFFetchByteArray(tif, dp, (u_short *)cp);
- break;
- case TIFF_SHORT:
- case TIFF_SSHORT:
- cp = CheckMalloc(tif,
- dp->tdir_count * sizeof (u_short), mesg);
- ok = cp && TIFFFetchShortArray(tif, dp, (u_short *)cp);
- break;
- case TIFF_LONG:
- case TIFF_SLONG:
- cp = CheckMalloc(tif,
- dp->tdir_count * sizeof (u_long), mesg);
- ok = cp && TIFFFetchLongArray(tif, dp, (u_long *)cp);
- break;
- case TIFF_RATIONAL:
- case TIFF_SRATIONAL:
- cp = CheckMalloc(tif,
- dp->tdir_count * sizeof (float), mesg);
- ok = cp && TIFFFetchRationalArray(tif, dp, (float *)cp);
- break;
- case TIFF_FLOAT:
- cp = CheckMalloc(tif,
- dp->tdir_count * sizeof (float), mesg);
- ok = cp && TIFFFetchFloatArray(tif, dp, (float *)cp);
- break;
- case TIFF_ASCII:
- /*
- * Some vendors write strings w/o the trailing
- * NULL byte, so always append one just in case.
- */
- cp = CheckMalloc(tif, dp->tdir_count+1, mesg);
- if (ok = (cp && TIFFFetchString(tif, dp, cp)))
- cp[dp->tdir_count] = '\0'; /* XXX */
- break;
- }
- if (ok)
- ok = TIFFSetField(tif, dp->tdir_tag, cp);
- if (cp != NULL)
- free(cp);
- } else if (CheckDirCount(tif, dp, 1)) { /* singleton value */
- char c[2];
- switch (dp->tdir_type) {
- case TIFF_BYTE:
- case TIFF_SBYTE:
- case TIFF_SHORT:
- case TIFF_SSHORT:
- ok = TIFFSetField(tif, dp->tdir_tag, (int)
- TIFFExtractData(tif, dp->tdir_type, dp->tdir_offset));
- break;
- case TIFF_LONG:
- case TIFF_SLONG:
- ok = TIFFSetField(tif, dp->tdir_tag, (u_long)
- TIFFExtractData(tif, dp->tdir_type, dp->tdir_offset));
- break;
- case TIFF_RATIONAL:
- case TIFF_SRATIONAL:
- ok = TIFFSetField(tif, dp->tdir_tag,
- TIFFFetchRational(tif, dp));
- break;
- case TIFF_FLOAT:
- ok = TIFFSetField(tif, dp->tdir_tag,
- TIFFFetchFloat(tif, dp));
- break;
- case TIFF_ASCII:
- if (ok = (TIFFFetchString(tif, dp, c))) {
- c[1] = '\0'; /* XXX paranoid */
- ok = TIFFSetField(tif, dp->tdir_tag, c);
- }
- break;
- }
- }
- return (ok);
- }
-
- /*
- * Fetch samples/pixel short values for
- * the specified tag and verify that
- * all values are the same.
- */
- static
- TIFFFetchPerSampleShorts(tif, dir, pl)
- TIFF *tif;
- TIFFDirEntry *dir;
- long *pl;
- {
- u_short v[4];
- int samples = tif->tif_dir.td_samplesperpixel;
-
- if (CheckDirCount(tif, dir, (u_long)samples) &&
- TIFFFetchShortArray(tif, dir, v)) {
- int i;
- for (i = 1; i < samples; i++)
- if (v[i] != v[0]) {
- TIFFError(tif->tif_name,
- "Cannot handle different per-sample values for field \"%s\"",
- TIFFFieldWithTag(dir->tdir_tag)->field_name);
- return (0);
- }
- *pl = v[0];
- return (1);
- }
- return (0);
- }
-
- /*
- * Fetch a set of offsets or lengths.
- * While this routine says "strips",
- * in fact it's also used for tiles.
- */
- static
- TIFFFetchStripThing(tif, dir, nstrips, lpp)
- TIFF *tif;
- TIFFDirEntry *dir;
- long nstrips;
- u_long **lpp;
- {
- register u_long *lp;
- int status;
-
- if (!CheckDirCount(tif, dir, nstrips))
- return (0);
- /*
- * Allocate space for strip information.
- */
- if (*lpp == NULL &&
- (*lpp = (u_long *)CheckMalloc(tif,
- nstrips * sizeof (u_long), "for strip array")) == NULL)
- return (0);
- lp = *lpp;
- if (dir->tdir_type == (int)TIFF_SHORT) {
- /*
- * Handle short->long expansion.
- */
- u_short *dp = (u_short *)CheckMalloc(tif,
- dir->tdir_count* sizeof (u_short), "to fetch strip tag");
- if (dp == NULL)
- return (0);
- if (status = TIFFFetchShortArray(tif, dir, dp)) {
- register u_short *wp = dp;
- while (nstrips-- > 0)
- *lp++ = *wp++;
- }
- free((char *)dp);
- } else
- status = TIFFFetchLongArray(tif, dir, lp);
- return (status);
- }
-
- #ifdef COLORIMETRY_SUPPORT
- static
- TIFFFetchRefBlackWhite(tif, dir)
- TIFF *tif;
- TIFFDirEntry *dir;
- {
- static char mesg[] = "for \"ReferenceBlackWhite\" array";
- char *cp;
- int ok;
-
- if (!CheckDirCount(tif, dir, 2*tif->tif_dir.td_samplesperpixel))
- return (0);
- if (dir->tdir_type == TIFF_RATIONAL)
- return (TIFFFetchNormalTag(tif, dir));
- /*
- * Handle LONG's for backward compatibility.
- */
- cp = CheckMalloc(tif, dir->tdir_count * sizeof (u_long), mesg);
- if (ok = (cp && TIFFFetchLongArray(tif, dir, (u_long *)cp))) {
- float *fp = (float *)
- CheckMalloc(tif, dir->tdir_count * sizeof (float), mesg);
- if (ok = (fp != NULL)) {
- int i;
- for (i = 0; i < dir->tdir_count; i++)
- fp[i] = (float)((u_long *)cp)[i];
- ok = TIFFSetField(tif, dir->tdir_tag, fp);
- free((char *)fp);
- }
- }
- if (cp)
- free(cp);
- return (ok);
- }
- #endif
-
- #ifdef JPEG_SUPPORT
- /*
- * Fetch the JPEG Quantization tables
- * for the specified directory entry.
- * Storage for the td_qtab array is
- * allocated as a side effect.
- */
- static
- TIFFFetchJPEGQTables(tif, dir)
- TIFF *tif;
- TIFFDirEntry *dir;
- {
- TIFFDirectory *td = &tif->tif_dir;
- long off[4];
- int i, j;
- TIFFDirEntry tdir;
- char *qmat;
-
- if (dir->tdir_count > 1) {
- /* XXX verify count <= 4 */
- if (!TIFFFetchData(tif, dir, (char *)off))
- return (0);
- } else
- off[0] = dir->tdir_offset;
- /*
- * We don't share per-component q matrices because
- * (besides complicating this logic even more), it
- * would make it very painful if the user does a ``set''.
- */
- td->td_qtab = (u_char **)CheckMalloc(tif,
- dir->tdir_count*(sizeof (u_char *) + 64*sizeof (u_char)),
- "for JPEG Q table");
- if (td->td_qtab == NULL)
- return (0);
- tdir.tdir_type = TIFF_BYTE;
- tdir.tdir_count = 64;
- qmat = (((char *)td->td_qtab) + dir->tdir_count*sizeof (u_char *));
- for (i = 0; i < dir->tdir_count; i++) {
- td->td_qtab[i] = (u_char *)qmat;
- tdir.tdir_offset = off[i];
- if (!TIFFFetchData(tif, &tdir, qmat))
- return (0);
- qmat += 64*sizeof (u_char);
- }
- return (1);
- }
-
- /*
- * Fetch JPEG Huffman code tables for the
- * specified directory entry. Storage for
- * the tables are allocated as a side effect.
- */
- static
- TIFFFetchJPEGCTables(tif, dir, ptab)
- TIFF *tif;
- TIFFDirEntry *dir;
- u_char ***ptab;
- {
- long off[4];
- int i, j, ncodes;
- TIFFDirEntry tdir;
- char *tab;
-
- if (dir->tdir_count > 1) {
- /* XXX verify count <= 4 */
- if (!TIFFFetchData(tif, dir, (char *)off))
- return (0);
- } else
- off[0] = dir->tdir_offset;
- /*
- * We don't share per-component tables because
- * (besides complicating this logic even more), it
- * would make it very painful if the user does a
- * ``set''. Note also that we don't try to optimize
- * storage of the tables -- we just allocate enough
- * space to hold the largest possible. All this
- * stuff is so complicated 'cuz the tag is defined
- * to be compatible with the JPEG table format,
- * rather than something that fits well into the
- * structure of TIFF -- argh!
- */
- *ptab = (u_char **)CheckMalloc(tif, dir->tdir_count*
- (sizeof (u_char *) + (16+256)*sizeof (u_char)),
- "for JPEG Huffman table");
- if (*ptab == NULL)
- return (0);
- tdir.tdir_type = TIFF_BYTE;
- tab = (((char *)*ptab) + dir->tdir_count*sizeof (u_char *));
- for (i = 0; i < dir->tdir_count; i++) {
- (*ptab)[i] = (u_char *)tab;
- tdir.tdir_offset = off[i];
- tdir.tdir_count = 16;
- /*
- * We must fetch the array that holds the
- * count of codes for each bit length first
- * and the count up the number of codes that
- * are in the variable length table. This
- * information is implicit in the JPEG format
- * 'cuz it's preceded by a length field.
- */
- if (!TIFFFetchData(tif, &tdir, tab)) /* count array */
- return (0);
- for (ncodes = 0, j = 0; j < 16; j++)
- ncodes += tab[j];
- /*
- * Adjust offsets and fetch codes separately.
- */
- tdir.tdir_offset += 16;
- tdir.tdir_count = ncodes;
- tab += 16;
- if (!TIFFFetchData(tif, &tdir, tab))
- return (0);
- tab += ncodes;
- }
- return (1);
- }
- #endif
-
- /*
- * Accept matteing-only ExtraSamples tag.
- */
- static
- TIFFFetchExtraSamples(tif, dp)
- TIFF *tif;
- TIFFDirEntry *dp;
- {
- int type;
-
- if (dp->tdir_count != 1) {
- TIFFError(tif->tif_name,
- "Can not handle more than 1 extra sample/pixel");
- return (0);
- }
- type = TIFFExtractData(tif, dp->tdir_type, dp->tdir_offset);
- if (type != EXTRASAMPLE_ASSOCALPHA) {
- TIFFError(tif->tif_name,
- "Can only handle associated-alpha extra samples");
- return (0);
- }
- return (TIFFSetField(tif, TIFFTAG_MATTEING, 1));
- }
-