blob: 5c1aca895b0c4316aee8a4a683b0c2a6b77e706c [file] [log] [blame]
cristy3ed852e2009-09-05 21:47:34 +00001/*
2%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3% %
4% %
5% %
6% PPPP SSSSS DDDD %
7% P P SS D D %
8% PPPP SSS D D %
9% P SS D D %
10% P SSSSS DDDD %
11% %
12% %
13% Read/Write Adobe Photoshop Image Format %
14% %
15% Software Design %
16% John Cristy %
17% Leonard Rosenthol %
18% July 1992 %
19% %
20% %
cristy7e41fe82010-12-04 23:12:08 +000021% Copyright 1999-2011 ImageMagick Studio LLC, a non-profit organization %
cristy3ed852e2009-09-05 21:47:34 +000022% dedicated to making software imaging solutions freely available. %
23% %
24% You may not use this file except in compliance with the License. You may %
25% obtain a copy of the License at %
26% %
27% http://www.imagemagick.org/script/license.php %
28% %
29% Unless required by applicable law or agreed to in writing, software %
30% distributed under the License is distributed on an "AS IS" BASIS, %
31% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %
32% See the License for the specific language governing permissions and %
33% limitations under the License. %
34% %
35%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
36%
37%
38*/
39
40/*
41 Include declarations.
42*/
43#include "magick/studio.h"
44#include "magick/artifact.h"
45#include "magick/blob.h"
46#include "magick/blob-private.h"
47#include "magick/cache.h"
cristye7e40552010-04-24 21:34:22 +000048#include "magick/colormap.h"
cristy3ed852e2009-09-05 21:47:34 +000049#include "magick/colorspace.h"
50#include "magick/constitute.h"
51#include "magick/enhance.h"
52#include "magick/exception.h"
53#include "magick/exception-private.h"
54#include "magick/image.h"
55#include "magick/image-private.h"
56#include "magick/list.h"
57#include "magick/log.h"
58#include "magick/magick.h"
59#include "magick/memory_.h"
60#include "magick/module.h"
cristy4689cf02010-02-17 21:15:45 +000061#include "magick/monitor-private.h"
cristy3ed852e2009-09-05 21:47:34 +000062#include "magick/profile.h"
63#include "magick/property.h"
64#include "magick/quantum-private.h"
65#include "magick/static.h"
66#include "magick/string_.h"
67
68/*
cristy2d3d87f2010-03-01 00:23:08 +000069 Define declaractions.
70*/
cristyaca3ec52010-03-02 14:55:01 +000071#define MaxPSDChannels 56
cristybb503372010-05-27 20:51:26 +000072#define PSDQuantum(x) (((ssize_t) (x)+1) & -2)
cristy2d3d87f2010-03-01 00:23:08 +000073
74/*
75 Enumerated declaractions.
76*/
77typedef enum
78{
79 BitmapMode = 0,
80 GrayscaleMode = 1,
81 IndexedMode = 2,
82 RGBMode = 3,
83 CMYKMode = 4,
84 MultichannelMode = 7,
85 DuotoneMode = 8,
86 LabMode = 9
87} PSDImageType;
88
89/*
90 Typedef declaractions.
91*/
92typedef struct _ChannelInfo
93{
94 short int
95 type;
96
cristybb503372010-05-27 20:51:26 +000097 size_t
cristy2d3d87f2010-03-01 00:23:08 +000098 size;
99} ChannelInfo;
100
101typedef struct _LayerInfo
102{
103 RectangleInfo
104 page,
105 mask;
106
107 unsigned short
108 channels;
109
110 ChannelInfo
111 channel_info[MaxPSDChannels];
112
113 char
114 blendkey[4];
115
116 Quantum
117 opacity;
118
119 unsigned char
120 clipping,
121 visible,
122 flags;
123
cristybb503372010-05-27 20:51:26 +0000124 size_t
cristy2d3d87f2010-03-01 00:23:08 +0000125 offset_x,
126 offset_y;
127
128 unsigned char
129 name[256];
130
131 Image
132 *image;
133} LayerInfo;
134
135typedef struct _PSDInfo
136{
137 char
138 signature[4];
139
140 unsigned short
141 channels,
142 version;
143
144 unsigned char
145 reserved[6];
146
cristybb503372010-05-27 20:51:26 +0000147 size_t
cristy2d3d87f2010-03-01 00:23:08 +0000148 rows,
149 columns;
150
151 unsigned short
152 depth,
153 mode;
154} PSDInfo;
155
156/*
cristy3ed852e2009-09-05 21:47:34 +0000157 Forward declarations.
158*/
159static MagickBooleanType
160 WritePSDImage(const ImageInfo *,Image *);
161
162/*
163%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
164% %
165% %
166% %
cristy3ed852e2009-09-05 21:47:34 +0000167% I s P S D %
168% %
169% %
170% %
171%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
172%
173% IsPSD()() returns MagickTrue if the image format type, identified by the
174% magick string, is PSD.
175%
176% The format of the IsPSD method is:
177%
178% MagickBooleanType IsPSD(const unsigned char *magick,const size_t length)
179%
180% A description of each parameter follows:
181%
182% o magick: compare image format pattern against these bytes.
183%
184% o length: Specifies the length of the magick string.
185%
186*/
187static MagickBooleanType IsPSD(const unsigned char *magick,const size_t length)
188{
189 if (length < 4)
190 return(MagickFalse);
191 if (LocaleNCompare((const char *) magick,"8BPS",4) == 0)
192 return(MagickTrue);
193 return(MagickFalse);
194}
195
196/*
197%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
198% %
199% %
200% %
201% R e a d P S D I m a g e %
202% %
203% %
204% %
205%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
206%
207% ReadPSDImage() reads an Adobe Photoshop image file and returns it. It
208% allocates the memory necessary for the new Image structure and returns a
209% pointer to the new image.
210%
211% The format of the ReadPSDImage method is:
212%
cristycd081772010-03-22 17:19:12 +0000213% Image *ReadPSDImage(image_info)
cristy3ed852e2009-09-05 21:47:34 +0000214%
215% A description of each parameter follows:
216%
217% o image_info: the image info.
218%
219% o exception: return any errors or warnings in this structure.
220%
221*/
222
cristy19eb6412010-04-23 14:42:29 +0000223static const char *CompositeOperatorToPSDBlendMode(CompositeOperator op)
cristycd081772010-03-22 17:19:12 +0000224{
225 const char
226 *blend_mode;
227
cristy19eb6412010-04-23 14:42:29 +0000228 switch (op)
cristycd081772010-03-22 17:19:12 +0000229 {
230 case OverCompositeOp: blend_mode = "norm"; break;
231 case MultiplyCompositeOp: blend_mode = "mul "; break;
232 case DissolveCompositeOp: blend_mode = "diss"; break;
233 case DifferenceCompositeOp: blend_mode = "diff"; break;
234 case DarkenCompositeOp: blend_mode = "dark"; break;
235 case LightenCompositeOp: blend_mode = "lite"; break;
236 case HueCompositeOp: blend_mode = "hue "; break;
237 case SaturateCompositeOp: blend_mode = "sat "; break;
238 case ColorizeCompositeOp: blend_mode = "colr"; break;
239 case LuminizeCompositeOp: blend_mode = "lum "; break;
240 case ScreenCompositeOp: blend_mode = "scrn"; break;
241 case OverlayCompositeOp: blend_mode = "over"; break;
242 default:
243 blend_mode = "norm";
244 }
245 return(blend_mode);
246}
247
248static ssize_t DecodePSDPixels(const size_t number_compact_pixels,
cristybb503372010-05-27 20:51:26 +0000249 const unsigned char *compact_pixels,const ssize_t depth,
cristycd081772010-03-22 17:19:12 +0000250 const size_t number_pixels,unsigned char *pixels)
251{
252 int
253 pixel;
254
255 register ssize_t
256 i,
257 j;
258
259 ssize_t
260 packets;
261
262 size_t
263 length;
264
265 packets=(ssize_t) number_compact_pixels;
266 for (i=0; (packets > 1) && (i < (ssize_t) number_pixels); )
267 {
268 length=(*compact_pixels++);
269 packets--;
270 if (length == 128)
271 continue;
272 if (length > 128)
273 {
274 length=256-length+1;
275 pixel=(*compact_pixels++);
276 packets--;
cristy284c7d82010-04-24 00:19:14 +0000277 for (j=0; j < (ssize_t) length; j++)
cristycd081772010-03-22 17:19:12 +0000278 {
279 switch (depth)
280 {
281 case 1:
282 {
cristy284c7d82010-04-24 00:19:14 +0000283 *pixels++=(pixel >> 7) & 0x01 ? 0U : 255U;
284 *pixels++=(pixel >> 6) & 0x01 ? 0U : 255U;
285 *pixels++=(pixel >> 5) & 0x01 ? 0U : 255U;
286 *pixels++=(pixel >> 4) & 0x01 ? 0U : 255U;
287 *pixels++=(pixel >> 3) & 0x01 ? 0U : 255U;
288 *pixels++=(pixel >> 2) & 0x01 ? 0U : 255U;
289 *pixels++=(pixel >> 1) & 0x01 ? 0U : 255U;
290 *pixels++=(pixel >> 0) & 0x01 ? 0U : 255U;
cristycd081772010-03-22 17:19:12 +0000291 i+=8;
292 break;
293 }
294 case 4:
295 {
cristy284c7d82010-04-24 00:19:14 +0000296 *pixels++=(unsigned char) ((pixel >> 4) & 0xff);
297 *pixels++=(unsigned char) ((pixel & 0x0f) & 0xff);
cristycd081772010-03-22 17:19:12 +0000298 i+=2;
299 break;
300 }
301 case 2:
302 {
cristy284c7d82010-04-24 00:19:14 +0000303 *pixels++=(unsigned char) ((pixel >> 6) & 0x03);
304 *pixels++=(unsigned char) ((pixel >> 4) & 0x03);
305 *pixels++=(unsigned char) ((pixel >> 2) & 0x03);
306 *pixels++=(unsigned char) ((pixel & 0x03) & 0x03);
cristycd081772010-03-22 17:19:12 +0000307 i+=4;
308 break;
309 }
310 default:
311 {
cristy284c7d82010-04-24 00:19:14 +0000312 *pixels++=(unsigned char) pixel;
cristycd081772010-03-22 17:19:12 +0000313 i++;
314 break;
315 }
316 }
317 }
318 continue;
319 }
320 length++;
cristy284c7d82010-04-24 00:19:14 +0000321 for (j=0; j < (ssize_t) length; j++)
cristycd081772010-03-22 17:19:12 +0000322 {
323 switch (depth)
324 {
325 case 1:
326 {
cristy284c7d82010-04-24 00:19:14 +0000327 *pixels++=(*compact_pixels >> 7) & 0x01 ? 0U : 255U;
328 *pixels++=(*compact_pixels >> 6) & 0x01 ? 0U : 255U;
329 *pixels++=(*compact_pixels >> 5) & 0x01 ? 0U : 255U;
330 *pixels++=(*compact_pixels >> 4) & 0x01 ? 0U : 255U;
331 *pixels++=(*compact_pixels >> 3) & 0x01 ? 0U : 255U;
332 *pixels++=(*compact_pixels >> 2) & 0x01 ? 0U : 255U;
333 *pixels++=(*compact_pixels >> 1) & 0x01 ? 0U : 255U;
334 *pixels++=(*compact_pixels >> 0) & 0x01 ? 0U : 255U;
cristycd081772010-03-22 17:19:12 +0000335 i+=8;
336 break;
337 }
338 case 4:
339 {
cristy618a9662010-03-23 13:59:54 +0000340 *pixels++=(*compact_pixels >> 4) & 0xff;
341 *pixels++=(*compact_pixels & 0x0f) & 0xff;
cristycd081772010-03-22 17:19:12 +0000342 i+=2;
343 break;
344 }
345 case 2:
346 {
cristy618a9662010-03-23 13:59:54 +0000347 *pixels++=(*compact_pixels >> 6) & 0x03;
348 *pixels++=(*compact_pixels >> 4) & 0x03;
349 *pixels++=(*compact_pixels >> 2) & 0x03;
350 *pixels++=(*compact_pixels & 0x03) & 0x03;
cristycd081772010-03-22 17:19:12 +0000351 i+=4;
352 break;
353 }
354 default:
355 {
356 *pixels++=(*compact_pixels);
357 i++;
358 break;
359 }
360 }
361 compact_pixels++;
362 }
363 }
364 return(i);
365}
366
cristy2d3d87f2010-03-01 00:23:08 +0000367static inline MagickOffsetType GetPSDOffset(PSDInfo *psd_info,Image *image)
368{
369 if (psd_info->version == 1)
370 return((MagickOffsetType) ReadBlobMSBShort(image));
371 return((MagickOffsetType) ReadBlobMSBLong(image));
372}
373
374static inline MagickSizeType GetPSDSize(PSDInfo *psd_info,Image *image)
375{
376 if (psd_info->version == 1)
377 return((MagickSizeType) ReadBlobMSBLong(image));
378 return((MagickSizeType) ReadBlobMSBLongLong(image));
379}
380
cristybb503372010-05-27 20:51:26 +0000381static inline ssize_t MagickAbsoluteValue(const ssize_t x)
cristy3ed852e2009-09-05 21:47:34 +0000382{
383 if (x < 0)
384 return(-x);
385 return(x);
386}
387
cristycd081772010-03-22 17:19:12 +0000388static const char *ModeToString(PSDImageType type)
cristy3ed852e2009-09-05 21:47:34 +0000389{
cristycd081772010-03-22 17:19:12 +0000390 switch (type)
cristy3ed852e2009-09-05 21:47:34 +0000391 {
392 case BitmapMode: return "Bitmap";
393 case GrayscaleMode: return "Grayscale";
394 case IndexedMode: return "Indexed";
395 case RGBMode: return "RGB";
396 case CMYKMode: return "CMYK";
397 case MultichannelMode: return "Multichannel";
398 case DuotoneMode: return "Duotone";
399 case LabMode: return "L*A*B";
400 default: return "unknown";
401 }
402}
403
404static MagickBooleanType ParseImageResourceBlocks(Image *image,
405 const unsigned char *blocks,size_t length)
406{
407 const unsigned char
408 *p;
409
410 StringInfo
411 *profile;
412
cristy6befb0f2010-05-31 14:33:15 +0000413 unsigned int
cristy3ed852e2009-09-05 21:47:34 +0000414 count,
cristy6befb0f2010-05-31 14:33:15 +0000415 long_sans;
cristy3ed852e2009-09-05 21:47:34 +0000416
417 unsigned short
418 id,
419 short_sans;
420
421 if (length < 16)
422 return(MagickFalse);
423 profile=AcquireStringInfo(length);
424 SetStringInfoDatum(profile,blocks);
425 (void) SetImageProfile(image,"8bim",profile);
426 profile=DestroyStringInfo(profile);
427 for (p=blocks; (p >= blocks) && (p < (blocks+length-16)); )
428 {
429 if (LocaleNCompare((const char *) p,"8BIM",4) != 0)
430 break;
cristy6befb0f2010-05-31 14:33:15 +0000431 p=PushLongPixel(MSBEndian,p,&long_sans);
cristyf11065e2010-05-14 13:26:59 +0000432 p=PushShortPixel(MSBEndian,p,&id);
433 p=PushShortPixel(MSBEndian,p,&short_sans);
434 p=PushLongPixel(MSBEndian,p,&count);
cristy3ed852e2009-09-05 21:47:34 +0000435 switch (id)
436 {
437 case 0x03ed:
438 {
cristy1e4a80b2010-05-14 16:18:26 +0000439 char
440 value[MaxTextExtent];
441
cristy3ed852e2009-09-05 21:47:34 +0000442 unsigned short
443 resolution;
444
445 /*
446 Resolution info.
447 */
cristyf11065e2010-05-14 13:26:59 +0000448 p=PushShortPixel(MSBEndian,p,&resolution);
cristy3ed852e2009-09-05 21:47:34 +0000449 image->x_resolution=(double) resolution;
cristy1e4a80b2010-05-14 16:18:26 +0000450 (void) FormatMagickString(value,MaxTextExtent,"%g",image->x_resolution);
451 (void) SetImageProperty(image,"tiff:XResolution",value);
cristyf11065e2010-05-14 13:26:59 +0000452 p=PushShortPixel(MSBEndian,p,&short_sans);
453 p=PushShortPixel(MSBEndian,p,&short_sans);
454 p=PushShortPixel(MSBEndian,p,&short_sans);
455 p=PushShortPixel(MSBEndian,p,&resolution);
cristy3ed852e2009-09-05 21:47:34 +0000456 image->y_resolution=(double) resolution;
cristy1e4a80b2010-05-14 16:18:26 +0000457 (void) FormatMagickString(value,MaxTextExtent,"%g",image->y_resolution);
458 (void) SetImageProperty(image,"tiff:YResolution",value);
cristyf11065e2010-05-14 13:26:59 +0000459 p=PushShortPixel(MSBEndian,p,&short_sans);
460 p=PushShortPixel(MSBEndian,p,&short_sans);
461 p=PushShortPixel(MSBEndian,p,&short_sans);
cristy3ed852e2009-09-05 21:47:34 +0000462 break;
463 }
464 default:
465 {
466 p+=count;
467 break;
468 }
469 }
470 if ((count & 0x01) != 0)
471 p++;
472 }
473 return(MagickTrue);
474}
475
cristycd081772010-03-22 17:19:12 +0000476static CompositeOperator PSDBlendModeToCompositeOperator(const char *mode)
cristy56ed31c2010-03-22 00:46:21 +0000477{
cristycd081772010-03-22 17:19:12 +0000478 if (mode == (const char *) NULL)
479 return(OverCompositeOp);
480 if (LocaleNCompare(mode,"norm",4) == 0)
481 return(OverCompositeOp);
482 if (LocaleNCompare(mode,"mul ",4) == 0)
483 return(MultiplyCompositeOp);
484 if (LocaleNCompare(mode,"diss",4) == 0)
485 return(DissolveCompositeOp);
486 if (LocaleNCompare(mode,"diff",4) == 0)
487 return(DifferenceCompositeOp);
488 if (LocaleNCompare(mode,"dark",4) == 0)
489 return(DarkenCompositeOp);
490 if (LocaleNCompare(mode,"lite",4) == 0)
491 return(LightenCompositeOp);
492 if (LocaleNCompare(mode,"hue ",4) == 0)
493 return(HueCompositeOp);
494 if (LocaleNCompare(mode,"sat ",4) == 0)
495 return(SaturateCompositeOp);
496 if (LocaleNCompare(mode,"colr",4) == 0)
497 return(ColorizeCompositeOp);
498 if (LocaleNCompare(mode,"lum ",4) == 0)
499 return(LuminizeCompositeOp);
500 if (LocaleNCompare(mode,"scrn",4) == 0)
501 return(ScreenCompositeOp);
502 if (LocaleNCompare(mode,"over",4) == 0)
503 return(OverlayCompositeOp);
504 if (LocaleNCompare(mode,"hLit",4) == 0)
505 return(OverCompositeOp);
506 if (LocaleNCompare(mode,"sLit",4) == 0)
507 return(OverCompositeOp);
508 if (LocaleNCompare(mode,"smud",4) == 0)
509 return(OverCompositeOp);
510 if (LocaleNCompare(mode,"div ",4) == 0)
511 return(OverCompositeOp);
512 if (LocaleNCompare(mode,"idiv",4) == 0)
513 return(OverCompositeOp);
514 return(OverCompositeOp);
cristy56ed31c2010-03-22 00:46:21 +0000515}
516
cristybb503372010-05-27 20:51:26 +0000517static MagickBooleanType ReadPSDLayer(Image *image,const size_t channels,
518 const ssize_t type,const MagickOffsetType *offsets,ExceptionInfo *exception)
cristy56ed31c2010-03-22 00:46:21 +0000519{
cristyd05dca12010-07-25 02:32:32 +0000520 ColorspaceType
521 colorspace;
cristy56ed31c2010-03-22 00:46:21 +0000522
523 Quantum
524 pixel;
525
cristy7753b2a2011-02-19 18:36:52 +0000526 register const unsigned char
527 *p;
528
cristy56ed31c2010-03-22 00:46:21 +0000529 register IndexPacket
530 *indexes;
531
cristy56ed31c2010-03-22 00:46:21 +0000532 register PixelPacket
533 *q;
534
cristy7753b2a2011-02-19 18:36:52 +0000535 register ssize_t
536 x;
cristy56ed31c2010-03-22 00:46:21 +0000537
538 size_t
539 packet_size;
540
541 ssize_t
cristyd05dca12010-07-25 02:32:32 +0000542 count,
543 y;
cristy56ed31c2010-03-22 00:46:21 +0000544
545 unsigned char
546 *compact_pixels,
547 *pixels;
548
549 unsigned short
550 nibble;
551
552 packet_size=1;
553 if (image->storage_class == PseudoClass)
554 {
555 if (image->colors > 256)
556 packet_size++;
557 else
558 if (image->depth > 8)
559 packet_size++;
560 }
561 else
562 if (image->depth > 8)
563 packet_size++;
cristycd081772010-03-22 17:19:12 +0000564 pixels=(unsigned char *) AcquireQuantumMemory(image->columns+256,packet_size*
cristy56ed31c2010-03-22 00:46:21 +0000565 sizeof(*pixels));
566 if (pixels == (unsigned char *) NULL)
567 ThrowBinaryException(ResourceLimitError,"MemoryAllocationFailed",
568 image->filename);
cristycd081772010-03-22 17:19:12 +0000569 (void) ResetMagickMemory(pixels,0,image->columns*packet_size*sizeof(*pixels));
570 compact_pixels=(unsigned char *) NULL;
cristy56ed31c2010-03-22 00:46:21 +0000571 if (image->compression == RLECompression)
572 {
cristye195f262010-04-16 18:12:35 +0000573 size_t
574 length;
575
576 length=0;
cristybb503372010-05-27 20:51:26 +0000577 for (y=0; y < (ssize_t) image->rows; y++)
cristye195f262010-04-16 18:12:35 +0000578 if ((MagickOffsetType) length < offsets[y])
cristy284c7d82010-04-24 00:19:14 +0000579 length=(size_t) offsets[y];
cristye195f262010-04-16 18:12:35 +0000580 compact_pixels=(unsigned char *) AcquireQuantumMemory(length,
581 sizeof(*pixels));
cristy56ed31c2010-03-22 00:46:21 +0000582 if (compact_pixels == (unsigned char *) NULL)
583 ThrowBinaryException(ResourceLimitError,"MemoryAllocationFailed",
584 image->filename);
cristy0132ba82010-04-16 23:34:02 +0000585 (void) ResetMagickMemory(compact_pixels,0,length*sizeof(*compact_pixels));
cristy56ed31c2010-03-22 00:46:21 +0000586 }
cristyd05dca12010-07-25 02:32:32 +0000587 colorspace=image->colorspace;
cristybb503372010-05-27 20:51:26 +0000588 for (y=0; y < (ssize_t) image->rows; y++)
cristy56ed31c2010-03-22 00:46:21 +0000589 {
cristy3b004082010-08-12 19:56:38 +0000590 if (image->depth == 1)
591 {
cristyb6cabe82011-03-18 13:25:59 +0000592 if (image->compression != RLECompression)
593 count=ReadBlob(image,(image->columns+7)/8,pixels);
594 else
595 {
596 count=ReadBlob(image,(size_t) offsets[y],compact_pixels);
597 if (count != (ssize_t) offsets[y])
598 break;
599 count=DecodePSDPixels((size_t) offsets[y],compact_pixels,
600 (ssize_t) 123456,(size_t) ((image->columns+7)/8),pixels);
601 }
cristy3b004082010-08-12 19:56:38 +0000602 if (count < (ssize_t) ((image->columns+7)/8))
603 break;
604 }
cristy56ed31c2010-03-22 00:46:21 +0000605 else
606 {
cristy3b004082010-08-12 19:56:38 +0000607 if (image->compression != RLECompression)
608 count=ReadBlob(image,packet_size*image->columns,pixels);
609 else
610 {
611 count=ReadBlob(image,(size_t) offsets[y],compact_pixels);
612 if (count != (ssize_t) offsets[y])
613 break;
614 count=DecodePSDPixels((size_t) offsets[y],compact_pixels,
615 (ssize_t) image->depth,packet_size*image->columns,pixels);
616 }
617 if (count < (ssize_t) (packet_size*image->columns))
cristy56ed31c2010-03-22 00:46:21 +0000618 break;
cristy56ed31c2010-03-22 00:46:21 +0000619 }
cristy56ed31c2010-03-22 00:46:21 +0000620 q=GetAuthenticPixels(image,0,y,image->columns,1,exception);
621 if (q == (PixelPacket *) NULL)
622 break;
623 indexes=GetAuthenticIndexQueue(image);
624 p=pixels;
cristybb503372010-05-27 20:51:26 +0000625 for (x=0; x < (ssize_t) image->columns; x++)
cristy56ed31c2010-03-22 00:46:21 +0000626 {
627 if (packet_size == 1)
628 pixel=ScaleCharToQuantum(*p++);
629 else
630 {
631 p=PushShortPixel(MSBEndian,p,&nibble);
632 pixel=ScaleShortToQuantum(nibble);
633 }
634 switch (type)
635 {
636 case -1:
637 {
638 q->opacity=(Quantum) (QuantumRange-pixel);
639 break;
640 }
641 case 0:
642 {
643 q->red=pixel;
644 if (channels == 1)
645 {
cristy5fde9aa2011-04-23 01:31:53 +0000646 SetGreenPixelComponent(q,GetRedPixelComponent(q));
647 SetBluePixelComponent(q,GetRedPixelComponent(q));
cristy56ed31c2010-03-22 00:46:21 +0000648 }
649 if (image->storage_class == PseudoClass)
650 {
651 if (packet_size == 1)
652 indexes[x]=(IndexPacket) ScaleQuantumToChar(pixel);
653 else
654 indexes[x]=(IndexPacket) ScaleQuantumToShort(pixel);
cristybb503372010-05-27 20:51:26 +0000655 q->red=image->colormap[(ssize_t) indexes[x]].red;
656 q->green=image->colormap[(ssize_t) indexes[x]].green;
657 q->blue=image->colormap[(ssize_t) indexes[x]].blue;
cristy3b004082010-08-12 19:56:38 +0000658 if (image->depth == 1)
659 {
660 ssize_t
cristyb6cabe82011-03-18 13:25:59 +0000661 bit,
662 number_bits;
cristy3b004082010-08-12 19:56:38 +0000663
cristyb6cabe82011-03-18 13:25:59 +0000664 number_bits=image->columns-x;
665 if (number_bits > 8)
666 number_bits=8;
667 for (bit=0; bit < number_bits; bit++)
cristy3b004082010-08-12 19:56:38 +0000668 {
cristy85421392010-08-14 02:41:27 +0000669 indexes[x]=((((unsigned char) pixel) & (0x01 << (7-bit)))
670 != 0 ? 0 : 255);
cristy3b004082010-08-12 19:56:38 +0000671 q->red=image->colormap[(ssize_t) indexes[x]].red;
672 q->green=image->colormap[(ssize_t) indexes[x]].green;
673 q->blue=image->colormap[(ssize_t) indexes[x]].blue;
674 q++;
675 x++;
676 }
677 q--;
678 x--;
679 }
cristy56ed31c2010-03-22 00:46:21 +0000680 }
681 break;
682 }
683 case 1:
684 {
685 if (image->storage_class == PseudoClass)
686 q->opacity=(Quantum) (QuantumRange-pixel);
687 else
688 q->green=pixel;
689 break;
690 }
691 case 2:
692 {
cristydede49f2010-08-20 20:26:26 +0000693 if (image->storage_class == PseudoClass)
694 q->opacity=(Quantum) (QuantumRange-pixel);
695 else
696 q->blue=pixel;
cristy56ed31c2010-03-22 00:46:21 +0000697 break;
698 }
699 case 3:
700 {
701 if (image->colorspace == CMYKColorspace)
702 indexes[x]=(IndexPacket) pixel;
703 else
704 q->opacity=(Quantum) (QuantumRange-pixel);
705 break;
706 }
707 case 4:
708 {
cristya60c0202010-07-31 21:36:45 +0000709 if ((image->colorspace == RGBColorspace) && (channels > 3))
710 break;
cristy56ed31c2010-03-22 00:46:21 +0000711 q->opacity=(Quantum) (QuantumRange-pixel);
712 break;
713 }
714 default:
715 break;
716 }
717 q++;
718 }
719 if (SyncAuthenticPixels(image,exception) == MagickFalse)
720 break;
721 }
cristyd05dca12010-07-25 02:32:32 +0000722 image->colorspace=colorspace;
cristy56ed31c2010-03-22 00:46:21 +0000723 if (image->compression == RLECompression)
724 compact_pixels=(unsigned char *) RelinquishMagickMemory(compact_pixels);
725 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
726 return(MagickTrue);
727}
728
cristy3ed852e2009-09-05 21:47:34 +0000729static Image *ReadPSDImage(const ImageInfo *image_info,ExceptionInfo *exception)
730{
cristy3ed852e2009-09-05 21:47:34 +0000731 char
cristy56ed31c2010-03-22 00:46:21 +0000732 message[MaxTextExtent],
cristy3ed852e2009-09-05 21:47:34 +0000733 type[4];
734
735 Image
736 *image;
737
cristy3ed852e2009-09-05 21:47:34 +0000738 LayerInfo
739 *layer_info;
740
cristy56ed31c2010-03-22 00:46:21 +0000741 MagickBooleanType
742 status;
743
744 MagickOffsetType
745 offset,
746 *offsets;
747
748 MagickSizeType
749 combinedlength,
750 length,
751 size;
752
cristy3ed852e2009-09-05 21:47:34 +0000753 PSDInfo
754 psd_info;
755
cristydede49f2010-08-20 20:26:26 +0000756 register PixelPacket
757 *q;
758
cristybb503372010-05-27 20:51:26 +0000759 register ssize_t
cristy56ed31c2010-03-22 00:46:21 +0000760 i,
cristy3ed852e2009-09-05 21:47:34 +0000761 x;
762
cristydc05fc22011-01-22 19:43:15 +0000763 size_t
764 mask_size,
765 skip_first_alpha = 0;
766
cristy3ed852e2009-09-05 21:47:34 +0000767 ssize_t
cristydc05fc22011-01-22 19:43:15 +0000768 count,
769 j,
770 number_layers,
771 y;
cristy3ed852e2009-09-05 21:47:34 +0000772
cristy3ed852e2009-09-05 21:47:34 +0000773 unsigned char
774 *data;
775
776 unsigned short
777 compression;
778
cristy3ed852e2009-09-05 21:47:34 +0000779 /*
780 Open image file.
781 */
782 assert(image_info != (const ImageInfo *) NULL);
783 assert(image_info->signature == MagickSignature);
784 if (image_info->debug != MagickFalse)
785 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
786 image_info->filename);
787 assert(exception != (ExceptionInfo *) NULL);
788 assert(exception->signature == MagickSignature);
789 image=AcquireImage(image_info);
790 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
791 if (status == MagickFalse)
792 {
793 image=DestroyImageList(image);
794 return((Image *) NULL);
795 }
796 /*
797 Read image header.
798 */
799 count=ReadBlob(image,4,(unsigned char *) psd_info.signature);
800 psd_info.version=ReadBlobMSBShort(image);
cristy50aea4a2010-03-09 17:37:44 +0000801 if ((count == 0) || (LocaleNCompare(psd_info.signature,"8BPS",4) != 0) ||
cristy2d3d87f2010-03-01 00:23:08 +0000802 ((psd_info.version != 1) && (psd_info.version != 2)))
cristy3ed852e2009-09-05 21:47:34 +0000803 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
804 count=ReadBlob(image,6,psd_info.reserved);
805 psd_info.channels=ReadBlobMSBShort(image);
806 if (psd_info.channels > MaxPSDChannels)
807 ThrowReaderException(CorruptImageError,"MaximumChannelsExceeded");
808 psd_info.rows=ReadBlobMSBLong(image);
809 psd_info.columns=ReadBlobMSBLong(image);
cristy2d3d87f2010-03-01 00:23:08 +0000810 if ((psd_info.version == 1) && ((psd_info.rows > 30000) ||
811 (psd_info.columns > 30000)))
812 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
cristy3ed852e2009-09-05 21:47:34 +0000813 psd_info.depth=ReadBlobMSBShort(image);
cristy2d3d87f2010-03-01 00:23:08 +0000814 if ((psd_info.depth != 1) && (psd_info.depth != 8) && (psd_info.depth != 16))
815 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
cristy3ed852e2009-09-05 21:47:34 +0000816 psd_info.mode=ReadBlobMSBShort(image);
817 if (image->debug != MagickFalse)
818 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +0000819 " Image is %.20g x %.20g with channels=%.20g, depth=%.20g, mode=%s",
820 (double) psd_info.columns,(double) psd_info.rows,(double)
821 psd_info.channels,(double) psd_info.depth,ModeToString((PSDImageType)
822 psd_info.mode));
cristy3ed852e2009-09-05 21:47:34 +0000823 /*
824 Initialize image.
825 */
826 image->depth=psd_info.depth;
827 image->columns=psd_info.columns;
828 image->rows=psd_info.rows;
cristyf6febd72010-11-06 15:54:54 +0000829 if (SetImageBackgroundColor(image) == MagickFalse)
cristy95524f92010-02-16 18:44:34 +0000830 {
831 InheritException(exception,&image->exception);
832 image=DestroyImageList(image);
833 return((Image *) NULL);
834 }
cristy3ed852e2009-09-05 21:47:34 +0000835 image->matte=psd_info.channels >= 4 ? MagickTrue : MagickFalse;
836 if (psd_info.mode == LabMode)
837 image->colorspace=LabColorspace;
838 if (psd_info.mode == CMYKMode)
839 {
840 image->colorspace=CMYKColorspace;
841 image->matte=psd_info.channels >= 5 ? MagickTrue : MagickFalse;
842 }
843 if ((psd_info.mode == BitmapMode) || (psd_info.mode == GrayscaleMode) ||
844 (psd_info.mode == DuotoneMode))
845 {
cristy52cf7f12010-02-07 18:07:56 +0000846 if (AcquireImageColormap(image,256) == MagickFalse)
847 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristy3ed852e2009-09-05 21:47:34 +0000848 image->matte=psd_info.channels >= 2 ? MagickTrue : MagickFalse;
849 if (image->debug != MagickFalse)
850 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristydede49f2010-08-20 20:26:26 +0000851 " Image colormap allocated");
cristy0132ba82010-04-16 23:34:02 +0000852 image->colorspace=GRAYColorspace;
cristy3ed852e2009-09-05 21:47:34 +0000853 }
854 if (image->debug != MagickFalse)
855 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
856 image->matte ? " image has matte" : " image has no matte");
857 /*
858 Read PSD raster colormap only present for indexed and duotone images.
859 */
860 length=ReadBlobMSBLong(image);
861 if (length != 0)
862 {
863 if (image->debug != MagickFalse)
864 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
865 " reading colormap");
866 if (psd_info.mode == DuotoneMode)
867 {
868 /*
869 Duotone image data; the format of this data is undocumented.
870 */
cristy56ed31c2010-03-22 00:46:21 +0000871 data=(unsigned char *) AcquireQuantumMemory((size_t) length,
872 sizeof(*data));
cristy3ed852e2009-09-05 21:47:34 +0000873 if (data == (unsigned char *) NULL)
874 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristy56ed31c2010-03-22 00:46:21 +0000875 count=ReadBlob(image,(size_t) length,data);
cristy3ed852e2009-09-05 21:47:34 +0000876 data=(unsigned char *) RelinquishMagickMemory(data);
877 }
878 else
879 {
880 /*
881 Read PSD raster colormap.
882 */
cristybb503372010-05-27 20:51:26 +0000883 if (AcquireImageColormap(image,(size_t) (length/3)) == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +0000884 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristybb503372010-05-27 20:51:26 +0000885 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +0000886 image->colormap[i].red=ScaleCharToQuantum((unsigned char)
887 ReadBlobByte(image));
cristybb503372010-05-27 20:51:26 +0000888 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +0000889 image->colormap[i].green=ScaleCharToQuantum((unsigned char)
890 ReadBlobByte(image));
cristybb503372010-05-27 20:51:26 +0000891 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +0000892 image->colormap[i].blue=ScaleCharToQuantum((unsigned char)
893 ReadBlobByte(image));
894 image->matte=psd_info.channels >= 2 ? MagickTrue : MagickFalse;
895 }
896 }
897 length=ReadBlobMSBLong(image);
898 if (length != 0)
899 {
900 unsigned char
901 *blocks;
902
903 /*
904 Image resources block.
905 */
906 if (image->debug != MagickFalse)
907 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +0000908 " reading image resource blocks - %.20g bytes",(double) length);
cristy56ed31c2010-03-22 00:46:21 +0000909 blocks=(unsigned char *) AcquireQuantumMemory((size_t) length,
910 sizeof(*blocks));
cristy3ed852e2009-09-05 21:47:34 +0000911 if (blocks == (unsigned char *) NULL)
912 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristy56ed31c2010-03-22 00:46:21 +0000913 count=ReadBlob(image,(size_t) length,blocks);
914 if ((count != (ssize_t) length) ||
cristy3ed852e2009-09-05 21:47:34 +0000915 (LocaleNCompare((char *) blocks,"8BIM",4) != 0))
916 {
917 blocks=(unsigned char *) RelinquishMagickMemory(blocks);
918 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
919 }
cristy56ed31c2010-03-22 00:46:21 +0000920 (void) ParseImageResourceBlocks(image,blocks,(size_t) length);
cristy3ed852e2009-09-05 21:47:34 +0000921 blocks=(unsigned char *) RelinquishMagickMemory(blocks);
922 }
923 /*
cristy3ed852e2009-09-05 21:47:34 +0000924 Layer and mask block.
925 */
926 layer_info=(LayerInfo *) NULL;
cristy4689cf02010-02-17 21:15:45 +0000927 number_layers=1;
cristy2d3d87f2010-03-01 00:23:08 +0000928 length=GetPSDSize(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +0000929 if (length == 8)
930 {
931 length=ReadBlobMSBLong(image);
932 length=ReadBlobMSBLong(image);
933 }
934 if ((image_info->number_scenes == 1) && (image_info->scene == 0))
cristyd4297022010-09-16 22:59:09 +0000935 {
936 if (DiscardBlobBytes(image,length) == MagickFalse)
937 ThrowFileException(exception,CorruptImageError,"UnexpectedEndOfFile",
938 image->filename);
939 length=0;
940 }
cristy3ed852e2009-09-05 21:47:34 +0000941 if (length == 0)
942 {
943 if (image->debug != MagickFalse)
944 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
945 " image has no layers");
946 }
947 else
948 {
949 offset=TellBlob(image);
cristy2d3d87f2010-03-01 00:23:08 +0000950 size=GetPSDSize(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +0000951 if (size == 0)
952 {
cristybb503372010-05-27 20:51:26 +0000953 size_t
cristy2d3d87f2010-03-01 00:23:08 +0000954 quantum;
955
cristy9fb55e82011-02-27 23:12:47 +0000956 unsigned long
957 tag;
958
cristy3ed852e2009-09-05 21:47:34 +0000959 /*
960 Skip layers & masks.
961 */
cristy56ed31c2010-03-22 00:46:21 +0000962 quantum=psd_info.version == 1 ? 4UL : 8UL;
cristy9fb55e82011-02-27 23:12:47 +0000963 tag=ReadBlobMSBLong(image);
cristy6c70aec2011-03-04 20:48:09 +0000964 (void) tag;
965 count=ReadBlob(image,4,(unsigned char *) type);
966 if ((count == 0) || (LocaleNCompare(type,"8BIM",4) != 0))
cristy9fb55e82011-02-27 23:12:47 +0000967 {
968 if (DiscardBlobBytes(image,length-quantum-8) == MagickFalse)
969 ThrowFileException(exception,CorruptImageError,
970 "UnexpectedEndOfFile",image->filename);
971 }
972 else
973 {
cristy6c70aec2011-03-04 20:48:09 +0000974 count=ReadBlob(image,4,(unsigned char *) type);
975 if ((count != 0) && (LocaleNCompare(type,"Lr16",4) == 0))
cristy9fb55e82011-02-27 23:12:47 +0000976 size=GetPSDSize(&psd_info,image);
977 else
978 if (DiscardBlobBytes(image,length-quantum-12) == MagickFalse)
979 ThrowFileException(exception,CorruptImageError,
980 "UnexpectedEndOfFile",image->filename);
981 }
cristy3ed852e2009-09-05 21:47:34 +0000982 }
cristy9fb55e82011-02-27 23:12:47 +0000983 if (size != 0)
cristy3ed852e2009-09-05 21:47:34 +0000984 {
985 MagickOffsetType
986 layer_offset;
987
988 layer_offset=offset+length;
989 number_layers=(short) ReadBlobMSBShort(image);
990 if (image->debug != MagickFalse)
991 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +0000992 " image contains %.20g layers",(double) number_layers);
cristy3ed852e2009-09-05 21:47:34 +0000993 if (number_layers < 0)
994 {
995 /*
996 Weird hack in PSD format to ignore first alpha channel.
997 */
998 skip_first_alpha=1;
cristyda16f162011-02-19 23:52:17 +0000999 (void) skip_first_alpha;
cristy3ed852e2009-09-05 21:47:34 +00001000 number_layers=MagickAbsoluteValue(number_layers);
1001 if (image->debug != MagickFalse)
1002 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1003 " negative layer count corrected for");
1004 }
1005 layer_info=(LayerInfo *) AcquireQuantumMemory((size_t) number_layers,
1006 sizeof(*layer_info));
1007 if (layer_info == (LayerInfo *) NULL)
1008 {
1009 if (image->debug != MagickFalse)
1010 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1011 " allocation of LayerInfo failed");
1012 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
1013 }
1014 (void) ResetMagickMemory(layer_info,0,(size_t) number_layers*
1015 sizeof(*layer_info));
1016 for (i=0; i < number_layers; i++)
1017 {
cristy4484e332010-10-08 23:46:21 +00001018 int
1019 x,
1020 y;
1021
cristy3ed852e2009-09-05 21:47:34 +00001022 if (image->debug != MagickFalse)
1023 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001024 " reading layer #%.20g",(double) i+1);
cristy6cff05d2010-09-02 11:22:46 +00001025 layer_info[i].page.y=(int) ReadBlobMSBLong(image);
1026 layer_info[i].page.x=(int) ReadBlobMSBLong(image);
cristy4484e332010-10-08 23:46:21 +00001027 y=(int) ReadBlobMSBLong(image);
1028 x=(int) ReadBlobMSBLong(image);
1029 layer_info[i].page.width=(ssize_t) (x-layer_info[i].page.x);
1030 layer_info[i].page.height=(ssize_t) (y-layer_info[i].page.y);
cristy3ed852e2009-09-05 21:47:34 +00001031 layer_info[i].channels=ReadBlobMSBShort(image);
1032 if (layer_info[i].channels > MaxPSDChannels)
1033 ThrowReaderException(CorruptImageError,"MaximumChannelsExceeded");
1034 if (image->debug != MagickFalse)
1035 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001036 " offset(%.20g,%.20g), size(%.20g,%.20g), channels=%.20g",
1037 (double) layer_info[i].page.x,(double) layer_info[i].page.y,
1038 (double) layer_info[i].page.height,(double)
1039 layer_info[i].page.width,(double) layer_info[i].channels);
cristybb503372010-05-27 20:51:26 +00001040 for (j=0; j < (ssize_t) layer_info[i].channels; j++)
cristy3ed852e2009-09-05 21:47:34 +00001041 {
cristy56ed31c2010-03-22 00:46:21 +00001042 layer_info[i].channel_info[j].type=(short)
1043 ReadBlobMSBShort(image);
cristybb503372010-05-27 20:51:26 +00001044 layer_info[i].channel_info[j].size=(size_t)
cristy56ed31c2010-03-22 00:46:21 +00001045 GetPSDSize(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +00001046 if (image->debug != MagickFalse)
1047 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001048 " channel[%.20g]: type=%.20g, size=%.20g",(double) j,
1049 (double) layer_info[i].channel_info[j].type,
1050 (double) layer_info[i].channel_info[j].size);
cristy3ed852e2009-09-05 21:47:34 +00001051 }
1052 count=ReadBlob(image,4,(unsigned char *) type);
1053 if ((count == 0) || (LocaleNCompare(type,"8BIM",4) != 0))
1054 {
1055 if (image->debug != MagickFalse)
1056 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1057 " layer type was %.4s instead of 8BIM", type);
1058 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
1059 }
1060 count=ReadBlob(image,4,(unsigned char *) layer_info[i].blendkey);
1061 layer_info[i].opacity=(Quantum) (QuantumRange-ScaleCharToQuantum(
1062 (unsigned char) ReadBlobByte(image)));
1063 layer_info[i].clipping=(unsigned char) ReadBlobByte(image);
1064 layer_info[i].flags=(unsigned char) ReadBlobByte(image);
1065 layer_info[i].visible=!(layer_info[i].flags & 0x02);
1066 if (image->debug != MagickFalse)
1067 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001068 " blend=%.4s, opacity=%.20g, clipping=%s, flags=%d, visible=%s",
1069 layer_info[i].blendkey,(double) layer_info[i].opacity,
cristy3ed852e2009-09-05 21:47:34 +00001070 layer_info[i].clipping ? "true" : "false",layer_info[i].flags,
1071 layer_info[i].visible ? "true" : "false");
1072 (void) ReadBlobByte(image); /* filler */
1073 combinedlength=0;
1074 size=ReadBlobMSBLong(image);
1075 if (size != 0)
1076 {
1077 if (image->debug != MagickFalse)
1078 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1079 " layer contains additional info");
1080 length=ReadBlobMSBLong(image);
1081 if (length != 0)
1082 {
1083 /*
1084 Layer mask info.
1085 */
cristy6cff05d2010-09-02 11:22:46 +00001086 layer_info[i].mask.y=(int) ReadBlobMSBLong(image);
1087 layer_info[i].mask.x=(int) ReadBlobMSBLong(image);
cristybb503372010-05-27 20:51:26 +00001088 layer_info[i].mask.height=(size_t)
cristy3ed852e2009-09-05 21:47:34 +00001089 (ReadBlobMSBLong(image)-layer_info[i].mask.y);
cristybb503372010-05-27 20:51:26 +00001090 layer_info[i].mask.width=(size_t)
cristy3ed852e2009-09-05 21:47:34 +00001091 (ReadBlobMSBLong(image)-layer_info[i].mask.x);
1092 if (image->debug != MagickFalse)
1093 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001094 " layer mask: offset(%.20g,%.20g), size(%.20g,%.20g), length=%.20g",
cristyd4297022010-09-16 22:59:09 +00001095 (double) layer_info[i].mask.x,(double) layer_info[i].mask.y,
cristye8c25f92010-06-03 00:53:06 +00001096 (double) layer_info[i].mask.width,(double)
1097 layer_info[i].mask.height,(double) length-16);
cristy3ed852e2009-09-05 21:47:34 +00001098 /*
1099 Skip over the rest of the layer mask information.
1100 */
cristyc1af14f2010-09-16 20:01:21 +00001101 if (DiscardBlobBytes(image,length-16) == MagickFalse)
cristyd4297022010-09-16 22:59:09 +00001102 ThrowFileException(exception,CorruptImageError,
1103 "UnexpectedEndOfFile",image->filename);
cristy3ed852e2009-09-05 21:47:34 +00001104 }
1105 combinedlength+=length+4; /* +4 for length */
1106 length=ReadBlobMSBLong(image);
1107 if (length != 0)
1108 {
1109 /*
1110 Layer blending ranges info.
1111 */
1112 if (image->debug != MagickFalse)
1113 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001114 " layer blending ranges: length=%.20g",(double)
cristyf2faecf2010-05-28 19:19:36 +00001115 length);
cristy3ed852e2009-09-05 21:47:34 +00001116 /*
1117 We read it, but don't use it...
1118 */
cristybb503372010-05-27 20:51:26 +00001119 for (j=0; j < (ssize_t) (length); j+=8)
cristy3ed852e2009-09-05 21:47:34 +00001120 {
1121 size_t blend_source=ReadBlobMSBLong(image);
1122 size_t blend_dest=ReadBlobMSBLong(image);
1123 if (image->debug != MagickFalse)
1124 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1125 " source(%x), dest(%x)",(unsigned int)
1126 blend_source,(unsigned int) blend_dest);
1127 }
1128 }
cristy2cc39842009-10-10 20:04:02 +00001129 combinedlength+=length+4;
1130 /*
1131 Layer name.
1132 */
cristy3ed852e2009-09-05 21:47:34 +00001133 length=(size_t) ReadBlobByte(image);
cristybb503372010-05-27 20:51:26 +00001134 for (j=0; j < (ssize_t) length; j++)
cristy2cc39842009-10-10 20:04:02 +00001135 layer_info[i].name[j]=(unsigned char) ReadBlobByte(image);
1136 layer_info[i].name[j]='\0';
1137 if (image->debug != MagickFalse)
1138 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1139 " layer name: %s",layer_info[i].name);
1140 combinedlength+=length+1;
cristy3ed852e2009-09-05 21:47:34 +00001141
1142#if 0 /* still in development */
1143 /*
1144 Adjustment layers and other stuff...
1145 */
1146 {
1147 char alsig[4],
1148 alkey[4];
1149
1150 count=ReadBlob(image,4,alsig);
1151 if ((count == 0) || (LocaleNCompare(alsig,"8BIM",4) != 0)) {
1152 if (debug != MagickFalse)
1153 {
1154 if (image->debug != MagickFalse)
1155 (void) LogMagickEvent(CoderEvent,GetMagickModule()," adjustment layer type was %.4s instead of 8BIM", alsig);
1156 }
1157 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
1158 }
1159 count=ReadBlob(image,4,alkey);
1160 length=ReadBlobMSBLong(image);
1161 if (debug != MagickFalse)
1162 {
1163 if (image->debug != MagickFalse)
1164 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001165 " adjustment layer key: %.4s, data length=%.20g",
1166 alkey, (double) length);
cristy3ed852e2009-09-05 21:47:34 +00001167 }
1168
1169 if ( length ) {
cristybb503372010-05-27 20:51:26 +00001170 for (j=0; j < (ssize_t) (length); j++)
cristy3ed852e2009-09-05 21:47:34 +00001171 (void) ReadBlobByte(image);
1172 }
1173
1174 }
1175 combinedlength += 12 + length; /* sig, key, length + the actual length*/
1176#endif
1177
1178 /*
1179 Skip the rest of the variable data until we support it.
1180 */
1181 if (image->debug != MagickFalse)
1182 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001183 " unsupported data: length=%.20g",(double)
cristyeec18db2010-03-03 21:15:45 +00001184 (size-combinedlength));
cristyc1af14f2010-09-16 20:01:21 +00001185 if (DiscardBlobBytes(image,size-combinedlength) == MagickFalse)
cristyd4297022010-09-16 22:59:09 +00001186 ThrowFileException(exception,CorruptImageError,
1187 "UnexpectedEndOfFile",image->filename);
cristy3ed852e2009-09-05 21:47:34 +00001188 }
1189 /*
1190 Allocate layered image.
1191 */
1192 layer_info[i].image=CloneImage(image,layer_info[i].page.width,
cristy390c39f2010-10-05 23:41:44 +00001193 layer_info[i].page.height == ~0U ? 1 : layer_info[i].page.height,
1194 MagickFalse,&image->exception);
cristy3ed852e2009-09-05 21:47:34 +00001195 if (layer_info[i].image == (Image *) NULL)
1196 {
1197 for (j=0; j < i; j++)
1198 layer_info[j].image=DestroyImage(layer_info[j].image);
1199 if (image->debug != MagickFalse)
1200 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001201 " allocation of image for layer %.20g failed",(double) i);
cristy3ed852e2009-09-05 21:47:34 +00001202 ThrowReaderException(ResourceLimitError,
1203 "MemoryAllocationFailed");
1204 }
1205 if (image->debug != MagickFalse)
1206 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1207 " setting up new layer image");
cristy2e2e46f2010-10-12 18:55:20 +00001208 if (image_info->ping != MagickFalse)
1209 (void) SetImageBackgroundColor(layer_info[i].image);
cristy3ed852e2009-09-05 21:47:34 +00001210 layer_info[i].image->compose=
1211 PSDBlendModeToCompositeOperator(layer_info[i].blendkey);
1212 if (layer_info[i].visible == MagickFalse)
1213 layer_info[i].image->compose=NoCompositeOp;
1214 if (psd_info.mode == CMYKMode)
cristy0132ba82010-04-16 23:34:02 +00001215 layer_info[i].image->colorspace=CMYKColorspace;
1216 if ((psd_info.mode == BitmapMode) ||
1217 (psd_info.mode == GrayscaleMode) ||
1218 (psd_info.mode == DuotoneMode))
1219 layer_info[i].image->colorspace=GRAYColorspace;
cristybb503372010-05-27 20:51:26 +00001220 for (j=0; j < (ssize_t) layer_info[i].channels; j++)
cristy3ed852e2009-09-05 21:47:34 +00001221 if (layer_info[i].channel_info[j].type == -1)
1222 layer_info[i].image->matte=MagickTrue;
1223 /*
1224 Set up some hidden attributes for folks that need them.
1225 */
cristye8c25f92010-06-03 00:53:06 +00001226 (void) FormatMagickString(message,MaxTextExtent,"%.20gld",
1227 (double) layer_info[i].page.x);
cristy56ed31c2010-03-22 00:46:21 +00001228 (void) SetImageArtifact(layer_info[i].image,"psd:layer.x",message);
cristye8c25f92010-06-03 00:53:06 +00001229 (void) FormatMagickString(message,MaxTextExtent,"%.20g",
1230 (double) layer_info[i].page.y);
cristy56ed31c2010-03-22 00:46:21 +00001231 (void) SetImageArtifact(layer_info[i].image,"psd:layer.y",message);
cristye8c25f92010-06-03 00:53:06 +00001232 (void) FormatMagickString(message,MaxTextExtent,"%.20g",
1233 (double) layer_info[i].opacity);
cristy56ed31c2010-03-22 00:46:21 +00001234 (void) SetImageArtifact(layer_info[i].image,"psd:layer.opacity",
1235 message);
cristy3ed852e2009-09-05 21:47:34 +00001236 (void) SetImageProperty(layer_info[i].image,"label",(char *)
1237 layer_info[i].name);
1238 }
1239 if (image->debug != MagickFalse)
1240 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1241 " reading image data for layers");
1242 /*
1243 Read pixel data for each layer.
1244 */
1245 for (i=0; i < number_layers; i++)
1246 {
1247 if (image->debug != MagickFalse)
1248 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001249 " reading data for layer %.20g",(double) i);
cristybb503372010-05-27 20:51:26 +00001250 for (j=0; j < (ssize_t) layer_info[i].channels; j++)
cristy3ed852e2009-09-05 21:47:34 +00001251 {
1252 if (image->debug != MagickFalse)
1253 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001254 " reading data for channel %.20g",(double) j);
cristy3ed852e2009-09-05 21:47:34 +00001255#if 1
1256 if (layer_info[i].channel_info[j].size <= (2*layer_info[i].image->rows))
1257 {
cristybb503372010-05-27 20:51:26 +00001258 ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001259 k;
1260
1261 if (image->debug != MagickFalse)
1262 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1263 " layer data is empty");
1264 /*
1265 A layer without data.
1266 */
cristybb503372010-05-27 20:51:26 +00001267 for (k=0; k < (ssize_t) layer_info[i].channel_info[j].size; k++)
cristy3ed852e2009-09-05 21:47:34 +00001268 (void) ReadBlobByte(layer_info[i].image);
1269 continue;
1270 }
1271#endif
cristy56ed31c2010-03-22 00:46:21 +00001272 offsets=(MagickOffsetType *) NULL;
1273 layer_info[i].image->compression=NoCompression;
cristy3ed852e2009-09-05 21:47:34 +00001274 compression=ReadBlobMSBShort(layer_info[i].image);
cristy2cc39842009-10-10 20:04:02 +00001275 if ((layer_info[i].page.height != 0) &&
1276 (layer_info[i].page.width != 0))
cristy3ed852e2009-09-05 21:47:34 +00001277 {
cristy2cc39842009-10-10 20:04:02 +00001278 if (compression == 1)
1279 {
1280 /*
1281 Read RLE compressed data.
1282 */
cristy56ed31c2010-03-22 00:46:21 +00001283 layer_info[i].image->compression=RLECompression;
cristy2cc39842009-10-10 20:04:02 +00001284 if (image->debug != MagickFalse)
1285 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1286 " layer data is RLE compressed");
cristy56ed31c2010-03-22 00:46:21 +00001287 offsets=(MagickOffsetType *) AcquireQuantumMemory(
1288 layer_info[i].image->rows,sizeof(*offsets));
1289 if (offsets == (MagickOffsetType *) NULL)
1290 ThrowReaderException(ResourceLimitError,
1291 "MemoryAllocationFailed");
cristybb503372010-05-27 20:51:26 +00001292 for (y=0; y < (ssize_t) layer_info[i].image->rows; y++)
cristy56ed31c2010-03-22 00:46:21 +00001293 offsets[y]=GetPSDOffset(&psd_info,layer_info[i].image);
cristy2cc39842009-10-10 20:04:02 +00001294 }
cristy56ed31c2010-03-22 00:46:21 +00001295 status=ReadPSDLayer(layer_info[i].image,
cristye3038982010-05-17 02:20:52 +00001296 layer_info[i].channels,layer_info[i].channel_info[j].type,
1297 offsets,exception);
cristy56ed31c2010-03-22 00:46:21 +00001298 if (compression == 1)
1299 offsets=(MagickOffsetType *) RelinquishMagickMemory(
1300 offsets);
1301 if (status == MagickFalse)
1302 break;
cristy3ed852e2009-09-05 21:47:34 +00001303 }
cristy56ed31c2010-03-22 00:46:21 +00001304 }
cristy3ed852e2009-09-05 21:47:34 +00001305 if (layer_info[i].opacity != OpaqueOpacity)
1306 {
1307 /*
1308 Correct for opacity level.
1309 */
cristybb503372010-05-27 20:51:26 +00001310 for (y=0; y < (ssize_t) layer_info[i].image->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001311 {
1312 q=GetAuthenticPixels(layer_info[i].image,0,y,
1313 layer_info[i].image->columns,1,exception);
1314 if (q == (PixelPacket *) NULL)
1315 break;
cristybb503372010-05-27 20:51:26 +00001316 for (x=0; x < (ssize_t) layer_info[i].image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001317 {
cristy07758662010-01-15 20:25:50 +00001318 q->opacity=(Quantum) (QuantumRange-(Quantum) (QuantumScale*
1319 ((QuantumRange-q->opacity)*(QuantumRange-
1320 layer_info[i].opacity))));
cristy3ed852e2009-09-05 21:47:34 +00001321 q++;
1322 }
1323 if (SyncAuthenticPixels(layer_info[i].image,exception) == MagickFalse)
1324 break;
1325 }
1326 }
1327 if (layer_info[i].image->colorspace == CMYKColorspace)
1328 (void) NegateImage(layer_info[i].image,MagickFalse);
cristy56ed31c2010-03-22 00:46:21 +00001329 status=SetImageProgress(image,LoadImagesTag,i,(MagickSizeType)
1330 number_layers);
cristy4689cf02010-02-17 21:15:45 +00001331 if (status == MagickFalse)
1332 break;
cristy3ed852e2009-09-05 21:47:34 +00001333 }
1334 /* added by palf -> invisible group layer make layer of this group
1335 invisible I consider that all layer with width and height null are
1336 layer for group layer */
1337 {
1338 short inside_layer = 0;
1339 short layer_visible = 0;
1340 for (i=number_layers-1; i >=0; i--)
1341 {
1342 if ((layer_info[i].page.width == 0) ||
1343 (layer_info[i].page.height == 0))
1344 {
1345 if (inside_layer == 0)
1346 {
1347 inside_layer=1;
1348 layer_visible=(short int) layer_info[i].visible;
1349 }
1350 else
1351 {
1352 inside_layer = 0;
1353 }
1354 }
1355 else
1356 if ((inside_layer == 1) && (layer_visible == 0))
1357 {
1358 layer_info[i].visible=(unsigned char) layer_visible;
1359 layer_info[i].image->compose=NoCompositeOp;
1360 }
1361 }
1362 }
1363 /* added by palf -> suppression of empty layer */
1364 /* I consider that all layer with width and height null are layer for group layer */
1365 for (i=0; i < number_layers; i++)
1366 {
1367 if ((layer_info[i].page.width == 0) ||
1368 (layer_info[i].page.height == 0))
1369 {
1370 if (layer_info[i].image != (Image *) NULL)
1371 layer_info[i].image=DestroyImage(layer_info[i].image);
1372 for (j=i; j < number_layers - 1; j++)
1373 layer_info[j] = layer_info[j+1];
1374 number_layers--;
1375 i--;
1376 }
cristy56ed31c2010-03-22 00:46:21 +00001377 }
cristy3ed852e2009-09-05 21:47:34 +00001378 mask_size = ReadBlobMSBLong(image); /* global mask size: currently ignored */
cristyda16f162011-02-19 23:52:17 +00001379 (void) mask_size;
cristy56ed31c2010-03-22 00:46:21 +00001380 if (number_layers > 0)
1381 {
1382 if (image->debug != MagickFalse)
1383 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1384 " putting layers into image list");
1385 for (i=0; i < number_layers; i++)
cristy3ed852e2009-09-05 21:47:34 +00001386 {
cristy56ed31c2010-03-22 00:46:21 +00001387 if (i > 0)
1388 layer_info[i].image->previous=layer_info[i-1].image;
1389 if (i < (number_layers-1))
1390 layer_info[i].image->next=layer_info[i+1].image;
1391 layer_info[i].image->page=layer_info[i].page;
1392 }
cristy3ed852e2009-09-05 21:47:34 +00001393 image->next=layer_info[0].image;
1394 layer_info[0].image->previous=image;
1395 layer_info=(LayerInfo *) RelinquishMagickMemory(layer_info);
cristy56ed31c2010-03-22 00:46:21 +00001396 }
1397 layer_offset-=TellBlob(image);
1398 offset=SeekBlob(image,layer_offset,SEEK_CUR);
1399 }
cristy3ed852e2009-09-05 21:47:34 +00001400 }
1401 /*
1402 Read the precombined layer, present for PSD < 4 compatibility
1403 */
1404 if (image->debug != MagickFalse)
1405 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1406 " reading the precombined layer");
cristy56ed31c2010-03-22 00:46:21 +00001407 offsets=(MagickOffsetType *) NULL;
1408 image->compression=NoCompression;
cristy3ed852e2009-09-05 21:47:34 +00001409 compression=ReadBlobMSBShort(image);
1410 if (compression == 1)
1411 {
1412 /*
cristy3f938c62010-03-07 00:53:52 +00001413 Read Packbit encoded pixel data as separate planes.
cristy3ed852e2009-09-05 21:47:34 +00001414 */
cristy56ed31c2010-03-22 00:46:21 +00001415 image->compression=RLECompression;
1416 offsets=(MagickOffsetType *) AcquireQuantumMemory(image->rows,
1417 psd_info.channels*sizeof(*offsets));
1418 if (offsets == (MagickOffsetType *) NULL)
1419 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristybb503372010-05-27 20:51:26 +00001420 for (i=0; i < (ssize_t) (image->rows*psd_info.channels); i++)
cristy56ed31c2010-03-22 00:46:21 +00001421 offsets[i]=GetPSDOffset(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +00001422 }
cristybb503372010-05-27 20:51:26 +00001423 for (i=0; i < (ssize_t) psd_info.channels; i++)
cristy56ed31c2010-03-22 00:46:21 +00001424 {
cristye3038982010-05-17 02:20:52 +00001425 status=ReadPSDLayer(image,psd_info.channels,i,offsets+i*image->rows,
1426 exception);
cristy56ed31c2010-03-22 00:46:21 +00001427 if (status == MagickFalse)
1428 break;
1429 status=SetImageProgress(image,LoadImagesTag,i,psd_info.channels);
1430 if (status == MagickFalse)
1431 break;
1432 }
1433 if (compression == 1)
1434 offsets=(MagickOffsetType *) RelinquishMagickMemory(offsets);
cristy3ed852e2009-09-05 21:47:34 +00001435 if (image->colorspace == CMYKColorspace)
1436 (void) NegateImage(image,MagickFalse);
1437 (void) CloseBlob(image);
1438 return(GetFirstImageInList(image));
1439}
1440
1441/*
1442%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1443% %
1444% %
1445% %
1446% R e g i s t e r P S D I m a g e %
1447% %
1448% %
1449% %
1450%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1451%
1452% RegisterPSDImage() adds properties for the PSD image format to
1453% the list of supported formats. The properties include the image format
1454% tag, a method to read and/or write the format, whether the format
1455% supports the saving of more than one frame to the same file or blob,
1456% whether the format supports native in-memory I/O, and a brief
1457% description of the format.
1458%
1459% The format of the RegisterPSDImage method is:
1460%
cristybb503372010-05-27 20:51:26 +00001461% size_t RegisterPSDImage(void)
cristy3ed852e2009-09-05 21:47:34 +00001462%
1463*/
cristybb503372010-05-27 20:51:26 +00001464ModuleExport size_t RegisterPSDImage(void)
cristy3ed852e2009-09-05 21:47:34 +00001465{
1466 MagickInfo
1467 *entry;
1468
cristyb4233012010-02-28 20:09:14 +00001469 entry=SetMagickInfo("PSB");
1470 entry->decoder=(DecodeImageHandler *) ReadPSDImage;
1471 entry->encoder=(EncodeImageHandler *) WritePSDImage;
1472 entry->magick=(IsImageFormatHandler *) IsPSD;
cristyffaf9782011-04-13 19:50:51 +00001473 entry->seekable_stream=MagickTrue;
cristyb4233012010-02-28 20:09:14 +00001474 entry->description=ConstantString("Adobe Large Document Format");
1475 entry->module=ConstantString("PSD");
1476 (void) RegisterMagickInfo(entry);
cristy3ed852e2009-09-05 21:47:34 +00001477 entry=SetMagickInfo("PSD");
1478 entry->decoder=(DecodeImageHandler *) ReadPSDImage;
1479 entry->encoder=(EncodeImageHandler *) WritePSDImage;
1480 entry->magick=(IsImageFormatHandler *) IsPSD;
cristyffaf9782011-04-13 19:50:51 +00001481 entry->seekable_stream=MagickTrue;
cristy3ed852e2009-09-05 21:47:34 +00001482 entry->description=ConstantString("Adobe Photoshop bitmap");
1483 entry->module=ConstantString("PSD");
1484 (void) RegisterMagickInfo(entry);
1485 return(MagickImageCoderSignature);
1486}
1487
1488/*
1489%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1490% %
1491% %
1492% %
1493% U n r e g i s t e r P S D I m a g e %
1494% %
1495% %
1496% %
1497%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1498%
1499% UnregisterPSDImage() removes format registrations made by the
1500% PSD module from the list of supported formats.
1501%
1502% The format of the UnregisterPSDImage method is:
1503%
1504% UnregisterPSDImage(void)
1505%
1506*/
1507ModuleExport void UnregisterPSDImage(void)
1508{
cristyb4233012010-02-28 20:09:14 +00001509 (void) UnregisterMagickInfo("PSB");
cristy3ed852e2009-09-05 21:47:34 +00001510 (void) UnregisterMagickInfo("PSD");
1511}
1512
1513/*
1514%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1515% %
1516% %
1517% %
1518% W r i t e P S D I m a g e %
1519% %
1520% %
1521% %
1522%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1523%
cristyb1459bc2010-03-23 21:41:43 +00001524% WritePSDImage() writes an image in the Adobe Photoshop encoded image format.
cristy3ed852e2009-09-05 21:47:34 +00001525%
1526% The format of the WritePSDImage method is:
1527%
1528% MagickBooleanType WritePSDImage(const ImageInfo *image_info,Image *image)
1529%
1530% A description of each parameter follows.
1531%
1532% o image_info: the image info.
1533%
1534% o image: The image.
1535%
1536%
1537*/
1538
cristy50aea4a2010-03-09 17:37:44 +00001539static inline ssize_t SetPSDOffset(const PSDInfo *psd_info,Image *image,
cristyf0460ee2010-03-06 02:55:11 +00001540 const size_t offset)
1541{
1542 if (psd_info->version == 1)
cristy56ed31c2010-03-22 00:46:21 +00001543 return(WriteBlobMSBShort(image,(unsigned short) offset));
1544 return(WriteBlobMSBLong(image,(unsigned short) offset));
cristyf0460ee2010-03-06 02:55:11 +00001545}
1546
cristy50aea4a2010-03-09 17:37:44 +00001547static inline ssize_t SetPSDSize(const PSDInfo *psd_info,Image *image,
cristyf0460ee2010-03-06 02:55:11 +00001548 const MagickSizeType size)
1549{
1550 if (psd_info->version == 1)
cristy56ed31c2010-03-22 00:46:21 +00001551 return(WriteBlobMSBLong(image,(unsigned int) size));
cristyf0460ee2010-03-06 02:55:11 +00001552 return(WriteBlobMSBLongLong(image,size));
1553}
1554
cristy4aff1572010-02-15 13:34:01 +00001555static size_t PSDPackbitsEncodeImage(Image *image,const size_t length,
cristyb1459bc2010-03-23 21:41:43 +00001556 const unsigned char *pixels,unsigned char *compact_pixels)
cristy4aff1572010-02-15 13:34:01 +00001557{
1558 int
1559 count;
1560
cristybb503372010-05-27 20:51:26 +00001561 register ssize_t
cristy4aff1572010-02-15 13:34:01 +00001562 i,
1563 j;
1564
1565 register unsigned char
1566 *q;
1567
1568 unsigned char
1569 *packbits;
1570
1571 /*
1572 Compress pixels with Packbits encoding.
1573 */
1574 assert(image != (Image *) NULL);
1575 assert(image->signature == MagickSignature);
1576 if (image->debug != MagickFalse)
1577 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
1578 assert(pixels != (unsigned char *) NULL);
1579 packbits=(unsigned char *) AcquireQuantumMemory(128UL,sizeof(*packbits));
1580 if (packbits == (unsigned char *) NULL)
1581 ThrowBinaryException(ResourceLimitError,"MemoryAllocationFailed",
1582 image->filename);
cristyb1459bc2010-03-23 21:41:43 +00001583 q=compact_pixels;
cristybb503372010-05-27 20:51:26 +00001584 for (i=(ssize_t) length; i != 0; )
cristy4aff1572010-02-15 13:34:01 +00001585 {
1586 switch (i)
1587 {
1588 case 1:
1589 {
1590 i--;
1591 *q++=(unsigned char) 0;
1592 *q++=(*pixels);
1593 break;
1594 }
1595 case 2:
1596 {
1597 i-=2;
1598 *q++=(unsigned char) 1;
1599 *q++=(*pixels);
1600 *q++=pixels[1];
1601 break;
1602 }
1603 case 3:
1604 {
1605 i-=3;
1606 if ((*pixels == *(pixels+1)) && (*(pixels+1) == *(pixels+2)))
1607 {
1608 *q++=(unsigned char) ((256-3)+1);
1609 *q++=(*pixels);
1610 break;
1611 }
1612 *q++=(unsigned char) 2;
1613 *q++=(*pixels);
1614 *q++=pixels[1];
1615 *q++=pixels[2];
1616 break;
1617 }
1618 default:
1619 {
1620 if ((*pixels == *(pixels+1)) && (*(pixels+1) == *(pixels+2)))
1621 {
1622 /*
1623 Packed run.
1624 */
1625 count=3;
cristybb503372010-05-27 20:51:26 +00001626 while (((ssize_t) count < i) && (*pixels == *(pixels+count)))
cristy4aff1572010-02-15 13:34:01 +00001627 {
1628 count++;
1629 if (count >= 127)
1630 break;
1631 }
1632 i-=count;
1633 *q++=(unsigned char) ((256-count)+1);
1634 *q++=(*pixels);
1635 pixels+=count;
1636 break;
1637 }
1638 /*
1639 Literal run.
1640 */
1641 count=0;
1642 while ((*(pixels+count) != *(pixels+count+1)) ||
1643 (*(pixels+count+1) != *(pixels+count+2)))
1644 {
1645 packbits[count+1]=pixels[count];
1646 count++;
cristybb503372010-05-27 20:51:26 +00001647 if (((ssize_t) count >= (i-3)) || (count >= 127))
cristy4aff1572010-02-15 13:34:01 +00001648 break;
1649 }
1650 i-=count;
1651 *packbits=(unsigned char) (count-1);
cristybb503372010-05-27 20:51:26 +00001652 for (j=0; j <= (ssize_t) count; j++)
cristy4aff1572010-02-15 13:34:01 +00001653 *q++=packbits[j];
1654 pixels+=count;
1655 break;
1656 }
1657 }
1658 }
1659 *q++=(unsigned char) 128; /* EOD marker */
1660 packbits=(unsigned char *) RelinquishMagickMemory(packbits);
cristyb1459bc2010-03-23 21:41:43 +00001661 return((size_t) (q-compact_pixels));
cristy4aff1572010-02-15 13:34:01 +00001662}
1663
cristy875e28a2010-03-06 19:46:55 +00001664static void WritePackbitsLength(const PSDInfo *psd_info,
cristya20214e2010-08-28 16:52:13 +00001665 const ImageInfo *image_info,Image *image,Image *next_image,
1666 unsigned char *compact_pixels,const QuantumType quantum_type)
cristy3ed852e2009-09-05 21:47:34 +00001667{
cristy3ed852e2009-09-05 21:47:34 +00001668 QuantumInfo
1669 *quantum_info;
1670
1671 register const PixelPacket
1672 *p;
1673
1674 size_t
cristy4aff1572010-02-15 13:34:01 +00001675 length,
cristy3ed852e2009-09-05 21:47:34 +00001676 packet_size;
1677
cristy75f85ae2010-09-25 03:01:06 +00001678 ssize_t
1679 y;
1680
cristya20214e2010-08-28 16:52:13 +00001681 unsigned char
1682 *pixels;
1683
1684 if (next_image->depth > 8)
1685 next_image->depth=16;
1686 packet_size=next_image->depth > 8UL ? 2UL : 1UL;
cristyda16f162011-02-19 23:52:17 +00001687 (void) packet_size;
cristy4aff1572010-02-15 13:34:01 +00001688 quantum_info=AcquireQuantumInfo(image_info,image);
cristya20214e2010-08-28 16:52:13 +00001689 pixels=GetQuantumPixels(quantum_info);
1690 for (y=0; y < (ssize_t) next_image->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001691 {
cristya20214e2010-08-28 16:52:13 +00001692 p=GetVirtualPixels(next_image,0,y,next_image->columns,1,&image->exception);
cristy3ed852e2009-09-05 21:47:34 +00001693 if (p == (const PixelPacket *) NULL)
cristy4aff1572010-02-15 13:34:01 +00001694 break;
cristya20214e2010-08-28 16:52:13 +00001695 length=ExportQuantumPixels(next_image,(CacheView *) NULL,quantum_info,
cristy3ed852e2009-09-05 21:47:34 +00001696 quantum_type,pixels,&image->exception);
cristyb1459bc2010-03-23 21:41:43 +00001697 length=PSDPackbitsEncodeImage(image,length,pixels,compact_pixels);
cristy875e28a2010-03-06 19:46:55 +00001698 (void) SetPSDOffset(psd_info,image,length);
cristy4aff1572010-02-15 13:34:01 +00001699 }
1700 quantum_info=DestroyQuantumInfo(quantum_info);
1701}
1702
cristy875e28a2010-03-06 19:46:55 +00001703static void WriteOneChannel(const PSDInfo *psd_info,const ImageInfo *image_info,
cristya20214e2010-08-28 16:52:13 +00001704 Image *image,Image *next_image,unsigned char *compact_pixels,
1705 const QuantumType quantum_type,const MagickBooleanType compression_flag)
cristy4aff1572010-02-15 13:34:01 +00001706{
1707 int
1708 y;
1709
cristy0910f242010-04-01 18:55:09 +00001710 MagickBooleanType
1711 monochrome;
1712
cristy4aff1572010-02-15 13:34:01 +00001713 QuantumInfo
1714 *quantum_info;
1715
1716 register const PixelPacket
1717 *p;
1718
cristybb503372010-05-27 20:51:26 +00001719 register ssize_t
cristy0910f242010-04-01 18:55:09 +00001720 i;
1721
cristy4aff1572010-02-15 13:34:01 +00001722 size_t
1723 length,
1724 packet_size;
1725
cristya20214e2010-08-28 16:52:13 +00001726 unsigned char
1727 *pixels;
1728
cristy50aea4a2010-03-09 17:37:44 +00001729 (void) psd_info;
cristy4aff1572010-02-15 13:34:01 +00001730 if ((compression_flag != MagickFalse) &&
cristya20214e2010-08-28 16:52:13 +00001731 (next_image->compression != RLECompression))
cristy4aff1572010-02-15 13:34:01 +00001732 (void) WriteBlobMSBShort(image,0);
cristya20214e2010-08-28 16:52:13 +00001733 if (next_image->depth > 8)
1734 next_image->depth=16;
cristybf9f4ba2010-08-20 18:13:29 +00001735 monochrome=IsMonochromeImage(image,&image->exception) && (image->depth == 1)
1736 ? MagickTrue : MagickFalse;
cristya20214e2010-08-28 16:52:13 +00001737 packet_size=next_image->depth > 8UL ? 2UL : 1UL;
cristyda16f162011-02-19 23:52:17 +00001738 (void) packet_size;
cristy4aff1572010-02-15 13:34:01 +00001739 quantum_info=AcquireQuantumInfo(image_info,image);
cristya20214e2010-08-28 16:52:13 +00001740 pixels=GetQuantumPixels(quantum_info);
1741 for (y=0; y < (ssize_t) next_image->rows; y++)
cristy4aff1572010-02-15 13:34:01 +00001742 {
cristya20214e2010-08-28 16:52:13 +00001743 p=GetVirtualPixels(next_image,0,y,next_image->columns,1,&image->exception);
cristy4aff1572010-02-15 13:34:01 +00001744 if (p == (const PixelPacket *) NULL)
1745 break;
cristya20214e2010-08-28 16:52:13 +00001746 length=ExportQuantumPixels(next_image,(CacheView *) NULL,quantum_info,
cristy4aff1572010-02-15 13:34:01 +00001747 quantum_type,pixels,&image->exception);
cristy0910f242010-04-01 18:55:09 +00001748 if (monochrome != MagickFalse)
cristybb503372010-05-27 20:51:26 +00001749 for (i=0; i < (ssize_t) length; i++)
cristy0910f242010-04-01 18:55:09 +00001750 pixels[i]=(~pixels[i]);
cristya20214e2010-08-28 16:52:13 +00001751 if (next_image->compression != RLECompression)
cristy4aff1572010-02-15 13:34:01 +00001752 (void) WriteBlob(image,length,pixels);
1753 else
1754 {
cristyb1459bc2010-03-23 21:41:43 +00001755 length=PSDPackbitsEncodeImage(image,length,pixels,compact_pixels);
1756 (void) WriteBlob(image,length,compact_pixels);
cristy4aff1572010-02-15 13:34:01 +00001757 }
cristy3ed852e2009-09-05 21:47:34 +00001758 }
1759 quantum_info=DestroyQuantumInfo(quantum_info);
1760}
1761
cristy875e28a2010-03-06 19:46:55 +00001762static MagickBooleanType WriteImageChannels(const PSDInfo *psd_info,
cristya20214e2010-08-28 16:52:13 +00001763 const ImageInfo *image_info,Image *image,Image *next_image,
cristy875e28a2010-03-06 19:46:55 +00001764 const MagickBooleanType separate)
cristy3ed852e2009-09-05 21:47:34 +00001765{
1766 int
1767 i;
1768
1769 size_t
1770 channels,
1771 packet_size;
1772
1773 unsigned char
cristya20214e2010-08-28 16:52:13 +00001774 *compact_pixels;
cristy3ed852e2009-09-05 21:47:34 +00001775
1776 /*
cristy875e28a2010-03-06 19:46:55 +00001777 Write uncompressed pixels as separate planes.
cristy3ed852e2009-09-05 21:47:34 +00001778 */
1779 channels=1;
cristya20214e2010-08-28 16:52:13 +00001780 packet_size=next_image->depth > 8UL ? 2UL : 1UL;
1781 compact_pixels=(unsigned char *) NULL;
1782 if (next_image->compression == RLECompression)
cristy4aff1572010-02-15 13:34:01 +00001783 {
cristya20214e2010-08-28 16:52:13 +00001784 compact_pixels=(unsigned char *) AcquireQuantumMemory(2*channels*
1785 next_image->columns,packet_size*sizeof(*compact_pixels));
1786 if (compact_pixels == (unsigned char *) NULL)
1787 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
cristy4aff1572010-02-15 13:34:01 +00001788 }
cristy3ed852e2009-09-05 21:47:34 +00001789 i=0;
cristya20214e2010-08-28 16:52:13 +00001790 if (IsGrayImage(next_image,&next_image->exception) != MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001791 {
cristya20214e2010-08-28 16:52:13 +00001792 if (next_image->compression == RLECompression)
cristy4aff1572010-02-15 13:34:01 +00001793 {
1794 /*
1795 Packbits compression.
1796 */
1797 (void) WriteBlobMSBShort(image,1);
cristya20214e2010-08-28 16:52:13 +00001798 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy95119782010-06-01 17:40:58 +00001799 compact_pixels,GrayQuantum);
cristya20214e2010-08-28 16:52:13 +00001800 if (next_image->matte != MagickFalse)
1801 WritePackbitsLength(psd_info,image_info,image,next_image,
cristyb1459bc2010-03-23 21:41:43 +00001802 compact_pixels,AlphaQuantum);
cristy4aff1572010-02-15 13:34:01 +00001803 }
cristya20214e2010-08-28 16:52:13 +00001804 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1805 GrayQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1806 MagickFalse);
1807 if (next_image->matte != MagickFalse)
1808 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1809 AlphaQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1810 MagickFalse);
cristy6886a752010-04-23 18:23:20 +00001811 (void) SetImageProgress(image,SaveImagesTag,0,1);
cristy3ed852e2009-09-05 21:47:34 +00001812 }
cristy0910f242010-04-01 18:55:09 +00001813 else
cristya20214e2010-08-28 16:52:13 +00001814 if (next_image->storage_class == PseudoClass)
cristy0910f242010-04-01 18:55:09 +00001815 {
cristya20214e2010-08-28 16:52:13 +00001816 if (next_image->compression == RLECompression)
cristy0910f242010-04-01 18:55:09 +00001817 {
1818 /*
1819 Packbits compression.
1820 */
1821 (void) WriteBlobMSBShort(image,1);
cristya20214e2010-08-28 16:52:13 +00001822 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy95119782010-06-01 17:40:58 +00001823 compact_pixels,IndexQuantum);
cristya20214e2010-08-28 16:52:13 +00001824 if (next_image->matte != MagickFalse)
1825 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy6886a752010-04-23 18:23:20 +00001826 compact_pixels,AlphaQuantum);
cristy0910f242010-04-01 18:55:09 +00001827 }
cristya20214e2010-08-28 16:52:13 +00001828 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1829 IndexQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1830 MagickFalse);
1831 if (next_image->matte != MagickFalse)
1832 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1833 AlphaQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1834 MagickFalse);
cristy0910f242010-04-01 18:55:09 +00001835 (void) SetImageProgress(image,SaveImagesTag,0,1);
1836 }
1837 else
1838 {
cristya20214e2010-08-28 16:52:13 +00001839 if (next_image->colorspace == CMYKColorspace)
1840 (void) NegateImage(next_image,MagickFalse);
1841 if (next_image->compression == RLECompression)
cristy0910f242010-04-01 18:55:09 +00001842 {
1843 /*
1844 Packbits compression.
1845 */
1846 (void) WriteBlobMSBShort(image,1);
cristya20214e2010-08-28 16:52:13 +00001847 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001848 compact_pixels,RedQuantum);
cristya20214e2010-08-28 16:52:13 +00001849 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001850 compact_pixels,GreenQuantum);
cristya20214e2010-08-28 16:52:13 +00001851 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001852 compact_pixels,BlueQuantum);
cristya20214e2010-08-28 16:52:13 +00001853 if (next_image->colorspace == CMYKColorspace)
1854 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001855 compact_pixels,BlackQuantum);
cristya20214e2010-08-28 16:52:13 +00001856 if (next_image->matte != MagickFalse)
1857 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy860cc732010-05-19 16:38:37 +00001858 compact_pixels,AlphaQuantum);
cristy0910f242010-04-01 18:55:09 +00001859 }
1860 (void) SetImageProgress(image,SaveImagesTag,0,6);
cristya20214e2010-08-28 16:52:13 +00001861 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1862 RedQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1863 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001864 (void) SetImageProgress(image,SaveImagesTag,1,6);
cristya20214e2010-08-28 16:52:13 +00001865 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1866 GreenQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1867 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001868 (void) SetImageProgress(image,SaveImagesTag,2,6);
cristya20214e2010-08-28 16:52:13 +00001869 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1870 BlueQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1871 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001872 (void) SetImageProgress(image,SaveImagesTag,3,6);
cristya20214e2010-08-28 16:52:13 +00001873 if (next_image->colorspace == CMYKColorspace)
1874 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1875 BlackQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1876 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001877 (void) SetImageProgress(image,SaveImagesTag,4,6);
cristya20214e2010-08-28 16:52:13 +00001878 if (next_image->matte != MagickFalse)
1879 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1880 AlphaQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1881 MagickFalse);
cristy0910f242010-04-01 18:55:09 +00001882 (void) SetImageProgress(image,SaveImagesTag,5,6);
cristya20214e2010-08-28 16:52:13 +00001883 if (next_image->colorspace == CMYKColorspace)
1884 (void) NegateImage(next_image,MagickFalse);
cristy0910f242010-04-01 18:55:09 +00001885 }
cristya20214e2010-08-28 16:52:13 +00001886 if (next_image->compression == RLECompression)
1887 compact_pixels=(unsigned char *) RelinquishMagickMemory(compact_pixels);
cristy3ed852e2009-09-05 21:47:34 +00001888 return(MagickTrue);
1889}
1890
cristy3ed852e2009-09-05 21:47:34 +00001891static void WritePascalString(Image* inImage,const char *inString,int inPad)
1892{
1893 size_t
cristya20214e2010-08-28 16:52:13 +00001894 length;
cristy3ed852e2009-09-05 21:47:34 +00001895
cristya20214e2010-08-28 16:52:13 +00001896 register ssize_t
1897 i;
cristy3ed852e2009-09-05 21:47:34 +00001898
cristya20214e2010-08-28 16:52:13 +00001899 /*
1900 Max length is 255.
1901 */
1902 length=(strlen(inString) > 255UL ) ? 255UL : strlen(inString);
1903 if (length == 0)
1904 (void) WriteBlobByte(inImage,0);
cristy3ed852e2009-09-05 21:47:34 +00001905 else
cristya20214e2010-08-28 16:52:13 +00001906 {
1907 (void) WriteBlobByte(inImage,(unsigned char) length);
1908 (void) WriteBlob(inImage, length, (const unsigned char *) inString);
1909 }
1910 length++;
1911 if ((length % inPad) == 0)
cristy3ed852e2009-09-05 21:47:34 +00001912 return;
cristya20214e2010-08-28 16:52:13 +00001913 for (i=0; i < (ssize_t) (inPad-(length % inPad)); i++)
cristy3ed852e2009-09-05 21:47:34 +00001914 (void) WriteBlobByte(inImage,0);
1915}
1916
1917static void WriteResolutionResourceBlock(Image *image)
1918{
cristy56ed31c2010-03-22 00:46:21 +00001919 double
1920 x_resolution,
1921 y_resolution;
cristy3ed852e2009-09-05 21:47:34 +00001922
1923 unsigned short
1924 units;
1925
1926 x_resolution=65536.0*image->x_resolution+0.5;
1927 y_resolution=65536.0*image->y_resolution+0.5;
1928 units=1;
1929 if (image->units == PixelsPerCentimeterResolution)
1930 {
1931 x_resolution=2.54*65536.0*image->x_resolution*0.5;
1932 y_resolution=2.54*65536.0*image->y_resolution+0.5;
1933 units=2;
1934 }
1935 (void) WriteBlob(image,4,(const unsigned char *) "8BIM");
1936 (void) WriteBlobMSBShort(image,0x03ED);
1937 (void) WriteBlobMSBShort(image,0);
1938 (void) WriteBlobMSBLong(image,16); /* resource size */
cristy56ed31c2010-03-22 00:46:21 +00001939 (void) WriteBlobMSBLong(image,(unsigned int) (x_resolution+0.5));
cristy3ed852e2009-09-05 21:47:34 +00001940 (void) WriteBlobMSBShort(image,units); /* horizontal resolution unit */
1941 (void) WriteBlobMSBShort(image,units); /* width unit */
cristy56ed31c2010-03-22 00:46:21 +00001942 (void) WriteBlobMSBLong(image,(unsigned int) (y_resolution+0.5));
cristy3ed852e2009-09-05 21:47:34 +00001943 (void) WriteBlobMSBShort(image,units); /* vertical resolution unit */
1944 (void) WriteBlobMSBShort(image,units); /* height unit */
1945}
1946
cristyd4d3f742010-04-25 20:36:50 +00001947static void RemoveICCProfileFromResourceBlock(StringInfo *bim_profile)
1948{
cristy0b796e62010-04-29 00:38:45 +00001949 register const unsigned char
cristyd4d3f742010-04-25 20:36:50 +00001950 *p;
1951
1952 size_t
1953 length;
1954
1955 unsigned char
1956 *datum;
1957
cristy6befb0f2010-05-31 14:33:15 +00001958 unsigned int
cristyd4d3f742010-04-25 20:36:50 +00001959 count,
cristy6befb0f2010-05-31 14:33:15 +00001960 long_sans;
cristyd4d3f742010-04-25 20:36:50 +00001961
1962 unsigned short
1963 id,
1964 short_sans;
1965
1966 length=GetStringInfoLength(bim_profile);
1967 if (length < 16)
cristy0b796e62010-04-29 00:38:45 +00001968 return;
cristyd4d3f742010-04-25 20:36:50 +00001969 datum=GetStringInfoDatum(bim_profile);
1970 for (p=datum; (p >= datum) && (p < (datum+length-16)); )
1971 {
cristy4176bb22010-05-01 16:29:09 +00001972 register unsigned char
1973 *q;
1974
1975 q=(unsigned char *) p;
cristyd4d3f742010-04-25 20:36:50 +00001976 if (LocaleNCompare((const char *) p,"8BIM",4) != 0)
1977 break;
cristy6befb0f2010-05-31 14:33:15 +00001978 p=PushLongPixel(MSBEndian,p,&long_sans);
cristyd4d3f742010-04-25 20:36:50 +00001979 p=PushShortPixel(MSBEndian,p,&id);
1980 p=PushShortPixel(MSBEndian,p,&short_sans);
1981 p=PushLongPixel(MSBEndian,p,&count);
1982 if (id == 0x0000040f)
1983 {
cristy4176bb22010-05-01 16:29:09 +00001984 (void) CopyMagickMemory(q,q+PSDQuantum(count)+12,length-
1985 (PSDQuantum(count)+12)-(q-datum));
1986 SetStringInfoLength(bim_profile,length-(PSDQuantum(count)+12));
cristyd4d3f742010-04-25 20:36:50 +00001987 break;
1988 }
1989 p+=count;
1990 if ((count & 0x01) != 0)
1991 p++;
1992 }
1993}
1994
cristyf11065e2010-05-14 13:26:59 +00001995static void RemoveResolutionFromResourceBlock(StringInfo *bim_profile)
1996{
1997 register const unsigned char
1998 *p;
1999
2000 size_t
2001 length;
2002
2003 unsigned char
2004 *datum;
2005
cristy6befb0f2010-05-31 14:33:15 +00002006 unsigned int
cristyf11065e2010-05-14 13:26:59 +00002007 count,
cristy6befb0f2010-05-31 14:33:15 +00002008 long_sans;
cristyf11065e2010-05-14 13:26:59 +00002009
2010 unsigned short
2011 id,
2012 short_sans;
2013
2014 length=GetStringInfoLength(bim_profile);
2015 if (length < 16)
2016 return;
2017 datum=GetStringInfoDatum(bim_profile);
2018 for (p=datum; (p >= datum) && (p < (datum+length-16)); )
2019 {
2020 register unsigned char
2021 *q;
2022
2023 q=(unsigned char *) p;
2024 if (LocaleNCompare((const char *) p,"8BIM",4) != 0)
2025 break;
cristy6befb0f2010-05-31 14:33:15 +00002026 p=PushLongPixel(MSBEndian,p,&long_sans);
cristyf11065e2010-05-14 13:26:59 +00002027 p=PushShortPixel(MSBEndian,p,&id);
2028 p=PushShortPixel(MSBEndian,p,&short_sans);
2029 p=PushLongPixel(MSBEndian,p,&count);
2030 if (id == 0x000003ed)
2031 {
2032 (void) CopyMagickMemory(q,q+PSDQuantum(count)+12,length-
2033 (PSDQuantum(count)+12)-(q-datum));
2034 SetStringInfoLength(bim_profile,length-(PSDQuantum(count)+12));
2035 break;
2036 }
2037 p+=count;
2038 if ((count & 0x01) != 0)
2039 p++;
2040 }
2041}
2042
cristy3ed852e2009-09-05 21:47:34 +00002043static MagickBooleanType WritePSDImage(const ImageInfo *image_info,Image *image)
2044{
2045 const char
cristya20214e2010-08-28 16:52:13 +00002046 *property;
cristy3ed852e2009-09-05 21:47:34 +00002047
2048 const StringInfo
cristy749d2152010-04-04 23:47:33 +00002049 *icc_profile;
cristy3ed852e2009-09-05 21:47:34 +00002050
cristya20214e2010-08-28 16:52:13 +00002051 Image
2052 *base_image,
2053 *next_image;
2054
cristy3ed852e2009-09-05 21:47:34 +00002055 MagickBooleanType
cristy3ed852e2009-09-05 21:47:34 +00002056 status;
2057
cristy875e28a2010-03-06 19:46:55 +00002058 PSDInfo
2059 psd_info;
2060
cristybb503372010-05-27 20:51:26 +00002061 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00002062 i;
2063
2064 size_t
cristya20214e2010-08-28 16:52:13 +00002065 channel_size,
2066 channelLength,
2067 layer_count,
2068 layer_info_size,
cristy749d2152010-04-04 23:47:33 +00002069 length,
cristy3ed852e2009-09-05 21:47:34 +00002070 num_channels,
cristya20214e2010-08-28 16:52:13 +00002071 packet_size,
2072 rounded_layer_info_size;
cristy3ed852e2009-09-05 21:47:34 +00002073
cristy0b796e62010-04-29 00:38:45 +00002074 StringInfo
2075 *bim_profile;
2076
cristy3ed852e2009-09-05 21:47:34 +00002077 unsigned char
2078 layer_name[4];
2079
cristy3ed852e2009-09-05 21:47:34 +00002080 /*
cristy56ed31c2010-03-22 00:46:21 +00002081 Open image file.
cristy3ed852e2009-09-05 21:47:34 +00002082 */
2083 assert(image_info != (const ImageInfo *) NULL);
2084 assert(image_info->signature == MagickSignature);
2085 assert(image != (Image *) NULL);
2086 assert(image->signature == MagickSignature);
2087 if (image->debug != MagickFalse)
2088 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
2089 status=OpenBlob(image_info,image,WriteBinaryBlobMode,&image->exception);
2090 if (status == MagickFalse)
2091 return(status);
2092 packet_size=(size_t) (image->depth > 8 ? 6 : 3);
2093 if (image->matte != MagickFalse)
2094 packet_size+=image->depth > 8 ? 2 : 1;
cristy875e28a2010-03-06 19:46:55 +00002095 psd_info.version=1;
2096 if ((LocaleCompare(image_info->magick,"PSB") == 0) ||
2097 (image->columns > 30000) || (image->rows > 30000))
2098 psd_info.version=2;
cristy50aea4a2010-03-09 17:37:44 +00002099 (void) WriteBlob(image,4,(const unsigned char *) "8BPS");
cristy875e28a2010-03-06 19:46:55 +00002100 (void) WriteBlobMSBShort(image,psd_info.version); /* version */
2101 for (i=1; i <= 6; i++)
2102 (void) WriteBlobByte(image, 0); /* 6 bytes of reserved */
cristy2045da32010-04-16 00:59:35 +00002103 if (IsGrayImage(image,&image->exception) != MagickFalse)
cristy6886a752010-04-23 18:23:20 +00002104 num_channels=(image->matte != MagickFalse ? 2UL : 1UL);
cristy3ed852e2009-09-05 21:47:34 +00002105 else
cristy0910f242010-04-01 18:55:09 +00002106 if (image->storage_class == PseudoClass)
cristy6886a752010-04-23 18:23:20 +00002107 num_channels=(image->matte != MagickFalse ? 2UL : 1UL);
cristy0910f242010-04-01 18:55:09 +00002108 else
2109 {
2110 if (image->colorspace != CMYKColorspace)
cristy6886a752010-04-23 18:23:20 +00002111 num_channels=(image->matte != MagickFalse ? 4UL : 3UL);
cristy0910f242010-04-01 18:55:09 +00002112 else
cristy6886a752010-04-23 18:23:20 +00002113 num_channels=(image->matte != MagickFalse ? 5UL : 4UL);
cristy0910f242010-04-01 18:55:09 +00002114 }
cristy3ed852e2009-09-05 21:47:34 +00002115 (void) WriteBlobMSBShort(image,(unsigned short) num_channels);
cristy56ed31c2010-03-22 00:46:21 +00002116 (void) WriteBlobMSBLong(image,(unsigned int) image->rows);
2117 (void) WriteBlobMSBLong(image,(unsigned int) image->columns);
cristy2045da32010-04-16 00:59:35 +00002118 if (IsGrayImage(image,&image->exception) != MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00002119 {
cristy2045da32010-04-16 00:59:35 +00002120 MagickBooleanType
2121 monochrome;
2122
cristy0910f242010-04-01 18:55:09 +00002123 /*
2124 Write depth & mode.
2125 */
cristy0b55dd32010-06-13 19:04:44 +00002126 monochrome=IsMonochromeImage(image,&image->exception) &&
cristybf9f4ba2010-08-20 18:13:29 +00002127 (image->depth == 1) ? MagickTrue : MagickFalse;
cristy284c7d82010-04-24 00:19:14 +00002128 (void) WriteBlobMSBShort(image,(unsigned short)
2129 (monochrome != MagickFalse ? 1 : image->depth > 8 ? 16 : 8));
cristya20214e2010-08-28 16:52:13 +00002130 (void) WriteBlobMSBShort(image,monochrome != MagickFalse ? BitmapMode :
2131 GrayscaleMode);
cristy3ed852e2009-09-05 21:47:34 +00002132 }
2133 else
2134 {
cristya20214e2010-08-28 16:52:13 +00002135 (void) WriteBlobMSBShort(image,(unsigned short) (image->storage_class ==
2136 PseudoClass ? 8 : image->depth > 8 ? 16 : 8));
cristy48845392010-06-02 01:19:17 +00002137 if (((image_info->colorspace != UndefinedColorspace) ||
cristy0910f242010-04-01 18:55:09 +00002138 (image->colorspace != CMYKColorspace)) &&
cristy48845392010-06-02 01:19:17 +00002139 (image_info->colorspace != CMYKColorspace))
cristy0910f242010-04-01 18:55:09 +00002140 {
2141 if (image->colorspace != RGBColorspace)
2142 (void) TransformImageColorspace(image,RGBColorspace);
2143 (void) WriteBlobMSBShort(image,(unsigned short)
cristy2045da32010-04-16 00:59:35 +00002144 (image->storage_class == PseudoClass ? IndexedMode : RGBMode));
cristy0910f242010-04-01 18:55:09 +00002145 }
2146 else
2147 {
cristy48845392010-06-02 01:19:17 +00002148 if (image->colorspace != CMYKColorspace)
cristy0910f242010-04-01 18:55:09 +00002149 (void) TransformImageColorspace(image,CMYKColorspace);
cristy2045da32010-04-16 00:59:35 +00002150 (void) WriteBlobMSBShort(image,CMYKMode);
cristy0910f242010-04-01 18:55:09 +00002151 }
cristy3ed852e2009-09-05 21:47:34 +00002152 }
cristyca9ddb02010-04-16 01:01:18 +00002153 if ((IsGrayImage(image,&image->exception) != MagickFalse) ||
2154 (image->storage_class == DirectClass) || (image->colors > 256))
cristy3ed852e2009-09-05 21:47:34 +00002155 (void) WriteBlobMSBLong(image,0);
2156 else
2157 {
2158 /*
2159 Write PSD raster colormap.
2160 */
2161 (void) WriteBlobMSBLong(image,768);
cristybb503372010-05-27 20:51:26 +00002162 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +00002163 (void) WriteBlobByte(image,ScaleQuantumToChar(image->colormap[i].red));
2164 for ( ; i < 256; i++)
2165 (void) WriteBlobByte(image,0);
cristybb503372010-05-27 20:51:26 +00002166 for (i=0; i < (ssize_t) image->colors; i++)
cristya20214e2010-08-28 16:52:13 +00002167 (void) WriteBlobByte(image,ScaleQuantumToChar(
2168 image->colormap[i].green));
cristy3ed852e2009-09-05 21:47:34 +00002169 for ( ; i < 256; i++)
2170 (void) WriteBlobByte(image,0);
cristybb503372010-05-27 20:51:26 +00002171 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +00002172 (void) WriteBlobByte(image,ScaleQuantumToChar(image->colormap[i].blue));
2173 for ( ; i < 256; i++)
2174 (void) WriteBlobByte(image,0);
2175 }
2176 /*
2177 Image resource block.
2178 */
cristy749d2152010-04-04 23:47:33 +00002179 length=28; /* 0x03EB */
cristy0b796e62010-04-29 00:38:45 +00002180 bim_profile=(StringInfo *) GetImageProfile(image,"8bim");
cristy749d2152010-04-04 23:47:33 +00002181 icc_profile=GetImageProfile(image,"icc");
cristyd4d3f742010-04-25 20:36:50 +00002182 if (bim_profile != (StringInfo *) NULL)
2183 {
cristy0b796e62010-04-29 00:38:45 +00002184 bim_profile=CloneStringInfo(bim_profile);
cristyd4d3f742010-04-25 20:36:50 +00002185 if (icc_profile != (StringInfo *) NULL)
2186 RemoveICCProfileFromResourceBlock(bim_profile);
cristyf11065e2010-05-14 13:26:59 +00002187 RemoveResolutionFromResourceBlock(bim_profile);
cristyd4d3f742010-04-25 20:36:50 +00002188 length+=PSDQuantum(GetStringInfoLength(bim_profile));
2189 }
cristy0b796e62010-04-29 00:38:45 +00002190 if (icc_profile != (const StringInfo *) NULL)
cristy749d2152010-04-04 23:47:33 +00002191 length+=PSDQuantum(GetStringInfoLength(icc_profile))+12;
cristy284c7d82010-04-24 00:19:14 +00002192 (void) WriteBlobMSBLong(image,(unsigned int) length);
cristyf11065e2010-05-14 13:26:59 +00002193 WriteResolutionResourceBlock(image);
cristy4176bb22010-05-01 16:29:09 +00002194 if (bim_profile != (StringInfo *) NULL)
2195 {
2196 (void) WriteBlob(image,GetStringInfoLength(bim_profile),
2197 GetStringInfoDatum(bim_profile));
2198 bim_profile=DestroyStringInfo(bim_profile);
2199 }
cristy749d2152010-04-04 23:47:33 +00002200 if (icc_profile != (StringInfo *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00002201 {
cristy749d2152010-04-04 23:47:33 +00002202 (void) WriteBlob(image,4,(const unsigned char *) "8BIM");
cristy4176bb22010-05-01 16:29:09 +00002203 (void) WriteBlobMSBShort(image,0x0000040F);
cristy749d2152010-04-04 23:47:33 +00002204 (void) WriteBlobMSBShort(image,0);
cristy284c7d82010-04-24 00:19:14 +00002205 (void) WriteBlobMSBLong(image,(unsigned int) GetStringInfoLength(
2206 icc_profile));
cristy749d2152010-04-04 23:47:33 +00002207 (void) WriteBlob(image,GetStringInfoLength(icc_profile),
2208 GetStringInfoDatum(icc_profile));
cristye195f262010-04-16 18:12:35 +00002209 if ((MagickOffsetType) GetStringInfoLength(icc_profile) !=
cristy2045da32010-04-16 00:59:35 +00002210 PSDQuantum(GetStringInfoLength(icc_profile)))
cristy749d2152010-04-04 23:47:33 +00002211 (void) WriteBlobByte(image,0);
cristye6365592010-04-02 17:31:23 +00002212 }
cristy48845392010-06-02 01:19:17 +00002213 layer_count=0;
2214 layer_info_size=2;
cristy2837bcc2010-08-07 23:57:39 +00002215 base_image=GetNextImageInList(image);
2216 if ((image->matte != MagickFalse) && (base_image == (Image *) NULL))
2217 base_image=image;
cristya20214e2010-08-28 16:52:13 +00002218 next_image=base_image;
2219 while ( next_image != NULL )
2220 {
2221 packet_size=next_image->depth > 8 ? 2UL : 1UL;
2222 if (IsGrayImage(next_image,&image->exception) != MagickFalse)
2223 num_channels=next_image->matte != MagickFalse ? 2UL : 1UL;
cristy3ed852e2009-09-05 21:47:34 +00002224 else
cristya20214e2010-08-28 16:52:13 +00002225 if (next_image->storage_class == PseudoClass)
2226 num_channels=next_image->matte != MagickFalse ? 2UL : 1UL;
cristy2045da32010-04-16 00:59:35 +00002227 else
cristya20214e2010-08-28 16:52:13 +00002228 if (next_image->colorspace != CMYKColorspace)
2229 num_channels=next_image->matte != MagickFalse ? 4UL : 3UL;
cristy2045da32010-04-16 00:59:35 +00002230 else
cristya20214e2010-08-28 16:52:13 +00002231 num_channels=next_image->matte != MagickFalse ? 5UL : 4UL;
2232 channelLength=(size_t) (next_image->columns*next_image->rows*packet_size+2);
cristy48845392010-06-02 01:19:17 +00002233 layer_info_size+=(size_t) (4*4+2+num_channels*6+(psd_info.version == 1 ? 8 :
2234 16)+4*1+4+num_channels*channelLength);
cristya20214e2010-08-28 16:52:13 +00002235 property=(const char *) GetImageProperty(next_image,"label");
2236 if (property == (const char *) NULL)
2237 layer_info_size+=16;
cristyde9b8f52010-03-12 01:17:00 +00002238 else
2239 {
cristya20214e2010-08-28 16:52:13 +00002240 size_t
2241 length;
2242
2243 length=strlen(property);
2244 layer_info_size+=8+length+(4-(length % 4));
cristyde9b8f52010-03-12 01:17:00 +00002245 }
cristy4aff1572010-02-15 13:34:01 +00002246 layer_count++;
cristya20214e2010-08-28 16:52:13 +00002247 next_image=GetNextImageInList(next_image);
cristy3ed852e2009-09-05 21:47:34 +00002248 }
cristy144f1b62010-05-18 00:52:09 +00002249 if (layer_count == 0)
cristy875e28a2010-03-06 19:46:55 +00002250 (void) SetPSDSize(&psd_info,image,0);
cristy3ed852e2009-09-05 21:47:34 +00002251 else
cristya20214e2010-08-28 16:52:13 +00002252 {
cristy8e9bd3e2010-08-29 00:03:56 +00002253 CompressionType
2254 compression;
2255
2256 (void) SetPSDSize(&psd_info,image,layer_info_size+
2257 (psd_info.version == 1 ? 8 : 16));
2258 if ((layer_info_size/2) != ((layer_info_size+1)/2))
2259 rounded_layer_info_size=layer_info_size+1;
cristya20214e2010-08-28 16:52:13 +00002260 else
cristy8e9bd3e2010-08-29 00:03:56 +00002261 rounded_layer_info_size=layer_info_size;
2262 (void) SetPSDSize(&psd_info,image,rounded_layer_info_size);
2263 (void) WriteBlobMSBShort(image,(unsigned short) layer_count);
2264 layer_count=1;
2265 compression=base_image->compression;
2266 next_image=base_image;
2267 while (next_image != NULL)
2268 {
2269 next_image->compression=NoCompression;
2270 (void) WriteBlobMSBLong(image,0);
2271 (void) WriteBlobMSBLong(image,0);
2272 (void) WriteBlobMSBLong(image,(unsigned int) next_image->rows);
2273 (void) WriteBlobMSBLong(image,(unsigned int) next_image->columns);
2274 packet_size=next_image->depth > 8 ? 2UL : 1UL;
2275 channel_size=(unsigned int) ((packet_size*next_image->rows*
2276 next_image->columns)+2);
2277 if ((IsGrayImage(next_image,&image->exception) != MagickFalse) ||
2278 (next_image->storage_class == PseudoClass))
2279 {
2280 (void) WriteBlobMSBShort(image,(unsigned short)
2281 (next_image->matte != MagickFalse ? 2 : 1));
2282 (void) WriteBlobMSBShort(image,0);
2283 (void) SetPSDSize(&psd_info,image,channel_size);
2284 if (next_image->matte != MagickFalse)
2285 {
2286 (void) WriteBlobMSBShort(image,(unsigned short) -1);
2287 (void) SetPSDSize(&psd_info,image,channel_size);
2288 }
2289 }
2290 else
2291 if (next_image->colorspace != CMYKColorspace)
2292 {
2293 (void) WriteBlobMSBShort(image,(unsigned short)
2294 (next_image->matte != MagickFalse ? 4 : 3));
2295 (void) WriteBlobMSBShort(image,0);
2296 (void) SetPSDSize(&psd_info,image,channel_size);
2297 (void) WriteBlobMSBShort(image,1);
2298 (void) SetPSDSize(&psd_info,image,channel_size);
2299 (void) WriteBlobMSBShort(image,2);
2300 (void) SetPSDSize(&psd_info,image,channel_size);
2301 if (next_image->matte!= MagickFalse )
2302 {
2303 (void) WriteBlobMSBShort(image,(unsigned short) -1);
2304 (void) SetPSDSize(&psd_info,image,channel_size);
2305 }
2306 }
2307 else
2308 {
2309 (void) WriteBlobMSBShort(image,(unsigned short)
2310 (next_image->matte ? 5 : 4));
2311 (void) WriteBlobMSBShort(image,0);
2312 (void) SetPSDSize(&psd_info,image,channel_size);
2313 (void) WriteBlobMSBShort(image,1);
2314 (void) SetPSDSize(&psd_info,image,channel_size);
2315 (void) WriteBlobMSBShort(image,2);
2316 (void) SetPSDSize(&psd_info,image,channel_size);
2317 (void) WriteBlobMSBShort(image,3);
2318 (void) SetPSDSize(&psd_info,image,channel_size);
2319 if (next_image->matte)
2320 {
2321 (void) WriteBlobMSBShort(image,(unsigned short) -1);
2322 (void) SetPSDSize(&psd_info,image,channel_size);
2323 }
2324 }
2325 (void) WriteBlob(image,4,(const unsigned char *) "8BIM");
2326 (void) WriteBlob(image,4,(const unsigned char *)
2327 CompositeOperatorToPSDBlendMode(next_image->compose));
2328 (void) WriteBlobByte(image,255); /* layer opacity */
2329 (void) WriteBlobByte(image,0);
2330 (void) WriteBlobByte(image,1); /* layer propertys - visible, etc. */
2331 (void) WriteBlobByte(image,0);
2332 property=(const char *) GetImageProperty(next_image,"label");
2333 if (property == (const char *) NULL)
2334 {
2335 (void) WriteBlobMSBLong(image,16);
2336 (void) WriteBlobMSBLong(image,0);
2337 (void) WriteBlobMSBLong(image,0);
2338 (void) FormatMagickString((char *) layer_name,MaxTextExtent,
2339 "L%06ld",(long) layer_count++);
2340 WritePascalString( image, (char*)layer_name, 4 );
2341 }
2342 else
2343 {
2344 size_t
2345 length;
cristy3ed852e2009-09-05 21:47:34 +00002346
cristy8e9bd3e2010-08-29 00:03:56 +00002347 length=strlen(property);
2348 (void) WriteBlobMSBLong(image,(unsigned int) (length+(4-
2349 (length % 4))+8));
2350 (void) WriteBlobMSBLong(image,0);
2351 (void) WriteBlobMSBLong(image,0);
2352 WritePascalString(image,property,4);
2353 }
2354 next_image=GetNextImageInList(next_image);
2355 }
2356 /*
2357 Now the image data!
2358 */
2359 next_image=base_image;
2360 while (next_image != NULL)
2361 {
2362 status=WriteImageChannels(&psd_info,image_info,image,next_image,
2363 MagickTrue);
2364 next_image=GetNextImageInList(next_image);
2365 }
2366 (void) WriteBlobMSBLong(image,0); /* user mask data */
2367 base_image->compression=compression;
cristy144f1b62010-05-18 00:52:09 +00002368 }
cristy144f1b62010-05-18 00:52:09 +00002369 /*
2370 Write composite image.
2371 */
cristy15e25cb2010-03-13 20:31:29 +00002372 status=WriteImageChannels(&psd_info,image_info,image,image,MagickFalse);
cristy3ed852e2009-09-05 21:47:34 +00002373 (void) CloseBlob(image);
2374 return(status);
2375}