|  | /* | 
|  | * 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)); | 
|  | } |