]>
git.saurik.com Git - wxWidgets.git/blob - src/tiff/libtiff/tif_dirwrite.c
b5d888b3e1717bef3bd03b5a1bc96509d17978b8
4 * Copyright (c) 1988-1997 Sam Leffler
5 * Copyright (c) 1991-1997 Silicon Graphics, Inc.
7 * Permission to use, copy, modify, distribute, and sell this software and
8 * its documentation for any purpose is hereby granted without fee, provided
9 * that (i) the above copyright notices and this permission notice appear in
10 * all copies of the software and related documentation, and (ii) the names of
11 * Sam Leffler and Silicon Graphics may not be used in any advertising or
12 * publicity relating to the software without the specific, prior written
13 * permission of Sam Leffler and Silicon Graphics.
15 * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
16 * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
17 * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
19 * IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
20 * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
21 * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
22 * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF
23 * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
30 * Directory Write Support Routines.
35 # define TIFFCvtNativeToIEEEFloat(tif, n, fp)
36 # define TIFFCvtNativeToIEEEDouble(tif, n, dp)
38 extern void TIFFCvtNativeToIEEEFloat(TIFF
*, uint32
, float*);
39 extern void TIFFCvtNativeToIEEEDouble(TIFF
*, uint32
, double*);
42 static int TIFFWriteNormalTag(TIFF
*, TIFFDirEntry
*, const TIFFFieldInfo
*);
43 static void TIFFSetupShortLong(TIFF
*, ttag_t
, TIFFDirEntry
*, uint32
);
44 static void TIFFSetupShort(TIFF
*, ttag_t
, TIFFDirEntry
*, uint16
);
45 static int TIFFSetupShortPair(TIFF
*, ttag_t
, TIFFDirEntry
*);
46 static int TIFFWritePerSampleShorts(TIFF
*, ttag_t
, TIFFDirEntry
*);
47 static int TIFFWritePerSampleAnys(TIFF
*, TIFFDataType
, ttag_t
, TIFFDirEntry
*);
48 static int TIFFWriteShortTable(TIFF
*, ttag_t
, TIFFDirEntry
*, uint32
, uint16
**);
49 static int TIFFWriteShortArray(TIFF
*, TIFFDirEntry
*, uint16
*);
50 static int TIFFWriteLongArray(TIFF
*, TIFFDirEntry
*, uint32
*);
51 static int TIFFWriteRationalArray(TIFF
*, TIFFDirEntry
*, float*);
52 static int TIFFWriteFloatArray(TIFF
*, TIFFDirEntry
*, float*);
53 static int TIFFWriteDoubleArray(TIFF
*, TIFFDirEntry
*, double*);
54 static int TIFFWriteByteArray(TIFF
*, TIFFDirEntry
*, char*);
55 static int TIFFWriteAnyArray(TIFF
*,
56 TIFFDataType
, ttag_t
, TIFFDirEntry
*, uint32
, double*);
57 static int TIFFWriteTransferFunction(TIFF
*, TIFFDirEntry
*);
58 static int TIFFWriteInkNames(TIFF
*, TIFFDirEntry
*);
59 static int TIFFWriteData(TIFF
*, TIFFDirEntry
*, char*);
60 static int TIFFLinkDirectory(TIFF
*);
62 #define WriteRationalPair(type, tag1, v1, tag2, v2) { \
63 TIFFWriteRational((tif), (type), (tag1), (dir), (v1)) \
64 TIFFWriteRational((tif), (type), (tag2), (dir)+1, (v2)) \
67 #define TIFFWriteRational(tif, type, tag, dir, v) \
68 (dir)->tdir_tag = (tag); \
69 (dir)->tdir_type = (type); \
70 (dir)->tdir_count = 1; \
71 if (!TIFFWriteRationalArray((tif), (dir), &(v))) \
75 * Write the contents of the current directory
76 * to the specified file. This routine doesn't
77 * handle overwriting a directory with auxiliary
78 * storage that's been changed.
81 _TIFFWriteDirectory(TIFF
* tif
, int done
)
91 unsigned long b
, fields
[FIELD_SETLONGS
];
94 if (tif
->tif_mode
== O_RDONLY
)
97 * Clear write state so that subsequent images with
98 * different characteristics get the right buffers
103 if (tif
->tif_flags
& TIFF_POSTENCODE
) {
104 tif
->tif_flags
&= ~TIFF_POSTENCODE
;
105 if (!(*tif
->tif_postencode
)(tif
)) {
106 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
,
107 "Error post-encoding before directory write");
111 (*tif
->tif_close
)(tif
); /* shutdown encoder */
113 * Flush any data that might have been written
114 * by the compression close+cleanup routines.
116 if (tif
->tif_rawcc
> 0 && !TIFFFlushData1(tif
)) {
117 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
,
118 "Error flushing data before directory write");
121 if ((tif
->tif_flags
& TIFF_MYBUFFER
) && tif
->tif_rawdata
) {
122 _TIFFfree(tif
->tif_rawdata
);
123 tif
->tif_rawdata
= NULL
;
125 tif
->tif_rawdatasize
= 0;
127 tif
->tif_flags
&= ~(TIFF_BEENWRITING
|TIFF_BUFFERSETUP
);
132 * Size the directory so that we can calculate
133 * offsets for the data items that aren't kept
134 * in-place in each field.
137 for (b
= 0; b
<= FIELD_LAST
; b
++)
138 if (TIFFFieldSet(tif
, b
) && b
!= FIELD_CUSTOM
)
139 nfields
+= (b
< FIELD_SUBFILETYPE
? 2 : 1);
140 nfields
+= td
->td_customValueCount
;
141 dirsize
= nfields
* sizeof (TIFFDirEntry
);
142 data
= (char*) _TIFFmalloc(dirsize
);
144 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
,
145 "Cannot write directory, out of space");
149 * Directory hasn't been placed yet, put
150 * it at the end of the file and link it
151 * into the existing directory structure.
153 if (tif
->tif_diroff
== 0 && !TIFFLinkDirectory(tif
))
155 tif
->tif_dataoff
= (toff_t
)(
156 tif
->tif_diroff
+ sizeof (uint16
) + dirsize
+ sizeof (toff_t
));
157 if (tif
->tif_dataoff
& 1)
159 (void) TIFFSeekFile(tif
, tif
->tif_dataoff
, SEEK_SET
);
161 dir
= (TIFFDirEntry
*) data
;
163 * Setup external form of directory
164 * entries and write data items.
166 _TIFFmemcpy(fields
, td
->td_fieldsset
, sizeof (fields
));
168 * Write out ExtraSamples tag only if
169 * extra samples are present in the data.
171 if (FieldSet(fields
, FIELD_EXTRASAMPLES
) && !td
->td_extrasamples
) {
172 ResetFieldBit(fields
, FIELD_EXTRASAMPLES
);
174 dirsize
-= sizeof (TIFFDirEntry
);
176 for (fi
= 0, nfi
= tif
->tif_nfields
; nfi
> 0; nfi
--, fi
++) {
177 const TIFFFieldInfo
* fip
= tif
->tif_fieldinfo
[fi
];
180 ** For custom fields, we test to see if the custom field
181 ** is set or not. For normal fields, we just use the
184 if( fip
->field_bit
== FIELD_CUSTOM
)
186 int ci
, is_set
= FALSE
;
188 for( ci
= 0; ci
< td
->td_customValueCount
; ci
++ )
189 is_set
|= (td
->td_customValues
[ci
].info
== fip
);
194 else if (!FieldSet(fields
, fip
->field_bit
))
199 ** Handle other fields.
201 switch (fip
->field_bit
)
203 case FIELD_STRIPOFFSETS
:
205 * We use one field bit for both strip and tile
207 * offsets, and so must be careful in selecting
208 * the appropriate field descriptor (so that tags
209 * are written in sorted order).
212 TIFFTAG_TILEOFFSETS
: TIFFTAG_STRIPOFFSETS
;
213 if (tag
!= fip
->field_tag
)
216 dir
->tdir_tag
= (uint16
) tag
;
217 dir
->tdir_type
= (uint16
) TIFF_LONG
;
218 dir
->tdir_count
= (uint32
) td
->td_nstrips
;
219 if (!TIFFWriteLongArray(tif
, dir
, td
->td_stripoffset
))
222 case FIELD_STRIPBYTECOUNTS
:
224 * We use one field bit for both strip and tile
225 * byte counts, and so must be careful in selecting
226 * the appropriate field descriptor (so that tags
227 * are written in sorted order).
230 TIFFTAG_TILEBYTECOUNTS
: TIFFTAG_STRIPBYTECOUNTS
;
231 if (tag
!= fip
->field_tag
)
234 dir
->tdir_tag
= (uint16
) tag
;
235 dir
->tdir_type
= (uint16
) TIFF_LONG
;
236 dir
->tdir_count
= (uint32
) td
->td_nstrips
;
237 if (!TIFFWriteLongArray(tif
, dir
,
238 td
->td_stripbytecount
))
241 case FIELD_ROWSPERSTRIP
:
242 TIFFSetupShortLong(tif
, TIFFTAG_ROWSPERSTRIP
,
243 dir
, td
->td_rowsperstrip
);
246 if (!TIFFWriteShortTable(tif
, TIFFTAG_COLORMAP
, dir
,
250 case FIELD_IMAGEDIMENSIONS
:
251 TIFFSetupShortLong(tif
, TIFFTAG_IMAGEWIDTH
,
252 dir
++, td
->td_imagewidth
);
253 TIFFSetupShortLong(tif
, TIFFTAG_IMAGELENGTH
,
254 dir
, td
->td_imagelength
);
256 case FIELD_TILEDIMENSIONS
:
257 TIFFSetupShortLong(tif
, TIFFTAG_TILEWIDTH
,
258 dir
++, td
->td_tilewidth
);
259 TIFFSetupShortLong(tif
, TIFFTAG_TILELENGTH
,
260 dir
, td
->td_tilelength
);
262 case FIELD_COMPRESSION
:
263 TIFFSetupShort(tif
, TIFFTAG_COMPRESSION
,
264 dir
, td
->td_compression
);
266 case FIELD_PHOTOMETRIC
:
267 TIFFSetupShort(tif
, TIFFTAG_PHOTOMETRIC
,
268 dir
, td
->td_photometric
);
271 WriteRationalPair(TIFF_RATIONAL
,
272 TIFFTAG_XPOSITION
, td
->td_xposition
,
273 TIFFTAG_YPOSITION
, td
->td_yposition
);
275 case FIELD_RESOLUTION
:
276 WriteRationalPair(TIFF_RATIONAL
,
277 TIFFTAG_XRESOLUTION
, td
->td_xresolution
,
278 TIFFTAG_YRESOLUTION
, td
->td_yresolution
);
280 case FIELD_BITSPERSAMPLE
:
281 case FIELD_MINSAMPLEVALUE
:
282 case FIELD_MAXSAMPLEVALUE
:
283 case FIELD_SAMPLEFORMAT
:
284 if (!TIFFWritePerSampleShorts(tif
, fip
->field_tag
, dir
))
287 case FIELD_SMINSAMPLEVALUE
:
288 case FIELD_SMAXSAMPLEVALUE
:
289 if (!TIFFWritePerSampleAnys(tif
,
290 _TIFFSampleToTagType(tif
), fip
->field_tag
, dir
))
293 case FIELD_PAGENUMBER
:
294 case FIELD_HALFTONEHINTS
:
295 case FIELD_YCBCRSUBSAMPLING
:
296 if (!TIFFSetupShortPair(tif
, fip
->field_tag
, dir
))
300 if (!TIFFWriteInkNames(tif
, dir
))
303 case FIELD_TRANSFERFUNCTION
:
304 if (!TIFFWriteTransferFunction(tif
, dir
))
309 * XXX: Always write this field using LONG type
310 * for backward compatibility.
312 dir
->tdir_tag
= (uint16
) fip
->field_tag
;
313 dir
->tdir_type
= (uint16
) TIFF_LONG
;
314 dir
->tdir_count
= (uint32
) td
->td_nsubifd
;
315 if (!TIFFWriteLongArray(tif
, dir
, td
->td_subifd
))
318 * Total hack: if this directory includes a SubIFD
319 * tag then force the next <n> directories to be
320 * written as ``sub directories'' of this one. This
321 * is used to write things like thumbnails and
322 * image masks that one wants to keep out of the
323 * normal directory linkage access mechanism.
325 if (dir
->tdir_count
> 0) {
326 tif
->tif_flags
|= TIFF_INSUBIFD
;
327 tif
->tif_nsubifd
= (uint16
) dir
->tdir_count
;
328 if (dir
->tdir_count
> 1)
329 tif
->tif_subifdoff
= dir
->tdir_offset
;
331 tif
->tif_subifdoff
= (uint32
)(
334 + ((char*)&dir
->tdir_offset
-data
));
338 /* XXX: Should be fixed and removed. */
339 if (fip
->field_tag
== TIFFTAG_DOTRANGE
) {
340 if (!TIFFSetupShortPair(tif
, fip
->field_tag
, dir
))
343 else if (!TIFFWriteNormalTag(tif
, dir
, fip
))
349 if( fip
->field_bit
!= FIELD_CUSTOM
)
350 ResetFieldBit(fields
, fip
->field_bit
);
356 dircount
= (uint16
) nfields
;
357 diroff
= (uint32
) tif
->tif_nextdiroff
;
358 if (tif
->tif_flags
& TIFF_SWAB
) {
360 * The file's byte order is opposite to the
361 * native machine architecture. We overwrite
362 * the directory information with impunity
363 * because it'll be released below after we
364 * write it to the file. Note that all the
365 * other tag construction routines assume that
366 * we do this byte-swapping; i.e. they only
367 * byte-swap indirect data.
369 for (dir
= (TIFFDirEntry
*) data
; dircount
; dir
++, dircount
--) {
370 TIFFSwabArrayOfShort(&dir
->tdir_tag
, 2);
371 TIFFSwabArrayOfLong(&dir
->tdir_count
, 2);
373 dircount
= (uint16
) nfields
;
374 TIFFSwabShort(&dircount
);
375 TIFFSwabLong(&diroff
);
377 (void) TIFFSeekFile(tif
, tif
->tif_diroff
, SEEK_SET
);
378 if (!WriteOK(tif
, &dircount
, sizeof (dircount
))) {
379 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
, "Error writing directory count");
382 if (!WriteOK(tif
, data
, dirsize
)) {
383 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
, "Error writing directory contents");
386 if (!WriteOK(tif
, &diroff
, sizeof (diroff
))) {
387 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
, "Error writing directory link");
391 TIFFFreeDirectory(tif
);
392 tif
->tif_flags
&= ~TIFF_DIRTYDIRECT
;
393 (*tif
->tif_cleanup
)(tif
);
396 * Reset directory-related state for subsequent
399 TIFFCreateDirectory(tif
);
407 #undef WriteRationalPair
410 TIFFWriteDirectory(TIFF
* tif
)
412 return _TIFFWriteDirectory(tif
, TRUE
);
416 * Similar to TIFFWriteDirectory(), writes the directory out
417 * but leaves all data structures in memory so that it can be
418 * written again. This will make a partially written TIFF file
419 * readable before it is successfully completed/closed.
422 TIFFCheckpointDirectory(TIFF
* tif
)
425 /* Setup the strips arrays, if they haven't already been. */
426 if (tif
->tif_dir
.td_stripoffset
== NULL
)
427 (void) TIFFSetupStrips(tif
);
428 rc
= _TIFFWriteDirectory(tif
, FALSE
);
429 (void) TIFFSetWriteOffset(tif
, TIFFSeekFile(tif
, 0, SEEK_END
));
434 * Process tags that are not special cased.
437 TIFFWriteNormalTag(TIFF
* tif
, TIFFDirEntry
* dir
, const TIFFFieldInfo
* fip
)
439 uint16 wc
= (uint16
) fip
->field_writecount
;
442 dir
->tdir_tag
= (uint16
) fip
->field_tag
;
443 dir
->tdir_type
= (uint16
) fip
->field_type
;
444 dir
->tdir_count
= wc
;
446 switch (fip
->field_type
) {
449 if (fip
->field_passcount
) {
451 if (wc
== (uint16
) TIFF_VARIABLE2
) {
452 TIFFGetField(tif
, fip
->field_tag
, &wc2
, &wp
);
453 dir
->tdir_count
= wc2
;
454 } else { /* Assume TIFF_VARIABLE */
455 TIFFGetField(tif
, fip
->field_tag
, &wc
, &wp
);
456 dir
->tdir_count
= wc
;
458 if (!TIFFWriteShortArray(tif
, dir
, wp
))
463 TIFFGetField(tif
, fip
->field_tag
, &sv
);
465 TIFFInsertData(tif
, dir
->tdir_type
, sv
);
468 TIFFGetField(tif
, fip
->field_tag
, &wp
);
469 if (!TIFFWriteShortArray(tif
, dir
, wp
))
477 if (fip
->field_passcount
) {
479 if (wc
== (uint16
) TIFF_VARIABLE2
) {
480 TIFFGetField(tif
, fip
->field_tag
, &wc2
, &lp
);
481 dir
->tdir_count
= wc2
;
482 } else { /* Assume TIFF_VARIABLE */
483 TIFFGetField(tif
, fip
->field_tag
, &wc
, &lp
);
484 dir
->tdir_count
= wc
;
486 if (!TIFFWriteLongArray(tif
, dir
, lp
))
490 /* XXX handle LONG->SHORT conversion */
491 TIFFGetField(tif
, fip
->field_tag
,
495 TIFFGetField(tif
, fip
->field_tag
, &lp
);
496 if (!TIFFWriteLongArray(tif
, dir
, lp
))
503 if (fip
->field_passcount
) {
505 if (wc
== (uint16
) TIFF_VARIABLE2
) {
506 TIFFGetField(tif
, fip
->field_tag
, &wc2
, &fp
);
507 dir
->tdir_count
= wc2
;
508 } else { /* Assume TIFF_VARIABLE */
509 TIFFGetField(tif
, fip
->field_tag
, &wc
, &fp
);
510 dir
->tdir_count
= wc
;
512 if (!TIFFWriteRationalArray(tif
, dir
, fp
))
517 TIFFGetField(tif
, fip
->field_tag
, &fv
);
518 if (!TIFFWriteRationalArray(tif
, dir
, &fv
))
522 TIFFGetField(tif
, fip
->field_tag
, &fp
);
523 if (!TIFFWriteRationalArray(tif
, dir
, fp
))
529 if (fip
->field_passcount
) {
531 if (wc
== (uint16
) TIFF_VARIABLE2
) {
532 TIFFGetField(tif
, fip
->field_tag
, &wc2
, &fp
);
533 dir
->tdir_count
= wc2
;
534 } else { /* Assume TIFF_VARIABLE */
535 TIFFGetField(tif
, fip
->field_tag
, &wc
, &fp
);
536 dir
->tdir_count
= wc
;
538 if (!TIFFWriteFloatArray(tif
, dir
, fp
))
543 TIFFGetField(tif
, fip
->field_tag
, &fv
);
544 if (!TIFFWriteFloatArray(tif
, dir
, &fv
))
548 TIFFGetField(tif
, fip
->field_tag
, &fp
);
549 if (!TIFFWriteFloatArray(tif
, dir
, fp
))
555 if (fip
->field_passcount
) {
557 if (wc
== (uint16
) TIFF_VARIABLE2
) {
558 TIFFGetField(tif
, fip
->field_tag
, &wc2
, &dp
);
559 dir
->tdir_count
= wc2
;
560 } else { /* Assume TIFF_VARIABLE */
561 TIFFGetField(tif
, fip
->field_tag
, &wc
, &dp
);
562 dir
->tdir_count
= wc
;
564 if (!TIFFWriteDoubleArray(tif
, dir
, dp
))
569 TIFFGetField(tif
, fip
->field_tag
, &dv
);
570 if (!TIFFWriteDoubleArray(tif
, dir
, &dv
))
574 TIFFGetField(tif
, fip
->field_tag
, &dp
);
575 if (!TIFFWriteDoubleArray(tif
, dir
, dp
))
583 if (fip
->field_passcount
)
584 TIFFGetField(tif
, fip
->field_tag
, &wc
, &cp
);
586 TIFFGetField(tif
, fip
->field_tag
, &cp
);
588 dir
->tdir_count
= (uint32
) (strlen(cp
) + 1);
589 if (!TIFFWriteByteArray(tif
, dir
, cp
))
596 if (fip
->field_passcount
) {
598 if (wc
== (uint16
) TIFF_VARIABLE2
) {
599 TIFFGetField(tif
, fip
->field_tag
, &wc2
, &cp
);
600 dir
->tdir_count
= wc2
;
601 } else { /* Assume TIFF_VARIABLE */
602 TIFFGetField(tif
, fip
->field_tag
, &wc
, &cp
);
603 dir
->tdir_count
= wc
;
605 if (!TIFFWriteByteArray(tif
, dir
, cp
))
610 TIFFGetField(tif
, fip
->field_tag
, &cv
);
611 if (!TIFFWriteByteArray(tif
, dir
, &cv
))
615 TIFFGetField(tif
, fip
->field_tag
, &cp
);
616 if (!TIFFWriteByteArray(tif
, dir
, cp
))
624 if (wc
== (unsigned short) TIFF_VARIABLE
) {
625 TIFFGetField(tif
, fip
->field_tag
, &wc
, &cp
);
626 dir
->tdir_count
= wc
;
627 } else if (wc
== (unsigned short) TIFF_VARIABLE2
) {
628 TIFFGetField(tif
, fip
->field_tag
, &wc2
, &cp
);
629 dir
->tdir_count
= wc2
;
631 TIFFGetField(tif
, fip
->field_tag
, &cp
);
632 if (!TIFFWriteByteArray(tif
, dir
, cp
))
644 * Setup a directory entry with either a SHORT
645 * or LONG type according to the value.
648 TIFFSetupShortLong(TIFF
* tif
, ttag_t tag
, TIFFDirEntry
* dir
, uint32 v
)
650 dir
->tdir_tag
= (uint16
) tag
;
653 dir
->tdir_type
= (short) TIFF_LONG
;
654 dir
->tdir_offset
= v
;
656 dir
->tdir_type
= (short) TIFF_SHORT
;
657 dir
->tdir_offset
= TIFFInsertData(tif
, (int) TIFF_SHORT
, v
);
662 * Setup a SHORT directory entry
665 TIFFSetupShort(TIFF
* tif
, ttag_t tag
, TIFFDirEntry
* dir
, uint16 v
)
667 dir
->tdir_tag
= (uint16
) tag
;
669 dir
->tdir_type
= (short) TIFF_SHORT
;
670 dir
->tdir_offset
= TIFFInsertData(tif
, (int) TIFF_SHORT
, v
);
672 #undef MakeShortDirent
674 #define NITEMS(x) (sizeof (x) / sizeof (x[0]))
676 * Setup a directory entry that references a
677 * samples/pixel array of SHORT values and
678 * (potentially) write the associated indirect
682 TIFFWritePerSampleShorts(TIFF
* tif
, ttag_t tag
, TIFFDirEntry
* dir
)
686 uint16 i
, samples
= tif
->tif_dir
.td_samplesperpixel
;
689 if (samples
> NITEMS(buf
)) {
690 w
= (uint16
*) _TIFFmalloc(samples
* sizeof (uint16
));
692 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
,
693 "No space to write per-sample shorts");
697 TIFFGetField(tif
, tag
, &v
);
698 for (i
= 0; i
< samples
; i
++)
701 dir
->tdir_tag
= (uint16
) tag
;
702 dir
->tdir_type
= (uint16
) TIFF_SHORT
;
703 dir
->tdir_count
= samples
;
704 status
= TIFFWriteShortArray(tif
, dir
, w
);
706 _TIFFfree((char*) w
);
711 * Setup a directory entry that references a samples/pixel array of ``type''
712 * values and (potentially) write the associated indirect values. The source
713 * data from TIFFGetField() for the specified tag must be returned as double.
716 TIFFWritePerSampleAnys(TIFF
* tif
,
717 TIFFDataType type
, ttag_t tag
, TIFFDirEntry
* dir
)
721 uint16 i
, samples
= tif
->tif_dir
.td_samplesperpixel
;
724 if (samples
> NITEMS(buf
)) {
725 w
= (double*) _TIFFmalloc(samples
* sizeof (double));
727 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
,
728 "No space to write per-sample values");
732 TIFFGetField(tif
, tag
, &v
);
733 for (i
= 0; i
< samples
; i
++)
735 status
= TIFFWriteAnyArray(tif
, type
, tag
, dir
, samples
, w
);
743 * Setup a pair of shorts that are returned by
744 * value, rather than as a reference to an array.
747 TIFFSetupShortPair(TIFF
* tif
, ttag_t tag
, TIFFDirEntry
* dir
)
751 TIFFGetField(tif
, tag
, &v
[0], &v
[1]);
753 dir
->tdir_tag
= (uint16
) tag
;
754 dir
->tdir_type
= (uint16
) TIFF_SHORT
;
756 return (TIFFWriteShortArray(tif
, dir
, v
));
760 * Setup a directory entry for an NxM table of shorts,
761 * where M is known to be 2**bitspersample, and write
762 * the associated indirect data.
765 TIFFWriteShortTable(TIFF
* tif
,
766 ttag_t tag
, TIFFDirEntry
* dir
, uint32 n
, uint16
** table
)
770 dir
->tdir_tag
= (uint16
) tag
;
771 dir
->tdir_type
= (short) TIFF_SHORT
;
772 /* XXX -- yech, fool TIFFWriteData */
773 dir
->tdir_count
= (uint32
) (1L<<tif
->tif_dir
.td_bitspersample
);
774 off
= tif
->tif_dataoff
;
775 for (i
= 0; i
< n
; i
++)
776 if (!TIFFWriteData(tif
, dir
, (char *)table
[i
]))
778 dir
->tdir_count
*= n
;
779 dir
->tdir_offset
= off
;
784 * Write/copy data associated with an ASCII or opaque tag value.
787 TIFFWriteByteArray(TIFF
* tif
, TIFFDirEntry
* dir
, char* cp
)
789 if (dir
->tdir_count
> 4) {
790 if (!TIFFWriteData(tif
, dir
, cp
))
793 _TIFFmemcpy(&dir
->tdir_offset
, cp
, dir
->tdir_count
);
798 * Setup a directory entry of an array of SHORT
799 * or SSHORT and write the associated indirect values.
802 TIFFWriteShortArray(TIFF
* tif
, TIFFDirEntry
* dir
, uint16
* v
)
804 if (dir
->tdir_count
<= 2) {
805 if (tif
->tif_header
.tiff_magic
== TIFF_BIGENDIAN
) {
806 dir
->tdir_offset
= (uint32
) ((long) v
[0] << 16);
807 if (dir
->tdir_count
== 2)
808 dir
->tdir_offset
|= v
[1] & 0xffff;
810 dir
->tdir_offset
= v
[0] & 0xffff;
811 if (dir
->tdir_count
== 2)
812 dir
->tdir_offset
|= (long) v
[1] << 16;
816 return (TIFFWriteData(tif
, dir
, (char*) v
));
820 * Setup a directory entry of an array of LONG
821 * or SLONG and write the associated indirect values.
824 TIFFWriteLongArray(TIFF
* tif
, TIFFDirEntry
* dir
, uint32
* v
)
826 if (dir
->tdir_count
== 1) {
827 dir
->tdir_offset
= v
[0];
830 return (TIFFWriteData(tif
, dir
, (char*) v
));
834 * Setup a directory entry of an array of RATIONAL
835 * or SRATIONAL and write the associated indirect values.
838 TIFFWriteRationalArray(TIFF
* tif
, TIFFDirEntry
* dir
, float* v
)
844 t
= (uint32
*) _TIFFmalloc(2 * dir
->tdir_count
* sizeof (uint32
));
846 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
,
847 "No space to write RATIONAL array");
850 for (i
= 0; i
< dir
->tdir_count
; i
++) {
856 if (dir
->tdir_type
== TIFF_RATIONAL
) {
857 TIFFWarningExt(tif
->tif_clientdata
, tif
->tif_name
,
858 "\"%s\": Information lost writing value (%g) as (unsigned) RATIONAL",
859 _TIFFFieldWithTag(tif
,dir
->tdir_tag
)->field_name
,
867 while (fv
< 1L<<(31-3) && den
< 1L<<(31-3))
868 fv
*= 1<<3, den
*= 1L<<3;
870 t
[2*i
+0] = (uint32
) (sign
* (fv
+ 0.5));
873 status
= TIFFWriteData(tif
, dir
, (char *)t
);
874 _TIFFfree((char*) t
);
879 TIFFWriteFloatArray(TIFF
* tif
, TIFFDirEntry
* dir
, float* v
)
881 TIFFCvtNativeToIEEEFloat(tif
, dir
->tdir_count
, v
);
882 if (dir
->tdir_count
== 1) {
883 dir
->tdir_offset
= *(uint32
*) &v
[0];
886 return (TIFFWriteData(tif
, dir
, (char*) v
));
890 TIFFWriteDoubleArray(TIFF
* tif
, TIFFDirEntry
* dir
, double* v
)
892 TIFFCvtNativeToIEEEDouble(tif
, dir
->tdir_count
, v
);
893 return (TIFFWriteData(tif
, dir
, (char*) v
));
897 * Write an array of ``type'' values for a specified tag (i.e. this is a tag
898 * which is allowed to have different types, e.g. SMaxSampleType).
899 * Internally the data values are represented as double since a double can
900 * hold any of the TIFF tag types (yes, this should really be an abstract
901 * type tany_t for portability). The data is converted into the specified
902 * type in a temporary buffer and then handed off to the appropriate array
906 TIFFWriteAnyArray(TIFF
* tif
,
907 TIFFDataType type
, ttag_t tag
, TIFFDirEntry
* dir
, uint32 n
, double* v
)
909 char buf
[10 * sizeof(double)];
913 if (n
* TIFFDataWidth(type
) > sizeof buf
) {
914 w
= (char*) _TIFFmalloc(n
* TIFFDataWidth(type
));
916 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
,
917 "No space to write array");
922 dir
->tdir_tag
= (uint16
) tag
;
923 dir
->tdir_type
= (uint16
) type
;
929 uint8
* bp
= (uint8
*) w
;
930 for (i
= 0; i
< (int) n
; i
++)
931 bp
[i
] = (uint8
) v
[i
];
932 if (!TIFFWriteByteArray(tif
, dir
, (char*) bp
))
938 int8
* bp
= (int8
*) w
;
939 for (i
= 0; i
< (int) n
; i
++)
941 if (!TIFFWriteByteArray(tif
, dir
, (char*) bp
))
947 uint16
* bp
= (uint16
*) w
;
948 for (i
= 0; i
< (int) n
; i
++)
949 bp
[i
] = (uint16
) v
[i
];
950 if (!TIFFWriteShortArray(tif
, dir
, (uint16
*)bp
))
956 int16
* bp
= (int16
*) w
;
957 for (i
= 0; i
< (int) n
; i
++)
958 bp
[i
] = (int16
) v
[i
];
959 if (!TIFFWriteShortArray(tif
, dir
, (uint16
*)bp
))
965 uint32
* bp
= (uint32
*) w
;
966 for (i
= 0; i
< (int) n
; i
++)
967 bp
[i
] = (uint32
) v
[i
];
968 if (!TIFFWriteLongArray(tif
, dir
, bp
))
974 int32
* bp
= (int32
*) w
;
975 for (i
= 0; i
< (int) n
; i
++)
976 bp
[i
] = (int32
) v
[i
];
977 if (!TIFFWriteLongArray(tif
, dir
, (uint32
*) bp
))
983 float* bp
= (float*) w
;
984 for (i
= 0; i
< (int) n
; i
++)
985 bp
[i
] = (float) v
[i
];
986 if (!TIFFWriteFloatArray(tif
, dir
, bp
))
991 return (TIFFWriteDoubleArray(tif
, dir
, v
));
1008 TIFFWriteTransferFunction(TIFF
* tif
, TIFFDirEntry
* dir
)
1010 TIFFDirectory
* td
= &tif
->tif_dir
;
1011 tsize_t n
= (1L<<td
->td_bitspersample
) * sizeof (uint16
);
1012 uint16
** tf
= td
->td_transferfunction
;
1016 * Check if the table can be written as a single column,
1017 * or if it must be written as 3 columns. Note that we
1018 * write a 3-column tag if there are 2 samples/pixel and
1019 * a single column of data won't suffice--hmm.
1021 switch (td
->td_samplesperpixel
- td
->td_extrasamples
) {
1022 default: if (_TIFFmemcmp(tf
[0], tf
[2], n
)) { ncols
= 3; break; }
1023 case 2: if (_TIFFmemcmp(tf
[0], tf
[1], n
)) { ncols
= 3; break; }
1024 case 1: case 0: ncols
= 1;
1026 return (TIFFWriteShortTable(tif
,
1027 TIFFTAG_TRANSFERFUNCTION
, dir
, ncols
, tf
));
1031 TIFFWriteInkNames(TIFF
* tif
, TIFFDirEntry
* dir
)
1033 TIFFDirectory
* td
= &tif
->tif_dir
;
1035 dir
->tdir_tag
= TIFFTAG_INKNAMES
;
1036 dir
->tdir_type
= (short) TIFF_ASCII
;
1037 dir
->tdir_count
= td
->td_inknameslen
;
1038 return (TIFFWriteByteArray(tif
, dir
, td
->td_inknames
));
1042 * Write a contiguous directory item.
1045 TIFFWriteData(TIFF
* tif
, TIFFDirEntry
* dir
, char* cp
)
1049 if (tif
->tif_flags
& TIFF_SWAB
) {
1050 switch (dir
->tdir_type
) {
1053 TIFFSwabArrayOfShort((uint16
*) cp
, dir
->tdir_count
);
1058 TIFFSwabArrayOfLong((uint32
*) cp
, dir
->tdir_count
);
1061 case TIFF_SRATIONAL
:
1062 TIFFSwabArrayOfLong((uint32
*) cp
, 2*dir
->tdir_count
);
1065 TIFFSwabArrayOfDouble((double*) cp
, dir
->tdir_count
);
1069 dir
->tdir_offset
= tif
->tif_dataoff
;
1070 cc
= dir
->tdir_count
* TIFFDataWidth((TIFFDataType
) dir
->tdir_type
);
1071 if (SeekOK(tif
, dir
->tdir_offset
) &&
1072 WriteOK(tif
, cp
, cc
)) {
1073 tif
->tif_dataoff
+= (cc
+ 1) & ~1;
1076 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
, "Error writing data for field \"%s\"",
1077 _TIFFFieldWithTag(tif
, dir
->tdir_tag
)->field_name
);
1082 * Similar to TIFFWriteDirectory(), but if the directory has already
1083 * been written once, it is relocated to the end of the file, in case it
1084 * has changed in size. Note that this will result in the loss of the
1085 * previously used directory space.
1089 TIFFRewriteDirectory( TIFF
*tif
)
1091 static const char module[] = "TIFFRewriteDirectory";
1093 /* We don't need to do anything special if it hasn't been written. */
1094 if( tif
->tif_diroff
== 0 )
1095 return TIFFWriteDirectory( tif
);
1098 ** Find and zero the pointer to this directory, so that TIFFLinkDirectory
1099 ** will cause it to be added after this directories current pre-link.
1102 /* Is it the first directory in the file? */
1103 if (tif
->tif_header
.tiff_diroff
== tif
->tif_diroff
)
1105 tif
->tif_header
.tiff_diroff
= 0;
1106 tif
->tif_diroff
= 0;
1108 TIFFSeekFile(tif
, (toff_t
)(TIFF_MAGIC_SIZE
+TIFF_VERSION_SIZE
),
1110 if (!WriteOK(tif
, &(tif
->tif_header
.tiff_diroff
),
1111 sizeof (tif
->tif_diroff
)))
1113 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
, "Error updating TIFF header");
1119 toff_t nextdir
, off
;
1121 nextdir
= tif
->tif_header
.tiff_diroff
;
1125 if (!SeekOK(tif
, nextdir
) ||
1126 !ReadOK(tif
, &dircount
, sizeof (dircount
))) {
1127 TIFFErrorExt(tif
->tif_clientdata
, module, "Error fetching directory count");
1130 if (tif
->tif_flags
& TIFF_SWAB
)
1131 TIFFSwabShort(&dircount
);
1132 (void) TIFFSeekFile(tif
,
1133 dircount
* sizeof (TIFFDirEntry
), SEEK_CUR
);
1134 if (!ReadOK(tif
, &nextdir
, sizeof (nextdir
))) {
1135 TIFFErrorExt(tif
->tif_clientdata
, module, "Error fetching directory link");
1138 if (tif
->tif_flags
& TIFF_SWAB
)
1139 TIFFSwabLong(&nextdir
);
1140 } while (nextdir
!= tif
->tif_diroff
&& nextdir
!= 0);
1141 off
= TIFFSeekFile(tif
, 0, SEEK_CUR
); /* get current offset */
1142 (void) TIFFSeekFile(tif
, off
- (toff_t
)sizeof(nextdir
), SEEK_SET
);
1143 tif
->tif_diroff
= 0;
1144 if (!WriteOK(tif
, &(tif
->tif_diroff
), sizeof (nextdir
))) {
1145 TIFFErrorExt(tif
->tif_clientdata
, module, "Error writing directory link");
1151 ** Now use TIFFWriteDirectory() normally.
1154 return TIFFWriteDirectory( tif
);
1159 * Link the current directory into the
1160 * directory chain for the file.
1163 TIFFLinkDirectory(TIFF
* tif
)
1165 static const char module[] = "TIFFLinkDirectory";
1169 tif
->tif_diroff
= (TIFFSeekFile(tif
, (toff_t
) 0, SEEK_END
)+1) &~ 1;
1170 diroff
= tif
->tif_diroff
;
1171 if (tif
->tif_flags
& TIFF_SWAB
)
1172 TIFFSwabLong(&diroff
);
1177 if (tif
->tif_flags
& TIFF_INSUBIFD
) {
1178 (void) TIFFSeekFile(tif
, tif
->tif_subifdoff
, SEEK_SET
);
1179 if (!WriteOK(tif
, &diroff
, sizeof (diroff
))) {
1180 TIFFErrorExt(tif
->tif_clientdata
, module,
1181 "%s: Error writing SubIFD directory link",
1186 * Advance to the next SubIFD or, if this is
1187 * the last one configured, revert back to the
1188 * normal directory linkage.
1190 if (--tif
->tif_nsubifd
)
1191 tif
->tif_subifdoff
+= sizeof (diroff
);
1193 tif
->tif_flags
&= ~TIFF_INSUBIFD
;
1197 if (tif
->tif_header
.tiff_diroff
== 0) {
1199 * First directory, overwrite offset in header.
1201 tif
->tif_header
.tiff_diroff
= tif
->tif_diroff
;
1202 (void) TIFFSeekFile(tif
,
1203 (toff_t
)(TIFF_MAGIC_SIZE
+TIFF_VERSION_SIZE
),
1205 if (!WriteOK(tif
, &diroff
, sizeof (diroff
))) {
1206 TIFFErrorExt(tif
->tif_clientdata
, tif
->tif_name
, "Error writing TIFF header");
1212 * Not the first directory, search to the last and append.
1214 nextdir
= tif
->tif_header
.tiff_diroff
;
1218 if (!SeekOK(tif
, nextdir
) ||
1219 !ReadOK(tif
, &dircount
, sizeof (dircount
))) {
1220 TIFFErrorExt(tif
->tif_clientdata
, module, "Error fetching directory count");
1223 if (tif
->tif_flags
& TIFF_SWAB
)
1224 TIFFSwabShort(&dircount
);
1225 (void) TIFFSeekFile(tif
,
1226 dircount
* sizeof (TIFFDirEntry
), SEEK_CUR
);
1227 if (!ReadOK(tif
, &nextdir
, sizeof (nextdir
))) {
1228 TIFFErrorExt(tif
->tif_clientdata
, module, "Error fetching directory link");
1231 if (tif
->tif_flags
& TIFF_SWAB
)
1232 TIFFSwabLong(&nextdir
);
1233 } while (nextdir
!= 0);
1234 off
= TIFFSeekFile(tif
, 0, SEEK_CUR
); /* get current offset */
1235 (void) TIFFSeekFile(tif
, off
- (toff_t
)sizeof(nextdir
), SEEK_SET
);
1236 if (!WriteOK(tif
, &diroff
, sizeof (diroff
))) {
1237 TIFFErrorExt(tif
->tif_clientdata
, module, "Error writing directory link");
1243 /* vim: set ts=8 sts=8 sw=8 noet: */