1 /*
2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3 % %
4 % %
5 % %
6 % RRRR GGGG BBBB %
7 % R R G B B %
8 % RRRR G GG BBBB %
9 % R R G G B B %
10 % R R GGG BBBB %
11 % %
12 % %
13 % Read/Write Raw RGB Image Format %
14 % %
15 % Software Design %
16 % Cristy %
17 % July 1992 %
18 % %
19 % %
20 % Copyright 1999-2021 ImageMagick Studio LLC, a non-profit organization %
21 % dedicated to making software imaging solutions freely available. %
22 % %
23 % You may not use this file except in compliance with the License. You may %
24 % obtain a copy of the License at %
25 % %
26 % https://imagemagick.org/script/license.php %
27 % %
28 % Unless required by applicable law or agreed to in writing, software %
29 % distributed under the License is distributed on an "AS IS" BASIS, %
30 % WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %
31 % See the License for the specific language governing permissions and %
32 % limitations under the License. %
33 % %
34 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
35 %
36 %
37 */
38
39 /*
40 Include declarations.
41 */
42 #include "MagickCore/studio.h"
43 #include "MagickCore/blob.h"
44 #include "MagickCore/blob-private.h"
45 #include "MagickCore/cache.h"
46 #include "MagickCore/channel.h"
47 #include "MagickCore/colorspace.h"
48 #include "MagickCore/colorspace-private.h"
49 #include "MagickCore/constitute.h"
50 #include "MagickCore/exception.h"
51 #include "MagickCore/exception-private.h"
52 #include "MagickCore/image.h"
53 #include "MagickCore/image-private.h"
54 #include "MagickCore/list.h"
55 #include "MagickCore/magick.h"
56 #include "MagickCore/memory_.h"
57 #include "MagickCore/monitor.h"
58 #include "MagickCore/monitor-private.h"
59 #include "MagickCore/pixel-accessor.h"
60 #include "MagickCore/quantum-private.h"
61 #include "MagickCore/static.h"
62 #include "MagickCore/statistic.h"
63 #include "MagickCore/string_.h"
64 #include "MagickCore/module.h"
65 #include "MagickCore/utility.h"
66
67 /*
68 Forward declarations.
69 */
70 static MagickBooleanType
71 WriteRGBImage(const ImageInfo *,Image *,ExceptionInfo *);
72
73 /*
74 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
75 % %
76 % %
77 % %
78 % R e a d R G B I m a g e %
79 % %
80 % %
81 % %
82 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
83 %
84 % ReadRGBImage() reads an image of raw RGB, RGBA, or RGBO samples and returns
85 % it. It allocates the memory necessary for the new Image structure and
86 % returns a pointer to the new image.
87 %
88 % The format of the ReadRGBImage method is:
89 %
90 % Image *ReadRGBImage(const ImageInfo *image_info,
91 % ExceptionInfo *exception)
92 %
93 % A description of each parameter follows:
94 %
95 % o image_info: the image info.
96 %
97 % o exception: return any errors or warnings in this structure.
98 %
99 */
ReadRGBImage(const ImageInfo * image_info,ExceptionInfo * exception)100 static Image *ReadRGBImage(const ImageInfo *image_info,ExceptionInfo *exception)
101 {
102 const void
103 *stream;
104
105 Image
106 *canvas_image,
107 *image;
108
109 MagickBooleanType
110 status;
111
112 MagickOffsetType
113 scene;
114
115 QuantumInfo
116 *quantum_info;
117
118 QuantumType
119 quantum_type;
120
121 ssize_t
122 i;
123
124 size_t
125 length;
126
127 ssize_t
128 count,
129 y;
130
131 unsigned char
132 *pixels;
133
134 /*
135 Open image file.
136 */
137 assert(image_info != (const ImageInfo *) NULL);
138 assert(image_info->signature == MagickCoreSignature);
139 if (image_info->debug != MagickFalse)
140 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
141 image_info->filename);
142 assert(exception != (ExceptionInfo *) NULL);
143 assert(exception->signature == MagickCoreSignature);
144 image=AcquireImage(image_info,exception);
145 if ((image->columns == 0) || (image->rows == 0))
146 ThrowReaderException(OptionError,"MustSpecifyImageSize");
147 if (image_info->interlace != PartitionInterlace)
148 {
149 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
150 if (status == MagickFalse)
151 {
152 image=DestroyImageList(image);
153 return((Image *) NULL);
154 }
155 if (DiscardBlobBytes(image,(MagickSizeType) image->offset) == MagickFalse)
156 ThrowFileException(exception,CorruptImageError,"UnexpectedEndOfFile",
157 image->filename);
158 }
159 /*
160 Create virtual canvas to support cropping (i.e. image.rgb[100x100+10+20]).
161 */
162 canvas_image=CloneImage(image,image->extract_info.width,1,MagickFalse,
163 exception);
164 if(canvas_image == (Image *) NULL)
165 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
166 (void) SetImageVirtualPixelMethod(canvas_image,BlackVirtualPixelMethod,
167 exception);
168 quantum_info=AcquireQuantumInfo(image_info,canvas_image);
169 if (quantum_info == (QuantumInfo *) NULL)
170 {
171 canvas_image=DestroyImage(canvas_image);
172 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
173 }
174 quantum_type=RGBQuantum;
175 if (LocaleCompare(image_info->magick,"RGBA") == 0)
176 {
177 quantum_type=RGBAQuantum;
178 image->alpha_trait=BlendPixelTrait;
179 canvas_image->alpha_trait=BlendPixelTrait;
180 }
181 if (LocaleCompare(image_info->magick,"RGBO") == 0)
182 {
183 quantum_type=RGBOQuantum;
184 image->alpha_trait=BlendPixelTrait;
185 canvas_image->alpha_trait=BlendPixelTrait;
186 }
187 pixels=GetQuantumPixels(quantum_info);
188 if (image_info->number_scenes != 0)
189 while (image->scene < image_info->scene)
190 {
191 /*
192 Skip to next image.
193 */
194 image->scene++;
195 length=GetQuantumExtent(canvas_image,quantum_info,quantum_type);
196 for (y=0; y < (ssize_t) image->rows; y++)
197 {
198 stream=ReadBlobStream(image,length,pixels,&count);
199 if (count != (ssize_t) length)
200 break;
201 }
202 }
203 count=0;
204 length=0;
205 scene=0;
206 status=MagickTrue;
207 stream=NULL;
208 do
209 {
210 /*
211 Read pixels to virtual canvas image then push to image.
212 */
213 if ((image_info->ping != MagickFalse) && (image_info->number_scenes != 0))
214 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
215 break;
216 status=SetImageExtent(image,image->columns,image->rows,exception);
217 if (status == MagickFalse)
218 break;
219 switch (image_info->interlace)
220 {
221 case NoInterlace:
222 default:
223 {
224 /*
225 No interlacing: RGBRGBRGBRGBRGBRGB...
226 */
227 if (scene == 0)
228 {
229 length=GetQuantumExtent(canvas_image,quantum_info,quantum_type);
230 stream=ReadBlobStream(image,length,pixels,&count);
231 }
232 for (y=0; y < (ssize_t) image->extract_info.height; y++)
233 {
234 const Quantum
235 *magick_restrict p;
236
237 Quantum
238 *magick_restrict q;
239
240 ssize_t
241 x;
242
243 if (count != (ssize_t) length)
244 {
245 status=MagickFalse;
246 ThrowFileException(exception,CorruptImageError,
247 "UnexpectedEndOfFile",image->filename);
248 break;
249 }
250 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
251 exception);
252 if (q == (Quantum *) NULL)
253 break;
254 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
255 quantum_info,quantum_type,(unsigned char *) stream,exception);
256 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
257 break;
258 if (((y-image->extract_info.y) >= 0) &&
259 ((y-image->extract_info.y) < (ssize_t) image->rows))
260 {
261 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
262 canvas_image->columns,1,exception);
263 q=QueueAuthenticPixels(image,0,y-image->extract_info.y,
264 image->columns,1,exception);
265 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
266 break;
267 for (x=0; x < (ssize_t) image->columns; x++)
268 {
269 SetPixelRed(image,GetPixelRed(canvas_image,p),q);
270 SetPixelGreen(image,GetPixelGreen(canvas_image,p),q);
271 SetPixelBlue(image,GetPixelBlue(canvas_image,p),q);
272 SetPixelAlpha(image,OpaqueAlpha,q);
273 if (image->alpha_trait != UndefinedPixelTrait)
274 SetPixelAlpha(image,GetPixelAlpha(canvas_image,p),q);
275 p+=GetPixelChannels(canvas_image);
276 q+=GetPixelChannels(image);
277 }
278 if (SyncAuthenticPixels(image,exception) == MagickFalse)
279 break;
280 }
281 if (image->previous == (Image *) NULL)
282 {
283 status=SetImageProgress(image,LoadImageTag,(MagickOffsetType) y,
284 image->rows);
285 if (status == MagickFalse)
286 break;
287 }
288 stream=ReadBlobStream(image,length,pixels,&count);
289 }
290 break;
291 }
292 case LineInterlace:
293 {
294 static QuantumType
295 quantum_types[4] =
296 {
297 RedQuantum,
298 GreenQuantum,
299 BlueQuantum,
300 AlphaQuantum
301 };
302
303 /*
304 Line interlacing: RRR...GGG...BBB...RRR...GGG...BBB...
305 */
306 if (LocaleCompare(image_info->magick,"RGBO") == 0)
307 quantum_types[3]=OpacityQuantum;
308 if (scene == 0)
309 {
310 length=GetQuantumExtent(canvas_image,quantum_info,RedQuantum);
311 stream=ReadBlobStream(image,length,pixels,&count);
312 }
313 for (y=0; y < (ssize_t) image->extract_info.height; y++)
314 {
315 for (i=0; i < (ssize_t) (image->alpha_trait != UndefinedPixelTrait ? 4 : 3); i++)
316 {
317 const Quantum
318 *magick_restrict p;
319
320 Quantum
321 *magick_restrict q;
322
323 ssize_t
324 x;
325
326 if (count != (ssize_t) length)
327 {
328 status=MagickFalse;
329 ThrowFileException(exception,CorruptImageError,
330 "UnexpectedEndOfFile",image->filename);
331 break;
332 }
333 quantum_type=quantum_types[i];
334 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
335 exception);
336 if (q == (Quantum *) NULL)
337 break;
338 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
339 quantum_info,quantum_type,(unsigned char *) stream,exception);
340 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
341 break;
342 if (((y-image->extract_info.y) >= 0) &&
343 ((y-image->extract_info.y) < (ssize_t) image->rows))
344 {
345 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,
346 0,canvas_image->columns,1,exception);
347 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
348 image->columns,1,exception);
349 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
350 break;
351 for (x=0; x < (ssize_t) image->columns; x++)
352 {
353 switch (quantum_type)
354 {
355 case RedQuantum:
356 {
357 SetPixelRed(image,GetPixelRed(canvas_image,p),q);
358 break;
359 }
360 case GreenQuantum:
361 {
362 SetPixelGreen(image,GetPixelGreen(canvas_image,p),q);
363 break;
364 }
365 case BlueQuantum:
366 {
367 SetPixelBlue(image,GetPixelBlue(canvas_image,p),q);
368 break;
369 }
370 case OpacityQuantum:
371 {
372 SetPixelOpacity(image,GetPixelOpacity(canvas_image,p),q);
373 break;
374 }
375 case AlphaQuantum:
376 {
377 SetPixelAlpha(image,GetPixelAlpha(canvas_image,p),q);
378 break;
379 }
380 default:
381 break;
382 }
383 p+=GetPixelChannels(canvas_image);
384 q+=GetPixelChannels(image);
385 }
386 if (SyncAuthenticPixels(image,exception) == MagickFalse)
387 break;
388 }
389 stream=ReadBlobStream(image,length,pixels,&count);
390 }
391 if (image->previous == (Image *) NULL)
392 {
393 status=SetImageProgress(image,LoadImageTag,(MagickOffsetType) y,
394 image->rows);
395 if (status == MagickFalse)
396 break;
397 }
398 }
399 break;
400 }
401 case PlaneInterlace:
402 {
403 /*
404 Plane interlacing: RRRRRR...GGGGGG...BBBBBB...
405 */
406 if (scene == 0)
407 {
408 length=GetQuantumExtent(canvas_image,quantum_info,RedQuantum);
409 stream=ReadBlobStream(image,length,pixels,&count);
410 }
411 for (y=0; y < (ssize_t) image->extract_info.height; y++)
412 {
413 const Quantum
414 *magick_restrict p;
415
416 Quantum
417 *magick_restrict q;
418
419 ssize_t
420 x;
421
422 if (count != (ssize_t) length)
423 {
424 status=MagickFalse;
425 ThrowFileException(exception,CorruptImageError,
426 "UnexpectedEndOfFile",image->filename);
427 break;
428 }
429 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
430 exception);
431 if (q == (Quantum *) NULL)
432 break;
433 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
434 quantum_info,RedQuantum,(unsigned char *) stream,exception);
435 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
436 break;
437 if (((y-image->extract_info.y) >= 0) &&
438 ((y-image->extract_info.y) < (ssize_t) image->rows))
439 {
440 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
441 canvas_image->columns,1,exception);
442 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
443 image->columns,1,exception);
444 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
445 break;
446 for (x=0; x < (ssize_t) image->columns; x++)
447 {
448 SetPixelRed(image,GetPixelRed(canvas_image,p),q);
449 p+=GetPixelChannels(canvas_image);
450 q+=GetPixelChannels(image);
451 }
452 if (SyncAuthenticPixels(image,exception) == MagickFalse)
453 break;
454 }
455 stream=ReadBlobStream(image,length,pixels,&count);
456 }
457 if (image->previous == (Image *) NULL)
458 {
459 status=SetImageProgress(image,LoadImageTag,1,6);
460 if (status == MagickFalse)
461 break;
462 }
463 for (y=0; y < (ssize_t) image->extract_info.height; y++)
464 {
465 const Quantum
466 *magick_restrict p;
467
468 Quantum
469 *magick_restrict q;
470
471 ssize_t
472 x;
473
474 if (count != (ssize_t) length)
475 {
476 status=MagickFalse;
477 ThrowFileException(exception,CorruptImageError,
478 "UnexpectedEndOfFile",image->filename);
479 break;
480 }
481 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
482 exception);
483 if (q == (Quantum *) NULL)
484 break;
485 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
486 quantum_info,GreenQuantum,(unsigned char *) stream,exception);
487 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
488 break;
489 if (((y-image->extract_info.y) >= 0) &&
490 ((y-image->extract_info.y) < (ssize_t) image->rows))
491 {
492 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
493 canvas_image->columns,1,exception);
494 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
495 image->columns,1,exception);
496 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
497 break;
498 for (x=0; x < (ssize_t) image->columns; x++)
499 {
500 SetPixelGreen(image,GetPixelGreen(canvas_image,p),q);
501 p+=GetPixelChannels(canvas_image);
502 q+=GetPixelChannels(image);
503 }
504 if (SyncAuthenticPixels(image,exception) == MagickFalse)
505 break;
506 }
507 stream=ReadBlobStream(image,length,pixels,&count);
508 }
509 if (image->previous == (Image *) NULL)
510 {
511 status=SetImageProgress(image,LoadImageTag,2,6);
512 if (status == MagickFalse)
513 break;
514 }
515 for (y=0; y < (ssize_t) image->extract_info.height; y++)
516 {
517 const Quantum
518 *magick_restrict p;
519
520 Quantum
521 *magick_restrict q;
522
523 ssize_t
524 x;
525
526 if (count != (ssize_t) length)
527 {
528 status=MagickFalse;
529 ThrowFileException(exception,CorruptImageError,
530 "UnexpectedEndOfFile",image->filename);
531 break;
532 }
533 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
534 exception);
535 if (q == (Quantum *) NULL)
536 break;
537 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
538 quantum_info,BlueQuantum,(unsigned char *) stream,exception);
539 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
540 break;
541 if (((y-image->extract_info.y) >= 0) &&
542 ((y-image->extract_info.y) < (ssize_t) image->rows))
543 {
544 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
545 canvas_image->columns,1,exception);
546 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
547 image->columns,1,exception);
548 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
549 break;
550 for (x=0; x < (ssize_t) image->columns; x++)
551 {
552 SetPixelBlue(image,GetPixelBlue(canvas_image,p),q);
553 p+=GetPixelChannels(canvas_image);
554 q+=GetPixelChannels(image);
555 }
556 if (SyncAuthenticPixels(image,exception) == MagickFalse)
557 break;
558 }
559 stream=ReadBlobStream(image,length,pixels,&count);
560 }
561 if (image->previous == (Image *) NULL)
562 {
563 status=SetImageProgress(image,LoadImageTag,4,6);
564 if (status == MagickFalse)
565 break;
566 }
567 if (image->alpha_trait != UndefinedPixelTrait)
568 {
569 for (y=0; y < (ssize_t) image->extract_info.height; y++)
570 {
571 const Quantum
572 *magick_restrict p;
573
574 Quantum
575 *magick_restrict q;
576
577 ssize_t
578 x;
579
580 if (count != (ssize_t) length)
581 {
582 status=MagickFalse;
583 ThrowFileException(exception,CorruptImageError,
584 "UnexpectedEndOfFile",image->filename);
585 break;
586 }
587 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
588 exception);
589 if (q == (Quantum *) NULL)
590 break;
591 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
592 quantum_info,AlphaQuantum,(unsigned char *) stream,exception);
593 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
594 break;
595 if (((y-image->extract_info.y) >= 0) &&
596 ((y-image->extract_info.y) < (ssize_t) image->rows))
597 {
598 p=GetVirtualPixels(canvas_image,
599 canvas_image->extract_info.x,0,canvas_image->columns,1,
600 exception);
601 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
602 image->columns,1,exception);
603 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
604 break;
605 for (x=0; x < (ssize_t) image->columns; x++)
606 {
607 SetPixelAlpha(image,GetPixelAlpha(canvas_image,p),q);
608 p+=GetPixelChannels(canvas_image);
609 q+=GetPixelChannels(image);
610 }
611 if (SyncAuthenticPixels(image,exception) == MagickFalse)
612 break;
613 }
614 stream=ReadBlobStream(image,length,pixels,&count);
615 }
616 if (image->previous == (Image *) NULL)
617 {
618 status=SetImageProgress(image,LoadImageTag,5,6);
619 if (status == MagickFalse)
620 break;
621 }
622 }
623 if (image->previous == (Image *) NULL)
624 {
625 status=SetImageProgress(image,LoadImageTag,6,6);
626 if (status == MagickFalse)
627 break;
628 }
629 break;
630 }
631 case PartitionInterlace:
632 {
633 /*
634 Partition interlacing: RRRRRR..., GGGGGG..., BBBBBB...
635 */
636 AppendImageFormat("R",image->filename);
637 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
638 if (status == MagickFalse)
639 break;
640 if (DiscardBlobBytes(image,(MagickSizeType) image->offset) == MagickFalse)
641 {
642 status=MagickFalse;
643 ThrowFileException(exception,CorruptImageError,
644 "UnexpectedEndOfFile",image->filename);
645 break;
646 }
647 length=GetQuantumExtent(canvas_image,quantum_info,RedQuantum);
648 for (i=0; i < (ssize_t) scene; i++)
649 {
650 for (y=0; y < (ssize_t) image->extract_info.height; y++)
651 {
652 stream=ReadBlobStream(image,length,pixels,&count);
653 if (count != (ssize_t) length)
654 break;
655 }
656 if (count != (ssize_t) length)
657 break;
658 }
659 stream=ReadBlobStream(image,length,pixels,&count);
660 for (y=0; y < (ssize_t) image->extract_info.height; y++)
661 {
662 const Quantum
663 *magick_restrict p;
664
665 Quantum
666 *magick_restrict q;
667
668 ssize_t
669 x;
670
671 if (count != (ssize_t) length)
672 {
673 status=MagickFalse;
674 ThrowFileException(exception,CorruptImageError,
675 "UnexpectedEndOfFile",image->filename);
676 break;
677 }
678 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
679 exception);
680 if (q == (Quantum *) NULL)
681 break;
682 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
683 quantum_info,RedQuantum,(unsigned char *) stream,exception);
684 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
685 break;
686 if (((y-image->extract_info.y) >= 0) &&
687 ((y-image->extract_info.y) < (ssize_t) image->rows))
688 {
689 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
690 canvas_image->columns,1,exception);
691 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
692 image->columns,1,exception);
693 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
694 break;
695 for (x=0; x < (ssize_t) image->columns; x++)
696 {
697 SetPixelRed(image,GetPixelRed(canvas_image,p),q);
698 p+=GetPixelChannels(canvas_image);
699 q+=GetPixelChannels(image);
700 }
701 if (SyncAuthenticPixels(image,exception) == MagickFalse)
702 break;
703 }
704 stream=ReadBlobStream(image,length,pixels,&count);
705 }
706 if (image->previous == (Image *) NULL)
707 {
708 status=SetImageProgress(image,LoadImageTag,1,5);
709 if (status == MagickFalse)
710 break;
711 }
712 (void) CloseBlob(image);
713 AppendImageFormat("G",image->filename);
714 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
715 if (status == MagickFalse)
716 break;
717 length=GetQuantumExtent(canvas_image,quantum_info,GreenQuantum);
718 for (i=0; i < (ssize_t) scene; i++)
719 {
720 for (y=0; y < (ssize_t) image->extract_info.height; y++)
721 {
722 stream=ReadBlobStream(image,length,pixels,&count);
723 if (count != (ssize_t) length)
724 break;
725 }
726 if (count != (ssize_t) length)
727 break;
728 }
729 stream=ReadBlobStream(image,length,pixels,&count);
730 for (y=0; y < (ssize_t) image->extract_info.height; y++)
731 {
732 const Quantum
733 *magick_restrict p;
734
735 Quantum
736 *magick_restrict q;
737
738 ssize_t
739 x;
740
741 if (count != (ssize_t) length)
742 {
743 status=MagickFalse;
744 ThrowFileException(exception,CorruptImageError,
745 "UnexpectedEndOfFile",image->filename);
746 break;
747 }
748 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
749 exception);
750 if (q == (Quantum *) NULL)
751 break;
752 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
753 quantum_info,GreenQuantum,(unsigned char *) stream,exception);
754 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
755 break;
756 if (((y-image->extract_info.y) >= 0) &&
757 ((y-image->extract_info.y) < (ssize_t) image->rows))
758 {
759 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
760 canvas_image->columns,1,exception);
761 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
762 image->columns,1,exception);
763 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
764 break;
765 for (x=0; x < (ssize_t) image->columns; x++)
766 {
767 SetPixelGreen(image,GetPixelGreen(canvas_image,p),q);
768 p+=GetPixelChannels(canvas_image);
769 q+=GetPixelChannels(image);
770 }
771 if (SyncAuthenticPixels(image,exception) == MagickFalse)
772 break;
773 }
774 stream=ReadBlobStream(image,length,pixels,&count);
775 }
776 if (image->previous == (Image *) NULL)
777 {
778 status=SetImageProgress(image,LoadImageTag,2,5);
779 if (status == MagickFalse)
780 break;
781 }
782 (void) CloseBlob(image);
783 AppendImageFormat("B",image->filename);
784 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
785 if (status == MagickFalse)
786 break;
787 length=GetQuantumExtent(canvas_image,quantum_info,BlueQuantum);
788 for (i=0; i < (ssize_t) scene; i++)
789 {
790 for (y=0; y < (ssize_t) image->extract_info.height; y++)
791 {
792 stream=ReadBlobStream(image,length,pixels,&count);
793 if (count != (ssize_t) length)
794 break;
795 }
796 if (count != (ssize_t) length)
797 break;
798 }
799 stream=ReadBlobStream(image,length,pixels,&count);
800 for (y=0; y < (ssize_t) image->extract_info.height; y++)
801 {
802 const Quantum
803 *magick_restrict p;
804
805 Quantum
806 *magick_restrict q;
807
808 ssize_t
809 x;
810
811 if (count != (ssize_t) length)
812 {
813 status=MagickFalse;
814 ThrowFileException(exception,CorruptImageError,
815 "UnexpectedEndOfFile",image->filename);
816 break;
817 }
818 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
819 exception);
820 if (q == (Quantum *) NULL)
821 break;
822 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
823 quantum_info,BlueQuantum,(unsigned char *) stream,exception);
824 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
825 break;
826 if (((y-image->extract_info.y) >= 0) &&
827 ((y-image->extract_info.y) < (ssize_t) image->rows))
828 {
829 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
830 canvas_image->columns,1,exception);
831 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
832 image->columns,1,exception);
833 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
834 break;
835 for (x=0; x < (ssize_t) image->columns; x++)
836 {
837 SetPixelBlue(image,GetPixelBlue(canvas_image,p),q);
838 p+=GetPixelChannels(canvas_image);
839 q+=GetPixelChannels(image);
840 }
841 if (SyncAuthenticPixels(image,exception) == MagickFalse)
842 break;
843 }
844 stream=ReadBlobStream(image,length,pixels,&count);
845 }
846 if (image->previous == (Image *) NULL)
847 {
848 status=SetImageProgress(image,LoadImageTag,3,5);
849 if (status == MagickFalse)
850 break;
851 }
852 if (image->alpha_trait != UndefinedPixelTrait)
853 {
854 (void) CloseBlob(image);
855 AppendImageFormat("A",image->filename);
856 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
857 if (status == MagickFalse)
858 break;
859 length=GetQuantumExtent(canvas_image,quantum_info,AlphaQuantum);
860 for (i=0; i < (ssize_t) scene; i++)
861 {
862 for (y=0; y < (ssize_t) image->extract_info.height; y++)
863 {
864 stream=ReadBlobStream(image,length,pixels,&count);
865 if (count != (ssize_t) length)
866 break;
867 }
868 if (count != (ssize_t) length)
869 break;
870 }
871 stream=ReadBlobStream(image,length,pixels,&count);
872 for (y=0; y < (ssize_t) image->extract_info.height; y++)
873 {
874 const Quantum
875 *magick_restrict p;
876
877 Quantum
878 *magick_restrict q;
879
880 ssize_t
881 x;
882
883 if (count != (ssize_t) length)
884 {
885 status=MagickFalse;
886 ThrowFileException(exception,CorruptImageError,
887 "UnexpectedEndOfFile",image->filename);
888 break;
889 }
890 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,
891 exception);
892 if (q == (Quantum *) NULL)
893 break;
894 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
895 quantum_info,BlueQuantum,(unsigned char *) stream,exception);
896 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
897 break;
898 if (((y-image->extract_info.y) >= 0) &&
899 ((y-image->extract_info.y) < (ssize_t) image->rows))
900 {
901 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,
902 0,canvas_image->columns,1,exception);
903 q=GetAuthenticPixels(image,0,y-image->extract_info.y,
904 image->columns,1,exception);
905 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
906 break;
907 for (x=0; x < (ssize_t) image->columns; x++)
908 {
909 SetPixelAlpha(image,GetPixelAlpha(canvas_image,p),q);
910 p+=GetPixelChannels(canvas_image);
911 q+=GetPixelChannels(image);
912 }
913 if (SyncAuthenticPixels(image,exception) == MagickFalse)
914 break;
915 }
916 stream=ReadBlobStream(image,length,pixels,&count);
917 }
918 if (image->previous == (Image *) NULL)
919 {
920 status=SetImageProgress(image,LoadImageTag,4,5);
921 if (status == MagickFalse)
922 break;
923 }
924 }
925 (void) CloseBlob(image);
926 if (image->previous == (Image *) NULL)
927 {
928 status=SetImageProgress(image,LoadImageTag,5,5);
929 if (status == MagickFalse)
930 break;
931 }
932 break;
933 }
934 }
935 if (status == MagickFalse)
936 break;
937 SetQuantumImageType(image,quantum_type);
938 /*
939 Proceed to next image.
940 */
941 if (image_info->number_scenes != 0)
942 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
943 break;
944 if (count == (ssize_t) length)
945 {
946 /*
947 Allocate next image structure.
948 */
949 AcquireNextImage(image_info,image,exception);
950 if (GetNextImageInList(image) == (Image *) NULL)
951 {
952 status=MagickFalse;
953 break;
954 }
955 image=SyncNextImageInList(image);
956 status=SetImageProgress(image,LoadImagesTag,TellBlob(image),
957 GetBlobSize(image));
958 if (status == MagickFalse)
959 break;
960 }
961 scene++;
962 } while (count == (ssize_t) length);
963 quantum_info=DestroyQuantumInfo(quantum_info);
964 canvas_image=DestroyImage(canvas_image);
965 (void) CloseBlob(image);
966 if (status == MagickFalse)
967 return(DestroyImageList(image));
968 return(GetFirstImageInList(image));
969 }
970
971 /*
972 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
973 % %
974 % %
975 % %
976 % R e a d R G B 5 6 5 I m a g e %
977 % %
978 % %
979 % %
980 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
981 %
982 % ReadGRAYImage() reads an image of raw RGB 5-6-5 samples and returns it. It
983 % allocates the memory necessary for the new Image structure and returns a
984 % pointer to the new image.
985 %
986 % The format of the ReadGRAYImage method is:
987 %
988 % Image *ReadGRAYImage(const ImageInfo *image_info,
989 % ExceptionInfo *exception)
990 %
991 % A description of each parameter follows:
992 %
993 % o image_info: the image info.
994 %
995 % o exception: return any errors or warnings in this structure.
996 %
997 */
ReadRGB565Image(const ImageInfo * image_info,ExceptionInfo * exception)998 static Image *ReadRGB565Image(const ImageInfo *image_info,
999 ExceptionInfo *exception)
1000 {
1001 const void
1002 *stream;
1003
1004 Image
1005 *canvas_image,
1006 *image;
1007
1008 MagickBooleanType
1009 status;
1010
1011 MagickOffsetType
1012 scene;
1013
1014 QuantumInfo
1015 *quantum_info;
1016
1017 QuantumType
1018 quantum_type;
1019
1020 size_t
1021 length;
1022
1023 ssize_t
1024 count,
1025 y;
1026
1027 unsigned char
1028 *pixels;
1029
1030 /*
1031 Open image file.
1032 */
1033 assert(image_info != (const ImageInfo *) NULL);
1034 assert(image_info->signature == MagickCoreSignature);
1035 if (image_info->debug != MagickFalse)
1036 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
1037 image_info->filename);
1038 assert(exception != (ExceptionInfo *) NULL);
1039 assert(exception->signature == MagickCoreSignature);
1040 image=AcquireImage(image_info,exception);
1041 if ((image->columns == 0) || (image->rows == 0))
1042 ThrowReaderException(OptionError,"MustSpecifyImageSize");
1043 image->depth=16;
1044 if (image_info->interlace != PartitionInterlace)
1045 {
1046 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
1047 if (status == MagickFalse)
1048 {
1049 image=DestroyImageList(image);
1050 return((Image *) NULL);
1051 }
1052 if (DiscardBlobBytes(image,(MagickSizeType) image->offset) == MagickFalse)
1053 ThrowFileException(exception,CorruptImageError,"UnexpectedEndOfFile",
1054 image->filename);
1055 }
1056 /*
1057 Create virtual canvas to support cropping (i.e. image.rgb[100x100+10+20]).
1058 */
1059 canvas_image=CloneImage(image,image->extract_info.width,1,MagickFalse,
1060 exception);
1061 if(canvas_image == (Image *) NULL)
1062 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
1063 (void) SetImageVirtualPixelMethod(canvas_image,BlackVirtualPixelMethod,
1064 exception);
1065 quantum_info=AcquireQuantumInfo(image_info,canvas_image);
1066 if (quantum_info == (QuantumInfo *) NULL)
1067 {
1068 canvas_image=DestroyImage(canvas_image);
1069 ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
1070 }
1071 quantum_type=GrayQuantum;
1072 pixels=GetQuantumPixels(quantum_info);
1073 if (image_info->number_scenes != 0)
1074 while (image->scene < image_info->scene)
1075 {
1076 /*
1077 Skip to next image.
1078 */
1079 image->scene++;
1080 length=GetQuantumExtent(canvas_image,quantum_info,quantum_type);
1081 for (y=0; y < (ssize_t) image->rows; y++)
1082 {
1083 stream=ReadBlobStream(image,length,pixels,&count);
1084 if (count != (ssize_t) length)
1085 break;
1086 }
1087 }
1088 count=0;
1089 length=0;
1090 scene=0;
1091 status=MagickTrue;
1092 stream=NULL;
1093 do
1094 {
1095 /*
1096 Read pixels to virtual canvas image then push to image.
1097 */
1098 if ((image_info->ping != MagickFalse) && (image_info->number_scenes != 0))
1099 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
1100 break;
1101 status=SetImageExtent(image,image->columns,image->rows,exception);
1102 if (status == MagickFalse)
1103 break;
1104 /*
1105 No interlacing: GGG...
1106 */
1107 if (scene == 0)
1108 {
1109 length=GetQuantumExtent(canvas_image,quantum_info,quantum_type);
1110 stream=ReadBlobStream(image,length,pixels,&count);
1111 }
1112 for (y=0; y < (ssize_t) image->extract_info.height; y++)
1113 {
1114 const Quantum
1115 *magick_restrict p;
1116
1117 Quantum
1118 *magick_restrict q;
1119
1120 ssize_t
1121 x;
1122
1123 if (count != (ssize_t) length)
1124 {
1125 status=MagickFalse;
1126 ThrowFileException(exception,CorruptImageError,
1127 "UnexpectedEndOfFile",image->filename);
1128 break;
1129 }
1130 q=GetAuthenticPixels(canvas_image,0,0,canvas_image->columns,1,exception);
1131 if (q == (Quantum *) NULL)
1132 break;
1133 length=ImportQuantumPixels(canvas_image,(CacheView *) NULL,
1134 quantum_info,quantum_type,(unsigned char *) stream,exception);
1135 if (SyncAuthenticPixels(canvas_image,exception) == MagickFalse)
1136 break;
1137 if (((y-image->extract_info.y) >= 0) &&
1138 ((y-image->extract_info.y) < (ssize_t) image->rows))
1139 {
1140 p=GetVirtualPixels(canvas_image,canvas_image->extract_info.x,0,
1141 canvas_image->columns,1,exception);
1142 q=QueueAuthenticPixels(image,0,y-image->extract_info.y,
1143 image->columns,1,exception);
1144 if ((p == (const Quantum *) NULL) || (q == (Quantum *) NULL))
1145 break;
1146 for (x=0; x < (ssize_t) image->columns; x++)
1147 {
1148 unsigned short
1149 pixel;
1150
1151 pixel=(unsigned short) ScaleQuantumToShort(GetPixelGray(
1152 canvas_image,p));
1153 SetPixelRed(image,(Quantum) (((pixel >> 11) & 0x1f) << 11),q);
1154 SetPixelGreen(image,(Quantum) (((pixel >> 5) & 0x3f) << 10),q);
1155 SetPixelBlue(image,(Quantum) ((pixel & 0x1f) << 11),q);
1156 p+=GetPixelChannels(canvas_image);
1157 q+=GetPixelChannels(image);
1158 }
1159 if (SyncAuthenticPixels(image,exception) == MagickFalse)
1160 break;
1161 }
1162 if (image->previous == (Image *) NULL)
1163 {
1164 status=SetImageProgress(image,LoadImageTag,(MagickOffsetType) y,
1165 image->rows);
1166 if (status == MagickFalse)
1167 break;
1168 }
1169 stream=ReadBlobStream(image,length,pixels,&count);
1170 }
1171 if (status == MagickFalse)
1172 break;
1173 SetQuantumImageType(image,quantum_type);
1174 /*
1175 Proceed to next image.
1176 */
1177 if (image_info->number_scenes != 0)
1178 if (image->scene >= (image_info->scene+image_info->number_scenes-1))
1179 break;
1180 if (count == (ssize_t) length)
1181 {
1182 /*
1183 Allocate next image structure.
1184 */
1185 AcquireNextImage(image_info,image,exception);
1186 if (GetNextImageInList(image) == (Image *) NULL)
1187 {
1188 status=MagickFalse;
1189 break;
1190 }
1191 image=SyncNextImageInList(image);
1192 status=SetImageProgress(image,LoadImagesTag,TellBlob(image),
1193 GetBlobSize(image));
1194 if (status == MagickFalse)
1195 break;
1196 }
1197 scene++;
1198 } while (count == (ssize_t) length);
1199 quantum_info=DestroyQuantumInfo(quantum_info);
1200 canvas_image=DestroyImage(canvas_image);
1201 (void) CloseBlob(image);
1202 if (status == MagickFalse)
1203 return(DestroyImageList(image));
1204 return(GetFirstImageInList(image));
1205 }
1206
1207 /*
1208 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1209 % %
1210 % %
1211 % %
1212 % R e g i s t e r R G B I m a g e %
1213 % %
1214 % %
1215 % %
1216 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1217 %
1218 % RegisterRGBImage() adds attributes for the RGB image format to
1219 % the list of supported formats. The attributes include the image format
1220 % tag, a method to read and/or write the format, whether the format
1221 % supports the saving of more than one frame to the same file or blob,
1222 % whether the format supports native in-memory I/O, and a brief
1223 % description of the format.
1224 %
1225 % The format of the RegisterRGBImage method is:
1226 %
1227 % size_t RegisterRGBImage(void)
1228 %
1229 */
RegisterRGBImage(void)1230 ModuleExport size_t RegisterRGBImage(void)
1231 {
1232 MagickInfo
1233 *entry;
1234
1235 entry=AcquireMagickInfo("RGB","RGB",
1236 "Raw red, green, and blue samples");
1237 entry->decoder=(DecodeImageHandler *) ReadRGBImage;
1238 entry->encoder=(EncodeImageHandler *) WriteRGBImage;
1239 entry->flags|=CoderRawSupportFlag;
1240 entry->flags|=CoderEndianSupportFlag;
1241 (void) RegisterMagickInfo(entry);
1242 entry=AcquireMagickInfo("RGB","RGBA",
1243 "Raw red, green, blue, and alpha samples");
1244 entry->decoder=(DecodeImageHandler *) ReadRGBImage;
1245 entry->encoder=(EncodeImageHandler *) WriteRGBImage;
1246 entry->flags|=CoderRawSupportFlag;
1247 entry->flags|=CoderEndianSupportFlag;
1248 (void) RegisterMagickInfo(entry);
1249 entry=AcquireMagickInfo("RGB","RGBO",
1250 "Raw red, green, blue, and opacity samples");
1251 entry->decoder=(DecodeImageHandler *) ReadRGBImage;
1252 entry->encoder=(EncodeImageHandler *) WriteRGBImage;
1253 entry->flags|=CoderRawSupportFlag;
1254 entry->flags|=CoderEndianSupportFlag;
1255 (void) RegisterMagickInfo(entry);
1256 entry=AcquireMagickInfo("RGB","RGB565",
1257 "Raw red, green, blue samples in 565 format");
1258 entry->decoder=(DecodeImageHandler *) ReadRGB565Image;
1259 entry->flags|=CoderRawSupportFlag;
1260 entry->flags|=CoderEndianSupportFlag;
1261 (void) RegisterMagickInfo(entry);
1262 return(MagickImageCoderSignature);
1263 }
1264
1265 /*
1266 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1267 % %
1268 % %
1269 % %
1270 % U n r e g i s t e r R G B I m a g e %
1271 % %
1272 % %
1273 % %
1274 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1275 %
1276 % UnregisterRGBImage() removes format registrations made by the RGB module
1277 % from the list of supported formats.
1278 %
1279 % The format of the UnregisterRGBImage method is:
1280 %
1281 % UnregisterRGBImage(void)
1282 %
1283 */
UnregisterRGBImage(void)1284 ModuleExport void UnregisterRGBImage(void)
1285 {
1286 (void) UnregisterMagickInfo("RGBO");
1287 (void) UnregisterMagickInfo("RGBA");
1288 (void) UnregisterMagickInfo("RGB");
1289 }
1290
1291 /*
1292 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1293 % %
1294 % %
1295 % %
1296 % W r i t e R G B I m a g e %
1297 % %
1298 % %
1299 % %
1300 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1301 %
1302 % WriteRGBImage() writes an image to a file in the RGB, RGBA, or RGBO
1303 % rasterfile format.
1304 %
1305 % The format of the WriteRGBImage method is:
1306 %
1307 % MagickBooleanType WriteRGBImage(const ImageInfo *image_info,
1308 % Image *image,ExceptionInfo *exception)
1309 %
1310 % A description of each parameter follows.
1311 %
1312 % o image_info: the image info.
1313 %
1314 % o image: The image.
1315 %
1316 % o exception: return any errors or warnings in this structure.
1317 %
1318 */
WriteRGBImage(const ImageInfo * image_info,Image * image,ExceptionInfo * exception)1319 static MagickBooleanType WriteRGBImage(const ImageInfo *image_info,
1320 Image *image,ExceptionInfo *exception)
1321 {
1322 MagickBooleanType
1323 status;
1324
1325 MagickOffsetType
1326 scene;
1327
1328 QuantumInfo
1329 *quantum_info;
1330
1331 QuantumType
1332 quantum_type;
1333
1334 size_t
1335 imageListLength,
1336 length;
1337
1338 ssize_t
1339 count,
1340 y;
1341
1342 unsigned char
1343 *pixels;
1344
1345 /*
1346 Allocate memory for pixels.
1347 */
1348 assert(image_info != (const ImageInfo *) NULL);
1349 assert(image_info->signature == MagickCoreSignature);
1350 assert(image != (Image *) NULL);
1351 assert(image->signature == MagickCoreSignature);
1352 if (image->debug != MagickFalse)
1353 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
1354 if (image_info->interlace != PartitionInterlace)
1355 {
1356 /*
1357 Open output image file.
1358 */
1359 status=OpenBlob(image_info,image,WriteBinaryBlobMode,exception);
1360 if (status == MagickFalse)
1361 return(status);
1362 }
1363 quantum_type=RGBQuantum;
1364 if (LocaleCompare(image_info->magick,"RGBA") == 0)
1365 quantum_type=RGBAQuantum;
1366 if (LocaleCompare(image_info->magick,"RGBO") == 0)
1367 quantum_type=RGBOQuantum;
1368 scene=0;
1369 imageListLength=GetImageListLength(image);
1370 do
1371 {
1372 /*
1373 Convert MIFF to RGB raster pixels.
1374 */
1375 (void) TransformImageColorspace(image,sRGBColorspace,exception);
1376 if ((LocaleCompare(image_info->magick,"RGBA") == 0) &&
1377 (image->alpha_trait == UndefinedPixelTrait))
1378 (void) SetImageAlphaChannel(image,OpaqueAlphaChannel,exception);
1379 quantum_info=AcquireQuantumInfo(image_info,image);
1380 if (quantum_info == (QuantumInfo *) NULL)
1381 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
1382 pixels=(unsigned char *) GetQuantumPixels(quantum_info);
1383 switch (image_info->interlace)
1384 {
1385 case NoInterlace:
1386 default:
1387 {
1388 /*
1389 No interlacing: RGBRGBRGBRGBRGBRGB...
1390 */
1391 for (y=0; y < (ssize_t) image->rows; y++)
1392 {
1393 const Quantum
1394 *magick_restrict p;
1395
1396 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1397 if (p == (const Quantum *) NULL)
1398 break;
1399 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1400 quantum_type,pixels,exception);
1401 count=WriteBlob(image,length,pixels);
1402 if (count != (ssize_t) length)
1403 break;
1404 if (image->previous == (Image *) NULL)
1405 {
1406 status=SetImageProgress(image,SaveImageTag,(MagickOffsetType) y,
1407 image->rows);
1408 if (status == MagickFalse)
1409 break;
1410 }
1411 }
1412 break;
1413 }
1414 case LineInterlace:
1415 {
1416 /*
1417 Line interlacing: RRR...GGG...BBB...RRR...GGG...BBB...
1418 */
1419 for (y=0; y < (ssize_t) image->rows; y++)
1420 {
1421 const Quantum
1422 *magick_restrict p;
1423
1424 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1425 if (p == (const Quantum *) NULL)
1426 break;
1427 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1428 RedQuantum,pixels,exception);
1429 count=WriteBlob(image,length,pixels);
1430 if (count != (ssize_t) length)
1431 break;
1432 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1433 GreenQuantum,pixels,exception);
1434 count=WriteBlob(image,length,pixels);
1435 if (count != (ssize_t) length)
1436 break;
1437 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1438 BlueQuantum,pixels,exception);
1439 count=WriteBlob(image,length,pixels);
1440 if (count != (ssize_t) length)
1441 break;
1442 if (quantum_type == RGBAQuantum)
1443 {
1444 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1445 AlphaQuantum,pixels,exception);
1446 count=WriteBlob(image,length,pixels);
1447 if (count != (ssize_t) length)
1448 break;
1449 }
1450 if (quantum_type == RGBOQuantum)
1451 {
1452 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1453 OpacityQuantum,pixels,exception);
1454 count=WriteBlob(image,length,pixels);
1455 if (count != (ssize_t) length)
1456 break;
1457 }
1458 if (image->previous == (Image *) NULL)
1459 {
1460 status=SetImageProgress(image,SaveImageTag,(MagickOffsetType) y,
1461 image->rows);
1462 if (status == MagickFalse)
1463 break;
1464 }
1465 }
1466 break;
1467 }
1468 case PlaneInterlace:
1469 {
1470 /*
1471 Plane interlacing: RRRRRR...GGGGGG...BBBBBB...
1472 */
1473 for (y=0; y < (ssize_t) image->rows; y++)
1474 {
1475 const Quantum
1476 *magick_restrict p;
1477
1478 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1479 if (p == (const Quantum *) NULL)
1480 break;
1481 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1482 RedQuantum,pixels,exception);
1483 count=WriteBlob(image,length,pixels);
1484 if (count != (ssize_t) length)
1485 break;
1486 }
1487 if (image->previous == (Image *) NULL)
1488 {
1489 status=SetImageProgress(image,SaveImageTag,1,6);
1490 if (status == MagickFalse)
1491 break;
1492 }
1493 for (y=0; y < (ssize_t) image->rows; y++)
1494 {
1495 const Quantum
1496 *magick_restrict p;
1497
1498 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1499 if (p == (const Quantum *) NULL)
1500 break;
1501 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1502 GreenQuantum,pixels,exception);
1503 count=WriteBlob(image,length,pixels);
1504 if (count != (ssize_t) length)
1505 break;
1506 }
1507 if (image->previous == (Image *) NULL)
1508 {
1509 status=SetImageProgress(image,SaveImageTag,2,6);
1510 if (status == MagickFalse)
1511 break;
1512 }
1513 for (y=0; y < (ssize_t) image->rows; y++)
1514 {
1515 const Quantum
1516 *magick_restrict p;
1517
1518 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1519 if (p == (const Quantum *) NULL)
1520 break;
1521 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1522 BlueQuantum,pixels,exception);
1523 count=WriteBlob(image,length,pixels);
1524 if (count != (ssize_t) length)
1525 break;
1526 }
1527 if (image->previous == (Image *) NULL)
1528 {
1529 status=SetImageProgress(image,SaveImageTag,3,6);
1530 if (status == MagickFalse)
1531 break;
1532 }
1533 if (quantum_type == RGBAQuantum)
1534 {
1535 for (y=0; y < (ssize_t) image->rows; y++)
1536 {
1537 const Quantum
1538 *magick_restrict p;
1539
1540 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1541 if (p == (const Quantum *) NULL)
1542 break;
1543 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1544 AlphaQuantum,pixels,exception);
1545 count=WriteBlob(image,length,pixels);
1546 if (count != (ssize_t) length)
1547 break;
1548 }
1549 if (image->previous == (Image *) NULL)
1550 {
1551 status=SetImageProgress(image,SaveImageTag,5,6);
1552 if (status == MagickFalse)
1553 break;
1554 }
1555 }
1556 if (image_info->interlace == PartitionInterlace)
1557 (void) CopyMagickString(image->filename,image_info->filename,
1558 MagickPathExtent);
1559 if (image->previous == (Image *) NULL)
1560 {
1561 status=SetImageProgress(image,SaveImageTag,6,6);
1562 if (status == MagickFalse)
1563 break;
1564 }
1565 break;
1566 }
1567 case PartitionInterlace:
1568 {
1569 /*
1570 Partition interlacing: RRRRRR..., GGGGGG..., BBBBBB...
1571 */
1572 AppendImageFormat("R",image->filename);
1573 status=OpenBlob(image_info,image,scene == 0 ? WriteBinaryBlobMode :
1574 AppendBinaryBlobMode,exception);
1575 if (status == MagickFalse)
1576 return(status);
1577 for (y=0; y < (ssize_t) image->rows; y++)
1578 {
1579 const Quantum
1580 *magick_restrict p;
1581
1582 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1583 if (p == (const Quantum *) NULL)
1584 break;
1585 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1586 RedQuantum,pixels,exception);
1587 count=WriteBlob(image,length,pixels);
1588 if (count != (ssize_t) length)
1589 break;
1590 }
1591 if (image->previous == (Image *) NULL)
1592 {
1593 status=SetImageProgress(image,SaveImageTag,1,6);
1594 if (status == MagickFalse)
1595 break;
1596 }
1597 (void) CloseBlob(image);
1598 AppendImageFormat("G",image->filename);
1599 status=OpenBlob(image_info,image,scene == 0 ? WriteBinaryBlobMode :
1600 AppendBinaryBlobMode,exception);
1601 if (status == MagickFalse)
1602 return(status);
1603 for (y=0; y < (ssize_t) image->rows; y++)
1604 {
1605 const Quantum
1606 *magick_restrict p;
1607
1608 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1609 if (p == (const Quantum *) NULL)
1610 break;
1611 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1612 GreenQuantum,pixels,exception);
1613 count=WriteBlob(image,length,pixels);
1614 if (count != (ssize_t) length)
1615 break;
1616 }
1617 if (image->previous == (Image *) NULL)
1618 {
1619 status=SetImageProgress(image,SaveImageTag,2,6);
1620 if (status == MagickFalse)
1621 break;
1622 }
1623 (void) CloseBlob(image);
1624 AppendImageFormat("B",image->filename);
1625 status=OpenBlob(image_info,image,scene == 0 ? WriteBinaryBlobMode :
1626 AppendBinaryBlobMode,exception);
1627 if (status == MagickFalse)
1628 return(status);
1629 for (y=0; y < (ssize_t) image->rows; y++)
1630 {
1631 const Quantum
1632 *magick_restrict p;
1633
1634 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1635 if (p == (const Quantum *) NULL)
1636 break;
1637 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1638 BlueQuantum,pixels,exception);
1639 count=WriteBlob(image,length,pixels);
1640 if (count != (ssize_t) length)
1641 break;
1642 }
1643 if (image->previous == (Image *) NULL)
1644 {
1645 status=SetImageProgress(image,SaveImageTag,3,6);
1646 if (status == MagickFalse)
1647 break;
1648 }
1649 if (quantum_type == RGBAQuantum)
1650 {
1651 (void) CloseBlob(image);
1652 AppendImageFormat("A",image->filename);
1653 status=OpenBlob(image_info,image,scene == 0 ? WriteBinaryBlobMode :
1654 AppendBinaryBlobMode,exception);
1655 if (status == MagickFalse)
1656 return(status);
1657 for (y=0; y < (ssize_t) image->rows; y++)
1658 {
1659 const Quantum
1660 *magick_restrict p;
1661
1662 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
1663 if (p == (const Quantum *) NULL)
1664 break;
1665 length=ExportQuantumPixels(image,(CacheView *) NULL,quantum_info,
1666 AlphaQuantum,pixels,exception);
1667 count=WriteBlob(image,length,pixels);
1668 if (count != (ssize_t) length)
1669 break;
1670 }
1671 if (image->previous == (Image *) NULL)
1672 {
1673 status=SetImageProgress(image,SaveImageTag,5,6);
1674 if (status == MagickFalse)
1675 break;
1676 }
1677 }
1678 (void) CloseBlob(image);
1679 (void) CopyMagickString(image->filename,image_info->filename,
1680 MagickPathExtent);
1681 if (image->previous == (Image *) NULL)
1682 {
1683 status=SetImageProgress(image,SaveImageTag,6,6);
1684 if (status == MagickFalse)
1685 break;
1686 }
1687 break;
1688 }
1689 }
1690 quantum_info=DestroyQuantumInfo(quantum_info);
1691 if (GetNextImageInList(image) == (Image *) NULL)
1692 break;
1693 image=SyncNextImageInList(image);
1694 status=SetImageProgress(image,SaveImagesTag,scene++,imageListLength);
1695 if (status == MagickFalse)
1696 break;
1697 } while (image_info->adjoin != MagickFalse);
1698 (void) CloseBlob(image);
1699 return(MagickTrue);
1700 }
1701