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