175 lines
5.8 KiB
C
175 lines
5.8 KiB
C
/*
|
|
* 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)
|
|
{
|
|
TIFFErrorExt(tif->tif_clientdata, tif->tif_name,
|
|
"File opened in read-only mode");
|
|
return 0;
|
|
}
|
|
if( tif->tif_diroff == 0 )
|
|
{
|
|
TIFFErrorExt(tif->tif_clientdata, module,
|
|
"Directory has not yet been written");
|
|
return 0;
|
|
}
|
|
if( (tif->tif_flags & TIFF_DIRTYDIRECT) != 0 )
|
|
{
|
|
TIFFErrorExt(tif->tif_clientdata, 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) )
|
|
{
|
|
TIFFErrorExt(tif->tif_clientdata, 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));
|
|
}
|
|
|
|
/* vim: set ts=8 sts=8 sw=8 noet: */
|
|
/*
|
|
* Local Variables:
|
|
* mode: c
|
|
* c-basic-offset: 8
|
|
* fill-column: 78
|
|
* End:
|
|
*/
|