blob: 9739045b2531a93c1834a2778b08bc9627e5bc15 [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"
cristy802d3642011-04-27 02:02:41 +000062#include "magick/pixel.h"
cristy3ed852e2009-09-05 21:47:34 +000063#include "magick/profile.h"
64#include "magick/property.h"
65#include "magick/quantum-private.h"
66#include "magick/static.h"
67#include "magick/string_.h"
68
69/*
cristy2d3d87f2010-03-01 00:23:08 +000070 Define declaractions.
71*/
cristyaca3ec52010-03-02 14:55:01 +000072#define MaxPSDChannels 56
cristybb503372010-05-27 20:51:26 +000073#define PSDQuantum(x) (((ssize_t) (x)+1) & -2)
cristy2d3d87f2010-03-01 00:23:08 +000074
75/*
76 Enumerated declaractions.
77*/
78typedef enum
79{
80 BitmapMode = 0,
81 GrayscaleMode = 1,
82 IndexedMode = 2,
83 RGBMode = 3,
84 CMYKMode = 4,
85 MultichannelMode = 7,
86 DuotoneMode = 8,
87 LabMode = 9
88} PSDImageType;
89
90/*
91 Typedef declaractions.
92*/
93typedef struct _ChannelInfo
94{
95 short int
96 type;
97
cristybb503372010-05-27 20:51:26 +000098 size_t
cristy2d3d87f2010-03-01 00:23:08 +000099 size;
100} ChannelInfo;
101
102typedef struct _LayerInfo
103{
104 RectangleInfo
105 page,
106 mask;
107
108 unsigned short
109 channels;
110
111 ChannelInfo
112 channel_info[MaxPSDChannels];
113
114 char
115 blendkey[4];
116
117 Quantum
118 opacity;
119
120 unsigned char
121 clipping,
122 visible,
123 flags;
124
cristybb503372010-05-27 20:51:26 +0000125 size_t
cristy2d3d87f2010-03-01 00:23:08 +0000126 offset_x,
127 offset_y;
128
129 unsigned char
130 name[256];
131
132 Image
133 *image;
134} LayerInfo;
135
136typedef struct _PSDInfo
137{
138 char
139 signature[4];
140
141 unsigned short
142 channels,
143 version;
144
145 unsigned char
146 reserved[6];
147
cristybb503372010-05-27 20:51:26 +0000148 size_t
cristy2d3d87f2010-03-01 00:23:08 +0000149 rows,
150 columns;
151
152 unsigned short
153 depth,
154 mode;
155} PSDInfo;
156
157/*
cristy3ed852e2009-09-05 21:47:34 +0000158 Forward declarations.
159*/
160static MagickBooleanType
161 WritePSDImage(const ImageInfo *,Image *);
162
163/*
164%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
165% %
166% %
167% %
cristy3ed852e2009-09-05 21:47:34 +0000168% I s P S D %
169% %
170% %
171% %
172%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
173%
174% IsPSD()() returns MagickTrue if the image format type, identified by the
175% magick string, is PSD.
176%
177% The format of the IsPSD method is:
178%
179% MagickBooleanType IsPSD(const unsigned char *magick,const size_t length)
180%
181% A description of each parameter follows:
182%
183% o magick: compare image format pattern against these bytes.
184%
185% o length: Specifies the length of the magick string.
186%
187*/
188static MagickBooleanType IsPSD(const unsigned char *magick,const size_t length)
189{
190 if (length < 4)
191 return(MagickFalse);
192 if (LocaleNCompare((const char *) magick,"8BPS",4) == 0)
193 return(MagickTrue);
194 return(MagickFalse);
195}
196
197/*
198%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
199% %
200% %
201% %
202% R e a d P S D I m a g e %
203% %
204% %
205% %
206%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
207%
208% ReadPSDImage() reads an Adobe Photoshop image file and returns it. It
209% allocates the memory necessary for the new Image structure and returns a
210% pointer to the new image.
211%
212% The format of the ReadPSDImage method is:
213%
cristycd081772010-03-22 17:19:12 +0000214% Image *ReadPSDImage(image_info)
cristy3ed852e2009-09-05 21:47:34 +0000215%
216% A description of each parameter follows:
217%
218% o image_info: the image info.
219%
220% o exception: return any errors or warnings in this structure.
221%
222*/
223
cristy19eb6412010-04-23 14:42:29 +0000224static const char *CompositeOperatorToPSDBlendMode(CompositeOperator op)
cristycd081772010-03-22 17:19:12 +0000225{
226 const char
227 *blend_mode;
228
cristy19eb6412010-04-23 14:42:29 +0000229 switch (op)
cristycd081772010-03-22 17:19:12 +0000230 {
231 case OverCompositeOp: blend_mode = "norm"; break;
232 case MultiplyCompositeOp: blend_mode = "mul "; break;
233 case DissolveCompositeOp: blend_mode = "diss"; break;
234 case DifferenceCompositeOp: blend_mode = "diff"; break;
235 case DarkenCompositeOp: blend_mode = "dark"; break;
236 case LightenCompositeOp: blend_mode = "lite"; break;
237 case HueCompositeOp: blend_mode = "hue "; break;
238 case SaturateCompositeOp: blend_mode = "sat "; break;
239 case ColorizeCompositeOp: blend_mode = "colr"; break;
240 case LuminizeCompositeOp: blend_mode = "lum "; break;
241 case ScreenCompositeOp: blend_mode = "scrn"; break;
242 case OverlayCompositeOp: blend_mode = "over"; break;
243 default:
244 blend_mode = "norm";
245 }
246 return(blend_mode);
247}
248
249static ssize_t DecodePSDPixels(const size_t number_compact_pixels,
cristybb503372010-05-27 20:51:26 +0000250 const unsigned char *compact_pixels,const ssize_t depth,
cristycd081772010-03-22 17:19:12 +0000251 const size_t number_pixels,unsigned char *pixels)
252{
253 int
254 pixel;
255
256 register ssize_t
257 i,
258 j;
259
cristycd081772010-03-22 17:19:12 +0000260 size_t
261 length;
262
cristy802d3642011-04-27 02:02:41 +0000263 ssize_t
264 packets;
265
cristycd081772010-03-22 17:19:12 +0000266 packets=(ssize_t) number_compact_pixels;
267 for (i=0; (packets > 1) && (i < (ssize_t) number_pixels); )
268 {
269 length=(*compact_pixels++);
270 packets--;
271 if (length == 128)
272 continue;
273 if (length > 128)
274 {
275 length=256-length+1;
276 pixel=(*compact_pixels++);
277 packets--;
cristy284c7d82010-04-24 00:19:14 +0000278 for (j=0; j < (ssize_t) length; j++)
cristycd081772010-03-22 17:19:12 +0000279 {
280 switch (depth)
281 {
282 case 1:
283 {
cristy284c7d82010-04-24 00:19:14 +0000284 *pixels++=(pixel >> 7) & 0x01 ? 0U : 255U;
285 *pixels++=(pixel >> 6) & 0x01 ? 0U : 255U;
286 *pixels++=(pixel >> 5) & 0x01 ? 0U : 255U;
287 *pixels++=(pixel >> 4) & 0x01 ? 0U : 255U;
288 *pixels++=(pixel >> 3) & 0x01 ? 0U : 255U;
289 *pixels++=(pixel >> 2) & 0x01 ? 0U : 255U;
290 *pixels++=(pixel >> 1) & 0x01 ? 0U : 255U;
291 *pixels++=(pixel >> 0) & 0x01 ? 0U : 255U;
cristycd081772010-03-22 17:19:12 +0000292 i+=8;
293 break;
294 }
295 case 4:
296 {
cristy284c7d82010-04-24 00:19:14 +0000297 *pixels++=(unsigned char) ((pixel >> 4) & 0xff);
298 *pixels++=(unsigned char) ((pixel & 0x0f) & 0xff);
cristycd081772010-03-22 17:19:12 +0000299 i+=2;
300 break;
301 }
302 case 2:
303 {
cristy284c7d82010-04-24 00:19:14 +0000304 *pixels++=(unsigned char) ((pixel >> 6) & 0x03);
305 *pixels++=(unsigned char) ((pixel >> 4) & 0x03);
306 *pixels++=(unsigned char) ((pixel >> 2) & 0x03);
307 *pixels++=(unsigned char) ((pixel & 0x03) & 0x03);
cristycd081772010-03-22 17:19:12 +0000308 i+=4;
309 break;
310 }
311 default:
312 {
cristy284c7d82010-04-24 00:19:14 +0000313 *pixels++=(unsigned char) pixel;
cristycd081772010-03-22 17:19:12 +0000314 i++;
315 break;
316 }
317 }
318 }
319 continue;
320 }
321 length++;
cristy284c7d82010-04-24 00:19:14 +0000322 for (j=0; j < (ssize_t) length; j++)
cristycd081772010-03-22 17:19:12 +0000323 {
324 switch (depth)
325 {
326 case 1:
327 {
cristy284c7d82010-04-24 00:19:14 +0000328 *pixels++=(*compact_pixels >> 7) & 0x01 ? 0U : 255U;
329 *pixels++=(*compact_pixels >> 6) & 0x01 ? 0U : 255U;
330 *pixels++=(*compact_pixels >> 5) & 0x01 ? 0U : 255U;
331 *pixels++=(*compact_pixels >> 4) & 0x01 ? 0U : 255U;
332 *pixels++=(*compact_pixels >> 3) & 0x01 ? 0U : 255U;
333 *pixels++=(*compact_pixels >> 2) & 0x01 ? 0U : 255U;
334 *pixels++=(*compact_pixels >> 1) & 0x01 ? 0U : 255U;
335 *pixels++=(*compact_pixels >> 0) & 0x01 ? 0U : 255U;
cristycd081772010-03-22 17:19:12 +0000336 i+=8;
337 break;
338 }
339 case 4:
340 {
cristy618a9662010-03-23 13:59:54 +0000341 *pixels++=(*compact_pixels >> 4) & 0xff;
342 *pixels++=(*compact_pixels & 0x0f) & 0xff;
cristycd081772010-03-22 17:19:12 +0000343 i+=2;
344 break;
345 }
346 case 2:
347 {
cristy618a9662010-03-23 13:59:54 +0000348 *pixels++=(*compact_pixels >> 6) & 0x03;
349 *pixels++=(*compact_pixels >> 4) & 0x03;
350 *pixels++=(*compact_pixels >> 2) & 0x03;
351 *pixels++=(*compact_pixels & 0x03) & 0x03;
cristycd081772010-03-22 17:19:12 +0000352 i+=4;
353 break;
354 }
355 default:
356 {
357 *pixels++=(*compact_pixels);
358 i++;
359 break;
360 }
361 }
362 compact_pixels++;
363 }
364 }
365 return(i);
366}
367
cristy2d3d87f2010-03-01 00:23:08 +0000368static inline MagickOffsetType GetPSDOffset(PSDInfo *psd_info,Image *image)
369{
370 if (psd_info->version == 1)
371 return((MagickOffsetType) ReadBlobMSBShort(image));
372 return((MagickOffsetType) ReadBlobMSBLong(image));
373}
374
375static inline MagickSizeType GetPSDSize(PSDInfo *psd_info,Image *image)
376{
377 if (psd_info->version == 1)
378 return((MagickSizeType) ReadBlobMSBLong(image));
379 return((MagickSizeType) ReadBlobMSBLongLong(image));
380}
381
cristybb503372010-05-27 20:51:26 +0000382static inline ssize_t MagickAbsoluteValue(const ssize_t x)
cristy3ed852e2009-09-05 21:47:34 +0000383{
384 if (x < 0)
385 return(-x);
386 return(x);
387}
388
cristycd081772010-03-22 17:19:12 +0000389static const char *ModeToString(PSDImageType type)
cristy3ed852e2009-09-05 21:47:34 +0000390{
cristycd081772010-03-22 17:19:12 +0000391 switch (type)
cristy3ed852e2009-09-05 21:47:34 +0000392 {
393 case BitmapMode: return "Bitmap";
394 case GrayscaleMode: return "Grayscale";
395 case IndexedMode: return "Indexed";
396 case RGBMode: return "RGB";
397 case CMYKMode: return "CMYK";
398 case MultichannelMode: return "Multichannel";
399 case DuotoneMode: return "Duotone";
400 case LabMode: return "L*A*B";
401 default: return "unknown";
402 }
403}
404
405static MagickBooleanType ParseImageResourceBlocks(Image *image,
406 const unsigned char *blocks,size_t length)
407{
408 const unsigned char
409 *p;
410
411 StringInfo
412 *profile;
413
cristy6befb0f2010-05-31 14:33:15 +0000414 unsigned int
cristy3ed852e2009-09-05 21:47:34 +0000415 count,
cristy6befb0f2010-05-31 14:33:15 +0000416 long_sans;
cristy3ed852e2009-09-05 21:47:34 +0000417
418 unsigned short
419 id,
420 short_sans;
421
422 if (length < 16)
423 return(MagickFalse);
424 profile=AcquireStringInfo(length);
425 SetStringInfoDatum(profile,blocks);
426 (void) SetImageProfile(image,"8bim",profile);
427 profile=DestroyStringInfo(profile);
428 for (p=blocks; (p >= blocks) && (p < (blocks+length-16)); )
429 {
430 if (LocaleNCompare((const char *) p,"8BIM",4) != 0)
431 break;
cristy6befb0f2010-05-31 14:33:15 +0000432 p=PushLongPixel(MSBEndian,p,&long_sans);
cristyf11065e2010-05-14 13:26:59 +0000433 p=PushShortPixel(MSBEndian,p,&id);
434 p=PushShortPixel(MSBEndian,p,&short_sans);
435 p=PushLongPixel(MSBEndian,p,&count);
cristy3ed852e2009-09-05 21:47:34 +0000436 switch (id)
437 {
438 case 0x03ed:
439 {
cristy1e4a80b2010-05-14 16:18:26 +0000440 char
441 value[MaxTextExtent];
442
cristy3ed852e2009-09-05 21:47:34 +0000443 unsigned short
444 resolution;
445
446 /*
447 Resolution info.
448 */
cristyf11065e2010-05-14 13:26:59 +0000449 p=PushShortPixel(MSBEndian,p,&resolution);
cristy3ed852e2009-09-05 21:47:34 +0000450 image->x_resolution=(double) resolution;
cristyb51dff52011-05-19 16:55:47 +0000451 (void) FormatLocaleString(value,MaxTextExtent,"%g",image->x_resolution);
cristy1e4a80b2010-05-14 16:18:26 +0000452 (void) SetImageProperty(image,"tiff:XResolution",value);
cristyf11065e2010-05-14 13:26:59 +0000453 p=PushShortPixel(MSBEndian,p,&short_sans);
454 p=PushShortPixel(MSBEndian,p,&short_sans);
455 p=PushShortPixel(MSBEndian,p,&short_sans);
456 p=PushShortPixel(MSBEndian,p,&resolution);
cristy3ed852e2009-09-05 21:47:34 +0000457 image->y_resolution=(double) resolution;
cristyb51dff52011-05-19 16:55:47 +0000458 (void) FormatLocaleString(value,MaxTextExtent,"%g",image->y_resolution);
cristy1e4a80b2010-05-14 16:18:26 +0000459 (void) SetImageProperty(image,"tiff:YResolution",value);
cristyf11065e2010-05-14 13:26:59 +0000460 p=PushShortPixel(MSBEndian,p,&short_sans);
461 p=PushShortPixel(MSBEndian,p,&short_sans);
462 p=PushShortPixel(MSBEndian,p,&short_sans);
cristy3ed852e2009-09-05 21:47:34 +0000463 break;
464 }
465 default:
466 {
467 p+=count;
468 break;
469 }
470 }
471 if ((count & 0x01) != 0)
472 p++;
473 }
474 return(MagickTrue);
475}
476
cristycd081772010-03-22 17:19:12 +0000477static CompositeOperator PSDBlendModeToCompositeOperator(const char *mode)
cristy56ed31c2010-03-22 00:46:21 +0000478{
cristycd081772010-03-22 17:19:12 +0000479 if (mode == (const char *) NULL)
480 return(OverCompositeOp);
481 if (LocaleNCompare(mode,"norm",4) == 0)
482 return(OverCompositeOp);
483 if (LocaleNCompare(mode,"mul ",4) == 0)
484 return(MultiplyCompositeOp);
485 if (LocaleNCompare(mode,"diss",4) == 0)
486 return(DissolveCompositeOp);
487 if (LocaleNCompare(mode,"diff",4) == 0)
488 return(DifferenceCompositeOp);
489 if (LocaleNCompare(mode,"dark",4) == 0)
490 return(DarkenCompositeOp);
491 if (LocaleNCompare(mode,"lite",4) == 0)
492 return(LightenCompositeOp);
493 if (LocaleNCompare(mode,"hue ",4) == 0)
494 return(HueCompositeOp);
495 if (LocaleNCompare(mode,"sat ",4) == 0)
496 return(SaturateCompositeOp);
497 if (LocaleNCompare(mode,"colr",4) == 0)
498 return(ColorizeCompositeOp);
499 if (LocaleNCompare(mode,"lum ",4) == 0)
500 return(LuminizeCompositeOp);
501 if (LocaleNCompare(mode,"scrn",4) == 0)
502 return(ScreenCompositeOp);
503 if (LocaleNCompare(mode,"over",4) == 0)
504 return(OverlayCompositeOp);
505 if (LocaleNCompare(mode,"hLit",4) == 0)
506 return(OverCompositeOp);
507 if (LocaleNCompare(mode,"sLit",4) == 0)
508 return(OverCompositeOp);
509 if (LocaleNCompare(mode,"smud",4) == 0)
510 return(OverCompositeOp);
511 if (LocaleNCompare(mode,"div ",4) == 0)
512 return(OverCompositeOp);
513 if (LocaleNCompare(mode,"idiv",4) == 0)
514 return(OverCompositeOp);
515 return(OverCompositeOp);
cristy56ed31c2010-03-22 00:46:21 +0000516}
517
cristybb503372010-05-27 20:51:26 +0000518static MagickBooleanType ReadPSDLayer(Image *image,const size_t channels,
519 const ssize_t type,const MagickOffsetType *offsets,ExceptionInfo *exception)
cristy56ed31c2010-03-22 00:46:21 +0000520{
cristyd05dca12010-07-25 02:32:32 +0000521 ColorspaceType
522 colorspace;
cristy56ed31c2010-03-22 00:46:21 +0000523
524 Quantum
525 pixel;
526
cristy7753b2a2011-02-19 18:36:52 +0000527 register const unsigned char
528 *p;
529
cristy56ed31c2010-03-22 00:46:21 +0000530 register IndexPacket
531 *indexes;
532
cristy56ed31c2010-03-22 00:46:21 +0000533 register PixelPacket
534 *q;
535
cristy7753b2a2011-02-19 18:36:52 +0000536 register ssize_t
537 x;
cristy56ed31c2010-03-22 00:46:21 +0000538
539 size_t
540 packet_size;
541
542 ssize_t
cristyd05dca12010-07-25 02:32:32 +0000543 count,
544 y;
cristy56ed31c2010-03-22 00:46:21 +0000545
546 unsigned char
547 *compact_pixels,
548 *pixels;
549
550 unsigned short
551 nibble;
552
553 packet_size=1;
554 if (image->storage_class == PseudoClass)
555 {
556 if (image->colors > 256)
557 packet_size++;
558 else
559 if (image->depth > 8)
560 packet_size++;
561 }
562 else
563 if (image->depth > 8)
564 packet_size++;
cristycd081772010-03-22 17:19:12 +0000565 pixels=(unsigned char *) AcquireQuantumMemory(image->columns+256,packet_size*
cristy56ed31c2010-03-22 00:46:21 +0000566 sizeof(*pixels));
567 if (pixels == (unsigned char *) NULL)
568 ThrowBinaryException(ResourceLimitError,"MemoryAllocationFailed",
569 image->filename);
cristycd081772010-03-22 17:19:12 +0000570 (void) ResetMagickMemory(pixels,0,image->columns*packet_size*sizeof(*pixels));
571 compact_pixels=(unsigned char *) NULL;
cristy56ed31c2010-03-22 00:46:21 +0000572 if (image->compression == RLECompression)
573 {
cristye195f262010-04-16 18:12:35 +0000574 size_t
575 length;
576
577 length=0;
cristybb503372010-05-27 20:51:26 +0000578 for (y=0; y < (ssize_t) image->rows; y++)
cristye195f262010-04-16 18:12:35 +0000579 if ((MagickOffsetType) length < offsets[y])
cristy284c7d82010-04-24 00:19:14 +0000580 length=(size_t) offsets[y];
cristye195f262010-04-16 18:12:35 +0000581 compact_pixels=(unsigned char *) AcquireQuantumMemory(length,
582 sizeof(*pixels));
cristy56ed31c2010-03-22 00:46:21 +0000583 if (compact_pixels == (unsigned char *) NULL)
584 ThrowBinaryException(ResourceLimitError,"MemoryAllocationFailed",
585 image->filename);
cristy0132ba82010-04-16 23:34:02 +0000586 (void) ResetMagickMemory(compact_pixels,0,length*sizeof(*compact_pixels));
cristy56ed31c2010-03-22 00:46:21 +0000587 }
cristyd05dca12010-07-25 02:32:32 +0000588 colorspace=image->colorspace;
cristybb503372010-05-27 20:51:26 +0000589 for (y=0; y < (ssize_t) image->rows; y++)
cristy56ed31c2010-03-22 00:46:21 +0000590 {
cristy3b004082010-08-12 19:56:38 +0000591 if (image->depth == 1)
592 {
cristyb6cabe82011-03-18 13:25:59 +0000593 if (image->compression != RLECompression)
594 count=ReadBlob(image,(image->columns+7)/8,pixels);
595 else
596 {
597 count=ReadBlob(image,(size_t) offsets[y],compact_pixels);
598 if (count != (ssize_t) offsets[y])
599 break;
600 count=DecodePSDPixels((size_t) offsets[y],compact_pixels,
601 (ssize_t) 123456,(size_t) ((image->columns+7)/8),pixels);
602 }
cristy3b004082010-08-12 19:56:38 +0000603 if (count < (ssize_t) ((image->columns+7)/8))
604 break;
605 }
cristy56ed31c2010-03-22 00:46:21 +0000606 else
607 {
cristy3b004082010-08-12 19:56:38 +0000608 if (image->compression != RLECompression)
609 count=ReadBlob(image,packet_size*image->columns,pixels);
610 else
611 {
612 count=ReadBlob(image,(size_t) offsets[y],compact_pixels);
613 if (count != (ssize_t) offsets[y])
614 break;
615 count=DecodePSDPixels((size_t) offsets[y],compact_pixels,
616 (ssize_t) image->depth,packet_size*image->columns,pixels);
617 }
618 if (count < (ssize_t) (packet_size*image->columns))
cristy56ed31c2010-03-22 00:46:21 +0000619 break;
cristy56ed31c2010-03-22 00:46:21 +0000620 }
cristy56ed31c2010-03-22 00:46:21 +0000621 q=GetAuthenticPixels(image,0,y,image->columns,1,exception);
622 if (q == (PixelPacket *) NULL)
623 break;
624 indexes=GetAuthenticIndexQueue(image);
625 p=pixels;
cristybb503372010-05-27 20:51:26 +0000626 for (x=0; x < (ssize_t) image->columns; x++)
cristy56ed31c2010-03-22 00:46:21 +0000627 {
628 if (packet_size == 1)
629 pixel=ScaleCharToQuantum(*p++);
630 else
631 {
632 p=PushShortPixel(MSBEndian,p,&nibble);
633 pixel=ScaleShortToQuantum(nibble);
634 }
635 switch (type)
636 {
637 case -1:
638 {
cristy802d3642011-04-27 02:02:41 +0000639 SetOpacityPixelComponent(q,QuantumRange-pixel);
cristy56ed31c2010-03-22 00:46:21 +0000640 break;
641 }
642 case 0:
643 {
cristy802d3642011-04-27 02:02:41 +0000644 SetRedPixelComponent(q,pixel);
cristy56ed31c2010-03-22 00:46:21 +0000645 if (channels == 1)
646 {
cristy5fde9aa2011-04-23 01:31:53 +0000647 SetGreenPixelComponent(q,GetRedPixelComponent(q));
648 SetBluePixelComponent(q,GetRedPixelComponent(q));
cristy56ed31c2010-03-22 00:46:21 +0000649 }
650 if (image->storage_class == PseudoClass)
651 {
652 if (packet_size == 1)
cristy802d3642011-04-27 02:02:41 +0000653 SetIndexPixelComponent(indexes+x,ScaleQuantumToChar(pixel));
cristy56ed31c2010-03-22 00:46:21 +0000654 else
cristy802d3642011-04-27 02:02:41 +0000655 SetIndexPixelComponent(indexes+x,ScaleQuantumToShort(pixel));
cristy99327bb2011-05-21 21:33:06 +0000656 SetRGBOPixelComponent(q,image->colormap+
657 GetIndexPixelComponent(indexes+x));
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 {
cristy802d3642011-04-27 02:02:41 +0000669 SetIndexPixelComponent(indexes+x,(((unsigned char) pixel) &
670 (0x01 << (7-bit))) != 0 ? 0 : 255);
cristy99327bb2011-05-21 21:33:06 +0000671 SetRGBOPixelComponent(q,image->colormap+
672 GetIndexPixelComponent(indexes+x));
cristy3b004082010-08-12 19:56:38 +0000673 q++;
674 x++;
675 }
676 q--;
677 x--;
678 }
cristy56ed31c2010-03-22 00:46:21 +0000679 }
680 break;
681 }
682 case 1:
683 {
684 if (image->storage_class == PseudoClass)
cristy802d3642011-04-27 02:02:41 +0000685 SetOpacityPixelComponent(q,QuantumRange-pixel);
cristy56ed31c2010-03-22 00:46:21 +0000686 else
cristy802d3642011-04-27 02:02:41 +0000687 SetGreenPixelComponent(q,pixel);
cristy56ed31c2010-03-22 00:46:21 +0000688 break;
689 }
690 case 2:
691 {
cristydede49f2010-08-20 20:26:26 +0000692 if (image->storage_class == PseudoClass)
cristy802d3642011-04-27 02:02:41 +0000693 SetOpacityPixelComponent(q,QuantumRange-pixel);
cristydede49f2010-08-20 20:26:26 +0000694 else
cristy802d3642011-04-27 02:02:41 +0000695 SetBluePixelComponent(q,pixel);
cristy56ed31c2010-03-22 00:46:21 +0000696 break;
697 }
698 case 3:
699 {
700 if (image->colorspace == CMYKColorspace)
cristy802d3642011-04-27 02:02:41 +0000701 SetIndexPixelComponent(indexes+x,pixel);
cristy56ed31c2010-03-22 00:46:21 +0000702 else
cristy802d3642011-04-27 02:02:41 +0000703 SetOpacityPixelComponent(q,QuantumRange-pixel);
cristy56ed31c2010-03-22 00:46:21 +0000704 break;
705 }
706 case 4:
707 {
cristya60c0202010-07-31 21:36:45 +0000708 if ((image->colorspace == RGBColorspace) && (channels > 3))
709 break;
cristy802d3642011-04-27 02:02:41 +0000710 SetOpacityPixelComponent(q,QuantumRange-pixel);
cristy56ed31c2010-03-22 00:46:21 +0000711 break;
712 }
713 default:
714 break;
715 }
716 q++;
717 }
718 if (SyncAuthenticPixels(image,exception) == MagickFalse)
719 break;
720 }
cristyd05dca12010-07-25 02:32:32 +0000721 image->colorspace=colorspace;
cristy56ed31c2010-03-22 00:46:21 +0000722 if (image->compression == RLECompression)
723 compact_pixels=(unsigned char *) RelinquishMagickMemory(compact_pixels);
724 pixels=(unsigned char *) RelinquishMagickMemory(pixels);
725 return(MagickTrue);
726}
727
cristy3ed852e2009-09-05 21:47:34 +0000728static Image *ReadPSDImage(const ImageInfo *image_info,ExceptionInfo *exception)
729{
cristy3ed852e2009-09-05 21:47:34 +0000730 char
cristy56ed31c2010-03-22 00:46:21 +0000731 message[MaxTextExtent],
cristy3ed852e2009-09-05 21:47:34 +0000732 type[4];
733
734 Image
735 *image;
736
cristy3ed852e2009-09-05 21:47:34 +0000737 LayerInfo
738 *layer_info;
739
cristy56ed31c2010-03-22 00:46:21 +0000740 MagickBooleanType
741 status;
742
743 MagickOffsetType
744 offset,
745 *offsets;
746
747 MagickSizeType
748 combinedlength,
749 length,
750 size;
751
cristy3ed852e2009-09-05 21:47:34 +0000752 PSDInfo
753 psd_info;
754
cristydede49f2010-08-20 20:26:26 +0000755 register PixelPacket
756 *q;
757
cristybb503372010-05-27 20:51:26 +0000758 register ssize_t
cristy56ed31c2010-03-22 00:46:21 +0000759 i,
cristy3ed852e2009-09-05 21:47:34 +0000760 x;
761
cristydc05fc22011-01-22 19:43:15 +0000762 size_t
763 mask_size,
764 skip_first_alpha = 0;
765
cristy3ed852e2009-09-05 21:47:34 +0000766 ssize_t
cristydc05fc22011-01-22 19:43:15 +0000767 count,
768 j,
769 number_layers,
770 y;
cristy3ed852e2009-09-05 21:47:34 +0000771
cristy3ed852e2009-09-05 21:47:34 +0000772 unsigned char
773 *data;
774
775 unsigned short
776 compression;
777
cristy3ed852e2009-09-05 21:47:34 +0000778 /*
779 Open image file.
780 */
781 assert(image_info != (const ImageInfo *) NULL);
782 assert(image_info->signature == MagickSignature);
783 if (image_info->debug != MagickFalse)
784 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
785 image_info->filename);
786 assert(exception != (ExceptionInfo *) NULL);
787 assert(exception->signature == MagickSignature);
788 image=AcquireImage(image_info);
789 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
790 if (status == MagickFalse)
791 {
792 image=DestroyImageList(image);
793 return((Image *) NULL);
794 }
795 /*
796 Read image header.
797 */
798 count=ReadBlob(image,4,(unsigned char *) psd_info.signature);
799 psd_info.version=ReadBlobMSBShort(image);
cristy50aea4a2010-03-09 17:37:44 +0000800 if ((count == 0) || (LocaleNCompare(psd_info.signature,"8BPS",4) != 0) ||
cristy2d3d87f2010-03-01 00:23:08 +0000801 ((psd_info.version != 1) && (psd_info.version != 2)))
cristy3ed852e2009-09-05 21:47:34 +0000802 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
803 count=ReadBlob(image,6,psd_info.reserved);
804 psd_info.channels=ReadBlobMSBShort(image);
805 if (psd_info.channels > MaxPSDChannels)
806 ThrowReaderException(CorruptImageError,"MaximumChannelsExceeded");
807 psd_info.rows=ReadBlobMSBLong(image);
808 psd_info.columns=ReadBlobMSBLong(image);
cristy2d3d87f2010-03-01 00:23:08 +0000809 if ((psd_info.version == 1) && ((psd_info.rows > 30000) ||
810 (psd_info.columns > 30000)))
811 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
cristy3ed852e2009-09-05 21:47:34 +0000812 psd_info.depth=ReadBlobMSBShort(image);
cristy2d3d87f2010-03-01 00:23:08 +0000813 if ((psd_info.depth != 1) && (psd_info.depth != 8) && (psd_info.depth != 16))
814 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
cristy3ed852e2009-09-05 21:47:34 +0000815 psd_info.mode=ReadBlobMSBShort(image);
816 if (image->debug != MagickFalse)
817 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +0000818 " Image is %.20g x %.20g with channels=%.20g, depth=%.20g, mode=%s",
819 (double) psd_info.columns,(double) psd_info.rows,(double)
820 psd_info.channels,(double) psd_info.depth,ModeToString((PSDImageType)
821 psd_info.mode));
cristy3ed852e2009-09-05 21:47:34 +0000822 /*
823 Initialize image.
824 */
825 image->depth=psd_info.depth;
826 image->columns=psd_info.columns;
827 image->rows=psd_info.rows;
cristyf6febd72010-11-06 15:54:54 +0000828 if (SetImageBackgroundColor(image) == MagickFalse)
cristy95524f92010-02-16 18:44:34 +0000829 {
830 InheritException(exception,&image->exception);
831 image=DestroyImageList(image);
832 return((Image *) NULL);
833 }
cristy3ed852e2009-09-05 21:47:34 +0000834 image->matte=psd_info.channels >= 4 ? MagickTrue : MagickFalse;
835 if (psd_info.mode == LabMode)
836 image->colorspace=LabColorspace;
837 if (psd_info.mode == CMYKMode)
838 {
839 image->colorspace=CMYKColorspace;
840 image->matte=psd_info.channels >= 5 ? MagickTrue : MagickFalse;
841 }
842 if ((psd_info.mode == BitmapMode) || (psd_info.mode == GrayscaleMode) ||
843 (psd_info.mode == DuotoneMode))
844 {
cristy52cf7f12010-02-07 18:07:56 +0000845 if (AcquireImageColormap(image,256) == MagickFalse)
846 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristy3ed852e2009-09-05 21:47:34 +0000847 image->matte=psd_info.channels >= 2 ? MagickTrue : MagickFalse;
848 if (image->debug != MagickFalse)
849 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristydede49f2010-08-20 20:26:26 +0000850 " Image colormap allocated");
cristy0132ba82010-04-16 23:34:02 +0000851 image->colorspace=GRAYColorspace;
cristy3ed852e2009-09-05 21:47:34 +0000852 }
853 if (image->debug != MagickFalse)
854 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
855 image->matte ? " image has matte" : " image has no matte");
856 /*
857 Read PSD raster colormap only present for indexed and duotone images.
858 */
859 length=ReadBlobMSBLong(image);
860 if (length != 0)
861 {
862 if (image->debug != MagickFalse)
863 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
864 " reading colormap");
865 if (psd_info.mode == DuotoneMode)
866 {
867 /*
868 Duotone image data; the format of this data is undocumented.
869 */
cristy56ed31c2010-03-22 00:46:21 +0000870 data=(unsigned char *) AcquireQuantumMemory((size_t) length,
871 sizeof(*data));
cristy3ed852e2009-09-05 21:47:34 +0000872 if (data == (unsigned char *) NULL)
873 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristy56ed31c2010-03-22 00:46:21 +0000874 count=ReadBlob(image,(size_t) length,data);
cristy3ed852e2009-09-05 21:47:34 +0000875 data=(unsigned char *) RelinquishMagickMemory(data);
876 }
877 else
878 {
879 /*
880 Read PSD raster colormap.
881 */
cristybb503372010-05-27 20:51:26 +0000882 if (AcquireImageColormap(image,(size_t) (length/3)) == MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +0000883 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristybb503372010-05-27 20:51:26 +0000884 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +0000885 image->colormap[i].red=ScaleCharToQuantum((unsigned char)
886 ReadBlobByte(image));
cristybb503372010-05-27 20:51:26 +0000887 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +0000888 image->colormap[i].green=ScaleCharToQuantum((unsigned char)
889 ReadBlobByte(image));
cristybb503372010-05-27 20:51:26 +0000890 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +0000891 image->colormap[i].blue=ScaleCharToQuantum((unsigned char)
892 ReadBlobByte(image));
893 image->matte=psd_info.channels >= 2 ? MagickTrue : MagickFalse;
894 }
895 }
896 length=ReadBlobMSBLong(image);
897 if (length != 0)
898 {
899 unsigned char
900 *blocks;
901
902 /*
903 Image resources block.
904 */
905 if (image->debug != MagickFalse)
906 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +0000907 " reading image resource blocks - %.20g bytes",(double) length);
cristy56ed31c2010-03-22 00:46:21 +0000908 blocks=(unsigned char *) AcquireQuantumMemory((size_t) length,
909 sizeof(*blocks));
cristy3ed852e2009-09-05 21:47:34 +0000910 if (blocks == (unsigned char *) NULL)
911 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristy56ed31c2010-03-22 00:46:21 +0000912 count=ReadBlob(image,(size_t) length,blocks);
913 if ((count != (ssize_t) length) ||
cristy3ed852e2009-09-05 21:47:34 +0000914 (LocaleNCompare((char *) blocks,"8BIM",4) != 0))
915 {
916 blocks=(unsigned char *) RelinquishMagickMemory(blocks);
917 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
918 }
cristy56ed31c2010-03-22 00:46:21 +0000919 (void) ParseImageResourceBlocks(image,blocks,(size_t) length);
cristy3ed852e2009-09-05 21:47:34 +0000920 blocks=(unsigned char *) RelinquishMagickMemory(blocks);
921 }
922 /*
cristy3ed852e2009-09-05 21:47:34 +0000923 Layer and mask block.
924 */
925 layer_info=(LayerInfo *) NULL;
cristy4689cf02010-02-17 21:15:45 +0000926 number_layers=1;
cristy2d3d87f2010-03-01 00:23:08 +0000927 length=GetPSDSize(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +0000928 if (length == 8)
929 {
930 length=ReadBlobMSBLong(image);
931 length=ReadBlobMSBLong(image);
932 }
933 if ((image_info->number_scenes == 1) && (image_info->scene == 0))
cristyd4297022010-09-16 22:59:09 +0000934 {
935 if (DiscardBlobBytes(image,length) == MagickFalse)
936 ThrowFileException(exception,CorruptImageError,"UnexpectedEndOfFile",
937 image->filename);
938 length=0;
939 }
cristy3ed852e2009-09-05 21:47:34 +0000940 if (length == 0)
941 {
942 if (image->debug != MagickFalse)
943 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
944 " image has no layers");
945 }
946 else
947 {
948 offset=TellBlob(image);
cristy2d3d87f2010-03-01 00:23:08 +0000949 size=GetPSDSize(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +0000950 if (size == 0)
951 {
cristybb503372010-05-27 20:51:26 +0000952 size_t
cristy2d3d87f2010-03-01 00:23:08 +0000953 quantum;
954
cristy9fb55e82011-02-27 23:12:47 +0000955 unsigned long
956 tag;
957
cristy3ed852e2009-09-05 21:47:34 +0000958 /*
959 Skip layers & masks.
960 */
cristy56ed31c2010-03-22 00:46:21 +0000961 quantum=psd_info.version == 1 ? 4UL : 8UL;
cristy9fb55e82011-02-27 23:12:47 +0000962 tag=ReadBlobMSBLong(image);
cristy6c70aec2011-03-04 20:48:09 +0000963 (void) tag;
964 count=ReadBlob(image,4,(unsigned char *) type);
965 if ((count == 0) || (LocaleNCompare(type,"8BIM",4) != 0))
cristy9fb55e82011-02-27 23:12:47 +0000966 {
967 if (DiscardBlobBytes(image,length-quantum-8) == MagickFalse)
968 ThrowFileException(exception,CorruptImageError,
969 "UnexpectedEndOfFile",image->filename);
970 }
971 else
972 {
cristy6c70aec2011-03-04 20:48:09 +0000973 count=ReadBlob(image,4,(unsigned char *) type);
974 if ((count != 0) && (LocaleNCompare(type,"Lr16",4) == 0))
cristy9fb55e82011-02-27 23:12:47 +0000975 size=GetPSDSize(&psd_info,image);
976 else
977 if (DiscardBlobBytes(image,length-quantum-12) == MagickFalse)
978 ThrowFileException(exception,CorruptImageError,
979 "UnexpectedEndOfFile",image->filename);
980 }
cristy3ed852e2009-09-05 21:47:34 +0000981 }
cristy9fb55e82011-02-27 23:12:47 +0000982 if (size != 0)
cristy3ed852e2009-09-05 21:47:34 +0000983 {
984 MagickOffsetType
985 layer_offset;
986
987 layer_offset=offset+length;
988 number_layers=(short) ReadBlobMSBShort(image);
989 if (image->debug != MagickFalse)
990 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +0000991 " image contains %.20g layers",(double) number_layers);
cristy3ed852e2009-09-05 21:47:34 +0000992 if (number_layers < 0)
993 {
994 /*
995 Weird hack in PSD format to ignore first alpha channel.
996 */
997 skip_first_alpha=1;
cristyda16f162011-02-19 23:52:17 +0000998 (void) skip_first_alpha;
cristy3ed852e2009-09-05 21:47:34 +0000999 number_layers=MagickAbsoluteValue(number_layers);
1000 if (image->debug != MagickFalse)
1001 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1002 " negative layer count corrected for");
1003 }
1004 layer_info=(LayerInfo *) AcquireQuantumMemory((size_t) number_layers,
1005 sizeof(*layer_info));
1006 if (layer_info == (LayerInfo *) NULL)
1007 {
1008 if (image->debug != MagickFalse)
1009 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1010 " allocation of LayerInfo failed");
1011 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
1012 }
1013 (void) ResetMagickMemory(layer_info,0,(size_t) number_layers*
1014 sizeof(*layer_info));
1015 for (i=0; i < number_layers; i++)
1016 {
cristy4484e332010-10-08 23:46:21 +00001017 int
1018 x,
1019 y;
1020
cristy3ed852e2009-09-05 21:47:34 +00001021 if (image->debug != MagickFalse)
1022 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001023 " reading layer #%.20g",(double) i+1);
cristy6cff05d2010-09-02 11:22:46 +00001024 layer_info[i].page.y=(int) ReadBlobMSBLong(image);
1025 layer_info[i].page.x=(int) ReadBlobMSBLong(image);
cristy4484e332010-10-08 23:46:21 +00001026 y=(int) ReadBlobMSBLong(image);
1027 x=(int) ReadBlobMSBLong(image);
1028 layer_info[i].page.width=(ssize_t) (x-layer_info[i].page.x);
1029 layer_info[i].page.height=(ssize_t) (y-layer_info[i].page.y);
cristy3ed852e2009-09-05 21:47:34 +00001030 layer_info[i].channels=ReadBlobMSBShort(image);
1031 if (layer_info[i].channels > MaxPSDChannels)
1032 ThrowReaderException(CorruptImageError,"MaximumChannelsExceeded");
1033 if (image->debug != MagickFalse)
1034 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001035 " offset(%.20g,%.20g), size(%.20g,%.20g), channels=%.20g",
1036 (double) layer_info[i].page.x,(double) layer_info[i].page.y,
1037 (double) layer_info[i].page.height,(double)
1038 layer_info[i].page.width,(double) layer_info[i].channels);
cristybb503372010-05-27 20:51:26 +00001039 for (j=0; j < (ssize_t) layer_info[i].channels; j++)
cristy3ed852e2009-09-05 21:47:34 +00001040 {
cristy56ed31c2010-03-22 00:46:21 +00001041 layer_info[i].channel_info[j].type=(short)
1042 ReadBlobMSBShort(image);
cristybb503372010-05-27 20:51:26 +00001043 layer_info[i].channel_info[j].size=(size_t)
cristy56ed31c2010-03-22 00:46:21 +00001044 GetPSDSize(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +00001045 if (image->debug != MagickFalse)
1046 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001047 " channel[%.20g]: type=%.20g, size=%.20g",(double) j,
1048 (double) layer_info[i].channel_info[j].type,
1049 (double) layer_info[i].channel_info[j].size);
cristy3ed852e2009-09-05 21:47:34 +00001050 }
1051 count=ReadBlob(image,4,(unsigned char *) type);
1052 if ((count == 0) || (LocaleNCompare(type,"8BIM",4) != 0))
1053 {
1054 if (image->debug != MagickFalse)
1055 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1056 " layer type was %.4s instead of 8BIM", type);
1057 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
1058 }
1059 count=ReadBlob(image,4,(unsigned char *) layer_info[i].blendkey);
1060 layer_info[i].opacity=(Quantum) (QuantumRange-ScaleCharToQuantum(
1061 (unsigned char) ReadBlobByte(image)));
1062 layer_info[i].clipping=(unsigned char) ReadBlobByte(image);
1063 layer_info[i].flags=(unsigned char) ReadBlobByte(image);
1064 layer_info[i].visible=!(layer_info[i].flags & 0x02);
1065 if (image->debug != MagickFalse)
1066 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001067 " blend=%.4s, opacity=%.20g, clipping=%s, flags=%d, visible=%s",
1068 layer_info[i].blendkey,(double) layer_info[i].opacity,
cristy3ed852e2009-09-05 21:47:34 +00001069 layer_info[i].clipping ? "true" : "false",layer_info[i].flags,
1070 layer_info[i].visible ? "true" : "false");
1071 (void) ReadBlobByte(image); /* filler */
1072 combinedlength=0;
1073 size=ReadBlobMSBLong(image);
1074 if (size != 0)
1075 {
1076 if (image->debug != MagickFalse)
1077 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1078 " layer contains additional info");
1079 length=ReadBlobMSBLong(image);
1080 if (length != 0)
1081 {
1082 /*
1083 Layer mask info.
1084 */
cristy6cff05d2010-09-02 11:22:46 +00001085 layer_info[i].mask.y=(int) ReadBlobMSBLong(image);
1086 layer_info[i].mask.x=(int) ReadBlobMSBLong(image);
cristybb503372010-05-27 20:51:26 +00001087 layer_info[i].mask.height=(size_t)
cristy3ed852e2009-09-05 21:47:34 +00001088 (ReadBlobMSBLong(image)-layer_info[i].mask.y);
cristybb503372010-05-27 20:51:26 +00001089 layer_info[i].mask.width=(size_t)
cristy3ed852e2009-09-05 21:47:34 +00001090 (ReadBlobMSBLong(image)-layer_info[i].mask.x);
1091 if (image->debug != MagickFalse)
1092 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001093 " layer mask: offset(%.20g,%.20g), size(%.20g,%.20g), length=%.20g",
cristyd4297022010-09-16 22:59:09 +00001094 (double) layer_info[i].mask.x,(double) layer_info[i].mask.y,
cristye8c25f92010-06-03 00:53:06 +00001095 (double) layer_info[i].mask.width,(double)
1096 layer_info[i].mask.height,(double) length-16);
cristy3ed852e2009-09-05 21:47:34 +00001097 /*
1098 Skip over the rest of the layer mask information.
1099 */
cristyc1af14f2010-09-16 20:01:21 +00001100 if (DiscardBlobBytes(image,length-16) == MagickFalse)
cristyd4297022010-09-16 22:59:09 +00001101 ThrowFileException(exception,CorruptImageError,
1102 "UnexpectedEndOfFile",image->filename);
cristy3ed852e2009-09-05 21:47:34 +00001103 }
1104 combinedlength+=length+4; /* +4 for length */
1105 length=ReadBlobMSBLong(image);
1106 if (length != 0)
1107 {
1108 /*
1109 Layer blending ranges info.
1110 */
1111 if (image->debug != MagickFalse)
1112 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001113 " layer blending ranges: length=%.20g",(double)
cristyf2faecf2010-05-28 19:19:36 +00001114 length);
cristy3ed852e2009-09-05 21:47:34 +00001115 /*
1116 We read it, but don't use it...
1117 */
cristybb503372010-05-27 20:51:26 +00001118 for (j=0; j < (ssize_t) (length); j+=8)
cristy3ed852e2009-09-05 21:47:34 +00001119 {
1120 size_t blend_source=ReadBlobMSBLong(image);
1121 size_t blend_dest=ReadBlobMSBLong(image);
1122 if (image->debug != MagickFalse)
1123 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1124 " source(%x), dest(%x)",(unsigned int)
1125 blend_source,(unsigned int) blend_dest);
1126 }
1127 }
cristy2cc39842009-10-10 20:04:02 +00001128 combinedlength+=length+4;
1129 /*
1130 Layer name.
1131 */
cristy3ed852e2009-09-05 21:47:34 +00001132 length=(size_t) ReadBlobByte(image);
cristybb503372010-05-27 20:51:26 +00001133 for (j=0; j < (ssize_t) length; j++)
cristy2cc39842009-10-10 20:04:02 +00001134 layer_info[i].name[j]=(unsigned char) ReadBlobByte(image);
1135 layer_info[i].name[j]='\0';
1136 if (image->debug != MagickFalse)
1137 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1138 " layer name: %s",layer_info[i].name);
1139 combinedlength+=length+1;
cristy3ed852e2009-09-05 21:47:34 +00001140
1141#if 0 /* still in development */
1142 /*
1143 Adjustment layers and other stuff...
1144 */
1145 {
cristy802d3642011-04-27 02:02:41 +00001146 char alsig[4], alkey[4];
cristy3ed852e2009-09-05 21:47:34 +00001147
1148 count=ReadBlob(image,4,alsig);
1149 if ((count == 0) || (LocaleNCompare(alsig,"8BIM",4) != 0)) {
1150 if (debug != MagickFalse)
1151 {
cristy802d3642011-04-27 02:02:41 +00001152 if (image->debug != MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001153 (void) LogMagickEvent(CoderEvent,GetMagickModule()," adjustment layer type was %.4s instead of 8BIM", alsig);
1154 }
1155 ThrowReaderException(CorruptImageError,"ImproperImageHeader");
1156 }
1157 count=ReadBlob(image,4,alkey);
1158 length=ReadBlobMSBLong(image);
1159 if (debug != MagickFalse)
1160 {
cristy802d3642011-04-27 02:02:41 +00001161 if (image->debug != MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001162 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001163 " adjustment layer key: %.4s, data length=%.20g",
1164 alkey, (double) length);
cristy3ed852e2009-09-05 21:47:34 +00001165 }
1166
1167 if ( length ) {
cristybb503372010-05-27 20:51:26 +00001168 for (j=0; j < (ssize_t) (length); j++)
cristy3ed852e2009-09-05 21:47:34 +00001169 (void) ReadBlobByte(image);
1170 }
1171
1172 }
1173 combinedlength += 12 + length; /* sig, key, length + the actual length*/
1174#endif
1175
1176 /*
1177 Skip the rest of the variable data until we support it.
1178 */
1179 if (image->debug != MagickFalse)
1180 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001181 " unsupported data: length=%.20g",(double)
cristyeec18db2010-03-03 21:15:45 +00001182 (size-combinedlength));
cristyc1af14f2010-09-16 20:01:21 +00001183 if (DiscardBlobBytes(image,size-combinedlength) == MagickFalse)
cristyd4297022010-09-16 22:59:09 +00001184 ThrowFileException(exception,CorruptImageError,
1185 "UnexpectedEndOfFile",image->filename);
cristy3ed852e2009-09-05 21:47:34 +00001186 }
1187 /*
1188 Allocate layered image.
1189 */
1190 layer_info[i].image=CloneImage(image,layer_info[i].page.width,
cristy390c39f2010-10-05 23:41:44 +00001191 layer_info[i].page.height == ~0U ? 1 : layer_info[i].page.height,
1192 MagickFalse,&image->exception);
cristy3ed852e2009-09-05 21:47:34 +00001193 if (layer_info[i].image == (Image *) NULL)
1194 {
1195 for (j=0; j < i; j++)
1196 layer_info[j].image=DestroyImage(layer_info[j].image);
1197 if (image->debug != MagickFalse)
1198 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001199 " allocation of image for layer %.20g failed",(double) i);
cristy3ed852e2009-09-05 21:47:34 +00001200 ThrowReaderException(ResourceLimitError,
1201 "MemoryAllocationFailed");
1202 }
1203 if (image->debug != MagickFalse)
1204 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1205 " setting up new layer image");
cristy2e2e46f2010-10-12 18:55:20 +00001206 if (image_info->ping != MagickFalse)
1207 (void) SetImageBackgroundColor(layer_info[i].image);
cristy3ed852e2009-09-05 21:47:34 +00001208 layer_info[i].image->compose=
1209 PSDBlendModeToCompositeOperator(layer_info[i].blendkey);
1210 if (layer_info[i].visible == MagickFalse)
1211 layer_info[i].image->compose=NoCompositeOp;
1212 if (psd_info.mode == CMYKMode)
cristy0132ba82010-04-16 23:34:02 +00001213 layer_info[i].image->colorspace=CMYKColorspace;
1214 if ((psd_info.mode == BitmapMode) ||
1215 (psd_info.mode == GrayscaleMode) ||
1216 (psd_info.mode == DuotoneMode))
1217 layer_info[i].image->colorspace=GRAYColorspace;
cristybb503372010-05-27 20:51:26 +00001218 for (j=0; j < (ssize_t) layer_info[i].channels; j++)
cristy3ed852e2009-09-05 21:47:34 +00001219 if (layer_info[i].channel_info[j].type == -1)
1220 layer_info[i].image->matte=MagickTrue;
1221 /*
1222 Set up some hidden attributes for folks that need them.
1223 */
cristyb51dff52011-05-19 16:55:47 +00001224 (void) FormatLocaleString(message,MaxTextExtent,"%.20gld",
cristye8c25f92010-06-03 00:53:06 +00001225 (double) layer_info[i].page.x);
cristy56ed31c2010-03-22 00:46:21 +00001226 (void) SetImageArtifact(layer_info[i].image,"psd:layer.x",message);
cristyb51dff52011-05-19 16:55:47 +00001227 (void) FormatLocaleString(message,MaxTextExtent,"%.20g",
cristye8c25f92010-06-03 00:53:06 +00001228 (double) layer_info[i].page.y);
cristy56ed31c2010-03-22 00:46:21 +00001229 (void) SetImageArtifact(layer_info[i].image,"psd:layer.y",message);
cristyb51dff52011-05-19 16:55:47 +00001230 (void) FormatLocaleString(message,MaxTextExtent,"%.20g",
cristye8c25f92010-06-03 00:53:06 +00001231 (double) layer_info[i].opacity);
cristy56ed31c2010-03-22 00:46:21 +00001232 (void) SetImageArtifact(layer_info[i].image,"psd:layer.opacity",
1233 message);
cristy3ed852e2009-09-05 21:47:34 +00001234 (void) SetImageProperty(layer_info[i].image,"label",(char *)
1235 layer_info[i].name);
1236 }
1237 if (image->debug != MagickFalse)
1238 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1239 " reading image data for layers");
1240 /*
1241 Read pixel data for each layer.
1242 */
1243 for (i=0; i < number_layers; i++)
1244 {
1245 if (image->debug != MagickFalse)
1246 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001247 " reading data for layer %.20g",(double) i);
cristybb503372010-05-27 20:51:26 +00001248 for (j=0; j < (ssize_t) layer_info[i].channels; j++)
cristy3ed852e2009-09-05 21:47:34 +00001249 {
1250 if (image->debug != MagickFalse)
1251 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
cristye8c25f92010-06-03 00:53:06 +00001252 " reading data for channel %.20g",(double) j);
cristy3ed852e2009-09-05 21:47:34 +00001253#if 1
1254 if (layer_info[i].channel_info[j].size <= (2*layer_info[i].image->rows))
1255 {
cristybb503372010-05-27 20:51:26 +00001256 ssize_t
cristy3ed852e2009-09-05 21:47:34 +00001257 k;
1258
1259 if (image->debug != MagickFalse)
1260 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1261 " layer data is empty");
1262 /*
1263 A layer without data.
1264 */
cristybb503372010-05-27 20:51:26 +00001265 for (k=0; k < (ssize_t) layer_info[i].channel_info[j].size; k++)
cristy3ed852e2009-09-05 21:47:34 +00001266 (void) ReadBlobByte(layer_info[i].image);
1267 continue;
1268 }
1269#endif
cristy56ed31c2010-03-22 00:46:21 +00001270 offsets=(MagickOffsetType *) NULL;
1271 layer_info[i].image->compression=NoCompression;
cristy3ed852e2009-09-05 21:47:34 +00001272 compression=ReadBlobMSBShort(layer_info[i].image);
cristy2cc39842009-10-10 20:04:02 +00001273 if ((layer_info[i].page.height != 0) &&
1274 (layer_info[i].page.width != 0))
cristy3ed852e2009-09-05 21:47:34 +00001275 {
cristy2cc39842009-10-10 20:04:02 +00001276 if (compression == 1)
1277 {
1278 /*
1279 Read RLE compressed data.
1280 */
cristy56ed31c2010-03-22 00:46:21 +00001281 layer_info[i].image->compression=RLECompression;
cristy2cc39842009-10-10 20:04:02 +00001282 if (image->debug != MagickFalse)
1283 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1284 " layer data is RLE compressed");
cristy56ed31c2010-03-22 00:46:21 +00001285 offsets=(MagickOffsetType *) AcquireQuantumMemory(
1286 layer_info[i].image->rows,sizeof(*offsets));
1287 if (offsets == (MagickOffsetType *) NULL)
1288 ThrowReaderException(ResourceLimitError,
1289 "MemoryAllocationFailed");
cristybb503372010-05-27 20:51:26 +00001290 for (y=0; y < (ssize_t) layer_info[i].image->rows; y++)
cristy56ed31c2010-03-22 00:46:21 +00001291 offsets[y]=GetPSDOffset(&psd_info,layer_info[i].image);
cristy2cc39842009-10-10 20:04:02 +00001292 }
cristy56ed31c2010-03-22 00:46:21 +00001293 status=ReadPSDLayer(layer_info[i].image,
cristye3038982010-05-17 02:20:52 +00001294 layer_info[i].channels,layer_info[i].channel_info[j].type,
1295 offsets,exception);
cristy56ed31c2010-03-22 00:46:21 +00001296 if (compression == 1)
1297 offsets=(MagickOffsetType *) RelinquishMagickMemory(
1298 offsets);
1299 if (status == MagickFalse)
1300 break;
cristy3ed852e2009-09-05 21:47:34 +00001301 }
cristy56ed31c2010-03-22 00:46:21 +00001302 }
cristy3ed852e2009-09-05 21:47:34 +00001303 if (layer_info[i].opacity != OpaqueOpacity)
1304 {
1305 /*
1306 Correct for opacity level.
1307 */
cristybb503372010-05-27 20:51:26 +00001308 for (y=0; y < (ssize_t) layer_info[i].image->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001309 {
1310 q=GetAuthenticPixels(layer_info[i].image,0,y,
1311 layer_info[i].image->columns,1,exception);
1312 if (q == (PixelPacket *) NULL)
1313 break;
cristybb503372010-05-27 20:51:26 +00001314 for (x=0; x < (ssize_t) layer_info[i].image->columns; x++)
cristy3ed852e2009-09-05 21:47:34 +00001315 {
cristy07758662010-01-15 20:25:50 +00001316 q->opacity=(Quantum) (QuantumRange-(Quantum) (QuantumScale*
1317 ((QuantumRange-q->opacity)*(QuantumRange-
1318 layer_info[i].opacity))));
cristy3ed852e2009-09-05 21:47:34 +00001319 q++;
1320 }
1321 if (SyncAuthenticPixels(layer_info[i].image,exception) == MagickFalse)
1322 break;
1323 }
1324 }
1325 if (layer_info[i].image->colorspace == CMYKColorspace)
1326 (void) NegateImage(layer_info[i].image,MagickFalse);
cristy56ed31c2010-03-22 00:46:21 +00001327 status=SetImageProgress(image,LoadImagesTag,i,(MagickSizeType)
1328 number_layers);
cristy4689cf02010-02-17 21:15:45 +00001329 if (status == MagickFalse)
1330 break;
cristy3ed852e2009-09-05 21:47:34 +00001331 }
1332 /* added by palf -> invisible group layer make layer of this group
1333 invisible I consider that all layer with width and height null are
1334 layer for group layer */
1335 {
1336 short inside_layer = 0;
1337 short layer_visible = 0;
1338 for (i=number_layers-1; i >=0; i--)
1339 {
1340 if ((layer_info[i].page.width == 0) ||
1341 (layer_info[i].page.height == 0))
1342 {
1343 if (inside_layer == 0)
1344 {
1345 inside_layer=1;
1346 layer_visible=(short int) layer_info[i].visible;
1347 }
1348 else
1349 {
1350 inside_layer = 0;
1351 }
1352 }
1353 else
1354 if ((inside_layer == 1) && (layer_visible == 0))
1355 {
1356 layer_info[i].visible=(unsigned char) layer_visible;
1357 layer_info[i].image->compose=NoCompositeOp;
1358 }
1359 }
1360 }
1361 /* added by palf -> suppression of empty layer */
1362 /* I consider that all layer with width and height null are layer for group layer */
1363 for (i=0; i < number_layers; i++)
1364 {
1365 if ((layer_info[i].page.width == 0) ||
1366 (layer_info[i].page.height == 0))
1367 {
1368 if (layer_info[i].image != (Image *) NULL)
1369 layer_info[i].image=DestroyImage(layer_info[i].image);
1370 for (j=i; j < number_layers - 1; j++)
1371 layer_info[j] = layer_info[j+1];
1372 number_layers--;
1373 i--;
1374 }
cristy56ed31c2010-03-22 00:46:21 +00001375 }
cristy3ed852e2009-09-05 21:47:34 +00001376 mask_size = ReadBlobMSBLong(image); /* global mask size: currently ignored */
cristyda16f162011-02-19 23:52:17 +00001377 (void) mask_size;
cristy56ed31c2010-03-22 00:46:21 +00001378 if (number_layers > 0)
1379 {
1380 if (image->debug != MagickFalse)
1381 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1382 " putting layers into image list");
1383 for (i=0; i < number_layers; i++)
cristy3ed852e2009-09-05 21:47:34 +00001384 {
cristy56ed31c2010-03-22 00:46:21 +00001385 if (i > 0)
1386 layer_info[i].image->previous=layer_info[i-1].image;
1387 if (i < (number_layers-1))
1388 layer_info[i].image->next=layer_info[i+1].image;
1389 layer_info[i].image->page=layer_info[i].page;
1390 }
cristy3ed852e2009-09-05 21:47:34 +00001391 image->next=layer_info[0].image;
1392 layer_info[0].image->previous=image;
1393 layer_info=(LayerInfo *) RelinquishMagickMemory(layer_info);
cristy56ed31c2010-03-22 00:46:21 +00001394 }
1395 layer_offset-=TellBlob(image);
1396 offset=SeekBlob(image,layer_offset,SEEK_CUR);
1397 }
cristy3ed852e2009-09-05 21:47:34 +00001398 }
1399 /*
1400 Read the precombined layer, present for PSD < 4 compatibility
1401 */
1402 if (image->debug != MagickFalse)
1403 (void) LogMagickEvent(CoderEvent,GetMagickModule(),
1404 " reading the precombined layer");
cristy56ed31c2010-03-22 00:46:21 +00001405 offsets=(MagickOffsetType *) NULL;
1406 image->compression=NoCompression;
cristy3ed852e2009-09-05 21:47:34 +00001407 compression=ReadBlobMSBShort(image);
1408 if (compression == 1)
1409 {
1410 /*
cristy3f938c62010-03-07 00:53:52 +00001411 Read Packbit encoded pixel data as separate planes.
cristy3ed852e2009-09-05 21:47:34 +00001412 */
cristy56ed31c2010-03-22 00:46:21 +00001413 image->compression=RLECompression;
1414 offsets=(MagickOffsetType *) AcquireQuantumMemory(image->rows,
1415 psd_info.channels*sizeof(*offsets));
1416 if (offsets == (MagickOffsetType *) NULL)
1417 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
cristybb503372010-05-27 20:51:26 +00001418 for (i=0; i < (ssize_t) (image->rows*psd_info.channels); i++)
cristy56ed31c2010-03-22 00:46:21 +00001419 offsets[i]=GetPSDOffset(&psd_info,image);
cristy3ed852e2009-09-05 21:47:34 +00001420 }
cristybb503372010-05-27 20:51:26 +00001421 for (i=0; i < (ssize_t) psd_info.channels; i++)
cristy56ed31c2010-03-22 00:46:21 +00001422 {
cristye3038982010-05-17 02:20:52 +00001423 status=ReadPSDLayer(image,psd_info.channels,i,offsets+i*image->rows,
1424 exception);
cristy56ed31c2010-03-22 00:46:21 +00001425 if (status == MagickFalse)
1426 break;
1427 status=SetImageProgress(image,LoadImagesTag,i,psd_info.channels);
1428 if (status == MagickFalse)
1429 break;
1430 }
1431 if (compression == 1)
1432 offsets=(MagickOffsetType *) RelinquishMagickMemory(offsets);
cristy3ed852e2009-09-05 21:47:34 +00001433 if (image->colorspace == CMYKColorspace)
1434 (void) NegateImage(image,MagickFalse);
1435 (void) CloseBlob(image);
1436 return(GetFirstImageInList(image));
1437}
1438
1439/*
1440%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1441% %
1442% %
1443% %
1444% R e g i s t e r P S D I m a g e %
1445% %
1446% %
1447% %
1448%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1449%
1450% RegisterPSDImage() adds properties for the PSD image format to
1451% the list of supported formats. The properties include the image format
1452% tag, a method to read and/or write the format, whether the format
1453% supports the saving of more than one frame to the same file or blob,
1454% whether the format supports native in-memory I/O, and a brief
1455% description of the format.
1456%
1457% The format of the RegisterPSDImage method is:
1458%
cristybb503372010-05-27 20:51:26 +00001459% size_t RegisterPSDImage(void)
cristy3ed852e2009-09-05 21:47:34 +00001460%
1461*/
cristybb503372010-05-27 20:51:26 +00001462ModuleExport size_t RegisterPSDImage(void)
cristy3ed852e2009-09-05 21:47:34 +00001463{
1464 MagickInfo
1465 *entry;
1466
cristyb4233012010-02-28 20:09:14 +00001467 entry=SetMagickInfo("PSB");
1468 entry->decoder=(DecodeImageHandler *) ReadPSDImage;
1469 entry->encoder=(EncodeImageHandler *) WritePSDImage;
1470 entry->magick=(IsImageFormatHandler *) IsPSD;
cristyffaf9782011-04-13 19:50:51 +00001471 entry->seekable_stream=MagickTrue;
cristyb4233012010-02-28 20:09:14 +00001472 entry->description=ConstantString("Adobe Large Document Format");
1473 entry->module=ConstantString("PSD");
1474 (void) RegisterMagickInfo(entry);
cristy3ed852e2009-09-05 21:47:34 +00001475 entry=SetMagickInfo("PSD");
1476 entry->decoder=(DecodeImageHandler *) ReadPSDImage;
1477 entry->encoder=(EncodeImageHandler *) WritePSDImage;
1478 entry->magick=(IsImageFormatHandler *) IsPSD;
cristyffaf9782011-04-13 19:50:51 +00001479 entry->seekable_stream=MagickTrue;
cristy3ed852e2009-09-05 21:47:34 +00001480 entry->description=ConstantString("Adobe Photoshop bitmap");
1481 entry->module=ConstantString("PSD");
1482 (void) RegisterMagickInfo(entry);
1483 return(MagickImageCoderSignature);
1484}
1485
1486/*
1487%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1488% %
1489% %
1490% %
1491% U n r e g i s t e r P S D I m a g e %
1492% %
1493% %
1494% %
1495%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1496%
1497% UnregisterPSDImage() removes format registrations made by the
1498% PSD module from the list of supported formats.
1499%
1500% The format of the UnregisterPSDImage method is:
1501%
1502% UnregisterPSDImage(void)
1503%
1504*/
1505ModuleExport void UnregisterPSDImage(void)
1506{
cristyb4233012010-02-28 20:09:14 +00001507 (void) UnregisterMagickInfo("PSB");
cristy3ed852e2009-09-05 21:47:34 +00001508 (void) UnregisterMagickInfo("PSD");
1509}
1510
1511/*
1512%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1513% %
1514% %
1515% %
1516% W r i t e P S D I m a g e %
1517% %
1518% %
1519% %
1520%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1521%
cristyb1459bc2010-03-23 21:41:43 +00001522% WritePSDImage() writes an image in the Adobe Photoshop encoded image format.
cristy3ed852e2009-09-05 21:47:34 +00001523%
1524% The format of the WritePSDImage method is:
1525%
1526% MagickBooleanType WritePSDImage(const ImageInfo *image_info,Image *image)
1527%
1528% A description of each parameter follows.
1529%
1530% o image_info: the image info.
1531%
1532% o image: The image.
1533%
cristy3ed852e2009-09-05 21:47:34 +00001534*/
1535
cristy50aea4a2010-03-09 17:37:44 +00001536static inline ssize_t SetPSDOffset(const PSDInfo *psd_info,Image *image,
cristyf0460ee2010-03-06 02:55:11 +00001537 const size_t offset)
1538{
1539 if (psd_info->version == 1)
cristy56ed31c2010-03-22 00:46:21 +00001540 return(WriteBlobMSBShort(image,(unsigned short) offset));
1541 return(WriteBlobMSBLong(image,(unsigned short) offset));
cristyf0460ee2010-03-06 02:55:11 +00001542}
1543
cristy50aea4a2010-03-09 17:37:44 +00001544static inline ssize_t SetPSDSize(const PSDInfo *psd_info,Image *image,
cristyf0460ee2010-03-06 02:55:11 +00001545 const MagickSizeType size)
1546{
1547 if (psd_info->version == 1)
cristy56ed31c2010-03-22 00:46:21 +00001548 return(WriteBlobMSBLong(image,(unsigned int) size));
cristyf0460ee2010-03-06 02:55:11 +00001549 return(WriteBlobMSBLongLong(image,size));
1550}
1551
cristy4aff1572010-02-15 13:34:01 +00001552static size_t PSDPackbitsEncodeImage(Image *image,const size_t length,
cristyb1459bc2010-03-23 21:41:43 +00001553 const unsigned char *pixels,unsigned char *compact_pixels)
cristy4aff1572010-02-15 13:34:01 +00001554{
1555 int
1556 count;
1557
cristybb503372010-05-27 20:51:26 +00001558 register ssize_t
cristy4aff1572010-02-15 13:34:01 +00001559 i,
1560 j;
1561
1562 register unsigned char
1563 *q;
1564
1565 unsigned char
1566 *packbits;
1567
1568 /*
1569 Compress pixels with Packbits encoding.
1570 */
1571 assert(image != (Image *) NULL);
1572 assert(image->signature == MagickSignature);
1573 if (image->debug != MagickFalse)
1574 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
1575 assert(pixels != (unsigned char *) NULL);
1576 packbits=(unsigned char *) AcquireQuantumMemory(128UL,sizeof(*packbits));
1577 if (packbits == (unsigned char *) NULL)
1578 ThrowBinaryException(ResourceLimitError,"MemoryAllocationFailed",
1579 image->filename);
cristyb1459bc2010-03-23 21:41:43 +00001580 q=compact_pixels;
cristybb503372010-05-27 20:51:26 +00001581 for (i=(ssize_t) length; i != 0; )
cristy4aff1572010-02-15 13:34:01 +00001582 {
1583 switch (i)
1584 {
1585 case 1:
1586 {
1587 i--;
1588 *q++=(unsigned char) 0;
1589 *q++=(*pixels);
1590 break;
1591 }
1592 case 2:
1593 {
1594 i-=2;
1595 *q++=(unsigned char) 1;
1596 *q++=(*pixels);
1597 *q++=pixels[1];
1598 break;
1599 }
1600 case 3:
1601 {
1602 i-=3;
1603 if ((*pixels == *(pixels+1)) && (*(pixels+1) == *(pixels+2)))
1604 {
1605 *q++=(unsigned char) ((256-3)+1);
1606 *q++=(*pixels);
1607 break;
1608 }
1609 *q++=(unsigned char) 2;
1610 *q++=(*pixels);
1611 *q++=pixels[1];
1612 *q++=pixels[2];
1613 break;
1614 }
1615 default:
1616 {
1617 if ((*pixels == *(pixels+1)) && (*(pixels+1) == *(pixels+2)))
1618 {
1619 /*
1620 Packed run.
1621 */
1622 count=3;
cristybb503372010-05-27 20:51:26 +00001623 while (((ssize_t) count < i) && (*pixels == *(pixels+count)))
cristy4aff1572010-02-15 13:34:01 +00001624 {
1625 count++;
1626 if (count >= 127)
1627 break;
1628 }
1629 i-=count;
1630 *q++=(unsigned char) ((256-count)+1);
1631 *q++=(*pixels);
1632 pixels+=count;
1633 break;
1634 }
1635 /*
1636 Literal run.
1637 */
1638 count=0;
1639 while ((*(pixels+count) != *(pixels+count+1)) ||
1640 (*(pixels+count+1) != *(pixels+count+2)))
1641 {
1642 packbits[count+1]=pixels[count];
1643 count++;
cristybb503372010-05-27 20:51:26 +00001644 if (((ssize_t) count >= (i-3)) || (count >= 127))
cristy4aff1572010-02-15 13:34:01 +00001645 break;
1646 }
1647 i-=count;
1648 *packbits=(unsigned char) (count-1);
cristybb503372010-05-27 20:51:26 +00001649 for (j=0; j <= (ssize_t) count; j++)
cristy4aff1572010-02-15 13:34:01 +00001650 *q++=packbits[j];
1651 pixels+=count;
1652 break;
1653 }
1654 }
1655 }
1656 *q++=(unsigned char) 128; /* EOD marker */
1657 packbits=(unsigned char *) RelinquishMagickMemory(packbits);
cristyb1459bc2010-03-23 21:41:43 +00001658 return((size_t) (q-compact_pixels));
cristy4aff1572010-02-15 13:34:01 +00001659}
1660
cristy875e28a2010-03-06 19:46:55 +00001661static void WritePackbitsLength(const PSDInfo *psd_info,
cristya20214e2010-08-28 16:52:13 +00001662 const ImageInfo *image_info,Image *image,Image *next_image,
1663 unsigned char *compact_pixels,const QuantumType quantum_type)
cristy3ed852e2009-09-05 21:47:34 +00001664{
cristy3ed852e2009-09-05 21:47:34 +00001665 QuantumInfo
1666 *quantum_info;
1667
1668 register const PixelPacket
1669 *p;
1670
1671 size_t
cristy4aff1572010-02-15 13:34:01 +00001672 length,
cristy3ed852e2009-09-05 21:47:34 +00001673 packet_size;
1674
cristy75f85ae2010-09-25 03:01:06 +00001675 ssize_t
1676 y;
1677
cristya20214e2010-08-28 16:52:13 +00001678 unsigned char
1679 *pixels;
1680
1681 if (next_image->depth > 8)
1682 next_image->depth=16;
1683 packet_size=next_image->depth > 8UL ? 2UL : 1UL;
cristyda16f162011-02-19 23:52:17 +00001684 (void) packet_size;
cristy4aff1572010-02-15 13:34:01 +00001685 quantum_info=AcquireQuantumInfo(image_info,image);
cristya20214e2010-08-28 16:52:13 +00001686 pixels=GetQuantumPixels(quantum_info);
1687 for (y=0; y < (ssize_t) next_image->rows; y++)
cristy3ed852e2009-09-05 21:47:34 +00001688 {
cristya20214e2010-08-28 16:52:13 +00001689 p=GetVirtualPixels(next_image,0,y,next_image->columns,1,&image->exception);
cristy3ed852e2009-09-05 21:47:34 +00001690 if (p == (const PixelPacket *) NULL)
cristy4aff1572010-02-15 13:34:01 +00001691 break;
cristya20214e2010-08-28 16:52:13 +00001692 length=ExportQuantumPixels(next_image,(CacheView *) NULL,quantum_info,
cristy3ed852e2009-09-05 21:47:34 +00001693 quantum_type,pixels,&image->exception);
cristyb1459bc2010-03-23 21:41:43 +00001694 length=PSDPackbitsEncodeImage(image,length,pixels,compact_pixels);
cristy875e28a2010-03-06 19:46:55 +00001695 (void) SetPSDOffset(psd_info,image,length);
cristy4aff1572010-02-15 13:34:01 +00001696 }
1697 quantum_info=DestroyQuantumInfo(quantum_info);
1698}
1699
cristy875e28a2010-03-06 19:46:55 +00001700static void WriteOneChannel(const PSDInfo *psd_info,const ImageInfo *image_info,
cristya20214e2010-08-28 16:52:13 +00001701 Image *image,Image *next_image,unsigned char *compact_pixels,
1702 const QuantumType quantum_type,const MagickBooleanType compression_flag)
cristy4aff1572010-02-15 13:34:01 +00001703{
1704 int
1705 y;
1706
cristy0910f242010-04-01 18:55:09 +00001707 MagickBooleanType
1708 monochrome;
1709
cristy4aff1572010-02-15 13:34:01 +00001710 QuantumInfo
1711 *quantum_info;
1712
1713 register const PixelPacket
1714 *p;
1715
cristybb503372010-05-27 20:51:26 +00001716 register ssize_t
cristy0910f242010-04-01 18:55:09 +00001717 i;
1718
cristy4aff1572010-02-15 13:34:01 +00001719 size_t
1720 length,
1721 packet_size;
1722
cristya20214e2010-08-28 16:52:13 +00001723 unsigned char
1724 *pixels;
1725
cristy50aea4a2010-03-09 17:37:44 +00001726 (void) psd_info;
cristy4aff1572010-02-15 13:34:01 +00001727 if ((compression_flag != MagickFalse) &&
cristya20214e2010-08-28 16:52:13 +00001728 (next_image->compression != RLECompression))
cristy4aff1572010-02-15 13:34:01 +00001729 (void) WriteBlobMSBShort(image,0);
cristya20214e2010-08-28 16:52:13 +00001730 if (next_image->depth > 8)
1731 next_image->depth=16;
cristybf9f4ba2010-08-20 18:13:29 +00001732 monochrome=IsMonochromeImage(image,&image->exception) && (image->depth == 1)
1733 ? MagickTrue : MagickFalse;
cristya20214e2010-08-28 16:52:13 +00001734 packet_size=next_image->depth > 8UL ? 2UL : 1UL;
cristyda16f162011-02-19 23:52:17 +00001735 (void) packet_size;
cristy4aff1572010-02-15 13:34:01 +00001736 quantum_info=AcquireQuantumInfo(image_info,image);
cristya20214e2010-08-28 16:52:13 +00001737 pixels=GetQuantumPixels(quantum_info);
1738 for (y=0; y < (ssize_t) next_image->rows; y++)
cristy4aff1572010-02-15 13:34:01 +00001739 {
cristya20214e2010-08-28 16:52:13 +00001740 p=GetVirtualPixels(next_image,0,y,next_image->columns,1,&image->exception);
cristy4aff1572010-02-15 13:34:01 +00001741 if (p == (const PixelPacket *) NULL)
1742 break;
cristya20214e2010-08-28 16:52:13 +00001743 length=ExportQuantumPixels(next_image,(CacheView *) NULL,quantum_info,
cristy4aff1572010-02-15 13:34:01 +00001744 quantum_type,pixels,&image->exception);
cristy0910f242010-04-01 18:55:09 +00001745 if (monochrome != MagickFalse)
cristybb503372010-05-27 20:51:26 +00001746 for (i=0; i < (ssize_t) length; i++)
cristy0910f242010-04-01 18:55:09 +00001747 pixels[i]=(~pixels[i]);
cristya20214e2010-08-28 16:52:13 +00001748 if (next_image->compression != RLECompression)
cristy4aff1572010-02-15 13:34:01 +00001749 (void) WriteBlob(image,length,pixels);
1750 else
1751 {
cristyb1459bc2010-03-23 21:41:43 +00001752 length=PSDPackbitsEncodeImage(image,length,pixels,compact_pixels);
1753 (void) WriteBlob(image,length,compact_pixels);
cristy4aff1572010-02-15 13:34:01 +00001754 }
cristy3ed852e2009-09-05 21:47:34 +00001755 }
1756 quantum_info=DestroyQuantumInfo(quantum_info);
1757}
1758
cristy875e28a2010-03-06 19:46:55 +00001759static MagickBooleanType WriteImageChannels(const PSDInfo *psd_info,
cristya20214e2010-08-28 16:52:13 +00001760 const ImageInfo *image_info,Image *image,Image *next_image,
cristy875e28a2010-03-06 19:46:55 +00001761 const MagickBooleanType separate)
cristy3ed852e2009-09-05 21:47:34 +00001762{
1763 int
1764 i;
1765
1766 size_t
1767 channels,
1768 packet_size;
1769
1770 unsigned char
cristya20214e2010-08-28 16:52:13 +00001771 *compact_pixels;
cristy3ed852e2009-09-05 21:47:34 +00001772
1773 /*
cristy875e28a2010-03-06 19:46:55 +00001774 Write uncompressed pixels as separate planes.
cristy3ed852e2009-09-05 21:47:34 +00001775 */
1776 channels=1;
cristya20214e2010-08-28 16:52:13 +00001777 packet_size=next_image->depth > 8UL ? 2UL : 1UL;
1778 compact_pixels=(unsigned char *) NULL;
1779 if (next_image->compression == RLECompression)
cristy4aff1572010-02-15 13:34:01 +00001780 {
cristya20214e2010-08-28 16:52:13 +00001781 compact_pixels=(unsigned char *) AcquireQuantumMemory(2*channels*
1782 next_image->columns,packet_size*sizeof(*compact_pixels));
1783 if (compact_pixels == (unsigned char *) NULL)
1784 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
cristy4aff1572010-02-15 13:34:01 +00001785 }
cristy3ed852e2009-09-05 21:47:34 +00001786 i=0;
cristya20214e2010-08-28 16:52:13 +00001787 if (IsGrayImage(next_image,&next_image->exception) != MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00001788 {
cristya20214e2010-08-28 16:52:13 +00001789 if (next_image->compression == RLECompression)
cristy4aff1572010-02-15 13:34:01 +00001790 {
1791 /*
1792 Packbits compression.
1793 */
1794 (void) WriteBlobMSBShort(image,1);
cristya20214e2010-08-28 16:52:13 +00001795 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy95119782010-06-01 17:40:58 +00001796 compact_pixels,GrayQuantum);
cristya20214e2010-08-28 16:52:13 +00001797 if (next_image->matte != MagickFalse)
1798 WritePackbitsLength(psd_info,image_info,image,next_image,
cristyb1459bc2010-03-23 21:41:43 +00001799 compact_pixels,AlphaQuantum);
cristy4aff1572010-02-15 13:34:01 +00001800 }
cristya20214e2010-08-28 16:52:13 +00001801 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1802 GrayQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1803 MagickFalse);
1804 if (next_image->matte != MagickFalse)
1805 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1806 AlphaQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1807 MagickFalse);
cristy6886a752010-04-23 18:23:20 +00001808 (void) SetImageProgress(image,SaveImagesTag,0,1);
cristy3ed852e2009-09-05 21:47:34 +00001809 }
cristy0910f242010-04-01 18:55:09 +00001810 else
cristya20214e2010-08-28 16:52:13 +00001811 if (next_image->storage_class == PseudoClass)
cristy0910f242010-04-01 18:55:09 +00001812 {
cristya20214e2010-08-28 16:52:13 +00001813 if (next_image->compression == RLECompression)
cristy0910f242010-04-01 18:55:09 +00001814 {
1815 /*
1816 Packbits compression.
1817 */
1818 (void) WriteBlobMSBShort(image,1);
cristya20214e2010-08-28 16:52:13 +00001819 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy95119782010-06-01 17:40:58 +00001820 compact_pixels,IndexQuantum);
cristya20214e2010-08-28 16:52:13 +00001821 if (next_image->matte != MagickFalse)
1822 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy6886a752010-04-23 18:23:20 +00001823 compact_pixels,AlphaQuantum);
cristy0910f242010-04-01 18:55:09 +00001824 }
cristya20214e2010-08-28 16:52:13 +00001825 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1826 IndexQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1827 MagickFalse);
1828 if (next_image->matte != MagickFalse)
1829 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1830 AlphaQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1831 MagickFalse);
cristy0910f242010-04-01 18:55:09 +00001832 (void) SetImageProgress(image,SaveImagesTag,0,1);
1833 }
1834 else
1835 {
cristya20214e2010-08-28 16:52:13 +00001836 if (next_image->colorspace == CMYKColorspace)
1837 (void) NegateImage(next_image,MagickFalse);
1838 if (next_image->compression == RLECompression)
cristy0910f242010-04-01 18:55:09 +00001839 {
1840 /*
1841 Packbits compression.
1842 */
1843 (void) WriteBlobMSBShort(image,1);
cristya20214e2010-08-28 16:52:13 +00001844 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001845 compact_pixels,RedQuantum);
cristya20214e2010-08-28 16:52:13 +00001846 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001847 compact_pixels,GreenQuantum);
cristya20214e2010-08-28 16:52:13 +00001848 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001849 compact_pixels,BlueQuantum);
cristya20214e2010-08-28 16:52:13 +00001850 if (next_image->colorspace == CMYKColorspace)
1851 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy0910f242010-04-01 18:55:09 +00001852 compact_pixels,BlackQuantum);
cristya20214e2010-08-28 16:52:13 +00001853 if (next_image->matte != MagickFalse)
1854 WritePackbitsLength(psd_info,image_info,image,next_image,
cristy860cc732010-05-19 16:38:37 +00001855 compact_pixels,AlphaQuantum);
cristy0910f242010-04-01 18:55:09 +00001856 }
1857 (void) SetImageProgress(image,SaveImagesTag,0,6);
cristya20214e2010-08-28 16:52:13 +00001858 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1859 RedQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1860 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001861 (void) SetImageProgress(image,SaveImagesTag,1,6);
cristya20214e2010-08-28 16:52:13 +00001862 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1863 GreenQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1864 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001865 (void) SetImageProgress(image,SaveImagesTag,2,6);
cristya20214e2010-08-28 16:52:13 +00001866 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1867 BlueQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1868 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001869 (void) SetImageProgress(image,SaveImagesTag,3,6);
cristya20214e2010-08-28 16:52:13 +00001870 if (next_image->colorspace == CMYKColorspace)
1871 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1872 BlackQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1873 MagickFalse);
cristy860cc732010-05-19 16:38:37 +00001874 (void) SetImageProgress(image,SaveImagesTag,4,6);
cristya20214e2010-08-28 16:52:13 +00001875 if (next_image->matte != MagickFalse)
1876 WriteOneChannel(psd_info,image_info,image,next_image,compact_pixels,
1877 AlphaQuantum,(i++ == 0) || (separate != MagickFalse) ? MagickTrue :
1878 MagickFalse);
cristy0910f242010-04-01 18:55:09 +00001879 (void) SetImageProgress(image,SaveImagesTag,5,6);
cristya20214e2010-08-28 16:52:13 +00001880 if (next_image->colorspace == CMYKColorspace)
1881 (void) NegateImage(next_image,MagickFalse);
cristy0910f242010-04-01 18:55:09 +00001882 }
cristya20214e2010-08-28 16:52:13 +00001883 if (next_image->compression == RLECompression)
1884 compact_pixels=(unsigned char *) RelinquishMagickMemory(compact_pixels);
cristy3ed852e2009-09-05 21:47:34 +00001885 return(MagickTrue);
1886}
1887
cristy3ed852e2009-09-05 21:47:34 +00001888static void WritePascalString(Image* inImage,const char *inString,int inPad)
1889{
1890 size_t
cristya20214e2010-08-28 16:52:13 +00001891 length;
cristy3ed852e2009-09-05 21:47:34 +00001892
cristya20214e2010-08-28 16:52:13 +00001893 register ssize_t
1894 i;
cristy3ed852e2009-09-05 21:47:34 +00001895
cristya20214e2010-08-28 16:52:13 +00001896 /*
1897 Max length is 255.
1898 */
1899 length=(strlen(inString) > 255UL ) ? 255UL : strlen(inString);
1900 if (length == 0)
1901 (void) WriteBlobByte(inImage,0);
cristy3ed852e2009-09-05 21:47:34 +00001902 else
cristya20214e2010-08-28 16:52:13 +00001903 {
1904 (void) WriteBlobByte(inImage,(unsigned char) length);
1905 (void) WriteBlob(inImage, length, (const unsigned char *) inString);
1906 }
1907 length++;
1908 if ((length % inPad) == 0)
cristy3ed852e2009-09-05 21:47:34 +00001909 return;
cristya20214e2010-08-28 16:52:13 +00001910 for (i=0; i < (ssize_t) (inPad-(length % inPad)); i++)
cristy3ed852e2009-09-05 21:47:34 +00001911 (void) WriteBlobByte(inImage,0);
1912}
1913
1914static void WriteResolutionResourceBlock(Image *image)
1915{
cristy56ed31c2010-03-22 00:46:21 +00001916 double
1917 x_resolution,
1918 y_resolution;
cristy3ed852e2009-09-05 21:47:34 +00001919
1920 unsigned short
1921 units;
1922
1923 x_resolution=65536.0*image->x_resolution+0.5;
1924 y_resolution=65536.0*image->y_resolution+0.5;
1925 units=1;
1926 if (image->units == PixelsPerCentimeterResolution)
1927 {
1928 x_resolution=2.54*65536.0*image->x_resolution*0.5;
1929 y_resolution=2.54*65536.0*image->y_resolution+0.5;
1930 units=2;
1931 }
1932 (void) WriteBlob(image,4,(const unsigned char *) "8BIM");
1933 (void) WriteBlobMSBShort(image,0x03ED);
1934 (void) WriteBlobMSBShort(image,0);
1935 (void) WriteBlobMSBLong(image,16); /* resource size */
cristy56ed31c2010-03-22 00:46:21 +00001936 (void) WriteBlobMSBLong(image,(unsigned int) (x_resolution+0.5));
cristy3ed852e2009-09-05 21:47:34 +00001937 (void) WriteBlobMSBShort(image,units); /* horizontal resolution unit */
1938 (void) WriteBlobMSBShort(image,units); /* width unit */
cristy56ed31c2010-03-22 00:46:21 +00001939 (void) WriteBlobMSBLong(image,(unsigned int) (y_resolution+0.5));
cristy3ed852e2009-09-05 21:47:34 +00001940 (void) WriteBlobMSBShort(image,units); /* vertical resolution unit */
1941 (void) WriteBlobMSBShort(image,units); /* height unit */
1942}
1943
cristyd4d3f742010-04-25 20:36:50 +00001944static void RemoveICCProfileFromResourceBlock(StringInfo *bim_profile)
1945{
cristy0b796e62010-04-29 00:38:45 +00001946 register const unsigned char
cristyd4d3f742010-04-25 20:36:50 +00001947 *p;
1948
1949 size_t
1950 length;
1951
1952 unsigned char
1953 *datum;
1954
cristy6befb0f2010-05-31 14:33:15 +00001955 unsigned int
cristyd4d3f742010-04-25 20:36:50 +00001956 count,
cristy6befb0f2010-05-31 14:33:15 +00001957 long_sans;
cristyd4d3f742010-04-25 20:36:50 +00001958
1959 unsigned short
1960 id,
1961 short_sans;
1962
1963 length=GetStringInfoLength(bim_profile);
1964 if (length < 16)
cristy0b796e62010-04-29 00:38:45 +00001965 return;
cristyd4d3f742010-04-25 20:36:50 +00001966 datum=GetStringInfoDatum(bim_profile);
1967 for (p=datum; (p >= datum) && (p < (datum+length-16)); )
1968 {
cristy4176bb22010-05-01 16:29:09 +00001969 register unsigned char
1970 *q;
1971
1972 q=(unsigned char *) p;
cristyd4d3f742010-04-25 20:36:50 +00001973 if (LocaleNCompare((const char *) p,"8BIM",4) != 0)
1974 break;
cristy6befb0f2010-05-31 14:33:15 +00001975 p=PushLongPixel(MSBEndian,p,&long_sans);
cristyd4d3f742010-04-25 20:36:50 +00001976 p=PushShortPixel(MSBEndian,p,&id);
1977 p=PushShortPixel(MSBEndian,p,&short_sans);
1978 p=PushLongPixel(MSBEndian,p,&count);
1979 if (id == 0x0000040f)
1980 {
cristy4176bb22010-05-01 16:29:09 +00001981 (void) CopyMagickMemory(q,q+PSDQuantum(count)+12,length-
1982 (PSDQuantum(count)+12)-(q-datum));
1983 SetStringInfoLength(bim_profile,length-(PSDQuantum(count)+12));
cristyd4d3f742010-04-25 20:36:50 +00001984 break;
1985 }
1986 p+=count;
1987 if ((count & 0x01) != 0)
1988 p++;
1989 }
1990}
1991
cristyf11065e2010-05-14 13:26:59 +00001992static void RemoveResolutionFromResourceBlock(StringInfo *bim_profile)
1993{
1994 register const unsigned char
1995 *p;
1996
1997 size_t
1998 length;
1999
2000 unsigned char
2001 *datum;
2002
cristy6befb0f2010-05-31 14:33:15 +00002003 unsigned int
cristyf11065e2010-05-14 13:26:59 +00002004 count,
cristy6befb0f2010-05-31 14:33:15 +00002005 long_sans;
cristyf11065e2010-05-14 13:26:59 +00002006
2007 unsigned short
2008 id,
2009 short_sans;
2010
2011 length=GetStringInfoLength(bim_profile);
2012 if (length < 16)
2013 return;
2014 datum=GetStringInfoDatum(bim_profile);
2015 for (p=datum; (p >= datum) && (p < (datum+length-16)); )
2016 {
2017 register unsigned char
2018 *q;
2019
2020 q=(unsigned char *) p;
2021 if (LocaleNCompare((const char *) p,"8BIM",4) != 0)
2022 break;
cristy6befb0f2010-05-31 14:33:15 +00002023 p=PushLongPixel(MSBEndian,p,&long_sans);
cristyf11065e2010-05-14 13:26:59 +00002024 p=PushShortPixel(MSBEndian,p,&id);
2025 p=PushShortPixel(MSBEndian,p,&short_sans);
2026 p=PushLongPixel(MSBEndian,p,&count);
2027 if (id == 0x000003ed)
2028 {
2029 (void) CopyMagickMemory(q,q+PSDQuantum(count)+12,length-
2030 (PSDQuantum(count)+12)-(q-datum));
2031 SetStringInfoLength(bim_profile,length-(PSDQuantum(count)+12));
2032 break;
2033 }
2034 p+=count;
2035 if ((count & 0x01) != 0)
2036 p++;
2037 }
2038}
2039
cristy3ed852e2009-09-05 21:47:34 +00002040static MagickBooleanType WritePSDImage(const ImageInfo *image_info,Image *image)
2041{
2042 const char
cristya20214e2010-08-28 16:52:13 +00002043 *property;
cristy3ed852e2009-09-05 21:47:34 +00002044
2045 const StringInfo
cristy749d2152010-04-04 23:47:33 +00002046 *icc_profile;
cristy3ed852e2009-09-05 21:47:34 +00002047
cristya20214e2010-08-28 16:52:13 +00002048 Image
2049 *base_image,
2050 *next_image;
2051
cristy3ed852e2009-09-05 21:47:34 +00002052 MagickBooleanType
cristy3ed852e2009-09-05 21:47:34 +00002053 status;
2054
cristy875e28a2010-03-06 19:46:55 +00002055 PSDInfo
2056 psd_info;
2057
cristybb503372010-05-27 20:51:26 +00002058 register ssize_t
cristy3ed852e2009-09-05 21:47:34 +00002059 i;
2060
2061 size_t
cristya20214e2010-08-28 16:52:13 +00002062 channel_size,
2063 channelLength,
2064 layer_count,
2065 layer_info_size,
cristy749d2152010-04-04 23:47:33 +00002066 length,
cristy3ed852e2009-09-05 21:47:34 +00002067 num_channels,
cristya20214e2010-08-28 16:52:13 +00002068 packet_size,
2069 rounded_layer_info_size;
cristy3ed852e2009-09-05 21:47:34 +00002070
cristy0b796e62010-04-29 00:38:45 +00002071 StringInfo
2072 *bim_profile;
2073
cristy3ed852e2009-09-05 21:47:34 +00002074 unsigned char
2075 layer_name[4];
2076
cristy3ed852e2009-09-05 21:47:34 +00002077 /*
cristy56ed31c2010-03-22 00:46:21 +00002078 Open image file.
cristy3ed852e2009-09-05 21:47:34 +00002079 */
2080 assert(image_info != (const ImageInfo *) NULL);
2081 assert(image_info->signature == MagickSignature);
2082 assert(image != (Image *) NULL);
2083 assert(image->signature == MagickSignature);
2084 if (image->debug != MagickFalse)
2085 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
2086 status=OpenBlob(image_info,image,WriteBinaryBlobMode,&image->exception);
2087 if (status == MagickFalse)
2088 return(status);
2089 packet_size=(size_t) (image->depth > 8 ? 6 : 3);
2090 if (image->matte != MagickFalse)
2091 packet_size+=image->depth > 8 ? 2 : 1;
cristy875e28a2010-03-06 19:46:55 +00002092 psd_info.version=1;
2093 if ((LocaleCompare(image_info->magick,"PSB") == 0) ||
2094 (image->columns > 30000) || (image->rows > 30000))
2095 psd_info.version=2;
cristy50aea4a2010-03-09 17:37:44 +00002096 (void) WriteBlob(image,4,(const unsigned char *) "8BPS");
cristy875e28a2010-03-06 19:46:55 +00002097 (void) WriteBlobMSBShort(image,psd_info.version); /* version */
2098 for (i=1; i <= 6; i++)
2099 (void) WriteBlobByte(image, 0); /* 6 bytes of reserved */
cristy2045da32010-04-16 00:59:35 +00002100 if (IsGrayImage(image,&image->exception) != MagickFalse)
cristy6886a752010-04-23 18:23:20 +00002101 num_channels=(image->matte != MagickFalse ? 2UL : 1UL);
cristy3ed852e2009-09-05 21:47:34 +00002102 else
cristy0910f242010-04-01 18:55:09 +00002103 if (image->storage_class == PseudoClass)
cristy6886a752010-04-23 18:23:20 +00002104 num_channels=(image->matte != MagickFalse ? 2UL : 1UL);
cristy0910f242010-04-01 18:55:09 +00002105 else
2106 {
2107 if (image->colorspace != CMYKColorspace)
cristy6886a752010-04-23 18:23:20 +00002108 num_channels=(image->matte != MagickFalse ? 4UL : 3UL);
cristy0910f242010-04-01 18:55:09 +00002109 else
cristy6886a752010-04-23 18:23:20 +00002110 num_channels=(image->matte != MagickFalse ? 5UL : 4UL);
cristy0910f242010-04-01 18:55:09 +00002111 }
cristy3ed852e2009-09-05 21:47:34 +00002112 (void) WriteBlobMSBShort(image,(unsigned short) num_channels);
cristy56ed31c2010-03-22 00:46:21 +00002113 (void) WriteBlobMSBLong(image,(unsigned int) image->rows);
2114 (void) WriteBlobMSBLong(image,(unsigned int) image->columns);
cristy2045da32010-04-16 00:59:35 +00002115 if (IsGrayImage(image,&image->exception) != MagickFalse)
cristy3ed852e2009-09-05 21:47:34 +00002116 {
cristy2045da32010-04-16 00:59:35 +00002117 MagickBooleanType
2118 monochrome;
2119
cristy0910f242010-04-01 18:55:09 +00002120 /*
2121 Write depth & mode.
2122 */
cristy0b55dd32010-06-13 19:04:44 +00002123 monochrome=IsMonochromeImage(image,&image->exception) &&
cristybf9f4ba2010-08-20 18:13:29 +00002124 (image->depth == 1) ? MagickTrue : MagickFalse;
cristy284c7d82010-04-24 00:19:14 +00002125 (void) WriteBlobMSBShort(image,(unsigned short)
2126 (monochrome != MagickFalse ? 1 : image->depth > 8 ? 16 : 8));
cristya20214e2010-08-28 16:52:13 +00002127 (void) WriteBlobMSBShort(image,monochrome != MagickFalse ? BitmapMode :
2128 GrayscaleMode);
cristy3ed852e2009-09-05 21:47:34 +00002129 }
2130 else
2131 {
cristya20214e2010-08-28 16:52:13 +00002132 (void) WriteBlobMSBShort(image,(unsigned short) (image->storage_class ==
2133 PseudoClass ? 8 : image->depth > 8 ? 16 : 8));
cristy48845392010-06-02 01:19:17 +00002134 if (((image_info->colorspace != UndefinedColorspace) ||
cristy0910f242010-04-01 18:55:09 +00002135 (image->colorspace != CMYKColorspace)) &&
cristy48845392010-06-02 01:19:17 +00002136 (image_info->colorspace != CMYKColorspace))
cristy0910f242010-04-01 18:55:09 +00002137 {
2138 if (image->colorspace != RGBColorspace)
2139 (void) TransformImageColorspace(image,RGBColorspace);
2140 (void) WriteBlobMSBShort(image,(unsigned short)
cristy2045da32010-04-16 00:59:35 +00002141 (image->storage_class == PseudoClass ? IndexedMode : RGBMode));
cristy0910f242010-04-01 18:55:09 +00002142 }
2143 else
2144 {
cristy48845392010-06-02 01:19:17 +00002145 if (image->colorspace != CMYKColorspace)
cristy0910f242010-04-01 18:55:09 +00002146 (void) TransformImageColorspace(image,CMYKColorspace);
cristy2045da32010-04-16 00:59:35 +00002147 (void) WriteBlobMSBShort(image,CMYKMode);
cristy0910f242010-04-01 18:55:09 +00002148 }
cristy3ed852e2009-09-05 21:47:34 +00002149 }
cristyca9ddb02010-04-16 01:01:18 +00002150 if ((IsGrayImage(image,&image->exception) != MagickFalse) ||
2151 (image->storage_class == DirectClass) || (image->colors > 256))
cristy3ed852e2009-09-05 21:47:34 +00002152 (void) WriteBlobMSBLong(image,0);
2153 else
2154 {
2155 /*
2156 Write PSD raster colormap.
2157 */
2158 (void) WriteBlobMSBLong(image,768);
cristybb503372010-05-27 20:51:26 +00002159 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +00002160 (void) WriteBlobByte(image,ScaleQuantumToChar(image->colormap[i].red));
2161 for ( ; i < 256; i++)
2162 (void) WriteBlobByte(image,0);
cristybb503372010-05-27 20:51:26 +00002163 for (i=0; i < (ssize_t) image->colors; i++)
cristya20214e2010-08-28 16:52:13 +00002164 (void) WriteBlobByte(image,ScaleQuantumToChar(
2165 image->colormap[i].green));
cristy3ed852e2009-09-05 21:47:34 +00002166 for ( ; i < 256; i++)
2167 (void) WriteBlobByte(image,0);
cristybb503372010-05-27 20:51:26 +00002168 for (i=0; i < (ssize_t) image->colors; i++)
cristy3ed852e2009-09-05 21:47:34 +00002169 (void) WriteBlobByte(image,ScaleQuantumToChar(image->colormap[i].blue));
2170 for ( ; i < 256; i++)
2171 (void) WriteBlobByte(image,0);
2172 }
2173 /*
2174 Image resource block.
2175 */
cristy749d2152010-04-04 23:47:33 +00002176 length=28; /* 0x03EB */
cristy0b796e62010-04-29 00:38:45 +00002177 bim_profile=(StringInfo *) GetImageProfile(image,"8bim");
cristy749d2152010-04-04 23:47:33 +00002178 icc_profile=GetImageProfile(image,"icc");
cristyd4d3f742010-04-25 20:36:50 +00002179 if (bim_profile != (StringInfo *) NULL)
2180 {
cristy0b796e62010-04-29 00:38:45 +00002181 bim_profile=CloneStringInfo(bim_profile);
cristyd4d3f742010-04-25 20:36:50 +00002182 if (icc_profile != (StringInfo *) NULL)
2183 RemoveICCProfileFromResourceBlock(bim_profile);
cristyf11065e2010-05-14 13:26:59 +00002184 RemoveResolutionFromResourceBlock(bim_profile);
cristyd4d3f742010-04-25 20:36:50 +00002185 length+=PSDQuantum(GetStringInfoLength(bim_profile));
2186 }
cristy0b796e62010-04-29 00:38:45 +00002187 if (icc_profile != (const StringInfo *) NULL)
cristy749d2152010-04-04 23:47:33 +00002188 length+=PSDQuantum(GetStringInfoLength(icc_profile))+12;
cristy284c7d82010-04-24 00:19:14 +00002189 (void) WriteBlobMSBLong(image,(unsigned int) length);
cristyf11065e2010-05-14 13:26:59 +00002190 WriteResolutionResourceBlock(image);
cristy4176bb22010-05-01 16:29:09 +00002191 if (bim_profile != (StringInfo *) NULL)
2192 {
2193 (void) WriteBlob(image,GetStringInfoLength(bim_profile),
2194 GetStringInfoDatum(bim_profile));
2195 bim_profile=DestroyStringInfo(bim_profile);
2196 }
cristy749d2152010-04-04 23:47:33 +00002197 if (icc_profile != (StringInfo *) NULL)
cristy3ed852e2009-09-05 21:47:34 +00002198 {
cristy749d2152010-04-04 23:47:33 +00002199 (void) WriteBlob(image,4,(const unsigned char *) "8BIM");
cristy4176bb22010-05-01 16:29:09 +00002200 (void) WriteBlobMSBShort(image,0x0000040F);
cristy749d2152010-04-04 23:47:33 +00002201 (void) WriteBlobMSBShort(image,0);
cristy284c7d82010-04-24 00:19:14 +00002202 (void) WriteBlobMSBLong(image,(unsigned int) GetStringInfoLength(
2203 icc_profile));
cristy749d2152010-04-04 23:47:33 +00002204 (void) WriteBlob(image,GetStringInfoLength(icc_profile),
2205 GetStringInfoDatum(icc_profile));
cristye195f262010-04-16 18:12:35 +00002206 if ((MagickOffsetType) GetStringInfoLength(icc_profile) !=
cristy2045da32010-04-16 00:59:35 +00002207 PSDQuantum(GetStringInfoLength(icc_profile)))
cristy749d2152010-04-04 23:47:33 +00002208 (void) WriteBlobByte(image,0);
cristye6365592010-04-02 17:31:23 +00002209 }
cristy48845392010-06-02 01:19:17 +00002210 layer_count=0;
2211 layer_info_size=2;
cristy2837bcc2010-08-07 23:57:39 +00002212 base_image=GetNextImageInList(image);
2213 if ((image->matte != MagickFalse) && (base_image == (Image *) NULL))
2214 base_image=image;
cristya20214e2010-08-28 16:52:13 +00002215 next_image=base_image;
2216 while ( next_image != NULL )
2217 {
2218 packet_size=next_image->depth > 8 ? 2UL : 1UL;
2219 if (IsGrayImage(next_image,&image->exception) != MagickFalse)
2220 num_channels=next_image->matte != MagickFalse ? 2UL : 1UL;
cristy3ed852e2009-09-05 21:47:34 +00002221 else
cristya20214e2010-08-28 16:52:13 +00002222 if (next_image->storage_class == PseudoClass)
2223 num_channels=next_image->matte != MagickFalse ? 2UL : 1UL;
cristy2045da32010-04-16 00:59:35 +00002224 else
cristya20214e2010-08-28 16:52:13 +00002225 if (next_image->colorspace != CMYKColorspace)
2226 num_channels=next_image->matte != MagickFalse ? 4UL : 3UL;
cristy2045da32010-04-16 00:59:35 +00002227 else
cristya20214e2010-08-28 16:52:13 +00002228 num_channels=next_image->matte != MagickFalse ? 5UL : 4UL;
2229 channelLength=(size_t) (next_image->columns*next_image->rows*packet_size+2);
cristy48845392010-06-02 01:19:17 +00002230 layer_info_size+=(size_t) (4*4+2+num_channels*6+(psd_info.version == 1 ? 8 :
2231 16)+4*1+4+num_channels*channelLength);
cristya20214e2010-08-28 16:52:13 +00002232 property=(const char *) GetImageProperty(next_image,"label");
2233 if (property == (const char *) NULL)
2234 layer_info_size+=16;
cristyde9b8f52010-03-12 01:17:00 +00002235 else
2236 {
cristya20214e2010-08-28 16:52:13 +00002237 size_t
2238 length;
2239
2240 length=strlen(property);
2241 layer_info_size+=8+length+(4-(length % 4));
cristyde9b8f52010-03-12 01:17:00 +00002242 }
cristy4aff1572010-02-15 13:34:01 +00002243 layer_count++;
cristya20214e2010-08-28 16:52:13 +00002244 next_image=GetNextImageInList(next_image);
cristy3ed852e2009-09-05 21:47:34 +00002245 }
cristy144f1b62010-05-18 00:52:09 +00002246 if (layer_count == 0)
cristy875e28a2010-03-06 19:46:55 +00002247 (void) SetPSDSize(&psd_info,image,0);
cristy3ed852e2009-09-05 21:47:34 +00002248 else
cristya20214e2010-08-28 16:52:13 +00002249 {
cristy8e9bd3e2010-08-29 00:03:56 +00002250 CompressionType
2251 compression;
2252
2253 (void) SetPSDSize(&psd_info,image,layer_info_size+
2254 (psd_info.version == 1 ? 8 : 16));
2255 if ((layer_info_size/2) != ((layer_info_size+1)/2))
2256 rounded_layer_info_size=layer_info_size+1;
cristya20214e2010-08-28 16:52:13 +00002257 else
cristy8e9bd3e2010-08-29 00:03:56 +00002258 rounded_layer_info_size=layer_info_size;
2259 (void) SetPSDSize(&psd_info,image,rounded_layer_info_size);
2260 (void) WriteBlobMSBShort(image,(unsigned short) layer_count);
2261 layer_count=1;
2262 compression=base_image->compression;
2263 next_image=base_image;
2264 while (next_image != NULL)
2265 {
2266 next_image->compression=NoCompression;
2267 (void) WriteBlobMSBLong(image,0);
2268 (void) WriteBlobMSBLong(image,0);
2269 (void) WriteBlobMSBLong(image,(unsigned int) next_image->rows);
2270 (void) WriteBlobMSBLong(image,(unsigned int) next_image->columns);
2271 packet_size=next_image->depth > 8 ? 2UL : 1UL;
2272 channel_size=(unsigned int) ((packet_size*next_image->rows*
2273 next_image->columns)+2);
2274 if ((IsGrayImage(next_image,&image->exception) != MagickFalse) ||
2275 (next_image->storage_class == PseudoClass))
2276 {
2277 (void) WriteBlobMSBShort(image,(unsigned short)
2278 (next_image->matte != MagickFalse ? 2 : 1));
2279 (void) WriteBlobMSBShort(image,0);
2280 (void) SetPSDSize(&psd_info,image,channel_size);
2281 if (next_image->matte != MagickFalse)
2282 {
2283 (void) WriteBlobMSBShort(image,(unsigned short) -1);
2284 (void) SetPSDSize(&psd_info,image,channel_size);
2285 }
2286 }
2287 else
2288 if (next_image->colorspace != CMYKColorspace)
2289 {
2290 (void) WriteBlobMSBShort(image,(unsigned short)
2291 (next_image->matte != MagickFalse ? 4 : 3));
2292 (void) WriteBlobMSBShort(image,0);
2293 (void) SetPSDSize(&psd_info,image,channel_size);
2294 (void) WriteBlobMSBShort(image,1);
2295 (void) SetPSDSize(&psd_info,image,channel_size);
2296 (void) WriteBlobMSBShort(image,2);
2297 (void) SetPSDSize(&psd_info,image,channel_size);
2298 if (next_image->matte!= MagickFalse )
2299 {
2300 (void) WriteBlobMSBShort(image,(unsigned short) -1);
2301 (void) SetPSDSize(&psd_info,image,channel_size);
2302 }
2303 }
2304 else
2305 {
2306 (void) WriteBlobMSBShort(image,(unsigned short)
2307 (next_image->matte ? 5 : 4));
2308 (void) WriteBlobMSBShort(image,0);
2309 (void) SetPSDSize(&psd_info,image,channel_size);
2310 (void) WriteBlobMSBShort(image,1);
2311 (void) SetPSDSize(&psd_info,image,channel_size);
2312 (void) WriteBlobMSBShort(image,2);
2313 (void) SetPSDSize(&psd_info,image,channel_size);
2314 (void) WriteBlobMSBShort(image,3);
2315 (void) SetPSDSize(&psd_info,image,channel_size);
2316 if (next_image->matte)
2317 {
2318 (void) WriteBlobMSBShort(image,(unsigned short) -1);
2319 (void) SetPSDSize(&psd_info,image,channel_size);
2320 }
2321 }
2322 (void) WriteBlob(image,4,(const unsigned char *) "8BIM");
2323 (void) WriteBlob(image,4,(const unsigned char *)
2324 CompositeOperatorToPSDBlendMode(next_image->compose));
2325 (void) WriteBlobByte(image,255); /* layer opacity */
2326 (void) WriteBlobByte(image,0);
2327 (void) WriteBlobByte(image,1); /* layer propertys - visible, etc. */
2328 (void) WriteBlobByte(image,0);
2329 property=(const char *) GetImageProperty(next_image,"label");
2330 if (property == (const char *) NULL)
2331 {
2332 (void) WriteBlobMSBLong(image,16);
2333 (void) WriteBlobMSBLong(image,0);
2334 (void) WriteBlobMSBLong(image,0);
cristyb51dff52011-05-19 16:55:47 +00002335 (void) FormatLocaleString((char *) layer_name,MaxTextExtent,
cristy8e9bd3e2010-08-29 00:03:56 +00002336 "L%06ld",(long) layer_count++);
2337 WritePascalString( image, (char*)layer_name, 4 );
2338 }
2339 else
2340 {
2341 size_t
2342 length;
cristy3ed852e2009-09-05 21:47:34 +00002343
cristy8e9bd3e2010-08-29 00:03:56 +00002344 length=strlen(property);
2345 (void) WriteBlobMSBLong(image,(unsigned int) (length+(4-
2346 (length % 4))+8));
2347 (void) WriteBlobMSBLong(image,0);
2348 (void) WriteBlobMSBLong(image,0);
2349 WritePascalString(image,property,4);
2350 }
2351 next_image=GetNextImageInList(next_image);
2352 }
2353 /*
2354 Now the image data!
2355 */
2356 next_image=base_image;
2357 while (next_image != NULL)
2358 {
2359 status=WriteImageChannels(&psd_info,image_info,image,next_image,
2360 MagickTrue);
2361 next_image=GetNextImageInList(next_image);
2362 }
2363 (void) WriteBlobMSBLong(image,0); /* user mask data */
2364 base_image->compression=compression;
cristy144f1b62010-05-18 00:52:09 +00002365 }
cristy144f1b62010-05-18 00:52:09 +00002366 /*
2367 Write composite image.
2368 */
cristy15e25cb2010-03-13 20:31:29 +00002369 status=WriteImageChannels(&psd_info,image_info,image,image,MagickFalse);
cristy3ed852e2009-09-05 21:47:34 +00002370 (void) CloseBlob(image);
2371 return(status);
2372}