|  | /* | 
|  | * 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 delta, const char* module); | 
|  | static int TIFFAppendToStrip(TIFF* tif, uint32 strip, uint8* data, tmsize_t cc); | 
|  |  | 
|  | int | 
|  | TIFFWriteScanline(TIFF* tif, void* buf, uint32 row, uint16 sample) | 
|  | { | 
|  | static const char module[] = "TIFFWriteScanline"; | 
|  | register TIFFDirectory *td; | 
|  | int status, imagegrew = 0; | 
|  | uint32 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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; | 
|  |  | 
|  | if( td->td_stripbytecount_p[strip] > 0 ) | 
|  | { | 
|  | /* if we are writing over existing tiles, zero length */ | 
|  | td->td_stripbytecount_p[strip] = 0; | 
|  |  | 
|  | /* this forces TIFFAppendToStrip() to do a seek */ | 
|  | 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*) buf, tif->tif_scanlinesize ); | 
|  |  | 
|  | status = (*tif->tif_encoderow)(tif, (uint8*) 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 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 safe_buffer_size = (uint64)(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; | 
|  | } | 
|  |  | 
|  | /* Force TIFFAppendToStrip() to consider placing data at end | 
|  | of file. */ | 
|  | tif->tif_curoff = 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 strip, void* data, tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFWriteEncodedStrip"; | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  | uint16 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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; | 
|  |  | 
|  | if( !_TIFFReserveLargeEnoughWriteBuffer(tif, strip) ) { | 
|  | return ((tmsize_t)(-1)); | 
|  | } | 
|  |  | 
|  | tif->tif_rawcc = 0; | 
|  | tif->tif_rawcp = tif->tif_rawdata; | 
|  |  | 
|  | if (td->td_stripsperimage == 0) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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*) data, cc ); | 
|  |  | 
|  | if (!isFillOrder(tif, td->td_fillorder) && | 
|  | (tif->tif_flags & TIFF_NOBITREV) == 0) | 
|  | TIFFReverseBits((uint8*) data, cc); | 
|  |  | 
|  | if (cc > 0 && | 
|  | !TIFFAppendToStrip(tif, strip, (uint8*) data, cc)) | 
|  | return ((tmsize_t) -1); | 
|  | return (cc); | 
|  | } | 
|  |  | 
|  | sample = (uint16)(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*) data, cc ); | 
|  |  | 
|  | if (!(*tif->tif_encodestrip)(tif, (uint8*) 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 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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); | 
|  | } | 
|  | tif->tif_curstrip = strip; | 
|  | if (td->td_stripsperimage == 0) { | 
|  | TIFFErrorExt(tif->tif_clientdata, module,"Zero strips per image"); | 
|  | return ((tmsize_t) -1); | 
|  | } | 
|  | tif->tif_row = (strip % td->td_stripsperimage) * td->td_rowsperstrip; | 
|  | return (TIFFAppendToStrip(tif, strip, (uint8*) 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 x, uint32 y, uint32 z, uint16 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 tile, void* data, tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFWriteEncodedTile"; | 
|  | TIFFDirectory *td; | 
|  | uint16 sample; | 
|  | uint32 howmany32; | 
|  |  | 
|  | if (!WRITECHECKTILES(tif, module)) | 
|  | return ((tmsize_t)(-1)); | 
|  | td = &tif->tif_dir; | 
|  | if (tile >= td->td_nstrips) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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; | 
|  |  | 
|  | 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata,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) { | 
|  | TIFFErrorExt(tif->tif_clientdata,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*) data, cc ); | 
|  |  | 
|  | if (!isFillOrder(tif, td->td_fillorder) && | 
|  | (tif->tif_flags & TIFF_NOBITREV) == 0) | 
|  | TIFFReverseBits((uint8*) data, cc); | 
|  |  | 
|  | if (cc > 0 && | 
|  | !TIFFAppendToStrip(tif, tile, (uint8*) data, cc)) | 
|  | return ((tmsize_t) -1); | 
|  | return (cc); | 
|  | } | 
|  |  | 
|  | sample = (uint16)(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*) data, cc ); | 
|  |  | 
|  | if (!(*tif->tif_encodetile)(tif, (uint8*) 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*)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 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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*) 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) ) | 
|  | { | 
|  | TIFFErrorExt(tif->tif_clientdata, "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 *) | 
|  | _TIFFCheckMalloc(tif, td->td_nstrips, sizeof (uint64), | 
|  | "for \"StripOffsets\" array"); | 
|  | td->td_stripbytecount_p = (uint64 *) | 
|  | _TIFFCheckMalloc(tif, td->td_nstrips, sizeof (uint64), | 
|  | "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)); | 
|  | _TIFFmemset(td->td_stripbytecount_p, 0, td->td_nstrips*sizeof (uint64)); | 
|  | 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) { | 
|  | TIFFErrorExt(tif->tif_clientdata, module, "File not open for writing"); | 
|  | return (0); | 
|  | } | 
|  | if (tiles ^ isTiled(tif)) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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)) { | 
|  | TIFFErrorExt(tif->tif_clientdata, module, | 
|  | "Must set \"ImageWidth\" before writing data"); | 
|  | return (0); | 
|  | } | 
|  | if (tif->tif_dir.td_samplesperpixel == 1) { | 
|  | /* | 
|  | * Planarconfiguration is irrelevant in case of single band | 
|  | * images and need not be included. We will set it anyway, | 
|  | * because this field is used in other parts of library even | 
|  | * in the single band case. | 
|  | */ | 
|  | if (!TIFFFieldSet(tif, FIELD_PLANARCONFIG)) | 
|  | tif->tif_dir.td_planarconfig = PLANARCONFIG_CONTIG; | 
|  | } else { | 
|  | if (!TIFFFieldSet(tif, FIELD_PLANARCONFIG)) { | 
|  | TIFFErrorExt(tif->tif_clientdata, module, | 
|  | "Must set \"PlanarConfiguration\" before writing data"); | 
|  | return (0); | 
|  | } | 
|  | } | 
|  | if (tif->tif_dir.td_stripoffset_p == NULL && !TIFFSetupStrips(tif)) { | 
|  | tif->tif_dir.td_nstrips = 0; | 
|  | TIFFErrorExt(tif->tif_clientdata, 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) { | 
|  | _TIFFfree(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 = _TIFFmalloc(size); | 
|  | if (bp == NULL) { | 
|  | TIFFErrorExt(tif->tif_clientdata, module, "No space for output buffer"); | 
|  | return (0); | 
|  | } | 
|  | tif->tif_flags |= TIFF_MYBUFFER; | 
|  | } else | 
|  | tif->tif_flags &= ~TIFF_MYBUFFER; | 
|  | tif->tif_rawdata = (uint8*) 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 delta, const char* module) | 
|  | { | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  | uint64* new_stripoffset; | 
|  | uint64* new_stripbytecount; | 
|  |  | 
|  | assert(td->td_planarconfig == PLANARCONFIG_CONTIG); | 
|  | new_stripoffset = (uint64*)_TIFFrealloc(td->td_stripoffset_p, | 
|  | (td->td_nstrips + delta) * sizeof (uint64)); | 
|  | new_stripbytecount = (uint64*)_TIFFrealloc(td->td_stripbytecount_p, | 
|  | (td->td_nstrips + delta) * sizeof (uint64)); | 
|  | if (new_stripoffset == NULL || new_stripbytecount == NULL) { | 
|  | if (new_stripoffset) | 
|  | _TIFFfree(new_stripoffset); | 
|  | if (new_stripbytecount) | 
|  | _TIFFfree(new_stripbytecount); | 
|  | td->td_nstrips = 0; | 
|  | TIFFErrorExt(tif->tif_clientdata, 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)); | 
|  | _TIFFmemset(td->td_stripbytecount_p + td->td_nstrips, | 
|  | 0, delta*sizeof (uint64)); | 
|  | td->td_nstrips += delta; | 
|  | tif->tif_flags |= TIFF_DIRTYDIRECT; | 
|  |  | 
|  | return (1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Append the data to the specified strip. | 
|  | */ | 
|  | static int | 
|  | TIFFAppendToStrip(TIFF* tif, uint32 strip, uint8* data, tmsize_t cc) | 
|  | { | 
|  | static const char module[] = "TIFFAppendToStrip"; | 
|  | TIFFDirectory *td = &tif->tif_dir; | 
|  | uint64 m; | 
|  | int64 old_byte_count = -1; | 
|  |  | 
|  | 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) 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])) { | 
|  | TIFFErrorExt(tif->tif_clientdata, module, | 
|  | "Seek error at scanline %lu", | 
|  | (unsigned long)tif->tif_row); | 
|  | return (0); | 
|  | } | 
|  | } | 
|  | 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)m; | 
|  | if ((m<tif->tif_curoff)||(m<(uint64)cc)) | 
|  | { | 
|  | TIFFErrorExt(tif->tif_clientdata, module, "Maximum TIFF file size exceeded"); | 
|  | return (0); | 
|  | } | 
|  | if (!WriteOK(tif, data, cc)) { | 
|  | TIFFErrorExt(tif->tif_clientdata, 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) 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*)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; | 
|  | } | 
|  |  | 
|  | /* vim: set ts=8 sts=8 sw=8 noet: */ | 
|  | /* | 
|  | * Local Variables: | 
|  | * mode: c | 
|  | * c-basic-offset: 8 | 
|  | * fill-column: 78 | 
|  | * End: | 
|  | */ |