blob: 50bfed63c41d6a61fb78ef3c76f245075efc5fbe [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% %
cristyb56bb242014-11-25 17:12:48 +000020% Copyright 1999-2015 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);
dirk841f9152014-09-11 04:35:42 +0000186 if (p == q || size < 16 || size > 256)
dirk501a11e2014-01-19 16:22:18 +0000187 return((Image *) NULL);
188
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);
281 assert(image_info->signature == MagickSignature);
282 (void) LogMagickEvent(CoderEvent,GetMagickModule(),"%s",image_info->filename);
283 assert(exception != (ExceptionInfo *) NULL);
284 assert(exception->signature == MagickSignature);
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);
310 }
cristyeaedf062010-05-29 22:36:02 +0000311 one=1;
cristy3ed852e2009-09-05 21:47:34 +0000312 for (i=0; i < icon_file.count; i++)
313 {
314 /*
315 Verify Icon identifier.
316 */
317 offset=(ssize_t) SeekBlob(image,(MagickOffsetType)
318 icon_file.directory[i].offset,SEEK_SET);
319 if (offset < 0)
320 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
321 icon_info.size=ReadBlobLSBLong(image);
cristyd63c0502011-01-03 15:45:11 +0000322 icon_info.width=(unsigned char) ((int) ReadBlobLSBLong(image));
323 icon_info.height=(unsigned char) ((int) ReadBlobLSBLong(image)/2);
cristy4e09e032011-10-05 19:42:10 +0000324 icon_info.planes=ReadBlobLSBShort(image);
325 icon_info.bits_per_pixel=ReadBlobLSBShort(image);
cristy43602772014-01-29 14:07:29 +0000326 if (((icon_info.planes == 18505) && (icon_info.bits_per_pixel == 21060)) ||
327 (icon_info.size == 0x474e5089))
cristy3ed852e2009-09-05 21:47:34 +0000328 {
329 Image
330 *icon_image;
331
332 ImageInfo
333 *read_info;
334
cristyf054ee72011-10-05 17:04:08 +0000335 size_t
336 length;
337
338 unsigned char
339 *png;
340
cristy3ed852e2009-09-05 21:47:34 +0000341 /*
342 Icon image encoded as a compressed PNG image.
343 */
cristyf054ee72011-10-05 17:04:08 +0000344 length=icon_file.directory[i].size;
cristy4e09e032011-10-05 19:42:10 +0000345 png=(unsigned char *) AcquireQuantumMemory(length+16,sizeof(*png));
cristyf054ee72011-10-05 17:04:08 +0000346 if (png == (unsigned char *) NULL)
347 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
348 (void) CopyMagickMemory(png,"\211PNG\r\n\032\n\000\000\000\015",12);
cristy4e09e032011-10-05 19:42:10 +0000349 png[12]=(unsigned char) icon_info.planes;
350 png[13]=(unsigned char) (icon_info.planes >> 8);
351 png[14]=(unsigned char) icon_info.bits_per_pixel;
352 png[15]=(unsigned char) (icon_info.bits_per_pixel >> 8);
353 count=ReadBlob(image,length-16,png+16);
dirk833908d2014-01-28 19:26:32 +0000354 icon_image=(Image *) NULL;
355 if (count > 0)
cristyf054ee72011-10-05 17:04:08 +0000356 {
dirk833908d2014-01-28 19:26:32 +0000357 read_info=CloneImageInfo(image_info);
358 (void) CopyMagickString(read_info->magick,"PNG",MaxTextExtent);
359 icon_image=BlobToImage(read_info,png,length+16,exception);
360 read_info=DestroyImageInfo(read_info);
cristyf054ee72011-10-05 17:04:08 +0000361 }
cristyf054ee72011-10-05 17:04:08 +0000362 png=(unsigned char *) RelinquishMagickMemory(png);
cristy47de5a92011-10-05 01:47:42 +0000363 if (icon_image == (Image *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000364 {
dirk833908d2014-01-28 19:26:32 +0000365 if (count != (ssize_t) (length-16))
366 ThrowReaderException(CorruptImageError,
367 "InsufficientImageDataInFile");
cristy47de5a92011-10-05 01:47:42 +0000368 image=DestroyImageList(image);
369 return((Image *) NULL);
cristy3ed852e2009-09-05 21:47:34 +0000370 }
cristy47de5a92011-10-05 01:47:42 +0000371 DestroyBlob(icon_image);
372 icon_image->blob=ReferenceBlob(image->blob);
373 ReplaceImageInList(&image,icon_image);
cristy3ed852e2009-09-05 21:47:34 +0000374 }
375 else
376 {
cristyf054ee72011-10-05 17:04:08 +0000377 if (icon_info.bits_per_pixel > 32)
378 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
379 icon_info.compression=ReadBlobLSBLong(image);
380 icon_info.image_size=ReadBlobLSBLong(image);
381 icon_info.x_pixels=ReadBlobLSBLong(image);
382 icon_info.y_pixels=ReadBlobLSBLong(image);
383 icon_info.number_colors=ReadBlobLSBLong(image);
384 icon_info.colors_important=ReadBlobLSBLong(image);
cristy8a46d822012-08-28 23:32:39 +0000385 image->alpha_trait=BlendPixelTrait;
cristyf054ee72011-10-05 17:04:08 +0000386 image->columns=(size_t) icon_file.directory[i].width;
387 if ((ssize_t) image->columns > icon_info.width)
388 image->columns=(size_t) icon_info.width;
cristy4e09e032011-10-05 19:42:10 +0000389 if (image->columns == 0)
390 image->columns=256;
cristyf054ee72011-10-05 17:04:08 +0000391 image->rows=(size_t) icon_file.directory[i].height;
392 if ((ssize_t) image->rows > icon_info.height)
393 image->rows=(size_t) icon_info.height;
cristy4e09e032011-10-05 19:42:10 +0000394 if (image->rows == 0)
395 image->rows=256;
cristyf054ee72011-10-05 17:04:08 +0000396 image->depth=icon_info.bits_per_pixel;
397 if (image->debug != MagickFalse)
398 {
399 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
400 " scene = %.20g",(double) i);
401 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
402 " size = %.20g",(double) icon_info.size);
403 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
404 " width = %.20g",(double) icon_file.directory[i].width);
405 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
406 " height = %.20g",(double) icon_file.directory[i].height);
407 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
408 " colors = %.20g",(double ) icon_info.number_colors);
409 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
410 " planes = %.20g",(double) icon_info.planes);
411 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
412 " bpp = %.20g",(double) icon_info.bits_per_pixel);
413 }
cristya2fb29a2014-09-21 22:04:52 +0000414 if ((icon_info.number_colors != 0) || (icon_info.bits_per_pixel <= 16U))
cristyf054ee72011-10-05 17:04:08 +0000415 {
416 image->storage_class=PseudoClass;
417 image->colors=icon_info.number_colors;
418 if (image->colors == 0)
419 image->colors=one << icon_info.bits_per_pixel;
420 }
421 if (image->storage_class == PseudoClass)
422 {
423 register ssize_t
424 i;
cristy3ed852e2009-09-05 21:47:34 +0000425
cristyf054ee72011-10-05 17:04:08 +0000426 unsigned char
427 *icon_colormap;
428
429 /*
430 Read Icon raster colormap.
431 */
dirk833908d2014-01-28 19:26:32 +0000432 if (AcquireImageColormap(image,image->colors,exception) ==
433 MagickFalse)
cristyf054ee72011-10-05 17:04:08 +0000434 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
435 icon_colormap=(unsigned char *) AcquireQuantumMemory((size_t)
436 image->colors,4UL*sizeof(*icon_colormap));
437 if (icon_colormap == (unsigned char *) NULL)
438 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
439 count=ReadBlob(image,(size_t) (4*image->colors),icon_colormap);
440 if (count != (ssize_t) (4*image->colors))
441 ThrowReaderException(CorruptImageError,
442 "InsufficientImageDataInFile");
443 p=icon_colormap;
444 for (i=0; i < (ssize_t) image->colors; i++)
445 {
446 image->colormap[i].blue=(Quantum) ScaleCharToQuantum(*p++);
447 image->colormap[i].green=(Quantum) ScaleCharToQuantum(*p++);
448 image->colormap[i].red=(Quantum) ScaleCharToQuantum(*p++);
449 p++;
450 }
451 icon_colormap=(unsigned char *) RelinquishMagickMemory(icon_colormap);
452 }
cristy3ed852e2009-09-05 21:47:34 +0000453 /*
454 Convert Icon raster image to pixel packets.
455 */
cristyf054ee72011-10-05 17:04:08 +0000456 if ((image_info->ping != MagickFalse) &&
457 (image_info->number_scenes != 0))
458 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
459 break;
cristyacabb842014-12-14 23:36:33 +0000460 status=SetImageExtent(image,image->columns,image->rows,exception);
461 if (status == MagickFalse)
462 return(DestroyImageList(image));
cristy3ed852e2009-09-05 21:47:34 +0000463 bytes_per_line=(((image->columns*icon_info.bits_per_pixel)+31) &
464 ~31) >> 3;
cristyda16f162011-02-19 23:52:17 +0000465 (void) bytes_per_line;
cristy3ed852e2009-09-05 21:47:34 +0000466 scanline_pad=((((image->columns*icon_info.bits_per_pixel)+31) & ~31)-
467 (image->columns*icon_info.bits_per_pixel)) >> 3;
468 switch (icon_info.bits_per_pixel)
469 {
470 case 1:
471 {
472 /*
473 Convert bitmap scanline.
474 */
cristybb503372010-05-27 20:51:26 +0000475 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000476 {
477 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000478 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000479 break;
cristybb503372010-05-27 20:51:26 +0000480 for (x=0; x < (ssize_t) (image->columns-7); x+=8)
cristy3ed852e2009-09-05 21:47:34 +0000481 {
cristyf054ee72011-10-05 17:04:08 +0000482 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000483 for (bit=0; bit < 8; bit++)
cristy4c08aed2011-07-01 19:47:50 +0000484 {
485 SetPixelIndex(image,((byte & (0x80 >> bit)) != 0 ? 0x01 :
486 0x00),q);
cristyed231572011-07-14 02:18:59 +0000487 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000488 }
cristy3ed852e2009-09-05 21:47:34 +0000489 }
490 if ((image->columns % 8) != 0)
491 {
cristyf054ee72011-10-05 17:04:08 +0000492 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000493 for (bit=0; bit < (image->columns % 8); bit++)
cristy4c08aed2011-07-01 19:47:50 +0000494 {
495 SetPixelIndex(image,((byte & (0x80 >> bit)) != 0 ? 0x01 :
496 0x00),q);
cristyed231572011-07-14 02:18:59 +0000497 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000498 }
cristy3ed852e2009-09-05 21:47:34 +0000499 }
cristybb503372010-05-27 20:51:26 +0000500 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000501 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000502 if (SyncAuthenticPixels(image,exception) == MagickFalse)
503 break;
504 if (image->previous == (Image *) NULL)
505 {
506 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
507 image->rows);
508 if (status == MagickFalse)
509 break;
510 }
511 }
512 break;
513 }
514 case 4:
515 {
516 /*
517 Read 4-bit Icon scanline.
518 */
cristybb503372010-05-27 20:51:26 +0000519 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000520 {
521 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000522 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000523 break;
cristybb503372010-05-27 20:51:26 +0000524 for (x=0; x < ((ssize_t) image->columns-1); x+=2)
cristy3ed852e2009-09-05 21:47:34 +0000525 {
cristyf054ee72011-10-05 17:04:08 +0000526 byte=(size_t) ReadBlobByte(image);
cristy4c08aed2011-07-01 19:47:50 +0000527 SetPixelIndex(image,((byte >> 4) & 0xf),q);
cristyed231572011-07-14 02:18:59 +0000528 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000529 SetPixelIndex(image,((byte) & 0xf),q);
cristyed231572011-07-14 02:18:59 +0000530 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000531 }
532 if ((image->columns % 2) != 0)
533 {
cristyf054ee72011-10-05 17:04:08 +0000534 byte=(size_t) ReadBlobByte(image);
cristy4c08aed2011-07-01 19:47:50 +0000535 SetPixelIndex(image,((byte >> 4) & 0xf),q);
cristyed231572011-07-14 02:18:59 +0000536 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000537 }
cristybb503372010-05-27 20:51:26 +0000538 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000539 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000540 if (SyncAuthenticPixels(image,exception) == MagickFalse)
541 break;
542 if (image->previous == (Image *) NULL)
543 {
544 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
545 image->rows);
546 if (status == MagickFalse)
547 break;
548 }
549 }
550 break;
551 }
552 case 8:
553 {
554 /*
555 Convert PseudoColor scanline.
556 */
cristybb503372010-05-27 20:51:26 +0000557 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000558 {
559 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000560 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000561 break;
cristybb503372010-05-27 20:51:26 +0000562 for (x=0; x < (ssize_t) image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +0000563 {
cristyf054ee72011-10-05 17:04:08 +0000564 byte=(size_t) ReadBlobByte(image);
cristy4c08aed2011-07-01 19:47:50 +0000565 SetPixelIndex(image,byte,q);
cristyed231572011-07-14 02:18:59 +0000566 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000567 }
cristybb503372010-05-27 20:51:26 +0000568 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000569 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000570 if (SyncAuthenticPixels(image,exception) == MagickFalse)
571 break;
572 if (image->previous == (Image *) NULL)
573 {
574 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
575 image->rows);
576 if (status == MagickFalse)
577 break;
578 }
579 }
580 break;
581 }
582 case 16:
583 {
584 /*
585 Convert PseudoColor scanline.
586 */
cristybb503372010-05-27 20:51:26 +0000587 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000588 {
589 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000590 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000591 break;
cristybb503372010-05-27 20:51:26 +0000592 for (x=0; x < (ssize_t) image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +0000593 {
cristyf054ee72011-10-05 17:04:08 +0000594 byte=(size_t) ReadBlobByte(image);
595 byte|=(size_t) (ReadBlobByte(image) << 8);
cristy4c08aed2011-07-01 19:47:50 +0000596 SetPixelIndex(image,byte,q);
cristyed231572011-07-14 02:18:59 +0000597 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000598 }
cristybb503372010-05-27 20:51:26 +0000599 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000600 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000601 if (SyncAuthenticPixels(image,exception) == MagickFalse)
602 break;
603 if (image->previous == (Image *) NULL)
604 {
605 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
606 image->rows);
607 if (status == MagickFalse)
608 break;
609 }
610 }
611 break;
612 }
613 case 24:
614 case 32:
615 {
616 /*
617 Convert DirectColor scanline.
618 */
cristybb503372010-05-27 20:51:26 +0000619 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000620 {
621 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000622 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000623 break;
cristybb503372010-05-27 20:51:26 +0000624 for (x=0; x < (ssize_t) image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +0000625 {
cristyf054ee72011-10-05 17:04:08 +0000626 SetPixelBlue(image,ScaleCharToQuantum((unsigned char)
627 ReadBlobByte(image)),q);
628 SetPixelGreen(image,ScaleCharToQuantum((unsigned char)
629 ReadBlobByte(image)),q);
630 SetPixelRed(image,ScaleCharToQuantum((unsigned char)
631 ReadBlobByte(image)),q);
cristy3ed852e2009-09-05 21:47:34 +0000632 if (icon_info.bits_per_pixel == 32)
cristyf054ee72011-10-05 17:04:08 +0000633 SetPixelAlpha(image,ScaleCharToQuantum((unsigned char)
634 ReadBlobByte(image)),q);
cristyed231572011-07-14 02:18:59 +0000635 q+=GetPixelChannels(image);
cristy3ed852e2009-09-05 21:47:34 +0000636 }
637 if (icon_info.bits_per_pixel == 24)
cristybb503372010-05-27 20:51:26 +0000638 for (x=0; x < (ssize_t) scanline_pad; x++)
cristyf054ee72011-10-05 17:04:08 +0000639 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000640 if (SyncAuthenticPixels(image,exception) == MagickFalse)
641 break;
642 if (image->previous == (Image *) NULL)
643 {
644 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
645 image->rows);
646 if (status == MagickFalse)
647 break;
648 }
649 }
650 break;
651 }
652 default:
653 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
654 }
cristy5dd71882010-08-01 20:53:13 +0000655 if (image_info->ping == MagickFalse)
cristyea1a8aa2011-10-20 13:24:06 +0000656 (void) SyncImage(image,exception);
cristy3ed852e2009-09-05 21:47:34 +0000657 if (icon_info.bits_per_pixel != 32)
658 {
659 /*
660 Read the ICON alpha mask.
661 */
662 image->storage_class=DirectClass;
cristybb503372010-05-27 20:51:26 +0000663 for (y=(ssize_t) image->rows-1; y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +0000664 {
665 q=GetAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000666 if (q == (Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +0000667 break;
cristybb503372010-05-27 20:51:26 +0000668 for (x=0; x < ((ssize_t) image->columns-7); x+=8)
cristy3ed852e2009-09-05 21:47:34 +0000669 {
cristyf054ee72011-10-05 17:04:08 +0000670 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000671 for (bit=0; bit < 8; bit++)
cristy4c08aed2011-07-01 19:47:50 +0000672 {
673 SetPixelAlpha(image,(((byte & (0x80 >> bit)) != 0) ?
674 TransparentAlpha : OpaqueAlpha),q);
cristyed231572011-07-14 02:18:59 +0000675 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000676 }
cristy3ed852e2009-09-05 21:47:34 +0000677 }
678 if ((image->columns % 8) != 0)
679 {
cristyf054ee72011-10-05 17:04:08 +0000680 byte=(size_t) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000681 for (bit=0; bit < (image->columns % 8); bit++)
cristy4c08aed2011-07-01 19:47:50 +0000682 {
683 SetPixelAlpha(image,(((byte & (0x80 >> bit)) != 0) ?
684 TransparentAlpha : OpaqueAlpha),q);
cristyed231572011-07-14 02:18:59 +0000685 q+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +0000686 }
cristy3ed852e2009-09-05 21:47:34 +0000687 }
cristy0e8200f2009-09-13 21:10:06 +0000688 if ((image->columns % 32) != 0)
cristybb503372010-05-27 20:51:26 +0000689 for (x=0; x < (ssize_t) ((32-(image->columns % 32))/8); x++)
cristyf054ee72011-10-05 17:04:08 +0000690 (void) ReadBlobByte(image);
cristy3ed852e2009-09-05 21:47:34 +0000691 if (SyncAuthenticPixels(image,exception) == MagickFalse)
692 break;
693 }
694 }
695 if (EOFBlob(image) != MagickFalse)
696 {
697 ThrowFileException(exception,CorruptImageError,
698 "UnexpectedEndOfFile",image->filename);
699 break;
700 }
701 }
702 /*
703 Proceed to next image.
704 */
705 if (image_info->number_scenes != 0)
706 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
707 break;
cristybb503372010-05-27 20:51:26 +0000708 if (i < (ssize_t) (icon_file.count-1))
cristy3ed852e2009-09-05 21:47:34 +0000709 {
710 /*
711 Allocate next image structure.
712 */
cristy1f1de182011-10-05 18:41:34 +0000713 AcquireNextImage(image_info,image,exception);
cristy3ed852e2009-09-05 21:47:34 +0000714 if (GetNextImageInList(image) == (Image *) NULL)
715 {
716 image=DestroyImageList(image);
717 return((Image *) NULL);
718 }
719 image=SyncNextImageInList(image);
720 status=SetImageProgress(image,LoadImagesTag,TellBlob(image),
721 GetBlobSize(image));
722 if (status == MagickFalse)
723 break;
724 }
725 }
726 (void) CloseBlob(image);
727 return(GetFirstImageInList(image));
728}
729
730/*
731%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
732% %
733% %
734% %
735% R e g i s t e r I C O N I m a g e %
736% %
737% %
738% %
739%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
740%
741% RegisterICONImage() adds attributes for the Icon image format to
742% the list of supported formats. The attributes include the image format
743% tag, a method to read and/or write the format, whether the format
744% supports the saving of more than one frame to the same file or blob,
745% whether the format supports native in-memory I/O, and a brief
746% description of the format.
747%
748% The format of the RegisterICONImage method is:
749%
cristybb503372010-05-27 20:51:26 +0000750% size_t RegisterICONImage(void)
cristy3ed852e2009-09-05 21:47:34 +0000751%
752*/
cristybb503372010-05-27 20:51:26 +0000753ModuleExport size_t RegisterICONImage(void)
cristy3ed852e2009-09-05 21:47:34 +0000754{
755 MagickInfo
756 *entry;
757
758 entry=SetMagickInfo("CUR");
759 entry->decoder=(DecodeImageHandler *) ReadICONImage;
760 entry->encoder=(EncodeImageHandler *) WriteICONImage;
761 entry->adjoin=MagickFalse;
762 entry->seekable_stream=MagickTrue;
763 entry->description=ConstantString("Microsoft icon");
764 entry->module=ConstantString("CUR");
765 (void) RegisterMagickInfo(entry);
766 entry=SetMagickInfo("ICO");
767 entry->decoder=(DecodeImageHandler *) ReadICONImage;
768 entry->encoder=(EncodeImageHandler *) WriteICONImage;
769 entry->adjoin=MagickTrue;
770 entry->seekable_stream=MagickTrue;
771 entry->description=ConstantString("Microsoft icon");
772 entry->module=ConstantString("ICON");
773 (void) RegisterMagickInfo(entry);
774 entry=SetMagickInfo("ICON");
775 entry->decoder=(DecodeImageHandler *) ReadICONImage;
776 entry->encoder=(EncodeImageHandler *) WriteICONImage;
777 entry->adjoin=MagickFalse;
778 entry->seekable_stream=MagickTrue;
779 entry->description=ConstantString("Microsoft icon");
780 entry->module=ConstantString("ICON");
781 (void) RegisterMagickInfo(entry);
782 return(MagickImageCoderSignature);
783}
784
785/*
786%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
787% %
788% %
789% %
790% U n r e g i s t e r I C O N I m a g e %
791% %
792% %
793% %
794%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
795%
796% UnregisterICONImage() removes format registrations made by the
797% ICON module from the list of supported formats.
798%
799% The format of the UnregisterICONImage method is:
800%
801% UnregisterICONImage(void)
802%
803*/
804ModuleExport void UnregisterICONImage(void)
805{
806 (void) UnregisterMagickInfo("CUR");
807 (void) UnregisterMagickInfo("ICO");
808 (void) UnregisterMagickInfo("ICON");
809}
810
811/*
812%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
813% %
814% %
815% %
816% W r i t e I C O N I m a g e %
817% %
818% %
819% %
820%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
821%
822% WriteICONImage() writes an image in Microsoft Windows bitmap encoded
823% image format, version 3 for Windows or (if the image has a matte channel)
824% version 4.
825%
cristy83cb91e2012-10-31 00:58:22 +0000826% It encodes any subimage as a compressed PNG image ("BI_PNG)", only when its
827% dimensions are 256x256 and image->compression is undefined or is defined as
828% ZipCompression.
glennrp4b708b82012-07-17 19:15:58 +0000829%
cristy3ed852e2009-09-05 21:47:34 +0000830% The format of the WriteICONImage method is:
831%
832% MagickBooleanType WriteICONImage(const ImageInfo *image_info,
cristy1e178e72011-08-28 19:44:34 +0000833% Image *image,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +0000834%
835% A description of each parameter follows.
836%
837% o image_info: the image info.
838%
839% o image: The image.
840%
cristy1e178e72011-08-28 19:44:34 +0000841% o exception: return any errors or warnings in this structure.
842%
cristy3ed852e2009-09-05 21:47:34 +0000843*/
844static MagickBooleanType WriteICONImage(const ImageInfo *image_info,
cristy1e178e72011-08-28 19:44:34 +0000845 Image *image,ExceptionInfo *exception)
cristy3ed852e2009-09-05 21:47:34 +0000846{
dirk501a11e2014-01-19 16:22:18 +0000847 const char
848 *option;
849
cristy3ed852e2009-09-05 21:47:34 +0000850 IconFile
851 icon_file;
852
853 IconInfo
854 icon_info;
855
856 Image
dirk9e137702014-01-09 19:29:59 +0000857 *images,
cristy3ed852e2009-09-05 21:47:34 +0000858 *next;
859
cristy3ed852e2009-09-05 21:47:34 +0000860 MagickBooleanType
861 status;
862
863 MagickOffsetType
864 offset,
865 scene;
866
cristy4c08aed2011-07-01 19:47:50 +0000867 register const Quantum
cristy3ed852e2009-09-05 21:47:34 +0000868 *p;
869
cristybb503372010-05-27 20:51:26 +0000870 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +0000871 i,
872 x;
873
874 register unsigned char
875 *q;
876
cristyebc891a2011-04-24 23:04:16 +0000877 size_t
878 bytes_per_line,
879 scanline_pad;
880
881 ssize_t
882 y;
883
cristy3ed852e2009-09-05 21:47:34 +0000884 unsigned char
885 bit,
886 byte,
887 *pixels;
888
cristy3ed852e2009-09-05 21:47:34 +0000889 /*
890 Open output image file.
891 */
892 assert(image_info != (const ImageInfo *) NULL);
893 assert(image_info->signature == MagickSignature);
894 assert(image != (Image *) NULL);
895 assert(image->signature == MagickSignature);
896 (void) LogMagickEvent(CoderEvent,GetMagickModule(),"%s",image->filename);
cristy3a37efd2011-08-28 20:31:03 +0000897 assert(exception != (ExceptionInfo *) NULL);
898 assert(exception->signature == MagickSignature);
cristy1e178e72011-08-28 19:44:34 +0000899 status=OpenBlob(image_info,image,WriteBinaryBlobMode,exception);
cristy3ed852e2009-09-05 21:47:34 +0000900 if (status == MagickFalse)
901 return(status);
dirk9e137702014-01-09 19:29:59 +0000902 images=(Image *) NULL;
dirk501a11e2014-01-19 16:22:18 +0000903 option=GetImageOption(image_info,"icon:auto-resize");
904 if (option != (const char *) NULL)
dirk9e137702014-01-09 19:29:59 +0000905 {
dirk501a11e2014-01-19 16:22:18 +0000906 images=AutoResizeImage(image,option,&scene,exception);
dirk9e137702014-01-09 19:29:59 +0000907 if (images == (Image *) NULL)
dirk501a11e2014-01-19 16:22:18 +0000908 ThrowWriterException(ImageError,"InvalidDimensions");
dirk9e137702014-01-09 19:29:59 +0000909 }
910 else
911 {
912 scene=0;
913 next=image;
914 do
915 {
916 if ((image->columns > 256L) || (image->rows > 256L))
917 ThrowWriterException(ImageError,"WidthOrHeightExceedsLimit");
918 scene++;
919 next=SyncNextImageInList(next);
920 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
921 }
cristy3ed852e2009-09-05 21:47:34 +0000922 /*
923 Dump out a ICON header template to be properly initialized later.
924 */
925 (void) WriteBlobLSBShort(image,0);
926 (void) WriteBlobLSBShort(image,1);
927 (void) WriteBlobLSBShort(image,(unsigned char) scene);
928 (void) ResetMagickMemory(&icon_file,0,sizeof(icon_file));
929 (void) ResetMagickMemory(&icon_info,0,sizeof(icon_info));
930 scene=0;
dirk9e137702014-01-09 19:29:59 +0000931 next=(images != (Image *) NULL) ? images : image;
cristy3ed852e2009-09-05 21:47:34 +0000932 do
933 {
934 (void) WriteBlobByte(image,icon_file.directory[scene].width);
935 (void) WriteBlobByte(image,icon_file.directory[scene].height);
936 (void) WriteBlobByte(image,icon_file.directory[scene].colors);
937 (void) WriteBlobByte(image,icon_file.directory[scene].reserved);
938 (void) WriteBlobLSBShort(image,icon_file.directory[scene].planes);
939 (void) WriteBlobLSBShort(image,icon_file.directory[scene].bits_per_pixel);
cristy0b29b252010-05-30 01:59:46 +0000940 (void) WriteBlobLSBLong(image,(unsigned int)
941 icon_file.directory[scene].size);
942 (void) WriteBlobLSBLong(image,(unsigned int)
943 icon_file.directory[scene].offset);
cristy3ed852e2009-09-05 21:47:34 +0000944 scene++;
945 next=SyncNextImageInList(next);
946 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
947 scene=0;
dirk9e137702014-01-09 19:29:59 +0000948 next=(images != (Image *) NULL) ? images : image;
cristy3ed852e2009-09-05 21:47:34 +0000949 do
950 {
glennrp4b708b82012-07-17 19:15:58 +0000951 if ((next->columns > 255L) && (next->rows > 255L) &&
952 ((next->compression == UndefinedCompression) ||
953 (next->compression == ZipCompression)))
cristy3ed852e2009-09-05 21:47:34 +0000954 {
955 Image
956 *write_image;
957
958 ImageInfo
959 *write_info;
960
961 size_t
962 length;
963
964 unsigned char
965 *png;
966
cristy1e178e72011-08-28 19:44:34 +0000967 write_image=CloneImage(next,0,0,MagickTrue,exception);
dirkc1e52002014-01-09 20:22:42 +0000968 if (write_image == (Image *) NULL)
969 {
970 images=DestroyImageList(images);
971 return(MagickFalse);
dirk9e137702014-01-09 19:29:59 +0000972 }
cristy3ed852e2009-09-05 21:47:34 +0000973 write_info=CloneImageInfo(image_info);
974 (void) CopyMagickString(write_info->filename,"PNG:",MaxTextExtent);
glennrp4b708b82012-07-17 19:15:58 +0000975
976 /* Don't write any ancillary chunks except for gAMA */
977 (void) SetImageArtifact(write_image,"png:include-chunk","none,gama");
978
979 /* Only write PNG32 formatted PNG (32-bit RGBA), 8 bits per channel */
980 (void) SetImageArtifact(write_image,"png:format","png32");
981
cristy3ed852e2009-09-05 21:47:34 +0000982 png=(unsigned char *) ImageToBlob(write_info,write_image,&length,
cristy1e178e72011-08-28 19:44:34 +0000983 exception);
cristy3ed852e2009-09-05 21:47:34 +0000984 write_image=DestroyImage(write_image);
985 write_info=DestroyImageInfo(write_info);
dirkc1e52002014-01-09 20:22:42 +0000986 if (png == (unsigned char *) NULL)
987 {
988 images=DestroyImageList(images);
989 return(MagickFalse);
dirk9e137702014-01-09 19:29:59 +0000990 }
cristy3ed852e2009-09-05 21:47:34 +0000991 icon_file.directory[scene].width=0;
992 icon_file.directory[scene].height=0;
993 icon_file.directory[scene].colors=0;
994 icon_file.directory[scene].reserved=0;
995 icon_file.directory[scene].planes=1;
996 icon_file.directory[scene].bits_per_pixel=32;
cristybb503372010-05-27 20:51:26 +0000997 icon_file.directory[scene].size=(size_t) length;
998 icon_file.directory[scene].offset=(size_t) TellBlob(image);
cristy3ed852e2009-09-05 21:47:34 +0000999 (void) WriteBlob(image,(size_t) length,png);
1000 png=(unsigned char *) RelinquishMagickMemory(png);
1001 }
1002 else
1003 {
1004 /*
1005 Initialize ICON raster file header.
1006 */
cristyaf8d3912014-02-21 14:50:33 +00001007 (void) TransformImageColorspace(next,sRGBColorspace,exception);
cristy3ed852e2009-09-05 21:47:34 +00001008 icon_info.file_size=14+12+28;
1009 icon_info.offset_bits=icon_info.file_size;
1010 icon_info.compression=BI_RGB;
1011 if ((next->storage_class != DirectClass) && (next->colors > 256))
cristy1e178e72011-08-28 19:44:34 +00001012 (void) SetImageStorageClass(next,DirectClass,exception);
cristy3ed852e2009-09-05 21:47:34 +00001013 if (next->storage_class == DirectClass)
1014 {
1015 /*
1016 Full color ICON raster.
1017 */
1018 icon_info.number_colors=0;
1019 icon_info.bits_per_pixel=32;
cristybb503372010-05-27 20:51:26 +00001020 icon_info.compression=(size_t) BI_RGB;
cristy3ed852e2009-09-05 21:47:34 +00001021 }
1022 else
1023 {
cristy35ef8242010-06-03 16:24:13 +00001024 size_t
1025 one;
1026
cristy3ed852e2009-09-05 21:47:34 +00001027 /*
1028 Colormapped ICON raster.
1029 */
1030 icon_info.bits_per_pixel=8;
1031 if (next->colors <= 256)
1032 icon_info.bits_per_pixel=8;
1033 if (next->colors <= 16)
1034 icon_info.bits_per_pixel=4;
1035 if (next->colors <= 2)
1036 icon_info.bits_per_pixel=1;
cristy35ef8242010-06-03 16:24:13 +00001037 one=1;
1038 icon_info.number_colors=one << icon_info.bits_per_pixel;
cristy3ed852e2009-09-05 21:47:34 +00001039 if (icon_info.number_colors < next->colors)
1040 {
cristy1e178e72011-08-28 19:44:34 +00001041 (void) SetImageStorageClass(next,DirectClass,exception);
cristy3ed852e2009-09-05 21:47:34 +00001042 icon_info.number_colors=0;
1043 icon_info.bits_per_pixel=(unsigned short) 24;
cristybb503372010-05-27 20:51:26 +00001044 icon_info.compression=(size_t) BI_RGB;
cristy3ed852e2009-09-05 21:47:34 +00001045 }
1046 else
1047 {
cristy0b29b252010-05-30 01:59:46 +00001048 size_t
1049 one;
1050
1051 one=1;
1052 icon_info.file_size+=3*(one << icon_info.bits_per_pixel);
1053 icon_info.offset_bits+=3*(one << icon_info.bits_per_pixel);
1054 icon_info.file_size+=(one << icon_info.bits_per_pixel);
1055 icon_info.offset_bits+=(one << icon_info.bits_per_pixel);
cristy3ed852e2009-09-05 21:47:34 +00001056 }
1057 }
1058 bytes_per_line=(((next->columns*icon_info.bits_per_pixel)+31) &
1059 ~31) >> 3;
1060 icon_info.ba_offset=0;
cristybb503372010-05-27 20:51:26 +00001061 icon_info.width=(ssize_t) next->columns;
1062 icon_info.height=(ssize_t) next->rows;
cristy3ed852e2009-09-05 21:47:34 +00001063 icon_info.planes=1;
1064 icon_info.image_size=bytes_per_line*next->rows;
1065 icon_info.size=40;
1066 icon_info.size+=(4*icon_info.number_colors);
1067 icon_info.size+=icon_info.image_size;
1068 icon_info.size+=(((icon_info.width+31) & ~31) >> 3)*icon_info.height;
1069 icon_info.file_size+=icon_info.image_size;
1070 icon_info.x_pixels=0;
1071 icon_info.y_pixels=0;
1072 switch (next->units)
1073 {
1074 case UndefinedResolution:
1075 case PixelsPerInchResolution:
1076 {
cristy2a11bef2011-10-28 18:33:11 +00001077 icon_info.x_pixels=(size_t) (100.0*next->resolution.x/2.54);
1078 icon_info.y_pixels=(size_t) (100.0*next->resolution.y/2.54);
cristy3ed852e2009-09-05 21:47:34 +00001079 break;
1080 }
1081 case PixelsPerCentimeterResolution:
1082 {
cristy2a11bef2011-10-28 18:33:11 +00001083 icon_info.x_pixels=(size_t) (100.0*next->resolution.x);
1084 icon_info.y_pixels=(size_t) (100.0*next->resolution.y);
cristy3ed852e2009-09-05 21:47:34 +00001085 break;
1086 }
1087 }
1088 icon_info.colors_important=icon_info.number_colors;
1089 /*
1090 Convert MIFF to ICON raster pixels.
1091 */
1092 pixels=(unsigned char *) AcquireQuantumMemory((size_t)
1093 icon_info.image_size,sizeof(*pixels));
dirkc1e52002014-01-09 20:22:42 +00001094 if (pixels == (unsigned char *) NULL)
1095 {
dirk9e137702014-01-09 19:29:59 +00001096 images=DestroyImageList(images);
1097 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
dirkc1e52002014-01-09 20:22:42 +00001098 }
cristy3ed852e2009-09-05 21:47:34 +00001099 (void) ResetMagickMemory(pixels,0,(size_t) icon_info.image_size);
1100 switch (icon_info.bits_per_pixel)
1101 {
1102 case 1:
1103 {
cristybb503372010-05-27 20:51:26 +00001104 size_t
cristy3ed852e2009-09-05 21:47:34 +00001105 bit,
1106 byte;
1107
1108 /*
1109 Convert PseudoClass image to a ICON monochrome image.
1110 */
cristybb503372010-05-27 20:51:26 +00001111 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001112 {
cristy1e178e72011-08-28 19:44:34 +00001113 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001114 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001115 break;
cristy3ed852e2009-09-05 21:47:34 +00001116 q=pixels+(next->rows-y-1)*bytes_per_line;
1117 bit=0;
1118 byte=0;
cristybb503372010-05-27 20:51:26 +00001119 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001120 {
1121 byte<<=1;
cristy4c08aed2011-07-01 19:47:50 +00001122 byte|=GetPixelIndex(next,p) != 0 ? 0x01 : 0x00;
cristy3ed852e2009-09-05 21:47:34 +00001123 bit++;
1124 if (bit == 8)
1125 {
1126 *q++=(unsigned char) byte;
1127 bit=0;
1128 byte=0;
1129 }
cristyed231572011-07-14 02:18:59 +00001130 p+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +00001131 }
cristy3ed852e2009-09-05 21:47:34 +00001132 if (bit != 0)
1133 *q++=(unsigned char) (byte << (8-bit));
1134 if (next->previous == (Image *) NULL)
1135 {
1136 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1137 if (status == MagickFalse)
1138 break;
1139 }
1140 }
1141 break;
1142 }
1143 case 4:
1144 {
cristybb503372010-05-27 20:51:26 +00001145 size_t
cristy3ed852e2009-09-05 21:47:34 +00001146 nibble,
1147 byte;
1148
1149 /*
1150 Convert PseudoClass image to a ICON monochrome image.
1151 */
cristybb503372010-05-27 20:51:26 +00001152 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001153 {
cristy1e178e72011-08-28 19:44:34 +00001154 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001155 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001156 break;
cristy3ed852e2009-09-05 21:47:34 +00001157 q=pixels+(next->rows-y-1)*bytes_per_line;
1158 nibble=0;
1159 byte=0;
cristybb503372010-05-27 20:51:26 +00001160 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001161 {
1162 byte<<=4;
cristy4c08aed2011-07-01 19:47:50 +00001163 byte|=((size_t) GetPixelIndex(next,p) & 0x0f);
cristy3ed852e2009-09-05 21:47:34 +00001164 nibble++;
1165 if (nibble == 2)
1166 {
1167 *q++=(unsigned char) byte;
1168 nibble=0;
1169 byte=0;
1170 }
cristyed231572011-07-14 02:18:59 +00001171 p+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +00001172 }
cristy3ed852e2009-09-05 21:47:34 +00001173 if (nibble != 0)
1174 *q++=(unsigned char) (byte << 4);
1175 if (next->previous == (Image *) NULL)
1176 {
1177 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1178 if (status == MagickFalse)
1179 break;
1180 }
1181 }
1182 break;
1183 }
1184 case 8:
1185 {
1186 /*
1187 Convert PseudoClass packet to ICON pixel.
1188 */
cristybb503372010-05-27 20:51:26 +00001189 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001190 {
cristy1e178e72011-08-28 19:44:34 +00001191 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001192 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001193 break;
cristy3ed852e2009-09-05 21:47:34 +00001194 q=pixels+(next->rows-y-1)*bytes_per_line;
cristybb503372010-05-27 20:51:26 +00001195 for (x=0; x < (ssize_t) next->columns; x++)
cristy4c08aed2011-07-01 19:47:50 +00001196 {
1197 *q++=(unsigned char) GetPixelIndex(next,p);
cristyed231572011-07-14 02:18:59 +00001198 p+=GetPixelChannels(image);
cristy4c08aed2011-07-01 19:47:50 +00001199 }
cristy3ed852e2009-09-05 21:47:34 +00001200 if (next->previous == (Image *) NULL)
1201 {
1202 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1203 if (status == MagickFalse)
1204 break;
1205 }
1206 }
1207 break;
1208 }
1209 case 24:
1210 case 32:
1211 {
1212 /*
1213 Convert DirectClass packet to ICON BGR888 or BGRA8888 pixel.
1214 */
cristybb503372010-05-27 20:51:26 +00001215 for (y=0; y < (ssize_t) next->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001216 {
cristy1e178e72011-08-28 19:44:34 +00001217 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001218 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001219 break;
1220 q=pixels+(next->rows-y-1)*bytes_per_line;
cristybb503372010-05-27 20:51:26 +00001221 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001222 {
cristy4c08aed2011-07-01 19:47:50 +00001223 *q++=ScaleQuantumToChar(GetPixelBlue(next,p));
1224 *q++=ScaleQuantumToChar(GetPixelGreen(next,p));
1225 *q++=ScaleQuantumToChar(GetPixelRed(next,p));
cristy17f11b02014-12-20 19:37:04 +00001226 if (next->alpha_trait == UndefinedPixelTrait)
cristy3ed852e2009-09-05 21:47:34 +00001227 *q++=ScaleQuantumToChar(QuantumRange);
1228 else
cristy4c08aed2011-07-01 19:47:50 +00001229 *q++=ScaleQuantumToChar(GetPixelAlpha(next,p));
cristyed231572011-07-14 02:18:59 +00001230 p+=GetPixelChannels(next);
cristy3ed852e2009-09-05 21:47:34 +00001231 }
1232 if (icon_info.bits_per_pixel == 24)
cristybb503372010-05-27 20:51:26 +00001233 for (x=3L*(ssize_t) next->columns; x < (ssize_t) bytes_per_line; x++)
cristy3ed852e2009-09-05 21:47:34 +00001234 *q++=0x00;
1235 if (next->previous == (Image *) NULL)
1236 {
1237 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1238 if (status == MagickFalse)
1239 break;
1240 }
1241 }
1242 break;
1243 }
1244 }
1245 /*
1246 Write 40-byte version 3+ bitmap header.
1247 */
1248 icon_file.directory[scene].width=(unsigned char) icon_info.width;
1249 icon_file.directory[scene].height=(unsigned char) icon_info.height;
1250 icon_file.directory[scene].colors=(unsigned char)
1251 icon_info.number_colors;
1252 icon_file.directory[scene].reserved=0;
1253 icon_file.directory[scene].planes=icon_info.planes;
1254 icon_file.directory[scene].bits_per_pixel=icon_info.bits_per_pixel;
1255 icon_file.directory[scene].size=icon_info.size;
cristybb503372010-05-27 20:51:26 +00001256 icon_file.directory[scene].offset=(size_t) TellBlob(image);
cristy35ef8242010-06-03 16:24:13 +00001257 (void) WriteBlobLSBLong(image,(unsigned int) 40);
1258 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.width);
1259 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.height*2);
cristy3ed852e2009-09-05 21:47:34 +00001260 (void) WriteBlobLSBShort(image,icon_info.planes);
1261 (void) WriteBlobLSBShort(image,icon_info.bits_per_pixel);
cristy35ef8242010-06-03 16:24:13 +00001262 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.compression);
1263 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.image_size);
1264 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.x_pixels);
1265 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.y_pixels);
1266 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.number_colors);
1267 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.colors_important);
cristy3ed852e2009-09-05 21:47:34 +00001268 if (next->storage_class == PseudoClass)
1269 {
1270 unsigned char
1271 *icon_colormap;
1272
1273 /*
1274 Dump colormap to file.
1275 */
1276 icon_colormap=(unsigned char *) AcquireQuantumMemory((size_t)
1277 (1UL << icon_info.bits_per_pixel),4UL*sizeof(*icon_colormap));
dirkc1e52002014-01-09 20:22:42 +00001278 if (icon_colormap == (unsigned char *) NULL)
1279 {
1280 images=DestroyImageList(images);
1281 ThrowWriterException(ResourceLimitError,
1282 "MemoryAllocationFailed");
dirk9e137702014-01-09 19:29:59 +00001283 }
cristy3ed852e2009-09-05 21:47:34 +00001284 q=icon_colormap;
cristybb503372010-05-27 20:51:26 +00001285 for (i=0; i < (ssize_t) next->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +00001286 {
1287 *q++=ScaleQuantumToChar(next->colormap[i].blue);
1288 *q++=ScaleQuantumToChar(next->colormap[i].green);
1289 *q++=ScaleQuantumToChar(next->colormap[i].red);
1290 *q++=(unsigned char) 0x0;
1291 }
cristybb503372010-05-27 20:51:26 +00001292 for ( ; i < (ssize_t) (1UL << icon_info.bits_per_pixel); i++)
cristy3ed852e2009-09-05 21:47:34 +00001293 {
1294 *q++=(unsigned char) 0x00;
1295 *q++=(unsigned char) 0x00;
1296 *q++=(unsigned char) 0x00;
1297 *q++=(unsigned char) 0x00;
1298 }
1299 (void) WriteBlob(image,(size_t) (4UL*(1UL <<
1300 icon_info.bits_per_pixel)),icon_colormap);
1301 icon_colormap=(unsigned char *) RelinquishMagickMemory(
1302 icon_colormap);
1303 }
1304 (void) WriteBlob(image,(size_t) icon_info.image_size,pixels);
1305 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
1306 /*
1307 Write matte mask.
1308 */
1309 scanline_pad=(((next->columns+31) & ~31)-next->columns) >> 3;
cristybb503372010-05-27 20:51:26 +00001310 for (y=((ssize_t) next->rows - 1); y >= 0; y--)
cristy3ed852e2009-09-05 21:47:34 +00001311 {
cristy1e178e72011-08-28 19:44:34 +00001312 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
cristy4c08aed2011-07-01 19:47:50 +00001313 if (p == (const Quantum *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00001314 break;
1315 bit=0;
1316 byte=0;
cristybb503372010-05-27 20:51:26 +00001317 for (x=0; x < (ssize_t) next->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001318 {
1319 byte<<=1;
cristy17f11b02014-12-20 19:37:04 +00001320 if ((next->alpha_trait != UndefinedPixelTrait) &&
cristy4c08aed2011-07-01 19:47:50 +00001321 (GetPixelAlpha(next,p) == (Quantum) TransparentAlpha))
cristy3ed852e2009-09-05 21:47:34 +00001322 byte|=0x01;
1323 bit++;
1324 if (bit == 8)
1325 {
1326 (void) WriteBlobByte(image,(unsigned char) byte);
1327 bit=0;
1328 byte=0;
1329 }
cristyed231572011-07-14 02:18:59 +00001330 p+=GetPixelChannels(next);
cristy3ed852e2009-09-05 21:47:34 +00001331 }
1332 if (bit != 0)
1333 (void) WriteBlobByte(image,(unsigned char) (byte << (8-bit)));
cristybb503372010-05-27 20:51:26 +00001334 for (i=0; i < (ssize_t) scanline_pad; i++)
cristy3ed852e2009-09-05 21:47:34 +00001335 (void) WriteBlobByte(image,(unsigned char) 0);
1336 }
1337 }
1338 if (GetNextImageInList(next) == (Image *) NULL)
1339 break;
cristy3ed852e2009-09-05 21:47:34 +00001340 status=SetImageProgress(next,SaveImagesTag,scene++,
1341 GetImageListLength(next));
1342 if (status == MagickFalse)
1343 break;
cristy5467fcf2014-05-18 17:20:55 +00001344 next=SyncNextImageInList(next);
cristy3ed852e2009-09-05 21:47:34 +00001345 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
1346 offset=SeekBlob(image,0,SEEK_SET);
cristyda16f162011-02-19 23:52:17 +00001347 (void) offset;
cristy3ed852e2009-09-05 21:47:34 +00001348 (void) WriteBlobLSBShort(image,0);
1349 (void) WriteBlobLSBShort(image,1);
1350 (void) WriteBlobLSBShort(image,(unsigned short) (scene+1));
1351 scene=0;
dirk9e137702014-01-09 19:29:59 +00001352 next=(images != (Image *) NULL) ? images : image;
cristy3ed852e2009-09-05 21:47:34 +00001353 do
1354 {
1355 (void) WriteBlobByte(image,icon_file.directory[scene].width);
1356 (void) WriteBlobByte(image,icon_file.directory[scene].height);
1357 (void) WriteBlobByte(image,icon_file.directory[scene].colors);
1358 (void) WriteBlobByte(image,icon_file.directory[scene].reserved);
1359 (void) WriteBlobLSBShort(image,icon_file.directory[scene].planes);
1360 (void) WriteBlobLSBShort(image,icon_file.directory[scene].bits_per_pixel);
cristy0b29b252010-05-30 01:59:46 +00001361 (void) WriteBlobLSBLong(image,(unsigned int)
1362 icon_file.directory[scene].size);
1363 (void) WriteBlobLSBLong(image,(unsigned int)
1364 icon_file.directory[scene].offset);
cristy3ed852e2009-09-05 21:47:34 +00001365 scene++;
1366 next=SyncNextImageInList(next);
1367 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
1368 (void) CloseBlob(image);
dirk9e137702014-01-09 19:29:59 +00001369 images=DestroyImageList(images);
cristy3ed852e2009-09-05 21:47:34 +00001370 return(MagickTrue);
1371}