|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
#include "tiffiop.h"
|
|
|
|
|
|
|
|
int TIFFFlush(TIFF *tif)
|
|
|
|
{
|
|
|
|
if (tif->tif_mode == O_RDONLY)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (!TIFFFlushData(tif))
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
/* In update (r+) mode we try to detect the case where
|
|
|
|
only the strip/tile map has been altered, and we try to
|
|
|
|
rewrite only that portion of the directory without
|
|
|
|
making any other changes */
|
|
|
|
|
|
|
|
if ((tif->tif_flags & TIFF_DIRTYSTRIP) &&
|
|
|
|
!(tif->tif_flags & TIFF_DIRTYDIRECT) && tif->tif_mode == O_RDWR)
|
|
|
|
{
|
|
|
|
if (TIFFForceStrileArrayWriting(tif))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((tif->tif_flags & (TIFF_DIRTYDIRECT | TIFF_DIRTYSTRIP)) &&
|
|
|
|
!TIFFRewriteDirectory(tif))
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is an advanced writing function that must be used in a particular
|
|
|
|
* sequence, and together with TIFFDeferStrileArrayWriting(),
|
|
|
|
* to make its intended effect. Its aim is to force the writing of
|
|
|
|
* the [Strip/Tile][Offsets/ByteCounts] arrays at the end of the file, when
|
|
|
|
* they have not yet been rewritten.
|
|
|
|
*
|
|
|
|
* The typical sequence of calls is:
|
|
|
|
* TIFFOpen()
|
|
|
|
* [ TIFFCreateDirectory(tif) ]
|
|
|
|
* Set fields with calls to TIFFSetField(tif, ...)
|
|
|
|
* TIFFDeferStrileArrayWriting(tif)
|
|
|
|
* TIFFWriteCheck(tif, ...)
|
|
|
|
* TIFFWriteDirectory(tif)
|
|
|
|
* ... potentially create other directories and come back to the above directory
|
|
|
|
* TIFFForceStrileArrayWriting(tif)
|
|
|
|
*
|
|
|
|
* Returns 1 in case of success, 0 otherwise.
|
|
|
|
*/
|
|
|
|
int TIFFForceStrileArrayWriting(TIFF *tif)
|
|
|
|
{
|
|
|
|
static const char module[] = "TIFFForceStrileArrayWriting";
|
|
|
|
const int isTiled = TIFFIsTiled(tif);
|
|
|
|
|
|
|
|
if (tif->tif_mode == O_RDONLY)
|
|
|
|
{
|
|
|
|
TIFFErrorExtR(tif, tif->tif_name, "File opened in read-only mode");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (tif->tif_diroff == 0)
|
|
|
|
{
|
|
|
|
TIFFErrorExtR(tif, module, "Directory has not yet been written");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if ((tif->tif_flags & TIFF_DIRTYDIRECT) != 0)
|
|
|
|
{
|
|
|
|
TIFFErrorExtR(tif, module,
|
|
|
|
"Directory has changes other than the strile arrays. "
|
|
|
|
"TIFFRewriteDirectory() should be called instead");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(tif->tif_flags & TIFF_DIRTYSTRIP))
|
|
|
|
{
|
|
|
|
if (!(tif->tif_dir.td_stripoffset_entry.tdir_tag != 0 &&
|
|
|
|
tif->tif_dir.td_stripoffset_entry.tdir_count == 0 &&
|
|
|
|
tif->tif_dir.td_stripoffset_entry.tdir_type == 0 &&
|
|
|
|
tif->tif_dir.td_stripoffset_entry.tdir_offset.toff_long8 == 0 &&
|
|
|
|
tif->tif_dir.td_stripbytecount_entry.tdir_tag != 0 &&
|
|
|
|
tif->tif_dir.td_stripbytecount_entry.tdir_count == 0 &&
|
|
|
|
tif->tif_dir.td_stripbytecount_entry.tdir_type == 0 &&
|
|
|
|
tif->tif_dir.td_stripbytecount_entry.tdir_offset.toff_long8 == 0))
|
|
|
|
{
|
|
|
|
TIFFErrorExtR(tif, module,
|
|
|
|
"Function not called together with "
|
|
|
|
"TIFFDeferStrileArrayWriting()");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tif->tif_dir.td_stripoffset_p == NULL && !TIFFSetupStrips(tif))
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_TIFFRewriteField(tif,
|
|
|
|
isTiled ? TIFFTAG_TILEOFFSETS : TIFFTAG_STRIPOFFSETS,
|
|
|
|
TIFF_LONG8, tif->tif_dir.td_nstrips,
|
|
|
|
tif->tif_dir.td_stripoffset_p) &&
|
|
|
|
_TIFFRewriteField(
|
|
|
|
tif, isTiled ? TIFFTAG_TILEBYTECOUNTS : TIFFTAG_STRIPBYTECOUNTS,
|
|
|
|
TIFF_LONG8, tif->tif_dir.td_nstrips,
|
|
|
|
tif->tif_dir.td_stripbytecount_p))
|
|
|
|
{
|
|
|
|
tif->tif_flags &= ~TIFF_DIRTYSTRIP;
|
|
|
|
tif->tif_flags &= ~TIFF_BEENWRITING;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Flush buffered data to the file.
|
|
|
|
*
|
|
|
|
* Frank Warmerdam'2000: I modified this to return 1 if TIFF_BEENWRITING
|
|
|
|
* is not set, so that TIFFFlush() will proceed to write out the directory.
|
|
|
|
* The documentation says returning 1 is an error indicator, but not having
|
|
|
|
* been writing isn't exactly a an error. Hopefully this doesn't cause
|
|
|
|
* problems for other people.
|
|
|
|
*/
|
|
|
|
int TIFFFlushData(TIFF *tif)
|
|
|
|
{
|
|
|
|
if ((tif->tif_flags & TIFF_BEENWRITING) == 0)
|
|
|
|
return (1);
|
|
|
|
if (tif->tif_flags & TIFF_POSTENCODE)
|
|
|
|
{
|
|
|
|
tif->tif_flags &= ~TIFF_POSTENCODE;
|
|
|
|
if (!(*tif->tif_postencode)(tif))
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
return (TIFFFlushData1(tif));
|
|
|
|
}
|