blob: 8ae0df783d46d4179f09639abc6abfa3a42a76d4 [file] [log] [blame]
cristy3ed852e2009-09-05 21:47:34 +00001/*
2%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3% %
4% %
5% %
6% IIIII CCCC OOO N N %
7% I C O O NN N %
8% I C O O N N N %
9% I C O O N NN %
10% IIIII CCCC OOO N N %
11% %
12% %
13% Read Microsoft Windows Icon Format %
14% %
15% Software Design %
cristyde984cd2013-12-01 14:49:27 +000016% Cristy %
cristy3ed852e2009-09-05 21:47:34 +000017% July 1992 %
18% %
19% %
Cristy7ce65e72015-12-12 18:03:16 -050020% Copyright 1999-2016 ImageMagick Studio LLC, a non-profit organization %
cristy3ed852e2009-09-05 21:47:34 +000021% dedicated to making software imaging solutions freely available. %
22% %
23% You may not use this file except in compliance with the License. You may %
24% obtain a copy of the License at %
25% %
26% http://www.imagemagick.org/script/license.php %
27% %
28% Unless required by applicable law or agreed to in writing, software %
29% distributed under the License is distributed on an "AS IS" BASIS, %
30% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %
31% See the License for the specific language governing permissions and %
32% limitations under the License. %
33% %
34%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
35%
36%
37*/
38
39/*
40 Include declarations.
41*/
cristy4c08aed2011-07-01 19:47:50 +000042#include "MagickCore/studio.h"
glennrpaa192b12012-01-17 21:35:21 +000043#include "MagickCore/artifact.h"
cristy4c08aed2011-07-01 19:47:50 +000044#include "MagickCore/blob.h"
45#include "MagickCore/blob-private.h"
46#include "MagickCore/cache.h"
47#include "MagickCore/colormap.h"
48#include "MagickCore/colorspace.h"
cristy3d9f5ba2012-06-26 13:37:31 +000049#include "MagickCore/colorspace-private.h"
cristy4c08aed2011-07-01 19:47:50 +000050#include "MagickCore/exception.h"
51#include "MagickCore/exception-private.h"
52#include "MagickCore/image.h"
53#include "MagickCore/image-private.h"
54#include "MagickCore/list.h"
55#include "MagickCore/log.h"
56#include "MagickCore/magick.h"
57#include "MagickCore/memory_.h"
58#include "MagickCore/monitor.h"
59#include "MagickCore/monitor-private.h"
cristy2c5fc272012-02-22 01:27:46 +000060#include "MagickCore/nt-base-private.h"
dirk9e137702014-01-09 19:29:59 +000061#include "MagickCore/option.h"
cristy4c08aed2011-07-01 19:47:50 +000062#include "MagickCore/pixel-accessor.h"
63#include "MagickCore/quantize.h"
64#include "MagickCore/quantum-private.h"
65#include "MagickCore/static.h"
66#include "MagickCore/string_.h"
67#include "MagickCore/module.h"
cristy3ed852e2009-09-05 21:47:34 +000068
69/*
70 Define declarations.
71*/
cristy07a3cca2012-12-10 13:09:10 +000072#if !defined(MAGICKCORE_WINDOWS_SUPPORT) || defined(__MINGW32__) || defined(__MINGW64__)
cristy3ed852e2009-09-05 21:47:34 +000073#define BI_RGB 0
74#define BI_RLE8 1
75#define BI_BITFIELDS 3
76#endif
77#define MaxIcons 1024
78
79/*
80 Typedef declarations.
81*/
82typedef struct _IconEntry
83{
84 unsigned char
85 width,
86 height,
87 colors,
88 reserved;
89
90 unsigned short int
91 planes,
92 bits_per_pixel;
93
cristybb503372010-05-27 20:51:26 +000094 size_t
cristy3ed852e2009-09-05 21:47:34 +000095 size,
96 offset;
97} IconEntry;
98
99typedef struct _IconFile
100{
101 short
102 reserved,
103 resource_type,
104 count;
105
106 IconEntry
107 directory[MaxIcons];
108} IconFile;
109
110typedef struct _IconInfo
111{
cristybb503372010-05-27 20:51:26 +0000112 size_t
cristy3ed852e2009-09-05 21:47:34 +0000113 file_size,
114 ba_offset,
115 offset_bits,
116 size;
117
cristybb503372010-05-27 20:51:26 +0000118 ssize_t
cristy3ed852e2009-09-05 21:47:34 +0000119 width,
120 height;
121
122 unsigned short
123 planes,
124 bits_per_pixel;
125
cristybb503372010-05-27 20:51:26 +0000126 size_t
cristy3ed852e2009-09-05 21:47:34 +0000127 compression,
128 image_size,
129 x_pixels,
130 y_pixels,
131 number_colors,
132 red_mask,
133 green_mask,
134 blue_mask,
135 alpha_mask,
136 colors_important;
137
cristybb503372010-05-27 20:51:26 +0000138 ssize_t
cristy3ed852e2009-09-05 21:47:34 +0000139 colorspace;
140} IconInfo;
141
142/*
143 Forward declaractions.
144*/
dirk501a11e2014-01-19 16:22:18 +0000145static Image
146 *AutoResizeImage(const Image *,const char *,MagickOffsetType *,
147 ExceptionInfo *);
148
cristy3ed852e2009-09-05 21:47:34 +0000149static MagickBooleanType
cristy1e178e72011-08-28 19:44:34 +0000150 WriteICONImage(const ImageInfo *,Image *,ExceptionInfo *);
dirk501a11e2014-01-19 16:22:18 +0000151
152Image *AutoResizeImage(const Image *image,const char *option,
153 MagickOffsetType *count,ExceptionInfo *exception)
154{
155 #define MAX_SIZES 16
156
157 char
158 *q;
159
160 const char
161 *p;
162
163 Image
164 *resized,
165 *images;
166
167 register ssize_t
168 i;
169
170 size_t
171 sizes[MAX_SIZES]={256,192,128,96,64,48,40,32,24,16};
172
173 images=NULL;
174 *count=0;
175 i=0;
176 p=option;
177 while (*p != '\0' && i < MAX_SIZES)
178 {
179 size_t
180 size;
181
182 while ((isspace((int) ((unsigned char) *p)) != 0))
183 p++;
184
185 size=(size_t)strtol(p,&q,10);
cristy0f6fc2d2015-05-30 00:49:11 +0000186 if ((p == q) || (size < 16) || (size > 256))
187 return((Image *) NULL);
dirk501a11e2014-01-19 16:22:18 +0000188
189 p=q;
190 sizes[i++]=size;
191
192 while ((isspace((int) ((unsigned char) *p)) != 0) || (*p == ','))
193 p++;
194 }
195
196 if (i==0)
197 i=10;
198 *count=i;
199 for (i=0; i < *count; i++)
200 {
201 resized=ResizeImage(image,sizes[i],sizes[i],image->filter,exception);
202 if (resized == (Image *) NULL)
203 return(DestroyImageList(images));
204
205 if (images == (Image *) NULL)
206 images=resized;
207 else
208 AppendImageToList(&images,resized);
209 }
210 return(images);
211}
cristy3ed852e2009-09-05 21:47:34 +0000212
213/*
214%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
215% %
216% %
217% %
218% R e a d I C O N I m a g e %
219% %
220% %
221% %
222%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
223%
224% ReadICONImage() reads a Microsoft icon image file and returns it. It
225% allocates the memory necessary for the new Image structure and returns a
226% pointer to the new image.
227%
228% The format of the ReadICONImage method is:
229%
230% Image *ReadICONImage(const ImageInfo *image_info,
231% ExceptionInfo *exception)
232%
233% A description of each parameter follows:
234%
235% o image_info: the image info.
236%
237% o exception: return any errors or warnings in this structure.
238%
239*/
240static Image *ReadICONImage(const ImageInfo *image_info,
241 ExceptionInfo *exception)
242{
243 IconFile
244 icon_file;
245
246 IconInfo
247 icon_info;
248
249 Image
250 *image;
251
cristy3ed852e2009-09-05 21:47:34 +0000252 MagickBooleanType
253 status;
254
cristybb503372010-05-27 20:51:26 +0000255 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +0000256 i,
257 x;
258
cristy4c08aed2011-07-01 19:47:50 +0000259 register Quantum
cristy3ed852e2009-09-05 21:47:34 +0000260 *q;
261
262 register unsigned char
263 *p;
264
cristybb503372010-05-27 20:51:26 +0000265 size_t
cristy3ed852e2009-09-05 21:47:34 +0000266 bit,
cristyeaedf062010-05-29 22:36:02 +0000267 byte,
cristy3ed852e2009-09-05 21:47:34 +0000268 bytes_per_line,
cristyeaedf062010-05-29 22:36:02 +0000269 one,
cristy3ed852e2009-09-05 21:47:34 +0000270 scanline_pad;
271
cristyebc891a2011-04-24 23:04:16 +0000272 ssize_t
273 count,
274 offset,
275 y;
276
cristy3ed852e2009-09-05 21:47:34 +0000277 /*
278 Open image file.
279 */
280 assert(image_info != (const ImageInfo *) NULL);
cristye1c94d92015-06-28 12:16:33 +0000281 assert(image_info->signature == MagickCoreSignature);
cristy3ed852e2009-09-05 21:47:34 +0000282 (void) LogMagickEvent(CoderEvent,GetMagickModule(),"%s",image_info->filename);
283 assert(exception != (ExceptionInfo *) NULL);
cristye1c94d92015-06-28 12:16:33 +0000284 assert(exception->signature == MagickCoreSignature);
cristy1f1de182011-10-05 18:41:34 +0000285 image=AcquireImage(image_info,exception);
cristy3ed852e2009-09-05 21:47:34 +0000286 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
287 if (status == MagickFalse)
288 {
289 image=DestroyImageList(image);
290 return((Image *) NULL);
291 }
292 icon_file.reserved=(short) ReadBlobLSBShort(image);
293 icon_file.resource_type=(short) ReadBlobLSBShort(image);
294 icon_file.count=(short) ReadBlobLSBShort(image);
295 if ((icon_file.reserved != 0) ||
296 ((icon_file.resource_type != 1) && (icon_file.resource_type != 2)) ||
297 (icon_file.count > MaxIcons))
298 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
299 for (i=0; i < icon_file.count; i++)
300 {
301 icon_file.directory[i].width=(unsigned char) ReadBlobByte(image);
302 icon_file.directory[i].height=(unsigned char) ReadBlobByte(image);
303 icon_file.directory[i].colors=(unsigned char) ReadBlobByte(image);
304 icon_file.directory[i].reserved=(unsigned char) ReadBlobByte(image);
305 icon_file.directory[i].planes=(unsigned short) ReadBlobLSBShort(image);
306 icon_file.directory[i].bits_per_pixel=(unsigned short)
307 ReadBlobLSBShort(image);
308 icon_file.directory[i].size=ReadBlobLSBLong(image);
309 icon_file.directory[i].offset=ReadBlobLSBLong(image);
cristy9c4177b2015-01-06 12:53:14 +0000310 if (EOFBlob(image) != MagickFalse)
Cristyc6c70bb2016-04-04 16:22:30 -0400311 break;
cristy3ed852e2009-09-05 21:47:34 +0000312 }
Cristyc6c70bb2016-04-04 16:22:30 -0400313 if (EOFBlob(image) != MagickFalse)
314 ThrowReaderException(CorruptImageError,"UnexpectedEndOfFile");
cristyeaedf062010-05-29 22:36:02 +0000315 one=1;
cristy3ed852e2009-09-05 21:47:34 +0000316 for (i=0; i < icon_file.count; i++)
317 {
318 /*
319 Verify Icon identifier.
320 */
321 offset=(ssize_t) SeekBlob(image,(MagickOffsetType)
322 icon_file.directory[i].offset,SEEK_SET);
323 if (offset < 0)
324 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
325 icon_info.size=ReadBlobLSBLong(image);
cristyd63c0502011-01-03 15:45:11 +0000326 icon_info.width=(unsigned char) ((int) ReadBlobLSBLong(image));
327 icon_info.height=(unsigned char) ((int) ReadBlobLSBLong(image)/2);
cristy4e09e032011-10-05 19:42:10 +0000328 icon_info.planes=ReadBlobLSBShort(image);
329 icon_info.bits_per_pixel=ReadBlobLSBShort(image);
cristy995156f2015-01-05 22:42:52 +0000330 if (EOFBlob(image) != MagickFalse)
331 {
332 ThrowFileException(exception,CorruptImageError,"UnexpectedEndOfFile",
333 image->filename);
334 break;
335 }
cristy43602772014-01-29 14:07:29 +0000336 if (((icon_info.planes == 18505) && (icon_info.bits_per_pixel == 21060)) ||
337 (icon_info.size == 0x474e5089))
cristy3ed852e2009-09-05 21:47:34 +0000338 {
339 Image
340 *icon_image;
341
342 ImageInfo
343 *read_info;
344
cristyf054ee72011-10-05 17:04:08 +0000345 size_t
346 length;
347
348 unsigned char
349 *png;
350
cristy3ed852e2009-09-05 21:47:34 +0000351 /*
352 Icon image encoded as a compressed PNG image.
353 */
cristyf054ee72011-10-05 17:04:08 +0000354 length=icon_file.directory[i].size;
cristy0f6fc2d2015-05-30 00:49:11 +0000355 if (~length < 16)
356 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristy4e09e032011-10-05 19:42:10 +0000357 png=(unsigned char *) AcquireQuantumMemory(length+16,sizeof(*png));
cristyf054ee72011-10-05 17:04:08 +0000358 if (png == (unsigned char *) NULL)
359 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
360 (void) CopyMagickMemory(png,"\211PNG\r\n\032\n\000\000\000\015",12);
cristy4e09e032011-10-05 19:42:10 +0000361 png[12]=(unsigned char) icon_info.planes;
362 png[13]=(unsigned char) (icon_info.planes >> 8);
363 png[14]=(unsigned char) icon_info.bits_per_pixel;
364 png[15]=(unsigned char) (icon_info.bits_per_pixel >> 8);
365 count=ReadBlob(image,length-16,png+16);
dirk833908d2014-01-28 19:26:32 +0000366 icon_image=(Image *) NULL;
367 if (count > 0)
cristyf054ee72011-10-05 17:04:08 +0000368 {
dirk833908d2014-01-28 19:26:32 +0000369 read_info=CloneImageInfo(image_info);
cristy151b66d2015-04-15 10:50:31 +0000370 (void) CopyMagickString(read_info->magick,"PNG",MagickPathExtent);
dirk833908d2014-01-28 19:26:32 +0000371 icon_image=BlobToImage(read_info,png,length+16,exception);
372 read_info=DestroyImageInfo(read_info);
cristyf054ee72011-10-05 17:04:08 +0000373 }
cristyf054ee72011-10-05 17:04:08 +0000374 png=(unsigned char *) RelinquishMagickMemory(png);
cristy47de5a92011-10-05 01:47:42 +0000375 if (icon_image == (Image *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000376 {
dirk833908d2014-01-28 19:26:32 +0000377 if (count != (ssize_t) (length-16))
378 ThrowReaderException(CorruptImageError,
379 "InsufficientImageDataInFile");
cristy47de5a92011-10-05 01:47:42 +0000380 image=DestroyImageList(image);
381 return((Image *) NULL);
cristy3ed852e2009-09-05 21:47:34 +0000382 }
cristy47de5a92011-10-05 01:47:42 +0000383 DestroyBlob(icon_image);
384 icon_image->blob=ReferenceBlob(image->blob);
385 ReplaceImageInList(&image,icon_image);
cristy3ed852e2009-09-05 21:47:34 +0000386 }
387 else
388 {
cristyf054ee72011-10-05 17:04:08 +0000389 if (icon_info.bits_per_pixel > 32)
390 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
391 icon_info.compression=ReadBlobLSBLong(image);
392 icon_info.image_size=ReadBlobLSBLong(image);
393 icon_info.x_pixels=ReadBlobLSBLong(image);
394 icon_info.y_pixels=ReadBlobLSBLong(image);
395 icon_info.number_colors=ReadBlobLSBLong(image);
396 icon_info.colors_important=ReadBlobLSBLong(image);
cristy8a46d822012-08-28 23:32:39 +0000397 image->alpha_trait=BlendPixelTrait;
cristyf054ee72011-10-05 17:04:08 +0000398 image->columns=(size_t) icon_file.directory[i].width;
399 if ((ssize_t) image->columns > icon_info.width)
400 image->columns=(size_t) icon_info.width;
cristy4e09e032011-10-05 19:42:10 +0000401 if (image->columns == 0)
402 image->columns=256;
cristyf054ee72011-10-05 17:04:08 +0000403 image->rows=(size_t) icon_file.directory[i].height;
404 if ((ssize_t) image->rows > icon_info.height)
405 image->rows=(size_t) icon_info.height;
cristy4e09e032011-10-05 19:42:10 +0000406 if (image->rows == 0)
407 image->rows=256;
cristyf054ee72011-10-05 17:04:08 +0000408 image->depth=icon_info.bits_per_pixel;
409 if (image->debug != MagickFalse)
410 {
411 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
412 " scene = %.20g",(double) i);
413 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
414 " size = %.20g",(double) icon_info.size);
415 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
416 " width = %.20g",(double) icon_file.directory[i].width);
417 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
418 " height = %.20g",(double) icon_file.directory[i].height);
419 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
420 " colors = %.20g",(double ) icon_info.number_colors);
421 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
422 " planes = %.20g",(double) icon_info.planes);
423 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
424 " bpp = %.20g",(double) icon_info.bits_per_pixel);
425 }
cristya2fb29a2014-09-21 22:04:52 +0000426 if ((icon_info.number_colors != 0) || (icon_info.bits_per_pixel <= 16U))
cristyf054ee72011-10-05 17:04:08 +0000427 {
428 image->storage_class=PseudoClass;
429 image->colors=icon_info.number_colors;
430 if (image->colors == 0)
431 image->colors=one << icon_info.bits_per_pixel;
432 }
433 if (image->storage_class == PseudoClass)
434 {
435 register ssize_t
436 i;
cristy3ed852e2009-09-05 21:47:34 +0000437
cristyf054ee72011-10-05 17:04:08 +0000438 unsigned char
439 *icon_colormap;
440
441 /*
442 Read Icon raster colormap.
443 */
dirk833908d2014-01-28 19:26:32 +0000444 if (AcquireImageColormap(image,image->colors,exception) ==
445 MagickFalse)
cristyf054ee72011-10-05 17:04:08 +0000446 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
447 icon_colormap=(unsigned char *) AcquireQuantumMemory((size_t)
448 image->colors,4UL*sizeof(*icon_colormap));
449 if (icon_colormap == (unsigned char *) NULL)
450 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
451 count=ReadBlob(image,(size_t) (4*image->colors),icon_colormap);
452 if (count != (ssize_t) (4*image->colors))
453 ThrowReaderException(CorruptImageError,
454 "InsufficientImageDataInFile");
455 p=icon_colormap;
456 for (i=0; i < (ssize_t) image->colors; i++)
457 {
458 image->colormap[i].blue=(Quantum) ScaleCharToQuantum(*p++);
459 image->colormap[i].green=(Quantum) ScaleCharToQuantum(*p++);
460 image->colormap[i].red=(Quantum) ScaleCharToQuantum(*p++);
461 p++;
462 }
463 icon_colormap=(unsigned char *) RelinquishMagickMemory(icon_colormap);
464 }
cristy3ed852e2009-09-05 21:47:34 +0000465 /*
466 Convert Icon raster image to pixel packets.
467 */
cristyf054ee72011-10-05 17:04:08 +0000468 if ((image_info->ping != MagickFalse) &&
469 (image_info->number_scenes != 0))
470 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
471 break;
cristyacabb842014-12-14 23:36:33 +0000472 status=SetImageExtent(image,image->columns,image->rows,exception);
473 if (status == MagickFalse)
474 return(DestroyImageList(image));
cristy3ed852e2009-09-05 21:47:34 +0000475 bytes_per_line=(((image->columns*icon_info.bits_per_pixel)+31) &
476 ~31) >> 3;
cristyda16f162011-02-19 23:52:17 +0000477 (void) bytes_per_line;
cristy3ed852e2009-09-05 21:47:34 +0000478 scanline_pad=((((image->columns*icon_info.bits_per_pixel)+31) & ~31)-
479 (image->columns*icon_info.bits_per_pixel)) >> 3;
480 switch (icon_info.bits_per_pixel)
481 {
482 case 1:
483 {
484 /*
485 Convert bitmap scanline.
486 */
cristybb503372010-05-27 20:51:26 +0000487 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000488 {
489 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000490 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000491 break;
cristybb503372010-05-27 20:51:26 +0000492 for (x=0; x < (ssize_t) (image->columns-7); x+=8)
cristy3ed852e2009-09-05 21:47:34 +0000493 {
cristyf054ee72011-10-05 17:04:08 +0000494 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000495 for (bit=0; bit < 8; bit++)
cristy4c08aed2011-07-01 19:47:50 +0000496 {
497 SetPixelIndex(image,((byte & (0x80 >> bit)) != 0 ? 0x01 :
498 0x00),q);
cristyed231572011-07-14 02:18:59 +0000499 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000500 }
cristy3ed852e2009-09-05 21:47:34 +0000501 }
502 if ((image->columns % 8) != 0)
503 {
cristyf054ee72011-10-05 17:04:08 +0000504 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000505 for (bit=0; bit < (image->columns % 8); bit++)
cristy4c08aed2011-07-01 19:47:50 +0000506 {
507 SetPixelIndex(image,((byte & (0x80 >> bit)) != 0 ? 0x01 :
508 0x00),q);
cristyed231572011-07-14 02:18:59 +0000509 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000510 }
cristy3ed852e2009-09-05 21:47:34 +0000511 }
cristybb503372010-05-27 20:51:26 +0000512 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000513 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000514 if (SyncAuthenticPixels(image,exception) == MagickFalse)
515 break;
516 if (image->previous == (Image *) NULL)
517 {
518 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
519 image->rows);
520 if (status == MagickFalse)
521 break;
522 }
523 }
524 break;
525 }
526 case 4:
527 {
528 /*
529 Read 4-bit Icon scanline.
530 */
cristybb503372010-05-27 20:51:26 +0000531 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000532 {
533 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000534 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000535 break;
cristybb503372010-05-27 20:51:26 +0000536 for (x=0; x < ((ssize_t) image->columns-1); x+=2)
cristy3ed852e2009-09-05 21:47:34 +0000537 {
cristyf054ee72011-10-05 17:04:08 +0000538 byte=(size_t) ReadBlobByte(image);
cristy4c08aed2011-07-01 19:47:50 +0000539 SetPixelIndex(image,((byte >> 4) & 0xf),q);
cristyed231572011-07-14 02:18:59 +0000540 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000541 SetPixelIndex(image,((byte) & 0xf),q);
cristyed231572011-07-14 02:18:59 +0000542 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000543 }
544 if ((image->columns % 2) != 0)
545 {
cristyf054ee72011-10-05 17:04:08 +0000546 byte=(size_t) ReadBlobByte(image);
cristy4c08aed2011-07-01 19:47:50 +0000547 SetPixelIndex(image,((byte >> 4) & 0xf),q);
cristyed231572011-07-14 02:18:59 +0000548 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000549 }
cristybb503372010-05-27 20:51:26 +0000550 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000551 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000552 if (SyncAuthenticPixels(image,exception) == MagickFalse)
553 break;
554 if (image->previous == (Image *) NULL)
555 {
556 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
557 image->rows);
558 if (status == MagickFalse)
559 break;
560 }
561 }
562 break;
563 }
564 case 8:
565 {
566 /*
567 Convert PseudoColor scanline.
568 */
cristybb503372010-05-27 20:51:26 +0000569 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000570 {
571 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000572 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000573 break;
cristybb503372010-05-27 20:51:26 +0000574 for (x=0; x < (ssize_t) image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +0000575 {
cristyf054ee72011-10-05 17:04:08 +0000576 byte=(size_t) ReadBlobByte(image);
dirk1678aec2015-08-30 12:07:29 +0200577 SetPixelIndex(image,(Quantum) byte,q);
cristyed231572011-07-14 02:18:59 +0000578 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000579 }
cristybb503372010-05-27 20:51:26 +0000580 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000581 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000582 if (SyncAuthenticPixels(image,exception) == MagickFalse)
583 break;
584 if (image->previous == (Image *) NULL)
585 {
586 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
587 image->rows);
588 if (status == MagickFalse)
589 break;
590 }
591 }
592 break;
593 }
594 case 16:
595 {
596 /*
597 Convert PseudoColor scanline.
598 */
cristybb503372010-05-27 20:51:26 +0000599 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000600 {
601 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000602 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000603 break;
cristybb503372010-05-27 20:51:26 +0000604 for (x=0; x < (ssize_t) image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +0000605 {
cristyf054ee72011-10-05 17:04:08 +0000606 byte=(size_t) ReadBlobByte(image);
607 byte|=(size_t) (ReadBlobByte(image) << 8);
dirk1678aec2015-08-30 12:07:29 +0200608 SetPixelIndex(image,(Quantum) byte,q);
cristyed231572011-07-14 02:18:59 +0000609 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000610 }
cristybb503372010-05-27 20:51:26 +0000611 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000612 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000613 if (SyncAuthenticPixels(image,exception) == MagickFalse)
614 break;
615 if (image->previous == (Image *) NULL)
616 {
617 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
618 image->rows);
619 if (status == MagickFalse)
620 break;
621 }
622 }
623 break;
624 }
625 case 24:
626 case 32:
627 {
628 /*
629 Convert DirectColor scanline.
630 */
cristybb503372010-05-27 20:51:26 +0000631 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000632 {
633 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000634 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000635 break;
cristybb503372010-05-27 20:51:26 +0000636 for (x=0; x < (ssize_t) image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +0000637 {
cristyf054ee72011-10-05 17:04:08 +0000638 SetPixelBlue(image,ScaleCharToQuantum((unsigned char)
639 ReadBlobByte(image)),q);
640 SetPixelGreen(image,ScaleCharToQuantum((unsigned char)
641 ReadBlobByte(image)),q);
642 SetPixelRed(image,ScaleCharToQuantum((unsigned char)
643 ReadBlobByte(image)),q);
cristy3ed852e2009-09-05 21:47:34 +0000644 if (icon_info.bits_per_pixel == 32)
cristyf054ee72011-10-05 17:04:08 +0000645 SetPixelAlpha(image,ScaleCharToQuantum((unsigned char)
646 ReadBlobByte(image)),q);
cristyed231572011-07-14 02:18:59 +0000647 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000648 }
649 if (icon_info.bits_per_pixel == 24)
cristybb503372010-05-27 20:51:26 +0000650 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000651 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000652 if (SyncAuthenticPixels(image,exception) == MagickFalse)
653 break;
654 if (image->previous == (Image *) NULL)
655 {
656 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
657 image->rows);
658 if (status == MagickFalse)
659 break;
660 }
661 }
662 break;
663 }
664 default:
665 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
666 }
Cristye0fe9892016-04-05 09:55:07 -0400667 if ((image_info->ping == MagickFalse) &&
668 (icon_info.bits_per_pixel <= 16))
cristyea1a8aa2011-10-20 13:24:06 +0000669 (void) SyncImage(image,exception);
cristy3ed852e2009-09-05 21:47:34 +0000670 if (icon_info.bits_per_pixel != 32)
671 {
672 /*
673 Read the ICON alpha mask.
674 */
675 image->storage_class=DirectClass;
cristybb503372010-05-27 20:51:26 +0000676 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000677 {
678 q=GetAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000679 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000680 break;
cristybb503372010-05-27 20:51:26 +0000681 for (x=0; x < ((ssize_t) image->columns-7); x+=8)
cristy3ed852e2009-09-05 21:47:34 +0000682 {
cristyf054ee72011-10-05 17:04:08 +0000683 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000684 for (bit=0; bit < 8; bit++)
cristy4c08aed2011-07-01 19:47:50 +0000685 {
686 SetPixelAlpha(image,(((byte & (0x80 >> bit)) != 0) ?
687 TransparentAlpha : OpaqueAlpha),q);
cristyed231572011-07-14 02:18:59 +0000688 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000689 }
cristy3ed852e2009-09-05 21:47:34 +0000690 }
691 if ((image->columns % 8) != 0)
692 {
cristyf054ee72011-10-05 17:04:08 +0000693 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000694 for (bit=0; bit < (image->columns % 8); bit++)
cristy4c08aed2011-07-01 19:47:50 +0000695 {
696 SetPixelAlpha(image,(((byte & (0x80 >> bit)) != 0) ?
697 TransparentAlpha : OpaqueAlpha),q);
cristyed231572011-07-14 02:18:59 +0000698 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000699 }
cristy3ed852e2009-09-05 21:47:34 +0000700 }
cristy0e8200f2009-09-13 21:10:06 +0000701 if ((image->columns % 32) != 0)
cristybb503372010-05-27 20:51:26 +0000702 for (x=0; x < (ssize_t) ((32-(image->columns % 32))/8); x++)
cristyf054ee72011-10-05 17:04:08 +0000703 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000704 if (SyncAuthenticPixels(image,exception) == MagickFalse)
705 break;
706 }
707 }
708 if (EOFBlob(image) != MagickFalse)
709 {
710 ThrowFileException(exception,CorruptImageError,
711 "UnexpectedEndOfFile",image->filename);
712 break;
713 }
714 }
715 /*
716 Proceed to next image.
717 */
718 if (image_info->number_scenes != 0)
719 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
720 break;
cristybb503372010-05-27 20:51:26 +0000721 if (i < (ssize_t) (icon_file.count-1))
cristy3ed852e2009-09-05 21:47:34 +0000722 {
723 /*
724 Allocate next image structure.
725 */
cristy1f1de182011-10-05 18:41:34 +0000726 AcquireNextImage(image_info,image,exception);
cristy3ed852e2009-09-05 21:47:34 +0000727 if (GetNextImageInList(image) == (Image *) NULL)
728 {
729 image=DestroyImageList(image);
730 return((Image *) NULL);
731 }
732 image=SyncNextImageInList(image);
733 status=SetImageProgress(image,LoadImagesTag,TellBlob(image),
734 GetBlobSize(image));
735 if (status == MagickFalse)
736 break;
737 }
738 }
739 (void) CloseBlob(image);
740 return(GetFirstImageInList(image));
741}
742
743/*
744%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
745% %
746% %
747% %
748% R e g i s t e r I C O N I m a g e %
749% %
750% %
751% %
752%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
753%
754% RegisterICONImage() adds attributes for the Icon image format to
755% the list of supported formats. The attributes include the image format
756% tag, a method to read and/or write the format, whether the format
757% supports the saving of more than one frame to the same file or blob,
758% whether the format supports native in-memory I/O, and a brief
759% description of the format.
760%
761% The format of the RegisterICONImage method is:
762%
cristybb503372010-05-27 20:51:26 +0000763% size_t RegisterICONImage(void)
cristy3ed852e2009-09-05 21:47:34 +0000764%
765*/
cristybb503372010-05-27 20:51:26 +0000766ModuleExport size_t RegisterICONImage(void)
cristy3ed852e2009-09-05 21:47:34 +0000767{
768 MagickInfo
769 *entry;
770
dirk06b627a2015-04-06 18:59:17 +0000771 entry=AcquireMagickInfo("ICON","CUR","Microsoft icon");
cristy3ed852e2009-09-05 21:47:34 +0000772 entry->decoder=(DecodeImageHandler *) ReadICONImage;
773 entry->encoder=(EncodeImageHandler *) WriteICONImage;
dirk08e9a112015-02-22 01:51:41 +0000774 entry->flags^=CoderAdjoinFlag;
775 entry->flags|=CoderSeekableStreamFlag;
cristy3ed852e2009-09-05 21:47:34 +0000776 (void) RegisterMagickInfo(entry);
dirk06b627a2015-04-06 18:59:17 +0000777 entry=AcquireMagickInfo("ICON","ICO","Microsoft icon");
cristy3ed852e2009-09-05 21:47:34 +0000778 entry->decoder=(DecodeImageHandler *) ReadICONImage;
779 entry->encoder=(EncodeImageHandler *) WriteICONImage;
dirk08e9a112015-02-22 01:51:41 +0000780 entry->flags|=CoderSeekableStreamFlag;
cristy3ed852e2009-09-05 21:47:34 +0000781 (void) RegisterMagickInfo(entry);
dirk06b627a2015-04-06 18:59:17 +0000782 entry=AcquireMagickInfo("ICON","ICON","Microsoft icon");
cristy3ed852e2009-09-05 21:47:34 +0000783 entry->decoder=(DecodeImageHandler *) ReadICONImage;
784 entry->encoder=(EncodeImageHandler *) WriteICONImage;
dirk08e9a112015-02-22 01:51:41 +0000785 entry->flags^=CoderAdjoinFlag;
786 entry->flags|=CoderSeekableStreamFlag;
cristy3ed852e2009-09-05 21:47:34 +0000787 (void) RegisterMagickInfo(entry);
788 return(MagickImageCoderSignature);
789}
790
791/*
792%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
793% %
794% %
795% %
796% U n r e g i s t e r I C O N I m a g e %
797% %
798% %
799% %
800%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
801%
802% UnregisterICONImage() removes format registrations made by the
803% ICON module from the list of supported formats.
804%
805% The format of the UnregisterICONImage method is:
806%
807% UnregisterICONImage(void)
808%
809*/
810ModuleExport void UnregisterICONImage(void)
811{
812 (void) UnregisterMagickInfo("CUR");
813 (void) UnregisterMagickInfo("ICO");
814 (void) UnregisterMagickInfo("ICON");
815}
816
817/*
818%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
819% %
820% %
821% %
822% W r i t e I C O N I m a g e %
823% %
824% %
825% %
826%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
827%
828% WriteICONImage() writes an image in Microsoft Windows bitmap encoded
829% image format, version 3 for Windows or (if the image has a matte channel)
830% version 4.
831%
cristy83cb91e2012-10-31 00:58:22 +0000832% It encodes any subimage as a compressed PNG image ("BI_PNG)", only when its
833% dimensions are 256x256 and image->compression is undefined or is defined as
834% ZipCompression.
glennrp4b708b82012-07-17 19:15:58 +0000835%
cristy3ed852e2009-09-05 21:47:34 +0000836% The format of the WriteICONImage method is:
837%
838% MagickBooleanType WriteICONImage(const ImageInfo *image_info,
cristy1e178e72011-08-28 19:44:34 +0000839% Image *image,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +0000840%
841% A description of each parameter follows.
842%
843% o image_info: the image info.
844%
845% o image: The image.
846%
cristy1e178e72011-08-28 19:44:34 +0000847% o exception: return any errors or warnings in this structure.
848%
cristy3ed852e2009-09-05 21:47:34 +0000849*/
850static MagickBooleanType WriteICONImage(const ImageInfo *image_info,
cristy1e178e72011-08-28 19:44:34 +0000851 Image *image,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +0000852{
dirk501a11e2014-01-19 16:22:18 +0000853 const char
854 *option;
855
cristy3ed852e2009-09-05 21:47:34 +0000856 IconFile
857 icon_file;
858
859 IconInfo
860 icon_info;
861
862 Image
dirk9e137702014-01-09 19:29:59 +0000863 *images,
cristy3ed852e2009-09-05 21:47:34 +0000864 *next;
865
cristy3ed852e2009-09-05 21:47:34 +0000866 MagickBooleanType
867 status;
868
869 MagickOffsetType
870 offset,
871 scene;
872
cristy4c08aed2011-07-01 19:47:50 +0000873 register const Quantum
cristy3ed852e2009-09-05 21:47:34 +0000874 *p;
875
cristybb503372010-05-27 20:51:26 +0000876 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +0000877 i,
878 x;
879
880 register unsigned char
881 *q;
882
cristyebc891a2011-04-24 23:04:16 +0000883 size_t
884 bytes_per_line,
885 scanline_pad;
886
887 ssize_t
888 y;
889
cristy3ed852e2009-09-05 21:47:34 +0000890 unsigned char
891 bit,
892 byte,
893 *pixels;
894
cristy3ed852e2009-09-05 21:47:34 +0000895 /*
896 Open output image file.
897 */
898 assert(image_info != (const ImageInfo *) NULL);
cristye1c94d92015-06-28 12:16:33 +0000899 assert(image_info->signature == MagickCoreSignature);
cristy3ed852e2009-09-05 21:47:34 +0000900 assert(image != (Image *) NULL);
cristye1c94d92015-06-28 12:16:33 +0000901 assert(image->signature == MagickCoreSignature);
cristy3ed852e2009-09-05 21:47:34 +0000902 (void) LogMagickEvent(CoderEvent,GetMagickModule(),"%s",image->filename);
cristy3a37efd2011-08-28 20:31:03 +0000903 assert(exception != (ExceptionInfo *) NULL);
cristye1c94d92015-06-28 12:16:33 +0000904 assert(exception->signature == MagickCoreSignature);
cristy1e178e72011-08-28 19:44:34 +0000905 status=OpenBlob(image_info,image,WriteBinaryBlobMode,exception);
cristy3ed852e2009-09-05 21:47:34 +0000906 if (status == MagickFalse)
907 return(status);
dirk9e137702014-01-09 19:29:59 +0000908 images=(Image *) NULL;
dirk501a11e2014-01-19 16:22:18 +0000909 option=GetImageOption(image_info,"icon:auto-resize");
910 if (option != (const char *) NULL)
dirk9e137702014-01-09 19:29:59 +0000911 {
dirk501a11e2014-01-19 16:22:18 +0000912 images=AutoResizeImage(image,option,&scene,exception);
dirk9e137702014-01-09 19:29:59 +0000913 if (images == (Image *) NULL)
dirk501a11e2014-01-19 16:22:18 +0000914 ThrowWriterException(ImageError,"InvalidDimensions");
dirk9e137702014-01-09 19:29:59 +0000915 }
916 else
917 {
918 scene=0;
919 next=image;
920 do
921 {
922 if ((image->columns > 256L) || (image->rows > 256L))
923 ThrowWriterException(ImageError,"WidthOrHeightExceedsLimit");
924 scene++;
925 next=SyncNextImageInList(next);
926 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
927 }
cristy3ed852e2009-09-05 21:47:34 +0000928 /*
929 Dump out a ICON header template to be properly initialized later.
930 */
931 (void) WriteBlobLSBShort(image,0);
932 (void) WriteBlobLSBShort(image,1);
933 (void) WriteBlobLSBShort(image,(unsigned char) scene);
934 (void) ResetMagickMemory(&icon_file,0,sizeof(icon_file));
935 (void) ResetMagickMemory(&icon_info,0,sizeof(icon_info));
936 scene=0;
dirk9e137702014-01-09 19:29:59 +0000937 next=(images != (Image *) NULL) ? images : image;
cristy3ed852e2009-09-05 21:47:34 +0000938 do
939 {
940 (void) WriteBlobByte(image,icon_file.directory[scene].width);
941 (void) WriteBlobByte(image,icon_file.directory[scene].height);
942 (void) WriteBlobByte(image,icon_file.directory[scene].colors);
943 (void) WriteBlobByte(image,icon_file.directory[scene].reserved);
944 (void) WriteBlobLSBShort(image,icon_file.directory[scene].planes);
945 (void) WriteBlobLSBShort(image,icon_file.directory[scene].bits_per_pixel);
cristy0b29b252010-05-30 01:59:46 +0000946 (void) WriteBlobLSBLong(image,(unsigned int)
947 icon_file.directory[scene].size);
948 (void) WriteBlobLSBLong(image,(unsigned int)
949 icon_file.directory[scene].offset);
cristy3ed852e2009-09-05 21:47:34 +0000950 scene++;
951 next=SyncNextImageInList(next);
952 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
953 scene=0;
dirk9e137702014-01-09 19:29:59 +0000954 next=(images != (Image *) NULL) ? images : image;
cristy3ed852e2009-09-05 21:47:34 +0000955 do
956 {
glennrp4b708b82012-07-17 19:15:58 +0000957 if ((next->columns > 255L) && (next->rows > 255L) &&
958 ((next->compression == UndefinedCompression) ||
959 (next->compression == ZipCompression)))
cristy3ed852e2009-09-05 21:47:34 +0000960 {
961 Image
962 *write_image;
963
964 ImageInfo
965 *write_info;
966
967 size_t
968 length;
969
970 unsigned char
971 *png;
972
cristy1e178e72011-08-28 19:44:34 +0000973 write_image=CloneImage(next,0,0,MagickTrue,exception);
dirkc1e52002014-01-09 20:22:42 +0000974 if (write_image == (Image *) NULL)
975 {
976 images=DestroyImageList(images);
977 return(MagickFalse);
dirk9e137702014-01-09 19:29:59 +0000978 }
cristy3ed852e2009-09-05 21:47:34 +0000979 write_info=CloneImageInfo(image_info);
cristy151b66d2015-04-15 10:50:31 +0000980 (void) CopyMagickString(write_info->filename,"PNG:",MagickPathExtent);
glennrp4b708b82012-07-17 19:15:58 +0000981
982 /* Don't write any ancillary chunks except for gAMA */
983 (void) SetImageArtifact(write_image,"png:include-chunk","none,gama");
984
985 /* Only write PNG32 formatted PNG (32-bit RGBA), 8 bits per channel */
986 (void) SetImageArtifact(write_image,"png:format","png32");
987
cristy3ed852e2009-09-05 21:47:34 +0000988 png=(unsigned char *) ImageToBlob(write_info,write_image,&length,
cristy1e178e72011-08-28 19:44:34 +0000989 exception);
Cristy15bc0d62015-12-17 12:36:54 -0500990 write_image=DestroyImageList(write_image);
cristy3ed852e2009-09-05 21:47:34 +0000991 write_info=DestroyImageInfo(write_info);
dirkc1e52002014-01-09 20:22:42 +0000992 if (png == (unsigned char *) NULL)
993 {
994 images=DestroyImageList(images);
995 return(MagickFalse);
dirk9e137702014-01-09 19:29:59 +0000996 }
cristy3ed852e2009-09-05 21:47:34 +0000997 icon_file.directory[scene].width=0;
998 icon_file.directory[scene].height=0;
999 icon_file.directory[scene].colors=0;
1000 icon_file.directory[scene].reserved=0;
1001 icon_file.directory[scene].planes=1;
1002 icon_file.directory[scene].bits_per_pixel=32;
cristybb503372010-05-27 20:51:26 +00001003 icon_file.directory[scene].size=(size_t) length;
1004 icon_file.directory[scene].offset=(size_t) TellBlob(image);
cristy3ed852e2009-09-05 21:47:34 +00001005 (void) WriteBlob(image,(size_t) length,png);
1006 png=(unsigned char *) RelinquishMagickMemory(png);
1007 }
1008 else
1009 {
1010 /*
1011 Initialize ICON raster file header.
1012 */
cristyaf8d3912014-02-21 14:50:33 +00001013 (void) TransformImageColorspace(next,sRGBColorspace,exception);
cristy3ed852e2009-09-05 21:47:34 +00001014 icon_info.file_size=14+12+28;
1015 icon_info.offset_bits=icon_info.file_size;
1016 icon_info.compression=BI_RGB;
1017 if ((next->storage_class != DirectClass) && (next->colors > 256))
cristy1e178e72011-08-28 19:44:34 +00001018 (void) SetImageStorageClass(next,DirectClass,exception);
cristy3ed852e2009-09-05 21:47:34 +00001019 if (next->storage_class == DirectClass)
1020 {
1021 /*
1022 Full color ICON raster.
1023 */
1024 icon_info.number_colors=0;
1025 icon_info.bits_per_pixel=32;
cristybb503372010-05-27 20:51:26 +00001026 icon_info.compression=(size_t) BI_RGB;
cristy3ed852e2009-09-05 21:47:34 +00001027 }
1028 else
1029 {
cristy35ef8242010-06-03 16:24:13 +00001030 size_t
1031 one;
1032
cristy3ed852e2009-09-05 21:47:34 +00001033 /*
1034 Colormapped ICON raster.
1035 */
1036 icon_info.bits_per_pixel=8;
1037 if (next->colors <= 256)
1038 icon_info.bits_per_pixel=8;
1039 if (next->colors <= 16)
1040 icon_info.bits_per_pixel=4;
1041 if (next->colors <= 2)
1042 icon_info.bits_per_pixel=1;
cristy35ef8242010-06-03 16:24:13 +00001043 one=1;
1044 icon_info.number_colors=one << icon_info.bits_per_pixel;
cristy3ed852e2009-09-05 21:47:34 +00001045 if (icon_info.number_colors < next->colors)
1046 {
cristy1e178e72011-08-28 19:44:34 +00001047 (void) SetImageStorageClass(next,DirectClass,exception);
cristy3ed852e2009-09-05 21:47:34 +00001048 icon_info.number_colors=0;
1049 icon_info.bits_per_pixel=(unsigned short) 24;
cristybb503372010-05-27 20:51:26 +00001050 icon_info.compression=(size_t) BI_RGB;
cristy3ed852e2009-09-05 21:47:34 +00001051 }
1052 else
1053 {
cristy0b29b252010-05-30 01:59:46 +00001054 size_t
1055 one;
1056
1057 one=1;
1058 icon_info.file_size+=3*(one << icon_info.bits_per_pixel);
1059 icon_info.offset_bits+=3*(one << icon_info.bits_per_pixel);
1060 icon_info.file_size+=(one << icon_info.bits_per_pixel);
1061 icon_info.offset_bits+=(one << icon_info.bits_per_pixel);
cristy3ed852e2009-09-05 21:47:34 +00001062 }
1063 }
1064 bytes_per_line=(((next->columns*icon_info.bits_per_pixel)+31) &
1065 ~31) >> 3;
1066 icon_info.ba_offset=0;
cristybb503372010-05-27 20:51:26 +00001067 icon_info.width=(ssize_t) next->columns;
1068 icon_info.height=(ssize_t) next->rows;
cristy3ed852e2009-09-05 21:47:34 +00001069 icon_info.planes=1;
1070 icon_info.image_size=bytes_per_line*next->rows;
1071 icon_info.size=40;
1072 icon_info.size+=(4*icon_info.number_colors);
1073 icon_info.size+=icon_info.image_size;
1074 icon_info.size+=(((icon_info.width+31) & ~31) >> 3)*icon_info.height;
1075 icon_info.file_size+=icon_info.image_size;
1076 icon_info.x_pixels=0;
1077 icon_info.y_pixels=0;
1078 switch (next->units)
1079 {
1080 case UndefinedResolution:
1081 case PixelsPerInchResolution:
1082 {
cristy2a11bef2011-10-28 18:33:11 +00001083 icon_info.x_pixels=(size_t) (100.0*next->resolution.x/2.54);
1084 icon_info.y_pixels=(size_t) (100.0*next->resolution.y/2.54);
cristy3ed852e2009-09-05 21:47:34 +00001085 break;
1086 }
1087 case PixelsPerCentimeterResolution:
1088 {
cristy2a11bef2011-10-28 18:33:11 +00001089 icon_info.x_pixels=(size_t) (100.0*next->resolution.x);
1090 icon_info.y_pixels=(size_t) (100.0*next->resolution.y);
cristy3ed852e2009-09-05 21:47:34 +00001091 break;
1092 }
1093 }
1094 icon_info.colors_important=icon_info.number_colors;
1095 /*
1096 Convert MIFF to ICON raster pixels.
1097 */
1098 pixels=(unsigned char *) AcquireQuantumMemory((size_t)
1099 icon_info.image_size,sizeof(*pixels));
dirkc1e52002014-01-09 20:22:42 +00001100 if (pixels == (unsigned char *) NULL)
1101 {
dirk9e137702014-01-09 19:29:59 +00001102 images=DestroyImageList(images);
1103 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
dirkc1e52002014-01-09 20:22:42 +00001104 }
cristy3ed852e2009-09-05 21:47:34 +00001105 (void) ResetMagickMemory(pixels,0,(size_t) icon_info.image_size);
1106 switch (icon_info.bits_per_pixel)
1107 {
1108 case 1:
1109 {
cristybb503372010-05-27 20:51:26 +00001110 size_t
cristy3ed852e2009-09-05 21:47:34 +00001111 bit,
1112 byte;
1113
1114 /*
1115 Convert PseudoClass image to a ICON monochrome image.
1116 */
cristybb503372010-05-27 20:51:26 +00001117 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001118 {
cristy1e178e72011-08-28 19:44:34 +00001119 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001120 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001121 break;
cristy3ed852e2009-09-05 21:47:34 +00001122 q=pixels+(next->rows-y-1)*bytes_per_line;
1123 bit=0;
1124 byte=0;
cristybb503372010-05-27 20:51:26 +00001125 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001126 {
1127 byte<<=1;
cristy4c08aed2011-07-01 19:47:50 +00001128 byte|=GetPixelIndex(next,p) != 0 ? 0x01 : 0x00;
cristy3ed852e2009-09-05 21:47:34 +00001129 bit++;
1130 if (bit == 8)
1131 {
1132 *q++=(unsigned char) byte;
1133 bit=0;
1134 byte=0;
1135 }
cristyed231572011-07-14 02:18:59 +00001136 p+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +00001137 }
cristy3ed852e2009-09-05 21:47:34 +00001138 if (bit != 0)
1139 *q++=(unsigned char) (byte << (8-bit));
1140 if (next->previous == (Image *) NULL)
1141 {
1142 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1143 if (status == MagickFalse)
1144 break;
1145 }
1146 }
1147 break;
1148 }
1149 case 4:
1150 {
cristybb503372010-05-27 20:51:26 +00001151 size_t
cristy3ed852e2009-09-05 21:47:34 +00001152 nibble,
1153 byte;
1154
1155 /*
1156 Convert PseudoClass image to a ICON monochrome image.
1157 */
cristybb503372010-05-27 20:51:26 +00001158 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001159 {
cristy1e178e72011-08-28 19:44:34 +00001160 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001161 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001162 break;
cristy3ed852e2009-09-05 21:47:34 +00001163 q=pixels+(next->rows-y-1)*bytes_per_line;
1164 nibble=0;
1165 byte=0;
cristybb503372010-05-27 20:51:26 +00001166 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001167 {
1168 byte<<=4;
cristy4c08aed2011-07-01 19:47:50 +00001169 byte|=((size_t) GetPixelIndex(next,p) & 0x0f);
cristy3ed852e2009-09-05 21:47:34 +00001170 nibble++;
1171 if (nibble == 2)
1172 {
1173 *q++=(unsigned char) byte;
1174 nibble=0;
1175 byte=0;
1176 }
cristyed231572011-07-14 02:18:59 +00001177 p+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +00001178 }
cristy3ed852e2009-09-05 21:47:34 +00001179 if (nibble != 0)
1180 *q++=(unsigned char) (byte << 4);
1181 if (next->previous == (Image *) NULL)
1182 {
1183 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1184 if (status == MagickFalse)
1185 break;
1186 }
1187 }
1188 break;
1189 }
1190 case 8:
1191 {
1192 /*
1193 Convert PseudoClass packet to ICON pixel.
1194 */
cristybb503372010-05-27 20:51:26 +00001195 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001196 {
cristy1e178e72011-08-28 19:44:34 +00001197 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001198 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001199 break;
cristy3ed852e2009-09-05 21:47:34 +00001200 q=pixels+(next->rows-y-1)*bytes_per_line;
cristybb503372010-05-27 20:51:26 +00001201 for (x=0; x < (ssize_t) next->columns; x++)
cristy4c08aed2011-07-01 19:47:50 +00001202 {
1203 *q++=(unsigned char) GetPixelIndex(next,p);
cristyed231572011-07-14 02:18:59 +00001204 p+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +00001205 }
cristy3ed852e2009-09-05 21:47:34 +00001206 if (next->previous == (Image *) NULL)
1207 {
1208 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1209 if (status == MagickFalse)
1210 break;
1211 }
1212 }
1213 break;
1214 }
1215 case 24:
1216 case 32:
1217 {
1218 /*
1219 Convert DirectClass packet to ICON BGR888 or BGRA8888 pixel.
1220 */
cristybb503372010-05-27 20:51:26 +00001221 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001222 {
cristy1e178e72011-08-28 19:44:34 +00001223 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001224 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001225 break;
1226 q=pixels+(next->rows-y-1)*bytes_per_line;
cristybb503372010-05-27 20:51:26 +00001227 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001228 {
cristy4c08aed2011-07-01 19:47:50 +00001229 *q++=ScaleQuantumToChar(GetPixelBlue(next,p));
1230 *q++=ScaleQuantumToChar(GetPixelGreen(next,p));
1231 *q++=ScaleQuantumToChar(GetPixelRed(next,p));
cristy17f11b02014-12-20 19:37:04 +00001232 if (next->alpha_trait == UndefinedPixelTrait)
cristy3ed852e2009-09-05 21:47:34 +00001233 *q++=ScaleQuantumToChar(QuantumRange);
1234 else
cristy4c08aed2011-07-01 19:47:50 +00001235 *q++=ScaleQuantumToChar(GetPixelAlpha(next,p));
cristyed231572011-07-14 02:18:59 +00001236 p+=GetPixelChannels(next);
cristy3ed852e2009-09-05 21:47:34 +00001237 }
1238 if (icon_info.bits_per_pixel == 24)
cristybb503372010-05-27 20:51:26 +00001239 for (x=3L*(ssize_t) next->columns; x < (ssize_t) bytes_per_line; x++)
cristy3ed852e2009-09-05 21:47:34 +00001240 *q++=0x00;
1241 if (next->previous == (Image *) NULL)
1242 {
1243 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1244 if (status == MagickFalse)
1245 break;
1246 }
1247 }
1248 break;
1249 }
1250 }
1251 /*
1252 Write 40-byte version 3+ bitmap header.
1253 */
1254 icon_file.directory[scene].width=(unsigned char) icon_info.width;
1255 icon_file.directory[scene].height=(unsigned char) icon_info.height;
1256 icon_file.directory[scene].colors=(unsigned char)
1257 icon_info.number_colors;
1258 icon_file.directory[scene].reserved=0;
1259 icon_file.directory[scene].planes=icon_info.planes;
1260 icon_file.directory[scene].bits_per_pixel=icon_info.bits_per_pixel;
1261 icon_file.directory[scene].size=icon_info.size;
cristybb503372010-05-27 20:51:26 +00001262 icon_file.directory[scene].offset=(size_t) TellBlob(image);
cristy35ef8242010-06-03 16:24:13 +00001263 (void) WriteBlobLSBLong(image,(unsigned int) 40);
1264 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.width);
1265 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.height*2);
cristy3ed852e2009-09-05 21:47:34 +00001266 (void) WriteBlobLSBShort(image,icon_info.planes);
1267 (void) WriteBlobLSBShort(image,icon_info.bits_per_pixel);
cristy35ef8242010-06-03 16:24:13 +00001268 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.compression);
1269 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.image_size);
1270 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.x_pixels);
1271 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.y_pixels);
1272 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.number_colors);
1273 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.colors_important);
cristy3ed852e2009-09-05 21:47:34 +00001274 if (next->storage_class == PseudoClass)
1275 {
1276 unsigned char
1277 *icon_colormap;
1278
1279 /*
1280 Dump colormap to file.
1281 */
1282 icon_colormap=(unsigned char *) AcquireQuantumMemory((size_t)
1283 (1UL << icon_info.bits_per_pixel),4UL*sizeof(*icon_colormap));
dirkc1e52002014-01-09 20:22:42 +00001284 if (icon_colormap == (unsigned char *) NULL)
1285 {
1286 images=DestroyImageList(images);
1287 ThrowWriterException(ResourceLimitError,
1288 "MemoryAllocationFailed");
dirk9e137702014-01-09 19:29:59 +00001289 }
cristy3ed852e2009-09-05 21:47:34 +00001290 q=icon_colormap;
cristybb503372010-05-27 20:51:26 +00001291 for (i=0; i < (ssize_t) next->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +00001292 {
1293 *q++=ScaleQuantumToChar(next->colormap[i].blue);
1294 *q++=ScaleQuantumToChar(next->colormap[i].green);
1295 *q++=ScaleQuantumToChar(next->colormap[i].red);
1296 *q++=(unsigned char) 0x0;
1297 }
cristybb503372010-05-27 20:51:26 +00001298 for ( ; i < (ssize_t) (1UL << icon_info.bits_per_pixel); i++)
cristy3ed852e2009-09-05 21:47:34 +00001299 {
1300 *q++=(unsigned char) 0x00;
1301 *q++=(unsigned char) 0x00;
1302 *q++=(unsigned char) 0x00;
1303 *q++=(unsigned char) 0x00;
1304 }
1305 (void) WriteBlob(image,(size_t) (4UL*(1UL <<
1306 icon_info.bits_per_pixel)),icon_colormap);
1307 icon_colormap=(unsigned char *) RelinquishMagickMemory(
1308 icon_colormap);
1309 }
1310 (void) WriteBlob(image,(size_t) icon_info.image_size,pixels);
1311 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
1312 /*
1313 Write matte mask.
1314 */
1315 scanline_pad=(((next->columns+31) & ~31)-next->columns) >> 3;
cristybb503372010-05-27 20:51:26 +00001316 for (y=((ssize_t) next->rows - 1); y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +00001317 {
cristy1e178e72011-08-28 19:44:34 +00001318 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001319 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001320 break;
1321 bit=0;
1322 byte=0;
cristybb503372010-05-27 20:51:26 +00001323 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001324 {
1325 byte<<=1;
cristy17f11b02014-12-20 19:37:04 +00001326 if ((next->alpha_trait != UndefinedPixelTrait) &&
cristy4c08aed2011-07-01 19:47:50 +00001327 (GetPixelAlpha(next,p) == (Quantum) TransparentAlpha))
cristy3ed852e2009-09-05 21:47:34 +00001328 byte|=0x01;
1329 bit++;
1330 if (bit == 8)
1331 {
1332 (void) WriteBlobByte(image,(unsigned char) byte);
1333 bit=0;
1334 byte=0;
1335 }
cristyed231572011-07-14 02:18:59 +00001336 p+=GetPixelChannels(next);
cristy3ed852e2009-09-05 21:47:34 +00001337 }
1338 if (bit != 0)
1339 (void) WriteBlobByte(image,(unsigned char) (byte << (8-bit)));
cristybb503372010-05-27 20:51:26 +00001340 for (i=0; i < (ssize_t) scanline_pad; i++)
cristy3ed852e2009-09-05 21:47:34 +00001341 (void) WriteBlobByte(image,(unsigned char) 0);
1342 }
1343 }
1344 if (GetNextImageInList(next) == (Image *) NULL)
1345 break;
cristy3ed852e2009-09-05 21:47:34 +00001346 status=SetImageProgress(next,SaveImagesTag,scene++,
1347 GetImageListLength(next));
1348 if (status == MagickFalse)
1349 break;
cristy5467fcf2014-05-18 17:20:55 +00001350 next=SyncNextImageInList(next);
cristy3ed852e2009-09-05 21:47:34 +00001351 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
1352 offset=SeekBlob(image,0,SEEK_SET);
cristyda16f162011-02-19 23:52:17 +00001353 (void) offset;
cristy3ed852e2009-09-05 21:47:34 +00001354 (void) WriteBlobLSBShort(image,0);
1355 (void) WriteBlobLSBShort(image,1);
1356 (void) WriteBlobLSBShort(image,(unsigned short) (scene+1));
1357 scene=0;
dirk9e137702014-01-09 19:29:59 +00001358 next=(images != (Image *) NULL) ? images : image;
cristy3ed852e2009-09-05 21:47:34 +00001359 do
1360 {
1361 (void) WriteBlobByte(image,icon_file.directory[scene].width);
1362 (void) WriteBlobByte(image,icon_file.directory[scene].height);
1363 (void) WriteBlobByte(image,icon_file.directory[scene].colors);
1364 (void) WriteBlobByte(image,icon_file.directory[scene].reserved);
1365 (void) WriteBlobLSBShort(image,icon_file.directory[scene].planes);
1366 (void) WriteBlobLSBShort(image,icon_file.directory[scene].bits_per_pixel);
cristy0b29b252010-05-30 01:59:46 +00001367 (void) WriteBlobLSBLong(image,(unsigned int)
1368 icon_file.directory[scene].size);
1369 (void) WriteBlobLSBLong(image,(unsigned int)
1370 icon_file.directory[scene].offset);
cristy3ed852e2009-09-05 21:47:34 +00001371 scene++;
1372 next=SyncNextImageInList(next);
1373 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
1374 (void) CloseBlob(image);
dirk9e137702014-01-09 19:29:59 +00001375 images=DestroyImageList(images);
cristy3ed852e2009-09-05 21:47:34 +00001376 return(MagickTrue);
1377}