1 /*
2 * Copyright (c) 1988-1997 Sam Leffler
3 * Copyright (c) 1991-1997 Silicon Graphics, Inc.
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 /*
26 * TIFF Library
27 *
28 * Compression Scheme Configuration Support.
29 */
30 #include "tiffiop.h"
31
TIFFNoEncode(TIFF * tif,const char * method)32 static int TIFFNoEncode(TIFF *tif, const char *method)
33 {
34 const TIFFCodec *c = TIFFFindCODEC(tif->tif_dir.td_compression);
35
36 if (c)
37 {
38 TIFFErrorExtR(tif, tif->tif_name, "%s %s encoding is not implemented",
39 c->name, method);
40 }
41 else
42 {
43 TIFFErrorExtR(tif, tif->tif_name,
44 "Compression scheme %" PRIu16
45 " %s encoding is not implemented",
46 tif->tif_dir.td_compression, method);
47 }
48 return (-1);
49 }
50
_TIFFNoRowEncode(TIFF * tif,uint8_t * pp,tmsize_t cc,uint16_t s)51 int _TIFFNoRowEncode(TIFF *tif, uint8_t *pp, tmsize_t cc, uint16_t s)
52 {
53 (void)pp;
54 (void)cc;
55 (void)s;
56 return (TIFFNoEncode(tif, "scanline"));
57 }
58
_TIFFNoStripEncode(TIFF * tif,uint8_t * pp,tmsize_t cc,uint16_t s)59 int _TIFFNoStripEncode(TIFF *tif, uint8_t *pp, tmsize_t cc, uint16_t s)
60 {
61 (void)pp;
62 (void)cc;
63 (void)s;
64 return (TIFFNoEncode(tif, "strip"));
65 }
66
_TIFFNoTileEncode(TIFF * tif,uint8_t * pp,tmsize_t cc,uint16_t s)67 int _TIFFNoTileEncode(TIFF *tif, uint8_t *pp, tmsize_t cc, uint16_t s)
68 {
69 (void)pp;
70 (void)cc;
71 (void)s;
72 return (TIFFNoEncode(tif, "tile"));
73 }
74
TIFFNoDecode(TIFF * tif,const char * method)75 static int TIFFNoDecode(TIFF *tif, const char *method)
76 {
77 const TIFFCodec *c = TIFFFindCODEC(tif->tif_dir.td_compression);
78
79 if (c)
80 TIFFErrorExtR(tif, tif->tif_name, "%s %s decoding is not implemented",
81 c->name, method);
82 else
83 TIFFErrorExtR(tif, tif->tif_name,
84 "Compression scheme %" PRIu16
85 " %s decoding is not implemented",
86 tif->tif_dir.td_compression, method);
87 return (0);
88 }
89
_TIFFNoFixupTags(TIFF * tif)90 static int _TIFFNoFixupTags(TIFF *tif)
91 {
92 (void)tif;
93 return (1);
94 }
95
_TIFFNoRowDecode(TIFF * tif,uint8_t * pp,tmsize_t cc,uint16_t s)96 int _TIFFNoRowDecode(TIFF *tif, uint8_t *pp, tmsize_t cc, uint16_t s)
97 {
98 (void)pp;
99 (void)cc;
100 (void)s;
101 return (TIFFNoDecode(tif, "scanline"));
102 }
103
_TIFFNoStripDecode(TIFF * tif,uint8_t * pp,tmsize_t cc,uint16_t s)104 int _TIFFNoStripDecode(TIFF *tif, uint8_t *pp, tmsize_t cc, uint16_t s)
105 {
106 (void)pp;
107 (void)cc;
108 (void)s;
109 return (TIFFNoDecode(tif, "strip"));
110 }
111
_TIFFNoTileDecode(TIFF * tif,uint8_t * pp,tmsize_t cc,uint16_t s)112 int _TIFFNoTileDecode(TIFF *tif, uint8_t *pp, tmsize_t cc, uint16_t s)
113 {
114 (void)pp;
115 (void)cc;
116 (void)s;
117 return (TIFFNoDecode(tif, "tile"));
118 }
119
_TIFFNoSeek(TIFF * tif,uint32_t off)120 int _TIFFNoSeek(TIFF *tif, uint32_t off)
121 {
122 (void)off;
123 TIFFErrorExtR(tif, tif->tif_name,
124 "Compression algorithm does not support random access");
125 return (0);
126 }
127
_TIFFNoPreCode(TIFF * tif,uint16_t s)128 int _TIFFNoPreCode(TIFF *tif, uint16_t s)
129 {
130 (void)tif;
131 (void)s;
132 return (1);
133 }
134
_TIFFtrue(TIFF * tif)135 static int _TIFFtrue(TIFF *tif)
136 {
137 (void)tif;
138 return (1);
139 }
_TIFFvoid(TIFF * tif)140 static void _TIFFvoid(TIFF *tif) { (void)tif; }
141
_TIFFSetDefaultCompressionState(TIFF * tif)142 void _TIFFSetDefaultCompressionState(TIFF *tif)
143 {
144 tif->tif_fixuptags = _TIFFNoFixupTags;
145 tif->tif_decodestatus = TRUE;
146 tif->tif_setupdecode = _TIFFtrue;
147 tif->tif_predecode = _TIFFNoPreCode;
148 tif->tif_decoderow = _TIFFNoRowDecode;
149 tif->tif_decodestrip = _TIFFNoStripDecode;
150 tif->tif_decodetile = _TIFFNoTileDecode;
151 tif->tif_encodestatus = TRUE;
152 tif->tif_setupencode = _TIFFtrue;
153 tif->tif_preencode = _TIFFNoPreCode;
154 tif->tif_postencode = _TIFFtrue;
155 tif->tif_encoderow = _TIFFNoRowEncode;
156 tif->tif_encodestrip = _TIFFNoStripEncode;
157 tif->tif_encodetile = _TIFFNoTileEncode;
158 tif->tif_close = _TIFFvoid;
159 tif->tif_seek = _TIFFNoSeek;
160 tif->tif_cleanup = _TIFFvoid;
161 tif->tif_defstripsize = _TIFFDefaultStripSize;
162 tif->tif_deftilesize = _TIFFDefaultTileSize;
163 tif->tif_flags &= ~(TIFF_NOBITREV | TIFF_NOREADRAW);
164 }
165
TIFFSetCompressionScheme(TIFF * tif,int scheme)166 int TIFFSetCompressionScheme(TIFF *tif, int scheme)
167 {
168 const TIFFCodec *c = TIFFFindCODEC((uint16_t)scheme);
169
170 _TIFFSetDefaultCompressionState(tif);
171 /*
172 * Don't treat an unknown compression scheme as an error.
173 * This permits applications to open files with data that
174 * the library does not have builtin support for, but which
175 * may still be meaningful.
176 */
177 return (c ? (*c->init)(tif, scheme) : 1);
178 }
179
180 /*
181 * Other compression schemes may be registered. Registered
182 * schemes can also override the builtin versions provided
183 * by this library.
184 */
185 typedef struct _codec
186 {
187 struct _codec *next;
188 TIFFCodec *info;
189 } codec_t;
190 static codec_t *registeredCODECS = NULL;
191
TIFFFindCODEC(uint16_t scheme)192 const TIFFCodec *TIFFFindCODEC(uint16_t scheme)
193 {
194 const TIFFCodec *c;
195 codec_t *cd;
196
197 for (cd = registeredCODECS; cd; cd = cd->next)
198 if (cd->info->scheme == scheme)
199 return ((const TIFFCodec *)cd->info);
200 for (c = _TIFFBuiltinCODECS; c->name; c++)
201 if (c->scheme == scheme)
202 return (c);
203 return ((const TIFFCodec *)0);
204 }
205
TIFFRegisterCODEC(uint16_t scheme,const char * name,TIFFInitMethod init)206 TIFFCodec *TIFFRegisterCODEC(uint16_t scheme, const char *name,
207 TIFFInitMethod init)
208 {
209 codec_t *cd = (codec_t *)_TIFFmallocExt(
210 NULL,
211 (tmsize_t)(sizeof(codec_t) + sizeof(TIFFCodec) + strlen(name) + 1));
212
213 if (cd != NULL)
214 {
215 cd->info = (TIFFCodec *)((uint8_t *)cd + sizeof(codec_t));
216 cd->info->name = (char *)((uint8_t *)cd->info + sizeof(TIFFCodec));
217 strcpy(cd->info->name, name);
218 cd->info->scheme = scheme;
219 cd->info->init = init;
220 cd->next = registeredCODECS;
221 registeredCODECS = cd;
222 }
223 else
224 {
225 TIFFErrorExt(0, "TIFFRegisterCODEC",
226 "No space to register compression scheme %s", name);
227 return NULL;
228 }
229 return (cd->info);
230 }
231
TIFFUnRegisterCODEC(TIFFCodec * c)232 void TIFFUnRegisterCODEC(TIFFCodec *c)
233 {
234 codec_t *cd;
235 codec_t **pcd;
236
237 for (pcd = ®isteredCODECS; (cd = *pcd) != NULL; pcd = &cd->next)
238 if (cd->info == c)
239 {
240 *pcd = cd->next;
241 _TIFFfreeExt(NULL, cd);
242 return;
243 }
244 TIFFErrorExt(0, "TIFFUnRegisterCODEC",
245 "Cannot remove compression scheme %s; not registered",
246 c->name);
247 }
248
249 /************************************************************************/
250 /* TIFFGetConfisuredCODECs() */
251 /************************************************************************/
252
253 /**
254 * Get list of configured codecs, both built-in and registered by user.
255 * Caller is responsible to free this structure.
256 *
257 * @return returns array of TIFFCodec records (the last record should be NULL)
258 * or NULL if function failed.
259 */
260
TIFFGetConfiguredCODECs()261 TIFFCodec *TIFFGetConfiguredCODECs()
262 {
263 int i = 1;
264 codec_t *cd;
265 const TIFFCodec *c;
266 TIFFCodec *codecs = NULL;
267 TIFFCodec *new_codecs;
268
269 for (cd = registeredCODECS; cd; cd = cd->next)
270 {
271 new_codecs =
272 (TIFFCodec *)_TIFFreallocExt(NULL, codecs, i * sizeof(TIFFCodec));
273 if (!new_codecs)
274 {
275 _TIFFfreeExt(NULL, codecs);
276 return NULL;
277 }
278 codecs = new_codecs;
279 _TIFFmemcpy(codecs + i - 1, cd->info, sizeof(TIFFCodec));
280 i++;
281 }
282 for (c = _TIFFBuiltinCODECS; c->name; c++)
283 {
284 if (TIFFIsCODECConfigured(c->scheme))
285 {
286 new_codecs = (TIFFCodec *)_TIFFreallocExt(NULL, codecs,
287 i * sizeof(TIFFCodec));
288 if (!new_codecs)
289 {
290 _TIFFfreeExt(NULL, codecs);
291 return NULL;
292 }
293 codecs = new_codecs;
294 _TIFFmemcpy(codecs + i - 1, (const void *)c, sizeof(TIFFCodec));
295 i++;
296 }
297 }
298
299 new_codecs =
300 (TIFFCodec *)_TIFFreallocExt(NULL, codecs, i * sizeof(TIFFCodec));
301 if (!new_codecs)
302 {
303 _TIFFfreeExt(NULL, codecs);
304 return NULL;
305 }
306 codecs = new_codecs;
307 _TIFFmemset(codecs + i - 1, 0, sizeof(TIFFCodec));
308
309 return codecs;
310 }
311