Line data Source code
1 : /*
2 : * Copyright (c) 2017, Planet Labs
3 : * Author: <even.rouault at spatialys.com>
4 : *
5 : * Permission to use, copy, modify, distribute, and sell this software and
6 : * its documentation for any purpose is hereby granted without fee, provided
7 : * that (i) the above copyright notices and this permission notice appear in
8 : * all copies of the software and related documentation, and (ii) the names of
9 : * Sam Leffler and Silicon Graphics may not be used in any advertising or
10 : * publicity relating to the software without the specific, prior written
11 : * permission of Sam Leffler and Silicon Graphics.
12 : *
13 : * THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
14 : * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
15 : * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
16 : *
17 : * IN NO EVENT SHALL SAM LEFFLER OR SILICON GRAPHICS BE LIABLE FOR
18 : * ANY SPECIAL, INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND,
19 : * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
20 : * WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF
21 : * LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
22 : * OF THIS SOFTWARE.
23 : */
24 :
25 : #include "tiffiop.h"
26 : #ifdef ZSTD_SUPPORT
27 : /*
28 : * TIFF Library.
29 : *
30 : * ZSTD Compression Support
31 : *
32 : */
33 :
34 : #include "tif_predict.h"
35 : #include "zstd.h"
36 :
37 : #include <stdio.h>
38 :
39 : /*
40 : * State block for each open TIFF file using ZSTD compression/decompression.
41 : */
42 : typedef struct
43 : {
44 : TIFFPredictorState predict;
45 : ZSTD_DStream *dstream;
46 : ZSTD_CStream *cstream;
47 : int compression_level; /* compression level */
48 : ZSTD_outBuffer out_buffer;
49 : int state; /* state flags */
50 : #define LSTATE_INIT_DECODE 0x01
51 : #define LSTATE_INIT_ENCODE 0x02
52 :
53 : TIFFVGetMethod vgetparent; /* super-class method */
54 : TIFFVSetMethod vsetparent; /* super-class method */
55 : } ZSTDState;
56 :
57 : #define GetZSTDState(tif) ((ZSTDState *)(tif)->tif_data)
58 : #define ZSTDDecoderState(tif) GetZSTDState(tif)
59 : #define ZSTDEncoderState(tif) GetZSTDState(tif)
60 :
61 : static int ZSTDEncode(TIFF *tif, uint8_t *bp, tmsize_t cc, uint16_t s);
62 : static int ZSTDDecode(TIFF *tif, uint8_t *op, tmsize_t occ, uint16_t s);
63 :
64 705 : static int ZSTDFixupTags(TIFF *tif)
65 : {
66 : (void)tif;
67 705 : return 1;
68 : }
69 :
70 153 : static int ZSTDSetupDecode(TIFF *tif)
71 : {
72 153 : ZSTDState *sp = ZSTDDecoderState(tif);
73 :
74 153 : assert(sp != NULL);
75 :
76 : /* if we were last encoding, terminate this mode */
77 153 : if (sp->state & LSTATE_INIT_ENCODE)
78 : {
79 21 : ZSTD_freeCStream(sp->cstream);
80 21 : sp->cstream = NULL;
81 21 : sp->state = 0;
82 : }
83 :
84 153 : sp->state |= LSTATE_INIT_DECODE;
85 153 : return 1;
86 : }
87 :
88 : /*
89 : * Setup state for decoding a strip.
90 : */
91 495 : static int ZSTDPreDecode(TIFF *tif, uint16_t s)
92 : {
93 : static const char module[] = "ZSTDPreDecode";
94 495 : ZSTDState *sp = ZSTDDecoderState(tif);
95 : size_t zstd_ret;
96 :
97 : (void)s;
98 495 : assert(sp != NULL);
99 :
100 495 : if ((sp->state & LSTATE_INIT_DECODE) == 0)
101 21 : tif->tif_setupdecode(tif);
102 :
103 495 : if (sp->dstream == NULL)
104 : {
105 153 : sp->dstream = ZSTD_createDStream();
106 153 : if (sp->dstream == NULL)
107 : {
108 0 : TIFFErrorExtR(tif, module, "Cannot allocate decompression stream");
109 0 : return 0;
110 : }
111 : }
112 :
113 495 : zstd_ret = ZSTD_initDStream(sp->dstream);
114 495 : if (ZSTD_isError(zstd_ret))
115 : {
116 0 : TIFFErrorExtR(tif, module, "Error in ZSTD_initDStream(): %s",
117 : ZSTD_getErrorName(zstd_ret));
118 0 : return 0;
119 : }
120 :
121 495 : return 1;
122 : }
123 :
124 495 : static int ZSTDDecode(TIFF *tif, uint8_t *op, tmsize_t occ, uint16_t s)
125 : {
126 : static const char module[] = "ZSTDDecode";
127 495 : ZSTDState *sp = ZSTDDecoderState(tif);
128 : ZSTD_inBuffer in_buffer;
129 : ZSTD_outBuffer out_buffer;
130 : size_t zstd_ret;
131 :
132 : (void)s;
133 495 : assert(sp != NULL);
134 495 : assert(sp->state == LSTATE_INIT_DECODE);
135 :
136 495 : in_buffer.src = tif->tif_rawcp;
137 495 : in_buffer.size = (size_t)tif->tif_rawcc;
138 495 : in_buffer.pos = 0;
139 :
140 495 : out_buffer.dst = op;
141 495 : out_buffer.size = (size_t)occ;
142 495 : out_buffer.pos = 0;
143 :
144 : do
145 : {
146 495 : zstd_ret = ZSTD_decompressStream(sp->dstream, &out_buffer, &in_buffer);
147 495 : if (ZSTD_isError(zstd_ret))
148 : {
149 2 : memset(op + out_buffer.pos, 0, out_buffer.size - out_buffer.pos);
150 2 : TIFFErrorExtR(tif, module, "Error in ZSTD_decompressStream(): %s",
151 : ZSTD_getErrorName(zstd_ret));
152 2 : return 0;
153 : }
154 170 : } while (zstd_ret != 0 && in_buffer.pos < in_buffer.size &&
155 663 : out_buffer.pos < out_buffer.size);
156 :
157 493 : if (out_buffer.pos < (size_t)occ)
158 : {
159 0 : memset(op + out_buffer.pos, 0, out_buffer.size - out_buffer.pos);
160 0 : TIFFErrorExtR(tif, module,
161 : "Not enough data at scanline %lu (short %lu bytes)",
162 0 : (unsigned long)tif->tif_row,
163 0 : (unsigned long)((size_t)occ - out_buffer.pos));
164 0 : return 0;
165 : }
166 :
167 493 : tif->tif_rawcp += in_buffer.pos;
168 493 : tif->tif_rawcc -= in_buffer.pos;
169 :
170 493 : return 1;
171 : }
172 :
173 109 : static int ZSTDSetupEncode(TIFF *tif)
174 : {
175 109 : ZSTDState *sp = ZSTDEncoderState(tif);
176 :
177 109 : assert(sp != NULL);
178 109 : if (sp->state & LSTATE_INIT_DECODE)
179 : {
180 0 : ZSTD_freeDStream(sp->dstream);
181 0 : sp->dstream = NULL;
182 0 : sp->state = 0;
183 : }
184 :
185 109 : sp->state |= LSTATE_INIT_ENCODE;
186 109 : return 1;
187 : }
188 :
189 : /*
190 : * Reset encoding state at the start of a strip.
191 : */
192 282 : static int ZSTDPreEncode(TIFF *tif, uint16_t s)
193 : {
194 : static const char module[] = "ZSTDPreEncode";
195 282 : ZSTDState *sp = ZSTDEncoderState(tif);
196 : size_t zstd_ret;
197 :
198 : (void)s;
199 282 : assert(sp != NULL);
200 282 : if (sp->state != LSTATE_INIT_ENCODE)
201 0 : tif->tif_setupencode(tif);
202 :
203 282 : if (sp->cstream == NULL)
204 : {
205 109 : sp->cstream = ZSTD_createCStream();
206 109 : if (sp->cstream == NULL)
207 : {
208 0 : TIFFErrorExtR(tif, module, "Cannot allocate compression stream");
209 0 : return 0;
210 : }
211 : }
212 :
213 282 : zstd_ret = ZSTD_initCStream(sp->cstream, sp->compression_level);
214 282 : if (ZSTD_isError(zstd_ret))
215 : {
216 0 : TIFFErrorExtR(tif, module, "Error in ZSTD_initCStream(): %s",
217 : ZSTD_getErrorName(zstd_ret));
218 0 : return 0;
219 : }
220 :
221 282 : sp->out_buffer.dst = tif->tif_rawdata;
222 282 : sp->out_buffer.size = (size_t)tif->tif_rawdatasize;
223 282 : sp->out_buffer.pos = 0;
224 :
225 282 : return 1;
226 : }
227 :
228 : /*
229 : * Encode a chunk of pixels.
230 : */
231 282 : static int ZSTDEncode(TIFF *tif, uint8_t *bp, tmsize_t cc, uint16_t s)
232 : {
233 : static const char module[] = "ZSTDEncode";
234 282 : ZSTDState *sp = ZSTDEncoderState(tif);
235 : ZSTD_inBuffer in_buffer;
236 : size_t zstd_ret;
237 :
238 282 : assert(sp != NULL);
239 282 : assert(sp->state == LSTATE_INIT_ENCODE);
240 :
241 : (void)s;
242 :
243 282 : in_buffer.src = bp;
244 282 : in_buffer.size = (size_t)cc;
245 282 : in_buffer.pos = 0;
246 :
247 : do
248 : {
249 : zstd_ret =
250 282 : ZSTD_compressStream(sp->cstream, &sp->out_buffer, &in_buffer);
251 282 : if (ZSTD_isError(zstd_ret))
252 : {
253 0 : TIFFErrorExtR(tif, module, "Error in ZSTD_compressStream(): %s",
254 : ZSTD_getErrorName(zstd_ret));
255 0 : return 0;
256 : }
257 282 : if (sp->out_buffer.pos == sp->out_buffer.size)
258 : {
259 0 : tif->tif_rawcc = tif->tif_rawdatasize;
260 0 : if (!TIFFFlushData1(tif))
261 0 : return 0;
262 0 : sp->out_buffer.dst = tif->tif_rawcp;
263 0 : sp->out_buffer.pos = 0;
264 : }
265 282 : } while (in_buffer.pos < in_buffer.size);
266 :
267 282 : return 1;
268 : }
269 :
270 : /*
271 : * Finish off an encoded strip by flushing it.
272 : */
273 282 : static int ZSTDPostEncode(TIFF *tif)
274 : {
275 : static const char module[] = "ZSTDPostEncode";
276 282 : ZSTDState *sp = ZSTDEncoderState(tif);
277 : size_t zstd_ret;
278 :
279 : do
280 : {
281 282 : zstd_ret = ZSTD_endStream(sp->cstream, &sp->out_buffer);
282 282 : if (ZSTD_isError(zstd_ret))
283 : {
284 0 : TIFFErrorExtR(tif, module, "Error in ZSTD_endStream(): %s",
285 : ZSTD_getErrorName(zstd_ret));
286 0 : return 0;
287 : }
288 282 : if (sp->out_buffer.pos > 0)
289 : {
290 282 : tif->tif_rawcc = sp->out_buffer.pos;
291 282 : if (!TIFFFlushData1(tif))
292 0 : return 0;
293 282 : sp->out_buffer.dst = tif->tif_rawcp;
294 282 : sp->out_buffer.pos = 0;
295 : }
296 282 : } while (zstd_ret != 0);
297 282 : return 1;
298 : }
299 :
300 871 : static void ZSTDCleanup(TIFF *tif)
301 : {
302 871 : ZSTDState *sp = GetZSTDState(tif);
303 :
304 871 : assert(sp != 0);
305 :
306 871 : (void)TIFFPredictorCleanup(tif);
307 :
308 871 : tif->tif_tagmethods.vgetfield = sp->vgetparent;
309 871 : tif->tif_tagmethods.vsetfield = sp->vsetparent;
310 :
311 871 : if (sp->dstream)
312 : {
313 153 : ZSTD_freeDStream(sp->dstream);
314 153 : sp->dstream = NULL;
315 : }
316 871 : if (sp->cstream)
317 : {
318 88 : ZSTD_freeCStream(sp->cstream);
319 88 : sp->cstream = NULL;
320 : }
321 871 : _TIFFfreeExt(tif, sp);
322 871 : tif->tif_data = NULL;
323 :
324 871 : _TIFFSetDefaultCompressionState(tif);
325 871 : }
326 :
327 8265 : static int ZSTDVSetField(TIFF *tif, uint32_t tag, va_list ap)
328 : {
329 : static const char module[] = "ZSTDVSetField";
330 8265 : ZSTDState *sp = GetZSTDState(tif);
331 :
332 8265 : switch (tag)
333 : {
334 26 : case TIFFTAG_ZSTD_LEVEL:
335 26 : sp->compression_level = (int)va_arg(ap, int);
336 52 : if (sp->compression_level <= 0 ||
337 26 : sp->compression_level > ZSTD_maxCLevel())
338 : {
339 0 : TIFFWarningExtR(tif, module,
340 : "ZSTD_LEVEL should be between 1 and %d",
341 : ZSTD_maxCLevel());
342 : }
343 26 : return 1;
344 8239 : default:
345 8239 : return (*sp->vsetparent)(tif, tag, ap);
346 : }
347 : /*NOTREACHED*/
348 : }
349 :
350 8035 : static int ZSTDVGetField(TIFF *tif, uint32_t tag, va_list ap)
351 : {
352 8035 : ZSTDState *sp = GetZSTDState(tif);
353 :
354 8035 : switch (tag)
355 : {
356 0 : case TIFFTAG_ZSTD_LEVEL:
357 0 : *va_arg(ap, int *) = sp->compression_level;
358 0 : break;
359 8035 : default:
360 8035 : return (*sp->vgetparent)(tif, tag, ap);
361 : }
362 0 : return 1;
363 : }
364 :
365 : static const TIFFField ZSTDFields[] = {
366 : {TIFFTAG_ZSTD_LEVEL, 0, 0, TIFF_ANY, 0, TIFF_SETGET_INT, FIELD_PSEUDO, TRUE,
367 : FALSE, "ZSTD compression_level", NULL},
368 : };
369 :
370 871 : int TIFFInitZSTD(TIFF *tif, int scheme)
371 : {
372 : static const char module[] = "TIFFInitZSTD";
373 : ZSTDState *sp;
374 :
375 : (void)scheme;
376 871 : assert(scheme == COMPRESSION_ZSTD);
377 :
378 : /*
379 : * Merge codec-specific tag information.
380 : */
381 871 : if (!_TIFFMergeFields(tif, ZSTDFields, TIFFArrayCount(ZSTDFields)))
382 : {
383 0 : TIFFErrorExtR(tif, module, "Merging ZSTD codec-specific tags failed");
384 0 : return 0;
385 : }
386 :
387 : /*
388 : * Allocate state block so tag methods have storage to record values.
389 : */
390 871 : tif->tif_data = (uint8_t *)_TIFFmallocExt(tif, sizeof(ZSTDState));
391 871 : if (tif->tif_data == NULL)
392 0 : goto bad;
393 871 : sp = GetZSTDState(tif);
394 :
395 : /*
396 : * Override parent get/set field methods.
397 : */
398 871 : sp->vgetparent = tif->tif_tagmethods.vgetfield;
399 871 : tif->tif_tagmethods.vgetfield = ZSTDVGetField; /* hook for codec tags */
400 871 : sp->vsetparent = tif->tif_tagmethods.vsetfield;
401 871 : tif->tif_tagmethods.vsetfield = ZSTDVSetField; /* hook for codec tags */
402 :
403 : /* Default values for codec-specific fields */
404 871 : sp->compression_level = 9; /* default comp. level */
405 871 : sp->state = 0;
406 871 : sp->dstream = 0;
407 871 : sp->cstream = 0;
408 871 : sp->out_buffer.dst = NULL;
409 871 : sp->out_buffer.size = 0;
410 871 : sp->out_buffer.pos = 0;
411 :
412 : /*
413 : * Install codec methods.
414 : */
415 871 : tif->tif_fixuptags = ZSTDFixupTags;
416 871 : tif->tif_setupdecode = ZSTDSetupDecode;
417 871 : tif->tif_predecode = ZSTDPreDecode;
418 871 : tif->tif_decoderow = ZSTDDecode;
419 871 : tif->tif_decodestrip = ZSTDDecode;
420 871 : tif->tif_decodetile = ZSTDDecode;
421 871 : tif->tif_setupencode = ZSTDSetupEncode;
422 871 : tif->tif_preencode = ZSTDPreEncode;
423 871 : tif->tif_postencode = ZSTDPostEncode;
424 871 : tif->tif_encoderow = ZSTDEncode;
425 871 : tif->tif_encodestrip = ZSTDEncode;
426 871 : tif->tif_encodetile = ZSTDEncode;
427 871 : tif->tif_cleanup = ZSTDCleanup;
428 : /*
429 : * Setup predictor setup.
430 : */
431 871 : (void)TIFFPredictorInit(tif);
432 871 : return 1;
433 0 : bad:
434 0 : TIFFErrorExtR(tif, module, "No space for ZSTD state block");
435 0 : return 0;
436 : }
437 : #endif /* ZSTD_SUPPORT */
|