|  | /* | 
|  | * 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. | 
|  | * | 
|  | * Scanline-oriented Write Support | 
|  | */ | 
|  | #include "tiffiop.h" | 
|  | #include <stdio.h> | 
|  |  | 
|  | #define STRIPINCR 20 /* expansion factor on strip array */ | 
|  |  | 
|  | #define WRITECHECKSTRIPS(tif, module)                                          \ | 
|  | (((tif)->tif_flags & TIFF_BEENWRITING) || TIFFWriteCheck((tif), 0, module)) | 
|  | #define WRITECHECKTILES(tif, module)                                           \ | 
|  | (((tif)->tif_flags & TIFF_BEENWRITING) || TIFFWriteCheck((tif), 1, module)) | 
|  | #define BUFFERCHECK(tif)                                                       \ | 
|  | ((((tif)->tif_flags & TIFF_BUFFERSETUP) && tif->tif_rawdata) ||            \ | 
|  | TIFFWriteBufferSetup((tif), NULL, (tmsize_t)-1)) | 
|  |  | 
|  | static int TIFFGrowStrips(TIFF *tif, uint32_t delta, const char *module); | 
|  | static int TIFFAppendToStrip(TIFF *tif, uint32_t strip, uint8_t *data, | 
|  | tmsize_t cc); | 
|  |  | 
|  | int TIFFWriteScanline(TIFF *tif, void *buf, uint32_t row, uint16_t sample) | 
|  | { | 
|  | static const char module[] = "TIFFWriteScanline"; | 
|  | register TIFFDirectory *td; | 
|  | int status, imagegrew = 0; | 
|  | uint32_t strip; | 
|  |  | 
|  | if (!WRITECHECKSTRIPS(tif, module)) | 
|  | return (-1); | 
|  | /* | 
|  | * Handle delayed allocation of data buffer.  This | 
|  | * permits it to be sized more intelligently (using | 
|  | * directory information). | 
|  | */ | 
|  | if (!BUFFERCHECK(tif)) | 
|  | return (-1); | 
|  | tif->tif_flags |= TIFF_BUF4WRITE; /* not strictly sure this is right*/ | 
|  |  | 
|  | td = &tif->tif_dir; | 
|  | /* | 
|  | * Extend image length if needed | 
|  | * (but only for PlanarConfig=1). | 
|  | */ | 
|  | if (row >= td->td_imagelength) | 
|  | { /* extend image */ | 
|  | if (td->td_planarconfig == PLANARCONFIG_SEPARATE) | 
|  | { | 
|  | TIFFErrorExtR( | 
|  | tif, module, | 
|  | "Can not change \"ImageLength\" when using separate planes"); | 
|  | return (-1); | 
|  | } | 
|  | td->td_imagelength = row + 1; | 
|  | imagegrew = 1; | 
|  | } | 
|  | /* | 
|  | * Calculate strip and check for crossings. | 
|  | */ | 
|  | if (td->td_planarconfig == PLANARCONFIG_SEPARATE) | 
|  | { | 
|  | if (sample >= td->td_samplesperpixel) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "%lu: Sample out of range, max %lu", | 
|  | (unsigned long)sample, | 
|  | (unsigned long)td->td_samplesperpixel); | 
|  | return (-1); | 
|  | } | 
|  | strip = sample * td->td_stripsperimage + row / td->td_rowsperstrip; | 
|  | } | 
|  | else | 
|  | strip = row / td->td_rowsperstrip; | 
|  | /* | 
|  | * Check strip array to make sure there's space. We don't support | 
|  | * dynamically growing files that have data organized in separate | 
|  | * bitplanes because it's too painful.  In that case we require that | 
|  | * the imagelength be set properly before the first write (so that the | 
|  | * strips array will be fully allocated above). | 
|  | */ | 
|  | if (strip >= td->td_nstrips && !TIFFGrowStrips(tif, 1, module)) | 
|  | return (-1); | 
|  | if (strip != tif->tif_curstrip) | 
|  | { | 
|  | /* | 
|  | * Changing strips -- flush any data present. | 
|  | */ | 
|  | if (!TIFFFlushData(tif)) | 
|  | return (-1); | 
|  | tif->tif_curstrip = strip; | 
|  | /* | 
|  | * Watch out for a growing image.  The value of strips/image | 
|  | * will initially be 1 (since it can't be deduced until the | 
|  | * imagelength is known). | 
|  | */ | 
|  | if (strip >= td->td_stripsperimage && imagegrew) | 
|  | td->td_stripsperimage = | 
|  | TIFFhowmany_32(td->td_imagelength, td->td_rowsperstrip); | 
|  | if (td->td_stripsperimage == 0) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Zero strips per image"); | 
|  | return (-1); | 
|  | } | 
|  | tif->tif_row = (strip % td->td_stripsperimage) * td->td_rowsperstrip; | 
|  | if ((tif->tif_flags & TIFF_CODERSETUP) == 0) | 
|  | { | 
|  | if (!(*tif->tif_setupencode)(tif)) | 
|  | return (-1); | 
|  | tif->tif_flags |= TIFF_CODERSETUP; | 
|  | } | 
|  |  | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  |  | 
|  | /* this informs TIFFAppendToStrip() we have changed strip */ | 
|  | tif->tif_curoff = 0; | 
|  |  | 
|  | if (!(*tif->tif_preencode)(tif, sample)) | 
|  | return (-1); | 
|  | tif->tif_flags |= TIFF_POSTENCODE; | 
|  | } | 
|  | /* | 
|  | * Ensure the write is either sequential or at the | 
|  | * beginning of a strip (or that we can randomly | 
|  | * access the data -- i.e. no encoding). | 
|  | */ | 
|  | if (row != tif->tif_row) | 
|  | { | 
|  | if (row < tif->tif_row) | 
|  | { | 
|  | /* | 
|  | * Moving backwards within the same strip: | 
|  | * backup to the start and then decode | 
|  | * forward (below). | 
|  | */ | 
|  | tif->tif_row = | 
|  | (strip % td->td_stripsperimage) * td->td_rowsperstrip; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  | } | 
|  | /* | 
|  | * Seek forward to the desired row. | 
|  | */ | 
|  | if (!(*tif->tif_seek)(tif, row - tif->tif_row)) | 
|  | return (-1); | 
|  | tif->tif_row = row; | 
|  | } | 
|  |  | 
|  | /* swab if needed - note that source buffer will be altered */ | 
|  | tif->tif_postdecode(tif, (uint8_t *)buf, tif->tif_scanlinesize); | 
|  |  | 
|  | status = (*tif->tif_encoderow)(tif, (uint8_t *)buf, tif->tif_scanlinesize, | 
|  | sample); | 
|  |  | 
|  | /* we are now poised at the beginning of the next row */ | 
|  | tif->tif_row = row + 1; | 
|  | return (status); | 
|  | } | 
|  |  | 
|  | /* Make sure that at the first attempt of rewriting a tile/strip, we will have | 
|  | */ | 
|  | /* more bytes available in the output buffer than the previous byte count, */ | 
|  | /* so that TIFFAppendToStrip() will detect the overflow when it is called the | 
|  | * first */ | 
|  | /* time if the new compressed tile is bigger than the older one. (GDAL #4771) */ | 
|  | static int _TIFFReserveLargeEnoughWriteBuffer(TIFF *tif, uint32_t strip_or_tile) | 
|  | { | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  | if (td->td_stripbytecount_p[strip_or_tile] > 0) | 
|  | { | 
|  | /* The +1 is to ensure at least one extra bytes */ | 
|  | /* The +4 is because the LZW encoder flushes 4 bytes before the limit */ | 
|  | uint64_t safe_buffer_size = | 
|  | (uint64_t)(td->td_stripbytecount_p[strip_or_tile] + 1 + 4); | 
|  | if (tif->tif_rawdatasize <= (tmsize_t)safe_buffer_size) | 
|  | { | 
|  | if (!(TIFFWriteBufferSetup( | 
|  | tif, NULL, | 
|  | (tmsize_t)TIFFroundup_64(safe_buffer_size, 1024)))) | 
|  | return 0; | 
|  | } | 
|  | } | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Encode the supplied data and write it to the | 
|  | * specified strip. | 
|  | * | 
|  | * NB: Image length must be setup before writing. | 
|  | */ | 
|  | tmsize_t TIFFWriteEncodedStrip(TIFF *tif, uint32_t strip, void *data, | 
|  | tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFWriteEncodedStrip"; | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  | uint16_t sample; | 
|  |  | 
|  | if (!WRITECHECKSTRIPS(tif, module)) | 
|  | return ((tmsize_t)-1); | 
|  | /* | 
|  | * Check strip array to make sure there's space. | 
|  | * We don't support dynamically growing files that | 
|  | * have data organized in separate bitplanes because | 
|  | * it's too painful.  In that case we require that | 
|  | * the imagelength be set properly before the first | 
|  | * write (so that the strips array will be fully | 
|  | * allocated above). | 
|  | */ | 
|  | if (strip >= td->td_nstrips) | 
|  | { | 
|  | if (td->td_planarconfig == PLANARCONFIG_SEPARATE) | 
|  | { | 
|  | TIFFErrorExtR( | 
|  | tif, module, | 
|  | "Can not grow image by strips when using separate planes"); | 
|  | return ((tmsize_t)-1); | 
|  | } | 
|  | if (!TIFFGrowStrips(tif, 1, module)) | 
|  | return ((tmsize_t)-1); | 
|  | td->td_stripsperimage = | 
|  | TIFFhowmany_32(td->td_imagelength, td->td_rowsperstrip); | 
|  | } | 
|  | /* | 
|  | * Handle delayed allocation of data buffer.  This | 
|  | * permits it to be sized according to the directory | 
|  | * info. | 
|  | */ | 
|  | if (!BUFFERCHECK(tif)) | 
|  | return ((tmsize_t)-1); | 
|  |  | 
|  | tif->tif_flags |= TIFF_BUF4WRITE; | 
|  |  | 
|  | tif->tif_curstrip = strip; | 
|  |  | 
|  | /* this informs TIFFAppendToStrip() we have changed or reset strip */ | 
|  | tif->tif_curoff = 0; | 
|  |  | 
|  | if (!_TIFFReserveLargeEnoughWriteBuffer(tif, strip)) | 
|  | { | 
|  | return ((tmsize_t)(-1)); | 
|  | } | 
|  |  | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  |  | 
|  | if (td->td_stripsperimage == 0) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Zero strips per image"); | 
|  | return ((tmsize_t)-1); | 
|  | } | 
|  |  | 
|  | tif->tif_row = (strip % td->td_stripsperimage) * td->td_rowsperstrip; | 
|  | if ((tif->tif_flags & TIFF_CODERSETUP) == 0) | 
|  | { | 
|  | if (!(*tif->tif_setupencode)(tif)) | 
|  | return ((tmsize_t)-1); | 
|  | tif->tif_flags |= TIFF_CODERSETUP; | 
|  | } | 
|  |  | 
|  | tif->tif_flags &= ~TIFF_POSTENCODE; | 
|  |  | 
|  | /* shortcut to avoid an extra memcpy() */ | 
|  | if (td->td_compression == COMPRESSION_NONE) | 
|  | { | 
|  | /* swab if needed - note that source buffer will be altered */ | 
|  | tif->tif_postdecode(tif, (uint8_t *)data, cc); | 
|  |  | 
|  | if (!isFillOrder(tif, td->td_fillorder) && | 
|  | (tif->tif_flags & TIFF_NOBITREV) == 0) | 
|  | TIFFReverseBits((uint8_t *)data, cc); | 
|  |  | 
|  | if (cc > 0 && !TIFFAppendToStrip(tif, strip, (uint8_t *)data, cc)) | 
|  | return ((tmsize_t)-1); | 
|  | return (cc); | 
|  | } | 
|  |  | 
|  | sample = (uint16_t)(strip / td->td_stripsperimage); | 
|  | if (!(*tif->tif_preencode)(tif, sample)) | 
|  | return ((tmsize_t)-1); | 
|  |  | 
|  | /* swab if needed - note that source buffer will be altered */ | 
|  | tif->tif_postdecode(tif, (uint8_t *)data, cc); | 
|  |  | 
|  | if (!(*tif->tif_encodestrip)(tif, (uint8_t *)data, cc, sample)) | 
|  | return ((tmsize_t)-1); | 
|  | if (!(*tif->tif_postencode)(tif)) | 
|  | return ((tmsize_t)-1); | 
|  | if (!isFillOrder(tif, td->td_fillorder) && | 
|  | (tif->tif_flags & TIFF_NOBITREV) == 0) | 
|  | TIFFReverseBits(tif->tif_rawdata, tif->tif_rawcc); | 
|  | if (tif->tif_rawcc > 0 && | 
|  | !TIFFAppendToStrip(tif, strip, tif->tif_rawdata, tif->tif_rawcc)) | 
|  | return ((tmsize_t)-1); | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  | return (cc); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Write the supplied data to the specified strip. | 
|  | * | 
|  | * NB: Image length must be setup before writing. | 
|  | */ | 
|  | tmsize_t TIFFWriteRawStrip(TIFF *tif, uint32_t strip, void *data, tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFWriteRawStrip"; | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  |  | 
|  | if (!WRITECHECKSTRIPS(tif, module)) | 
|  | return ((tmsize_t)-1); | 
|  | /* | 
|  | * Check strip array to make sure there's space. | 
|  | * We don't support dynamically growing files that | 
|  | * have data organized in separate bitplanes because | 
|  | * it's too painful.  In that case we require that | 
|  | * the imagelength be set properly before the first | 
|  | * write (so that the strips array will be fully | 
|  | * allocated above). | 
|  | */ | 
|  | if (strip >= td->td_nstrips) | 
|  | { | 
|  | if (td->td_planarconfig == PLANARCONFIG_SEPARATE) | 
|  | { | 
|  | TIFFErrorExtR( | 
|  | tif, module, | 
|  | "Can not grow image by strips when using separate planes"); | 
|  | return ((tmsize_t)-1); | 
|  | } | 
|  | /* | 
|  | * Watch out for a growing image.  The value of | 
|  | * strips/image will initially be 1 (since it | 
|  | * can't be deduced until the imagelength is known). | 
|  | */ | 
|  | if (strip >= td->td_stripsperimage) | 
|  | td->td_stripsperimage = | 
|  | TIFFhowmany_32(td->td_imagelength, td->td_rowsperstrip); | 
|  | if (!TIFFGrowStrips(tif, 1, module)) | 
|  | return ((tmsize_t)-1); | 
|  | } | 
|  |  | 
|  | if (tif->tif_curstrip != strip) | 
|  | { | 
|  | tif->tif_curstrip = strip; | 
|  |  | 
|  | /* this informs TIFFAppendToStrip() we have changed or reset strip */ | 
|  | tif->tif_curoff = 0; | 
|  | } | 
|  |  | 
|  | if (td->td_stripsperimage == 0) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Zero strips per image"); | 
|  | return ((tmsize_t)-1); | 
|  | } | 
|  | tif->tif_row = (strip % td->td_stripsperimage) * td->td_rowsperstrip; | 
|  | return (TIFFAppendToStrip(tif, strip, (uint8_t *)data, cc) ? cc | 
|  | : (tmsize_t)-1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Write and compress a tile of data.  The | 
|  | * tile is selected by the (x,y,z,s) coordinates. | 
|  | */ | 
|  | tmsize_t TIFFWriteTile(TIFF *tif, void *buf, uint32_t x, uint32_t y, uint32_t z, | 
|  | uint16_t s) | 
|  | { | 
|  | if (!TIFFCheckTile(tif, x, y, z, s)) | 
|  | return ((tmsize_t)(-1)); | 
|  | /* | 
|  | * NB: A tile size of -1 is used instead of tif_tilesize knowing | 
|  | *     that TIFFWriteEncodedTile will clamp this to the tile size. | 
|  | *     This is done because the tile size may not be defined until | 
|  | *     after the output buffer is setup in TIFFWriteBufferSetup. | 
|  | */ | 
|  | return (TIFFWriteEncodedTile(tif, TIFFComputeTile(tif, x, y, z, s), buf, | 
|  | (tmsize_t)(-1))); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Encode the supplied data and write it to the | 
|  | * specified tile.  There must be space for the | 
|  | * data.  The function clamps individual writes | 
|  | * to a tile to the tile size, but does not (and | 
|  | * can not) check that multiple writes to the same | 
|  | * tile do not write more than tile size data. | 
|  | * | 
|  | * NB: Image length must be setup before writing; this | 
|  | *     interface does not support automatically growing | 
|  | *     the image on each write (as TIFFWriteScanline does). | 
|  | */ | 
|  | tmsize_t TIFFWriteEncodedTile(TIFF *tif, uint32_t tile, void *data, tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFWriteEncodedTile"; | 
|  | TIFFDirectory *td; | 
|  | uint16_t sample; | 
|  | uint32_t howmany32; | 
|  |  | 
|  | if (!WRITECHECKTILES(tif, module)) | 
|  | return ((tmsize_t)(-1)); | 
|  | td = &tif->tif_dir; | 
|  | if (tile >= td->td_nstrips) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Tile %lu out of range, max %lu", | 
|  | (unsigned long)tile, (unsigned long)td->td_nstrips); | 
|  | return ((tmsize_t)(-1)); | 
|  | } | 
|  | /* | 
|  | * Handle delayed allocation of data buffer.  This | 
|  | * permits it to be sized more intelligently (using | 
|  | * directory information). | 
|  | */ | 
|  | if (!BUFFERCHECK(tif)) | 
|  | return ((tmsize_t)(-1)); | 
|  |  | 
|  | tif->tif_flags |= TIFF_BUF4WRITE; | 
|  |  | 
|  | tif->tif_curtile = tile; | 
|  |  | 
|  | /* this informs TIFFAppendToStrip() we have changed or reset tile */ | 
|  | tif->tif_curoff = 0; | 
|  |  | 
|  | if (!_TIFFReserveLargeEnoughWriteBuffer(tif, tile)) | 
|  | { | 
|  | return ((tmsize_t)(-1)); | 
|  | } | 
|  |  | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  |  | 
|  | /* | 
|  | * Compute tiles per row & per column to compute | 
|  | * current row and column | 
|  | */ | 
|  | howmany32 = TIFFhowmany_32(td->td_imagelength, td->td_tilelength); | 
|  | if (howmany32 == 0) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Zero tiles"); | 
|  | return ((tmsize_t)(-1)); | 
|  | } | 
|  | tif->tif_row = (tile % howmany32) * td->td_tilelength; | 
|  | howmany32 = TIFFhowmany_32(td->td_imagewidth, td->td_tilewidth); | 
|  | if (howmany32 == 0) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Zero tiles"); | 
|  | return ((tmsize_t)(-1)); | 
|  | } | 
|  | tif->tif_col = (tile % howmany32) * td->td_tilewidth; | 
|  |  | 
|  | if ((tif->tif_flags & TIFF_CODERSETUP) == 0) | 
|  | { | 
|  | if (!(*tif->tif_setupencode)(tif)) | 
|  | return ((tmsize_t)(-1)); | 
|  | tif->tif_flags |= TIFF_CODERSETUP; | 
|  | } | 
|  | tif->tif_flags &= ~TIFF_POSTENCODE; | 
|  |  | 
|  | /* | 
|  | * Clamp write amount to the tile size.  This is mostly | 
|  | * done so that callers can pass in some large number | 
|  | * (e.g. -1) and have the tile size used instead. | 
|  | */ | 
|  | if (cc < 1 || cc > tif->tif_tilesize) | 
|  | cc = tif->tif_tilesize; | 
|  |  | 
|  | /* shortcut to avoid an extra memcpy() */ | 
|  | if (td->td_compression == COMPRESSION_NONE) | 
|  | { | 
|  | /* swab if needed - note that source buffer will be altered */ | 
|  | tif->tif_postdecode(tif, (uint8_t *)data, cc); | 
|  |  | 
|  | if (!isFillOrder(tif, td->td_fillorder) && | 
|  | (tif->tif_flags & TIFF_NOBITREV) == 0) | 
|  | TIFFReverseBits((uint8_t *)data, cc); | 
|  |  | 
|  | if (cc > 0 && !TIFFAppendToStrip(tif, tile, (uint8_t *)data, cc)) | 
|  | return ((tmsize_t)-1); | 
|  | return (cc); | 
|  | } | 
|  |  | 
|  | sample = (uint16_t)(tile / td->td_stripsperimage); | 
|  | if (!(*tif->tif_preencode)(tif, sample)) | 
|  | return ((tmsize_t)(-1)); | 
|  | /* swab if needed - note that source buffer will be altered */ | 
|  | tif->tif_postdecode(tif, (uint8_t *)data, cc); | 
|  |  | 
|  | if (!(*tif->tif_encodetile)(tif, (uint8_t *)data, cc, sample)) | 
|  | return ((tmsize_t)-1); | 
|  | if (!(*tif->tif_postencode)(tif)) | 
|  | return ((tmsize_t)(-1)); | 
|  | if (!isFillOrder(tif, td->td_fillorder) && | 
|  | (tif->tif_flags & TIFF_NOBITREV) == 0) | 
|  | TIFFReverseBits((uint8_t *)tif->tif_rawdata, tif->tif_rawcc); | 
|  | if (tif->tif_rawcc > 0 && | 
|  | !TIFFAppendToStrip(tif, tile, tif->tif_rawdata, tif->tif_rawcc)) | 
|  | return ((tmsize_t)(-1)); | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  | return (cc); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Write the supplied data to the specified strip. | 
|  | * There must be space for the data; we don't check | 
|  | * if strips overlap! | 
|  | * | 
|  | * NB: Image length must be setup before writing; this | 
|  | *     interface does not support automatically growing | 
|  | *     the image on each write (as TIFFWriteScanline does). | 
|  | */ | 
|  | tmsize_t TIFFWriteRawTile(TIFF *tif, uint32_t tile, void *data, tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFWriteRawTile"; | 
|  |  | 
|  | if (!WRITECHECKTILES(tif, module)) | 
|  | return ((tmsize_t)(-1)); | 
|  | if (tile >= tif->tif_dir.td_nstrips) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Tile %lu out of range, max %lu", | 
|  | (unsigned long)tile, | 
|  | (unsigned long)tif->tif_dir.td_nstrips); | 
|  | return ((tmsize_t)(-1)); | 
|  | } | 
|  | return (TIFFAppendToStrip(tif, tile, (uint8_t *)data, cc) ? cc | 
|  | : (tmsize_t)(-1)); | 
|  | } | 
|  |  | 
|  | #define isUnspecified(tif, f)                                                  \ | 
|  | (TIFFFieldSet(tif, f) && (tif)->tif_dir.td_imagelength == 0) | 
|  |  | 
|  | int TIFFSetupStrips(TIFF *tif) | 
|  | { | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  |  | 
|  | if (isTiled(tif)) | 
|  | td->td_stripsperimage = isUnspecified(tif, FIELD_TILEDIMENSIONS) | 
|  | ? td->td_samplesperpixel | 
|  | : TIFFNumberOfTiles(tif); | 
|  | else | 
|  | td->td_stripsperimage = isUnspecified(tif, FIELD_ROWSPERSTRIP) | 
|  | ? td->td_samplesperpixel | 
|  | : TIFFNumberOfStrips(tif); | 
|  | td->td_nstrips = td->td_stripsperimage; | 
|  | /* TIFFWriteDirectoryTagData has a limitation to 0x80000000U bytes */ | 
|  | if (td->td_nstrips >= | 
|  | 0x80000000U / ((tif->tif_flags & TIFF_BIGTIFF) ? 0x8U : 0x4U)) | 
|  | { | 
|  | TIFFErrorExtR(tif, "TIFFSetupStrips", | 
|  | "Too large Strip/Tile Offsets/ByteCounts arrays"); | 
|  | return 0; | 
|  | } | 
|  | if (td->td_planarconfig == PLANARCONFIG_SEPARATE) | 
|  | td->td_stripsperimage /= td->td_samplesperpixel; | 
|  | td->td_stripoffset_p = (uint64_t *)_TIFFCheckMalloc( | 
|  | tif, td->td_nstrips, sizeof(uint64_t), "for \"StripOffsets\" array"); | 
|  | td->td_stripbytecount_p = (uint64_t *)_TIFFCheckMalloc( | 
|  | tif, td->td_nstrips, sizeof(uint64_t), "for \"StripByteCounts\" array"); | 
|  | if (td->td_stripoffset_p == NULL || td->td_stripbytecount_p == NULL) | 
|  | return (0); | 
|  | /* | 
|  | * Place data at the end-of-file | 
|  | * (by setting offsets to zero). | 
|  | */ | 
|  | _TIFFmemset(td->td_stripoffset_p, 0, td->td_nstrips * sizeof(uint64_t)); | 
|  | _TIFFmemset(td->td_stripbytecount_p, 0, td->td_nstrips * sizeof(uint64_t)); | 
|  | TIFFSetFieldBit(tif, FIELD_STRIPOFFSETS); | 
|  | TIFFSetFieldBit(tif, FIELD_STRIPBYTECOUNTS); | 
|  | return (1); | 
|  | } | 
|  | #undef isUnspecified | 
|  |  | 
|  | /* | 
|  | * Verify file is writable and that the directory | 
|  | * information is setup properly.  In doing the latter | 
|  | * we also "freeze" the state of the directory so | 
|  | * that important information is not changed. | 
|  | */ | 
|  | int TIFFWriteCheck(TIFF *tif, int tiles, const char *module) | 
|  | { | 
|  | if (tif->tif_mode == O_RDONLY) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "File not open for writing"); | 
|  | return (0); | 
|  | } | 
|  | if (tiles ^ isTiled(tif)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, | 
|  | tiles ? "Can not write tiles to a striped image" | 
|  | : "Can not write scanlines to a tiled image"); | 
|  | return (0); | 
|  | } | 
|  |  | 
|  | _TIFFFillStriles(tif); | 
|  |  | 
|  | /* | 
|  | * On the first write verify all the required information | 
|  | * has been setup and initialize any data structures that | 
|  | * had to wait until directory information was set. | 
|  | * Note that a lot of our work is assumed to remain valid | 
|  | * because we disallow any of the important parameters | 
|  | * from changing after we start writing (i.e. once | 
|  | * TIFF_BEENWRITING is set, TIFFSetField will only allow | 
|  | * the image's length to be changed). | 
|  | */ | 
|  | if (!TIFFFieldSet(tif, FIELD_IMAGEDIMENSIONS)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, | 
|  | "Must set \"ImageWidth\" before writing data"); | 
|  | return (0); | 
|  | } | 
|  | if (tif->tif_dir.td_stripoffset_p == NULL && !TIFFSetupStrips(tif)) | 
|  | { | 
|  | tif->tif_dir.td_nstrips = 0; | 
|  | TIFFErrorExtR(tif, module, "No space for %s arrays", | 
|  | isTiled(tif) ? "tile" : "strip"); | 
|  | return (0); | 
|  | } | 
|  | if (isTiled(tif)) | 
|  | { | 
|  | tif->tif_tilesize = TIFFTileSize(tif); | 
|  | if (tif->tif_tilesize == 0) | 
|  | return (0); | 
|  | } | 
|  | else | 
|  | tif->tif_tilesize = (tmsize_t)(-1); | 
|  | tif->tif_scanlinesize = TIFFScanlineSize(tif); | 
|  | if (tif->tif_scanlinesize == 0) | 
|  | return (0); | 
|  | tif->tif_flags |= TIFF_BEENWRITING; | 
|  |  | 
|  | 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 && | 
|  | !(tif->tif_flags & TIFF_DIRTYDIRECT)) | 
|  | { | 
|  | TIFFForceStrileArrayWriting(tif); | 
|  | } | 
|  |  | 
|  | return (1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Setup the raw data buffer used for encoding. | 
|  | */ | 
|  | int TIFFWriteBufferSetup(TIFF *tif, void *bp, tmsize_t size) | 
|  | { | 
|  | static const char module[] = "TIFFWriteBufferSetup"; | 
|  |  | 
|  | if (tif->tif_rawdata) | 
|  | { | 
|  | if (tif->tif_flags & TIFF_MYBUFFER) | 
|  | { | 
|  | _TIFFfreeExt(tif, tif->tif_rawdata); | 
|  | tif->tif_flags &= ~TIFF_MYBUFFER; | 
|  | } | 
|  | tif->tif_rawdata = NULL; | 
|  | } | 
|  | if (size == (tmsize_t)(-1)) | 
|  | { | 
|  | size = (isTiled(tif) ? tif->tif_tilesize : TIFFStripSize(tif)); | 
|  |  | 
|  | /* Adds 10% margin for cases where compression would expand a bit */ | 
|  | if (size < TIFF_TMSIZE_T_MAX - size / 10) | 
|  | size += size / 10; | 
|  | /* | 
|  | * Make raw data buffer at least 8K | 
|  | */ | 
|  | if (size < 8 * 1024) | 
|  | size = 8 * 1024; | 
|  | bp = NULL; /* NB: force malloc */ | 
|  | } | 
|  | if (bp == NULL) | 
|  | { | 
|  | bp = _TIFFmallocExt(tif, size); | 
|  | if (bp == NULL) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "No space for output buffer"); | 
|  | return (0); | 
|  | } | 
|  | tif->tif_flags |= TIFF_MYBUFFER; | 
|  | } | 
|  | else | 
|  | tif->tif_flags &= ~TIFF_MYBUFFER; | 
|  | tif->tif_rawdata = (uint8_t *)bp; | 
|  | tif->tif_rawdatasize = size; | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  | tif->tif_flags |= TIFF_BUFFERSETUP; | 
|  | return (1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Grow the strip data structures by delta strips. | 
|  | */ | 
|  | static int TIFFGrowStrips(TIFF *tif, uint32_t delta, const char *module) | 
|  | { | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  | uint64_t *new_stripoffset; | 
|  | uint64_t *new_stripbytecount; | 
|  |  | 
|  | assert(td->td_planarconfig == PLANARCONFIG_CONTIG); | 
|  | new_stripoffset = (uint64_t *)_TIFFreallocExt( | 
|  | tif, td->td_stripoffset_p, (td->td_nstrips + delta) * sizeof(uint64_t)); | 
|  | new_stripbytecount = (uint64_t *)_TIFFreallocExt( | 
|  | tif, td->td_stripbytecount_p, | 
|  | (td->td_nstrips + delta) * sizeof(uint64_t)); | 
|  | if (new_stripoffset == NULL || new_stripbytecount == NULL) | 
|  | { | 
|  | if (new_stripoffset) | 
|  | _TIFFfreeExt(tif, new_stripoffset); | 
|  | if (new_stripbytecount) | 
|  | _TIFFfreeExt(tif, new_stripbytecount); | 
|  | td->td_nstrips = 0; | 
|  | TIFFErrorExtR(tif, module, "No space to expand strip arrays"); | 
|  | return (0); | 
|  | } | 
|  | td->td_stripoffset_p = new_stripoffset; | 
|  | td->td_stripbytecount_p = new_stripbytecount; | 
|  | _TIFFmemset(td->td_stripoffset_p + td->td_nstrips, 0, | 
|  | delta * sizeof(uint64_t)); | 
|  | _TIFFmemset(td->td_stripbytecount_p + td->td_nstrips, 0, | 
|  | delta * sizeof(uint64_t)); | 
|  | td->td_nstrips += delta; | 
|  | tif->tif_flags |= TIFF_DIRTYDIRECT; | 
|  |  | 
|  | return (1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Append the data to the specified strip. | 
|  | */ | 
|  | static int TIFFAppendToStrip(TIFF *tif, uint32_t strip, uint8_t *data, | 
|  | tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFAppendToStrip"; | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  | uint64_t m; | 
|  | int64_t old_byte_count = -1; | 
|  |  | 
|  | if (tif->tif_curoff == 0) | 
|  | tif->tif_lastvalidoff = 0; | 
|  |  | 
|  | if (td->td_stripoffset_p[strip] == 0 || tif->tif_curoff == 0) | 
|  | { | 
|  | assert(td->td_nstrips > 0); | 
|  |  | 
|  | if (td->td_stripbytecount_p[strip] != 0 && | 
|  | td->td_stripoffset_p[strip] != 0 && | 
|  | td->td_stripbytecount_p[strip] >= (uint64_t)cc) | 
|  | { | 
|  | /* | 
|  | * There is already tile data on disk, and the new tile | 
|  | * data we have will fit in the same space.  The only | 
|  | * aspect of this that is risky is that there could be | 
|  | * more data to append to this strip before we are done | 
|  | * depending on how we are getting called. | 
|  | */ | 
|  | if (!SeekOK(tif, td->td_stripoffset_p[strip])) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Seek error at scanline %lu", | 
|  | (unsigned long)tif->tif_row); | 
|  | return (0); | 
|  | } | 
|  |  | 
|  | tif->tif_lastvalidoff = | 
|  | td->td_stripoffset_p[strip] + td->td_stripbytecount_p[strip]; | 
|  | } | 
|  | else | 
|  | { | 
|  | /* | 
|  | * Seek to end of file, and set that as our location to | 
|  | * write this strip. | 
|  | */ | 
|  | td->td_stripoffset_p[strip] = TIFFSeekFile(tif, 0, SEEK_END); | 
|  | tif->tif_flags |= TIFF_DIRTYSTRIP; | 
|  | } | 
|  |  | 
|  | tif->tif_curoff = td->td_stripoffset_p[strip]; | 
|  |  | 
|  | /* | 
|  | * We are starting a fresh strip/tile, so set the size to zero. | 
|  | */ | 
|  | old_byte_count = td->td_stripbytecount_p[strip]; | 
|  | td->td_stripbytecount_p[strip] = 0; | 
|  | } | 
|  |  | 
|  | m = tif->tif_curoff + cc; | 
|  | if (!(tif->tif_flags & TIFF_BIGTIFF)) | 
|  | m = (uint32_t)m; | 
|  | if ((m < tif->tif_curoff) || (m < (uint64_t)cc)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Maximum TIFF file size exceeded"); | 
|  | return (0); | 
|  | } | 
|  |  | 
|  | if (tif->tif_lastvalidoff != 0 && m > tif->tif_lastvalidoff && | 
|  | td->td_stripbytecount_p[strip] > 0) | 
|  | { | 
|  | /* Ouch: we have detected that we are rewriting in place a strip/tile */ | 
|  | /* with several calls to TIFFAppendToStrip(). The first call was with */ | 
|  | /* a size smaller than the previous size of the strip/tile, so we */ | 
|  | /* opted to rewrite in place, but a following call causes us to go */ | 
|  | /* outsize of the strip/tile area, so we have to finally go for a */ | 
|  | /* append-at-end-of-file strategy, and start by moving what we already | 
|  | */ | 
|  | /* wrote. */ | 
|  | tmsize_t tempSize; | 
|  | void *temp; | 
|  | uint64_t offsetRead; | 
|  | uint64_t offsetWrite; | 
|  | uint64_t toCopy = td->td_stripbytecount_p[strip]; | 
|  |  | 
|  | if (toCopy < 1024 * 1024) | 
|  | tempSize = (tmsize_t)toCopy; | 
|  | else | 
|  | tempSize = 1024 * 1024; | 
|  |  | 
|  | offsetRead = td->td_stripoffset_p[strip]; | 
|  | offsetWrite = TIFFSeekFile(tif, 0, SEEK_END); | 
|  |  | 
|  | m = offsetWrite + toCopy + cc; | 
|  | if (!(tif->tif_flags & TIFF_BIGTIFF) && m != (uint32_t)m) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Maximum TIFF file size exceeded"); | 
|  | return (0); | 
|  | } | 
|  |  | 
|  | temp = _TIFFmallocExt(tif, tempSize); | 
|  | if (temp == NULL) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "No space for output buffer"); | 
|  | return (0); | 
|  | } | 
|  |  | 
|  | tif->tif_flags |= TIFF_DIRTYSTRIP; | 
|  |  | 
|  | td->td_stripoffset_p[strip] = offsetWrite; | 
|  | td->td_stripbytecount_p[strip] = 0; | 
|  |  | 
|  | /* Move data written by previous calls to us at end of file */ | 
|  | while (toCopy > 0) | 
|  | { | 
|  | if (!SeekOK(tif, offsetRead)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Seek error"); | 
|  | _TIFFfreeExt(tif, temp); | 
|  | return (0); | 
|  | } | 
|  | if (!ReadOK(tif, temp, tempSize)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Cannot read"); | 
|  | _TIFFfreeExt(tif, temp); | 
|  | return (0); | 
|  | } | 
|  | if (!SeekOK(tif, offsetWrite)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Seek error"); | 
|  | _TIFFfreeExt(tif, temp); | 
|  | return (0); | 
|  | } | 
|  | if (!WriteOK(tif, temp, tempSize)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Cannot write"); | 
|  | _TIFFfreeExt(tif, temp); | 
|  | return (0); | 
|  | } | 
|  | offsetRead += tempSize; | 
|  | offsetWrite += tempSize; | 
|  | td->td_stripbytecount_p[strip] += tempSize; | 
|  | toCopy -= tempSize; | 
|  | } | 
|  | _TIFFfreeExt(tif, temp); | 
|  |  | 
|  | /* Append the data of this call */ | 
|  | offsetWrite += cc; | 
|  | m = offsetWrite; | 
|  | } | 
|  |  | 
|  | if (!WriteOK(tif, data, cc)) | 
|  | { | 
|  | TIFFErrorExtR(tif, module, "Write error at scanline %lu", | 
|  | (unsigned long)tif->tif_row); | 
|  | return (0); | 
|  | } | 
|  | tif->tif_curoff = m; | 
|  | td->td_stripbytecount_p[strip] += cc; | 
|  |  | 
|  | if ((int64_t)td->td_stripbytecount_p[strip] != old_byte_count) | 
|  | tif->tif_flags |= TIFF_DIRTYSTRIP; | 
|  |  | 
|  | return (1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Internal version of TIFFFlushData that can be | 
|  | * called by ``encodestrip routines'' w/o concern | 
|  | * for infinite recursion. | 
|  | */ | 
|  | int TIFFFlushData1(TIFF *tif) | 
|  | { | 
|  | if (tif->tif_rawcc > 0 && tif->tif_flags & TIFF_BUF4WRITE) | 
|  | { | 
|  | if (!isFillOrder(tif, tif->tif_dir.td_fillorder) && | 
|  | (tif->tif_flags & TIFF_NOBITREV) == 0) | 
|  | TIFFReverseBits((uint8_t *)tif->tif_rawdata, tif->tif_rawcc); | 
|  | if (!TIFFAppendToStrip( | 
|  | tif, isTiled(tif) ? tif->tif_curtile : tif->tif_curstrip, | 
|  | tif->tif_rawdata, tif->tif_rawcc)) | 
|  | { | 
|  | /* We update those variables even in case of error since there's */ | 
|  | /* code that doesn't really check the return code of this */ | 
|  | /* function */ | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  | return (0); | 
|  | } | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  | } | 
|  | return (1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Set the current write offset.  This should only be | 
|  | * used to set the offset to a known previous location | 
|  | * (very carefully), or to 0 so that the next write gets | 
|  | * appended to the end of the file. | 
|  | */ | 
|  | void TIFFSetWriteOffset(TIFF *tif, toff_t off) | 
|  | { | 
|  | tif->tif_curoff = off; | 
|  | tif->tif_lastvalidoff = 0; | 
|  | } |