2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
10 % IIIII CCCC OOO N N %
13 % Read Microsoft Windows Icon Format %
20 % Copyright 1999-2012 ImageMagick Studio LLC, a non-profit organization %
21 % dedicated to making software imaging solutions freely available. %
23 % You may not use this file except in compliance with the License. You may %
24 % obtain a copy of the License at %
26 % http://www.imagemagick.org/script/license.php %
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. %
34 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
42 #include "MagickCore/studio.h"
43 #include "MagickCore/artifact.h"
44 #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"
49 #include "MagickCore/colorspace-private.h"
50 #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"
60 #include "MagickCore/nt-base-private.h"
61 #include "MagickCore/pixel-accessor.h"
62 #include "MagickCore/quantize.h"
63 #include "MagickCore/quantum-private.h"
64 #include "MagickCore/static.h"
65 #include "MagickCore/string_.h"
66 #include "MagickCore/module.h"
71 #if !defined(MAGICKCORE_WINDOWS_SUPPORT) || defined(__MINGW32__)
74 #define BI_BITFIELDS 3
81 typedef struct _IconEntry
98 typedef struct _IconFile
109 typedef struct _IconInfo
142 Forward declaractions.
144 static MagickBooleanType
145 WriteICONImage(const ImageInfo *,Image *,ExceptionInfo *);
148 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
152 % R e a d I C O N I m a g e %
156 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
158 % ReadICONImage() reads a Microsoft icon image file and returns it. It
159 % allocates the memory necessary for the new Image structure and returns a
160 % pointer to the new image.
162 % The format of the ReadICONImage method is:
164 % Image *ReadICONImage(const ImageInfo *image_info,
165 % ExceptionInfo *exception)
167 % A description of each parameter follows:
169 % o image_info: the image info.
171 % o exception: return any errors or warnings in this structure.
174 static Image *ReadICONImage(const ImageInfo *image_info,
175 ExceptionInfo *exception)
196 register unsigned char
214 assert(image_info != (const ImageInfo *) NULL);
215 assert(image_info->signature == MagickSignature);
216 (void) LogMagickEvent(CoderEvent,GetMagickModule(),"%s",image_info->filename);
217 assert(exception != (ExceptionInfo *) NULL);
218 assert(exception->signature == MagickSignature);
219 image=AcquireImage(image_info,exception);
220 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
221 if (status == MagickFalse)
223 image=DestroyImageList(image);
224 return((Image *) NULL);
226 icon_file.reserved=(short) ReadBlobLSBShort(image);
227 icon_file.resource_type=(short) ReadBlobLSBShort(image);
228 icon_file.count=(short) ReadBlobLSBShort(image);
229 if ((icon_file.reserved != 0) ||
230 ((icon_file.resource_type != 1) && (icon_file.resource_type != 2)) ||
231 (icon_file.count > MaxIcons))
232 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
233 for (i=0; i < icon_file.count; i++)
235 icon_file.directory[i].width=(unsigned char) ReadBlobByte(image);
236 icon_file.directory[i].height=(unsigned char) ReadBlobByte(image);
237 icon_file.directory[i].colors=(unsigned char) ReadBlobByte(image);
238 icon_file.directory[i].reserved=(unsigned char) ReadBlobByte(image);
239 icon_file.directory[i].planes=(unsigned short) ReadBlobLSBShort(image);
240 icon_file.directory[i].bits_per_pixel=(unsigned short)
241 ReadBlobLSBShort(image);
242 icon_file.directory[i].size=ReadBlobLSBLong(image);
243 icon_file.directory[i].offset=ReadBlobLSBLong(image);
246 for (i=0; i < icon_file.count; i++)
249 Verify Icon identifier.
251 offset=(ssize_t) SeekBlob(image,(MagickOffsetType)
252 icon_file.directory[i].offset,SEEK_SET);
254 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
255 icon_info.size=ReadBlobLSBLong(image);
256 icon_info.width=(unsigned char) ((int) ReadBlobLSBLong(image));
257 icon_info.height=(unsigned char) ((int) ReadBlobLSBLong(image)/2);
258 icon_info.planes=ReadBlobLSBShort(image);
259 icon_info.bits_per_pixel=ReadBlobLSBShort(image);
260 if ((icon_info.planes == 18505) && (icon_info.bits_per_pixel == 21060))
275 Icon image encoded as a compressed PNG image.
277 length=icon_file.directory[i].size;
278 png=(unsigned char *) AcquireQuantumMemory(length+16,sizeof(*png));
279 if (png == (unsigned char *) NULL)
280 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
281 (void) CopyMagickMemory(png,"\211PNG\r\n\032\n\000\000\000\015",12);
282 png[12]=(unsigned char) icon_info.planes;
283 png[13]=(unsigned char) (icon_info.planes >> 8);
284 png[14]=(unsigned char) icon_info.bits_per_pixel;
285 png[15]=(unsigned char) (icon_info.bits_per_pixel >> 8);
286 count=ReadBlob(image,length-16,png+16);
287 if (count != (ssize_t) (length-16))
289 png=(unsigned char *) RelinquishMagickMemory(png);
290 ThrowReaderException(CorruptImageError,
291 "InsufficientImageDataInFile");
293 read_info=CloneImageInfo(image_info);
294 (void) CopyMagickString(read_info->magick,"PNG",MaxTextExtent);
295 icon_image=BlobToImage(read_info,png,length+16,exception);
296 read_info=DestroyImageInfo(read_info);
297 png=(unsigned char *) RelinquishMagickMemory(png);
298 if (icon_image == (Image *) NULL)
300 image=DestroyImageList(image);
301 return((Image *) NULL);
303 DestroyBlob(icon_image);
304 icon_image->blob=ReferenceBlob(image->blob);
305 ReplaceImageInList(&image,icon_image);
309 if (icon_info.bits_per_pixel > 32)
310 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
311 icon_info.compression=ReadBlobLSBLong(image);
312 icon_info.image_size=ReadBlobLSBLong(image);
313 icon_info.x_pixels=ReadBlobLSBLong(image);
314 icon_info.y_pixels=ReadBlobLSBLong(image);
315 icon_info.number_colors=ReadBlobLSBLong(image);
316 icon_info.colors_important=ReadBlobLSBLong(image);
317 image->matte=MagickTrue;
318 image->columns=(size_t) icon_file.directory[i].width;
319 if ((ssize_t) image->columns > icon_info.width)
320 image->columns=(size_t) icon_info.width;
321 if (image->columns == 0)
323 image->rows=(size_t) icon_file.directory[i].height;
324 if ((ssize_t) image->rows > icon_info.height)
325 image->rows=(size_t) icon_info.height;
326 if (image->rows == 0)
328 image->depth=icon_info.bits_per_pixel;
329 if (image->debug != MagickFalse)
331 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
332 " scene = %.20g",(double) i);
333 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
334 " size = %.20g",(double) icon_info.size);
335 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
336 " width = %.20g",(double) icon_file.directory[i].width);
337 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
338 " height = %.20g",(double) icon_file.directory[i].height);
339 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
340 " colors = %.20g",(double ) icon_info.number_colors);
341 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
342 " planes = %.20g",(double) icon_info.planes);
343 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
344 " bpp = %.20g",(double) icon_info.bits_per_pixel);
346 if ((icon_info.number_colors != 0) || (icon_info.bits_per_pixel <= 16))
348 image->storage_class=PseudoClass;
349 image->colors=icon_info.number_colors;
350 if (image->colors == 0)
351 image->colors=one << icon_info.bits_per_pixel;
353 if (image->storage_class == PseudoClass)
366 Read Icon raster colormap.
369 number_colors=one << icon_info.bits_per_pixel;
370 if (AcquireImageColormap(image,number_colors,exception) == MagickFalse)
371 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
372 icon_colormap=(unsigned char *) AcquireQuantumMemory((size_t)
373 image->colors,4UL*sizeof(*icon_colormap));
374 if (icon_colormap == (unsigned char *) NULL)
375 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
376 count=ReadBlob(image,(size_t) (4*image->colors),icon_colormap);
377 if (count != (ssize_t) (4*image->colors))
378 ThrowReaderException(CorruptImageError,
379 "InsufficientImageDataInFile");
381 for (i=0; i < (ssize_t) image->colors; i++)
383 image->colormap[i].blue=(Quantum) ScaleCharToQuantum(*p++);
384 image->colormap[i].green=(Quantum) ScaleCharToQuantum(*p++);
385 image->colormap[i].red=(Quantum) ScaleCharToQuantum(*p++);
388 icon_colormap=(unsigned char *) RelinquishMagickMemory(icon_colormap);
391 Convert Icon raster image to pixel packets.
393 if ((image_info->ping != MagickFalse) &&
394 (image_info->number_scenes != 0))
395 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
397 bytes_per_line=(((image->columns*icon_info.bits_per_pixel)+31) &
399 (void) bytes_per_line;
400 scanline_pad=((((image->columns*icon_info.bits_per_pixel)+31) & ~31)-
401 (image->columns*icon_info.bits_per_pixel)) >> 3;
402 switch (icon_info.bits_per_pixel)
407 Convert bitmap scanline.
409 for (y=(ssize_t) image->rows-1; y >= 0; y--)
411 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
412 if (q == (Quantum *) NULL)
414 for (x=0; x < (ssize_t) (image->columns-7); x+=8)
416 byte=(size_t) ReadBlobByte(image);
417 for (bit=0; bit < 8; bit++)
419 SetPixelIndex(image,((byte & (0x80 >> bit)) != 0 ? 0x01 :
421 q+=GetPixelChannels(image);
424 if ((image->columns % 8) != 0)
426 byte=(size_t) ReadBlobByte(image);
427 for (bit=0; bit < (image->columns % 8); bit++)
429 SetPixelIndex(image,((byte & (0x80 >> bit)) != 0 ? 0x01 :
431 q+=GetPixelChannels(image);
434 for (x=0; x < (ssize_t) scanline_pad; x++)
435 (void) ReadBlobByte(image);
436 if (SyncAuthenticPixels(image,exception) == MagickFalse)
438 if (image->previous == (Image *) NULL)
440 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
442 if (status == MagickFalse)
451 Read 4-bit Icon scanline.
453 for (y=(ssize_t) image->rows-1; y >= 0; y--)
455 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
456 if (q == (Quantum *) NULL)
458 for (x=0; x < ((ssize_t) image->columns-1); x+=2)
460 byte=(size_t) ReadBlobByte(image);
461 SetPixelIndex(image,((byte >> 4) & 0xf),q);
462 q+=GetPixelChannels(image);
463 SetPixelIndex(image,((byte) & 0xf),q);
464 q+=GetPixelChannels(image);
466 if ((image->columns % 2) != 0)
468 byte=(size_t) ReadBlobByte(image);
469 SetPixelIndex(image,((byte >> 4) & 0xf),q);
470 q+=GetPixelChannels(image);
472 for (x=0; x < (ssize_t) scanline_pad; x++)
473 (void) ReadBlobByte(image);
474 if (SyncAuthenticPixels(image,exception) == MagickFalse)
476 if (image->previous == (Image *) NULL)
478 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
480 if (status == MagickFalse)
489 Convert PseudoColor scanline.
491 for (y=(ssize_t) image->rows-1; y >= 0; y--)
493 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
494 if (q == (Quantum *) NULL)
496 for (x=0; x < (ssize_t) image->columns; x++)
498 byte=(size_t) ReadBlobByte(image);
499 SetPixelIndex(image,byte,q);
500 q+=GetPixelChannels(image);
502 for (x=0; x < (ssize_t) scanline_pad; x++)
503 (void) ReadBlobByte(image);
504 if (SyncAuthenticPixels(image,exception) == MagickFalse)
506 if (image->previous == (Image *) NULL)
508 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
510 if (status == MagickFalse)
519 Convert PseudoColor scanline.
521 for (y=(ssize_t) image->rows-1; y >= 0; y--)
523 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
524 if (q == (Quantum *) NULL)
526 for (x=0; x < (ssize_t) image->columns; x++)
528 byte=(size_t) ReadBlobByte(image);
529 byte|=(size_t) (ReadBlobByte(image) << 8);
530 SetPixelIndex(image,byte,q);
531 q+=GetPixelChannels(image);
533 for (x=0; x < (ssize_t) scanline_pad; x++)
534 (void) ReadBlobByte(image);
535 if (SyncAuthenticPixels(image,exception) == MagickFalse)
537 if (image->previous == (Image *) NULL)
539 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
541 if (status == MagickFalse)
551 Convert DirectColor scanline.
553 for (y=(ssize_t) image->rows-1; y >= 0; y--)
555 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
556 if (q == (Quantum *) NULL)
558 for (x=0; x < (ssize_t) image->columns; x++)
560 SetPixelBlue(image,ScaleCharToQuantum((unsigned char)
561 ReadBlobByte(image)),q);
562 SetPixelGreen(image,ScaleCharToQuantum((unsigned char)
563 ReadBlobByte(image)),q);
564 SetPixelRed(image,ScaleCharToQuantum((unsigned char)
565 ReadBlobByte(image)),q);
566 if (icon_info.bits_per_pixel == 32)
567 SetPixelAlpha(image,ScaleCharToQuantum((unsigned char)
568 ReadBlobByte(image)),q);
569 q+=GetPixelChannels(image);
571 if (icon_info.bits_per_pixel == 24)
572 for (x=0; x < (ssize_t) scanline_pad; x++)
573 (void) ReadBlobByte(image);
574 if (SyncAuthenticPixels(image,exception) == MagickFalse)
576 if (image->previous == (Image *) NULL)
578 status=SetImageProgress(image,LoadImageTag,image->rows-y-1,
580 if (status == MagickFalse)
587 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
589 if (image_info->ping == MagickFalse)
590 (void) SyncImage(image,exception);
591 if (icon_info.bits_per_pixel != 32)
594 Read the ICON alpha mask.
596 image->storage_class=DirectClass;
597 for (y=(ssize_t) image->rows-1; y >= 0; y--)
599 q=GetAuthenticPixels(image,0,y,image->columns,1,exception);
600 if (q == (Quantum *) NULL)
602 for (x=0; x < ((ssize_t) image->columns-7); x+=8)
604 byte=(size_t) ReadBlobByte(image);
605 for (bit=0; bit < 8; bit++)
607 SetPixelAlpha(image,(((byte & (0x80 >> bit)) != 0) ?
608 TransparentAlpha : OpaqueAlpha),q);
609 q+=GetPixelChannels(image);
612 if ((image->columns % 8) != 0)
614 byte=(size_t) ReadBlobByte(image);
615 for (bit=0; bit < (image->columns % 8); bit++)
617 SetPixelAlpha(image,(((byte & (0x80 >> bit)) != 0) ?
618 TransparentAlpha : OpaqueAlpha),q);
619 q+=GetPixelChannels(image);
622 if ((image->columns % 32) != 0)
623 for (x=0; x < (ssize_t) ((32-(image->columns % 32))/8); x++)
624 (void) ReadBlobByte(image);
625 if (SyncAuthenticPixels(image,exception) == MagickFalse)
629 if (EOFBlob(image) != MagickFalse)
631 ThrowFileException(exception,CorruptImageError,
632 "UnexpectedEndOfFile",image->filename);
637 Proceed to next image.
639 if (image_info->number_scenes != 0)
640 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
642 if (i < (ssize_t) (icon_file.count-1))
645 Allocate next image structure.
647 AcquireNextImage(image_info,image,exception);
648 if (GetNextImageInList(image) == (Image *) NULL)
650 image=DestroyImageList(image);
651 return((Image *) NULL);
653 image=SyncNextImageInList(image);
654 status=SetImageProgress(image,LoadImagesTag,TellBlob(image),
656 if (status == MagickFalse)
660 (void) CloseBlob(image);
661 return(GetFirstImageInList(image));
665 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
669 % R e g i s t e r I C O N I m a g e %
673 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
675 % RegisterICONImage() adds attributes for the Icon image format to
676 % the list of supported formats. The attributes include the image format
677 % tag, a method to read and/or write the format, whether the format
678 % supports the saving of more than one frame to the same file or blob,
679 % whether the format supports native in-memory I/O, and a brief
680 % description of the format.
682 % The format of the RegisterICONImage method is:
684 % size_t RegisterICONImage(void)
687 ModuleExport size_t RegisterICONImage(void)
692 entry=SetMagickInfo("CUR");
693 entry->decoder=(DecodeImageHandler *) ReadICONImage;
694 entry->encoder=(EncodeImageHandler *) WriteICONImage;
695 entry->adjoin=MagickFalse;
696 entry->seekable_stream=MagickTrue;
697 entry->description=ConstantString("Microsoft icon");
698 entry->module=ConstantString("CUR");
699 (void) RegisterMagickInfo(entry);
700 entry=SetMagickInfo("ICO");
701 entry->decoder=(DecodeImageHandler *) ReadICONImage;
702 entry->encoder=(EncodeImageHandler *) WriteICONImage;
703 entry->adjoin=MagickTrue;
704 entry->seekable_stream=MagickTrue;
705 entry->description=ConstantString("Microsoft icon");
706 entry->module=ConstantString("ICON");
707 (void) RegisterMagickInfo(entry);
708 entry=SetMagickInfo("ICON");
709 entry->decoder=(DecodeImageHandler *) ReadICONImage;
710 entry->encoder=(EncodeImageHandler *) WriteICONImage;
711 entry->adjoin=MagickFalse;
712 entry->seekable_stream=MagickTrue;
713 entry->description=ConstantString("Microsoft icon");
714 entry->module=ConstantString("ICON");
715 (void) RegisterMagickInfo(entry);
716 return(MagickImageCoderSignature);
720 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
724 % U n r e g i s t e r I C O N I m a g e %
728 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
730 % UnregisterICONImage() removes format registrations made by the
731 % ICON module from the list of supported formats.
733 % The format of the UnregisterICONImage method is:
735 % UnregisterICONImage(void)
738 ModuleExport void UnregisterICONImage(void)
740 (void) UnregisterMagickInfo("CUR");
741 (void) UnregisterMagickInfo("ICO");
742 (void) UnregisterMagickInfo("ICON");
746 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
750 % W r i t e I C O N I m a g e %
754 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
756 % WriteICONImage() writes an image in Microsoft Windows bitmap encoded
757 % image format, version 3 for Windows or (if the image has a matte channel)
760 % It encodes any subimage as a compressed PNG image ("BI_PNG)",
761 % only when its dimensions are 256x256 and image->compression is
762 % undefined or is defined as ZipCompression.
764 % The format of the WriteICONImage method is:
766 % MagickBooleanType WriteICONImage(const ImageInfo *image_info,
767 % Image *image,ExceptionInfo *exception)
769 % A description of each parameter follows.
771 % o image_info: the image info.
773 % o image: The image.
775 % o exception: return any errors or warnings in this structure.
778 static MagickBooleanType WriteICONImage(const ImageInfo *image_info,
779 Image *image,ExceptionInfo *exception)
797 register const Quantum
804 register unsigned char
820 Open output image file.
822 assert(image_info != (const ImageInfo *) NULL);
823 assert(image_info->signature == MagickSignature);
824 assert(image != (Image *) NULL);
825 assert(image->signature == MagickSignature);
826 (void) LogMagickEvent(CoderEvent,GetMagickModule(),"%s",image->filename);
827 assert(exception != (ExceptionInfo *) NULL);
828 assert(exception->signature == MagickSignature);
829 status=OpenBlob(image_info,image,WriteBinaryBlobMode,exception);
830 if (status == MagickFalse)
836 if ((image->columns > 256L) || (image->rows > 256L))
837 ThrowWriterException(ImageError,"WidthOrHeightExceedsLimit");
839 next=SyncNextImageInList(next);
840 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
842 Dump out a ICON header template to be properly initialized later.
844 (void) WriteBlobLSBShort(image,0);
845 (void) WriteBlobLSBShort(image,1);
846 (void) WriteBlobLSBShort(image,(unsigned char) scene);
847 (void) ResetMagickMemory(&icon_file,0,sizeof(icon_file));
848 (void) ResetMagickMemory(&icon_info,0,sizeof(icon_info));
853 (void) WriteBlobByte(image,icon_file.directory[scene].width);
854 (void) WriteBlobByte(image,icon_file.directory[scene].height);
855 (void) WriteBlobByte(image,icon_file.directory[scene].colors);
856 (void) WriteBlobByte(image,icon_file.directory[scene].reserved);
857 (void) WriteBlobLSBShort(image,icon_file.directory[scene].planes);
858 (void) WriteBlobLSBShort(image,icon_file.directory[scene].bits_per_pixel);
859 (void) WriteBlobLSBLong(image,(unsigned int)
860 icon_file.directory[scene].size);
861 (void) WriteBlobLSBLong(image,(unsigned int)
862 icon_file.directory[scene].offset);
864 next=SyncNextImageInList(next);
865 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
870 if ((next->columns > 255L) && (next->rows > 255L) &&
871 ((next->compression == UndefinedCompression) ||
872 (next->compression == ZipCompression)))
886 write_image=CloneImage(next,0,0,MagickTrue,exception);
887 if (write_image == (Image *) NULL)
889 write_info=CloneImageInfo(image_info);
890 (void) CopyMagickString(write_info->filename,"PNG:",MaxTextExtent);
892 /* Don't write any ancillary chunks except for gAMA */
893 (void) SetImageArtifact(write_image,"png:include-chunk","none,gama");
895 /* Only write PNG32 formatted PNG (32-bit RGBA), 8 bits per channel */
896 (void) SetImageArtifact(write_image,"png:format","png32");
898 png=(unsigned char *) ImageToBlob(write_info,write_image,&length,
900 write_image=DestroyImage(write_image);
901 write_info=DestroyImageInfo(write_info);
902 if (png == (unsigned char *) NULL)
904 icon_file.directory[scene].width=0;
905 icon_file.directory[scene].height=0;
906 icon_file.directory[scene].colors=0;
907 icon_file.directory[scene].reserved=0;
908 icon_file.directory[scene].planes=1;
909 icon_file.directory[scene].bits_per_pixel=32;
910 icon_file.directory[scene].size=(size_t) length;
911 icon_file.directory[scene].offset=(size_t) TellBlob(image);
912 (void) WriteBlob(image,(size_t) length,png);
913 png=(unsigned char *) RelinquishMagickMemory(png);
918 Initialize ICON raster file header.
920 if (IssRGBCompatibleColorspace(next->colorspace) == MagickFalse)
921 (void) TransformImageColorspace(next,sRGBColorspace,exception);
922 icon_info.file_size=14+12+28;
923 icon_info.offset_bits=icon_info.file_size;
924 icon_info.compression=BI_RGB;
925 if ((next->storage_class != DirectClass) && (next->colors > 256))
926 (void) SetImageStorageClass(next,DirectClass,exception);
927 if (next->storage_class == DirectClass)
930 Full color ICON raster.
932 icon_info.number_colors=0;
933 icon_info.bits_per_pixel=32;
934 icon_info.compression=(size_t) BI_RGB;
942 Colormapped ICON raster.
944 icon_info.bits_per_pixel=8;
945 if (next->colors <= 256)
946 icon_info.bits_per_pixel=8;
947 if (next->colors <= 16)
948 icon_info.bits_per_pixel=4;
949 if (next->colors <= 2)
950 icon_info.bits_per_pixel=1;
952 icon_info.number_colors=one << icon_info.bits_per_pixel;
953 if (icon_info.number_colors < next->colors)
955 (void) SetImageStorageClass(next,DirectClass,exception);
956 icon_info.number_colors=0;
957 icon_info.bits_per_pixel=(unsigned short) 24;
958 icon_info.compression=(size_t) BI_RGB;
966 icon_info.file_size+=3*(one << icon_info.bits_per_pixel);
967 icon_info.offset_bits+=3*(one << icon_info.bits_per_pixel);
968 icon_info.file_size+=(one << icon_info.bits_per_pixel);
969 icon_info.offset_bits+=(one << icon_info.bits_per_pixel);
972 bytes_per_line=(((next->columns*icon_info.bits_per_pixel)+31) &
974 icon_info.ba_offset=0;
975 icon_info.width=(ssize_t) next->columns;
976 icon_info.height=(ssize_t) next->rows;
978 icon_info.image_size=bytes_per_line*next->rows;
980 icon_info.size+=(4*icon_info.number_colors);
981 icon_info.size+=icon_info.image_size;
982 icon_info.size+=(((icon_info.width+31) & ~31) >> 3)*icon_info.height;
983 icon_info.file_size+=icon_info.image_size;
984 icon_info.x_pixels=0;
985 icon_info.y_pixels=0;
988 case UndefinedResolution:
989 case PixelsPerInchResolution:
991 icon_info.x_pixels=(size_t) (100.0*next->resolution.x/2.54);
992 icon_info.y_pixels=(size_t) (100.0*next->resolution.y/2.54);
995 case PixelsPerCentimeterResolution:
997 icon_info.x_pixels=(size_t) (100.0*next->resolution.x);
998 icon_info.y_pixels=(size_t) (100.0*next->resolution.y);
1002 icon_info.colors_important=icon_info.number_colors;
1004 Convert MIFF to ICON raster pixels.
1006 pixels=(unsigned char *) AcquireQuantumMemory((size_t)
1007 icon_info.image_size,sizeof(*pixels));
1008 if (pixels == (unsigned char *) NULL)
1009 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
1010 (void) ResetMagickMemory(pixels,0,(size_t) icon_info.image_size);
1011 switch (icon_info.bits_per_pixel)
1020 Convert PseudoClass image to a ICON monochrome image.
1022 for (y=0; y < (ssize_t) next->rows; y++)
1024 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
1025 if (p == (const Quantum *) NULL)
1027 q=pixels+(next->rows-y-1)*bytes_per_line;
1030 for (x=0; x < (ssize_t) next->columns; x++)
1033 byte|=GetPixelIndex(next,p) != 0 ? 0x01 : 0x00;
1037 *q++=(unsigned char) byte;
1041 p+=GetPixelChannels(image);
1044 *q++=(unsigned char) (byte << (8-bit));
1045 if (next->previous == (Image *) NULL)
1047 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1048 if (status == MagickFalse)
1061 Convert PseudoClass image to a ICON monochrome image.
1063 for (y=0; y < (ssize_t) next->rows; y++)
1065 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
1066 if (p == (const Quantum *) NULL)
1068 q=pixels+(next->rows-y-1)*bytes_per_line;
1071 for (x=0; x < (ssize_t) next->columns; x++)
1074 byte|=((size_t) GetPixelIndex(next,p) & 0x0f);
1078 *q++=(unsigned char) byte;
1082 p+=GetPixelChannels(image);
1085 *q++=(unsigned char) (byte << 4);
1086 if (next->previous == (Image *) NULL)
1088 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1089 if (status == MagickFalse)
1098 Convert PseudoClass packet to ICON pixel.
1100 for (y=0; y < (ssize_t) next->rows; y++)
1102 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
1103 if (p == (const Quantum *) NULL)
1105 q=pixels+(next->rows-y-1)*bytes_per_line;
1106 for (x=0; x < (ssize_t) next->columns; x++)
1108 *q++=(unsigned char) GetPixelIndex(next,p);
1109 p+=GetPixelChannels(image);
1111 if (next->previous == (Image *) NULL)
1113 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1114 if (status == MagickFalse)
1124 Convert DirectClass packet to ICON BGR888 or BGRA8888 pixel.
1126 for (y=0; y < (ssize_t) next->rows; y++)
1128 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
1129 if (p == (const Quantum *) NULL)
1131 q=pixels+(next->rows-y-1)*bytes_per_line;
1132 for (x=0; x < (ssize_t) next->columns; x++)
1134 *q++=ScaleQuantumToChar(GetPixelBlue(next,p));
1135 *q++=ScaleQuantumToChar(GetPixelGreen(next,p));
1136 *q++=ScaleQuantumToChar(GetPixelRed(next,p));
1137 if (next->matte == MagickFalse)
1138 *q++=ScaleQuantumToChar(QuantumRange);
1140 *q++=ScaleQuantumToChar(GetPixelAlpha(next,p));
1141 p+=GetPixelChannels(next);
1143 if (icon_info.bits_per_pixel == 24)
1144 for (x=3L*(ssize_t) next->columns; x < (ssize_t) bytes_per_line; x++)
1146 if (next->previous == (Image *) NULL)
1148 status=SetImageProgress(next,SaveImageTag,y,next->rows);
1149 if (status == MagickFalse)
1157 Write 40-byte version 3+ bitmap header.
1159 icon_file.directory[scene].width=(unsigned char) icon_info.width;
1160 icon_file.directory[scene].height=(unsigned char) icon_info.height;
1161 icon_file.directory[scene].colors=(unsigned char)
1162 icon_info.number_colors;
1163 icon_file.directory[scene].reserved=0;
1164 icon_file.directory[scene].planes=icon_info.planes;
1165 icon_file.directory[scene].bits_per_pixel=icon_info.bits_per_pixel;
1166 icon_file.directory[scene].size=icon_info.size;
1167 icon_file.directory[scene].offset=(size_t) TellBlob(image);
1168 (void) WriteBlobLSBLong(image,(unsigned int) 40);
1169 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.width);
1170 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.height*2);
1171 (void) WriteBlobLSBShort(image,icon_info.planes);
1172 (void) WriteBlobLSBShort(image,icon_info.bits_per_pixel);
1173 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.compression);
1174 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.image_size);
1175 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.x_pixels);
1176 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.y_pixels);
1177 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.number_colors);
1178 (void) WriteBlobLSBLong(image,(unsigned int) icon_info.colors_important);
1179 if (next->storage_class == PseudoClass)
1185 Dump colormap to file.
1187 icon_colormap=(unsigned char *) AcquireQuantumMemory((size_t)
1188 (1UL << icon_info.bits_per_pixel),4UL*sizeof(*icon_colormap));
1189 if (icon_colormap == (unsigned char *) NULL)
1190 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
1192 for (i=0; i < (ssize_t) next->colors; i++)
1194 *q++=ScaleQuantumToChar(next->colormap[i].blue);
1195 *q++=ScaleQuantumToChar(next->colormap[i].green);
1196 *q++=ScaleQuantumToChar(next->colormap[i].red);
1197 *q++=(unsigned char) 0x0;
1199 for ( ; i < (ssize_t) (1UL << icon_info.bits_per_pixel); i++)
1201 *q++=(unsigned char) 0x00;
1202 *q++=(unsigned char) 0x00;
1203 *q++=(unsigned char) 0x00;
1204 *q++=(unsigned char) 0x00;
1206 (void) WriteBlob(image,(size_t) (4UL*(1UL <<
1207 icon_info.bits_per_pixel)),icon_colormap);
1208 icon_colormap=(unsigned char *) RelinquishMagickMemory(
1211 (void) WriteBlob(image,(size_t) icon_info.image_size,pixels);
1212 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
1216 scanline_pad=(((next->columns+31) & ~31)-next->columns) >> 3;
1217 for (y=((ssize_t) next->rows - 1); y >= 0; y--)
1219 p=GetVirtualPixels(next,0,y,next->columns,1,exception);
1220 if (p == (const Quantum *) NULL)
1224 for (x=0; x < (ssize_t) next->columns; x++)
1227 if ((next->matte == MagickTrue) &&
1228 (GetPixelAlpha(next,p) == (Quantum) TransparentAlpha))
1233 (void) WriteBlobByte(image,(unsigned char) byte);
1237 p+=GetPixelChannels(next);
1240 (void) WriteBlobByte(image,(unsigned char) (byte << (8-bit)));
1241 for (i=0; i < (ssize_t) scanline_pad; i++)
1242 (void) WriteBlobByte(image,(unsigned char) 0);
1245 if (GetNextImageInList(next) == (Image *) NULL)
1247 next=SyncNextImageInList(next);
1248 status=SetImageProgress(next,SaveImagesTag,scene++,
1249 GetImageListLength(next));
1250 if (status == MagickFalse)
1252 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
1253 offset=SeekBlob(image,0,SEEK_SET);
1255 (void) WriteBlobLSBShort(image,0);
1256 (void) WriteBlobLSBShort(image,1);
1257 (void) WriteBlobLSBShort(image,(unsigned short) (scene+1));
1262 (void) WriteBlobByte(image,icon_file.directory[scene].width);
1263 (void) WriteBlobByte(image,icon_file.directory[scene].height);
1264 (void) WriteBlobByte(image,icon_file.directory[scene].colors);
1265 (void) WriteBlobByte(image,icon_file.directory[scene].reserved);
1266 (void) WriteBlobLSBShort(image,icon_file.directory[scene].planes);
1267 (void) WriteBlobLSBShort(image,icon_file.directory[scene].bits_per_pixel);
1268 (void) WriteBlobLSBLong(image,(unsigned int)
1269 icon_file.directory[scene].size);
1270 (void) WriteBlobLSBLong(image,(unsigned int)
1271 icon_file.directory[scene].offset);
1273 next=SyncNextImageInList(next);
1274 } while ((next != (Image *) NULL) && (image_info->adjoin != MagickFalse));
1275 (void) CloseBlob(image);