1 /*
2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3 %                                                                             %
4 %                                                                             %
5 %                                                                             %
6 %                            RRRR   L       AAA                               %
7 %                            R   R  L      A   A                              %
8 %                            RRRR   L      AAAAA                              %
9 %                            R R    L      A   A                              %
10 %                            R  R   LLLLL  A   A                              %
11 %                                                                             %
12 %                                                                             %
13 %                      Read Alias/Wavefront 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/property.h"
44 #include "MagickCore/blob.h"
45 #include "MagickCore/blob-private.h"
46 #include "MagickCore/cache.h"
47 #include "MagickCore/exception.h"
48 #include "MagickCore/exception-private.h"
49 #include "MagickCore/image.h"
50 #include "MagickCore/image-private.h"
51 #include "MagickCore/list.h"
52 #include "MagickCore/magick.h"
53 #include "MagickCore/memory_.h"
54 #include "MagickCore/monitor.h"
55 #include "MagickCore/monitor-private.h"
56 #include "MagickCore/pixel-accessor.h"
57 #include "MagickCore/quantum-private.h"
58 #include "MagickCore/static.h"
59 #include "MagickCore/string_.h"
60 #include "MagickCore/module.h"
61 
62 /*
63 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
64 %                                                                             %
65 %                                                                             %
66 %                                                                             %
67 %   R e a d R L A I m a g e                                                   %
68 %                                                                             %
69 %                                                                             %
70 %                                                                             %
71 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
72 %
73 %  ReadRLAImage() reads a run-length encoded Wavefront RLA image file
74 %  and returns it.  It allocates the memory necessary for the new Image
75 %  structure and returns a pointer to the new image.
76 %
77 %  Note:  This module was contributed by Lester Vecsey (master@internexus.net).
78 %
79 %  The format of the ReadRLAImage method is:
80 %
81 %      Image *ReadRLAImage(const ImageInfo *image_info,ExceptionInfo *exception)
82 %
83 %  A description of each parameter follows:
84 %
85 %    o image_info: the image info.
86 %
87 %    o exception: return any errors or warnings in this structure.
88 %
89 */
ReadRLAImage(const ImageInfo * image_info,ExceptionInfo * exception)90 static Image *ReadRLAImage(const ImageInfo *image_info,ExceptionInfo *exception)
91 {
92   typedef struct _WindowFrame
93   {
94     short
95       left,
96       right,
97       bottom,
98       top;
99   } WindowFrame;
100 
101   typedef struct _RLAInfo
102   {
103     WindowFrame
104       window,
105       active_window;
106 
107     short
108       frame,
109       storage_type,
110       number_channels,
111       number_matte_channels,
112       number_auxiliary_channels,
113       revision;
114 
115     char
116       gamma[16+1],
117       red_primary[24+1],
118       green_primary[24+1],
119       blue_primary[24+1],
120       white_point[24+1];
121 
122     int
123       job_number;
124 
125     char
126       name[128+1],
127       description[128+1],
128       program[64+1],
129       machine[32+1],
130       user[32+1],
131       date[20+1],
132       aspect[24+1],
133       aspect_ratio[8+1],
134       chan[32+1];
135 
136     short
137       field;
138 
139     char
140       time[12],
141       filter[32];
142 
143     short
144       bits_per_channel,
145       matte_type,
146       matte_bits,
147       auxiliary_type,
148       auxiliary_bits;
149 
150     char
151       auxiliary[32+1],
152       space[36+1];
153 
154     int
155       next;
156   } RLAInfo;
157 
158   Image
159     *image;
160 
161   int
162     channel,
163     length,
164     runlength;
165 
166   MagickBooleanType
167     status;
168 
169   MagickOffsetType
170     offset,
171     *scanlines;
172 
173   ssize_t
174     i,
175     x;
176 
177   Quantum
178     *q;
179 
180   ssize_t
181     count,
182     y;
183 
184   RLAInfo
185     rla_info;
186 
187   unsigned char
188     byte;
189 
190   /*
191     Open image file.
192   */
193   assert(image_info != (const ImageInfo *) NULL);
194   assert(image_info->signature == MagickCoreSignature);
195   if (image_info->debug != MagickFalse)
196     (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
197       image_info->filename);
198   assert(exception != (ExceptionInfo *) NULL);
199   assert(exception->signature == MagickCoreSignature);
200   image=AcquireImage(image_info,exception);
201   status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
202   if (status == MagickFalse)
203     {
204       image=DestroyImageList(image);
205       return((Image *) NULL);
206     }
207   (void) memset(&rla_info,0,sizeof(rla_info));
208   rla_info.window.left=(short) ReadBlobMSBShort(image);
209   rla_info.window.right=(short) ReadBlobMSBShort(image);
210   rla_info.window.bottom=(short) ReadBlobMSBShort(image);
211   rla_info.window.top=(short) ReadBlobMSBShort(image);
212   rla_info.active_window.left=(short) ReadBlobMSBShort(image);
213   rla_info.active_window.right=(short) ReadBlobMSBShort(image);
214   rla_info.active_window.bottom=(short) ReadBlobMSBShort(image);
215   rla_info.active_window.top=(short) ReadBlobMSBShort(image);
216   rla_info.frame=(short) ReadBlobMSBShort(image);
217   rla_info.storage_type=(short) ReadBlobMSBShort(image);
218   rla_info.number_channels=(short) ReadBlobMSBShort(image);
219   if (rla_info.number_channels < 0)
220     ThrowReaderException(CorruptImageError,"ImproperImageHeader");
221   rla_info.number_matte_channels=(short) ReadBlobMSBShort(image);
222   if (rla_info.number_matte_channels < 0)
223     ThrowReaderException(CorruptImageError,"ImproperImageHeader");
224   if ((rla_info.number_channels > 3) || (rla_info.number_matte_channels > 3))
225     ThrowReaderException(CoderError,"Unsupported number of channels");
226   if (rla_info.number_channels == 0)
227     rla_info.number_channels=3;
228   rla_info.number_channels+=rla_info.number_matte_channels;
229   rla_info.number_auxiliary_channels=(short) ReadBlobMSBShort(image);
230   rla_info.revision=(short) ReadBlobMSBShort(image);
231   (void) ReadBlob(image,16,(unsigned char *) rla_info.gamma);
232   (void) ReadBlob(image,24,(unsigned char *) rla_info.red_primary);
233   (void) ReadBlob(image,24,(unsigned char *) rla_info.green_primary);
234   (void) ReadBlob(image,24,(unsigned char *) rla_info.blue_primary);
235   (void) ReadBlob(image,24,(unsigned char *) rla_info.white_point);
236   rla_info.job_number=ReadBlobMSBSignedLong(image);
237   (void) ReadBlob(image,128,(unsigned char *) rla_info.name);
238   (void) ReadBlob(image,128,(unsigned char *) rla_info.description);
239   rla_info.description[127]='\0';
240   (void) ReadBlob(image,64,(unsigned char *) rla_info.program);
241   (void) ReadBlob(image,32,(unsigned char *) rla_info.machine);
242   (void) ReadBlob(image,32,(unsigned char *) rla_info.user);
243   (void) ReadBlob(image,20,(unsigned char *) rla_info.date);
244   (void) ReadBlob(image,24,(unsigned char *) rla_info.aspect);
245   (void) ReadBlob(image,8,(unsigned char *) rla_info.aspect_ratio);
246   (void) ReadBlob(image,32,(unsigned char *) rla_info.chan);
247   rla_info.field=(short) ReadBlobMSBShort(image);
248   (void) ReadBlob(image,12,(unsigned char *) rla_info.time);
249   (void) ReadBlob(image,32,(unsigned char *) rla_info.filter);
250   rla_info.bits_per_channel=(short) ReadBlobMSBShort(image);
251   rla_info.matte_type=(short) ReadBlobMSBShort(image);
252   rla_info.matte_bits=(short) ReadBlobMSBShort(image);
253   rla_info.auxiliary_type=(short) ReadBlobMSBShort(image);
254   rla_info.auxiliary_bits=(short) ReadBlobMSBShort(image);
255   (void) ReadBlob(image,32,(unsigned char *) rla_info.auxiliary);
256   count=ReadBlob(image,36,(unsigned char *) rla_info.space);
257   if ((size_t) count != 36)
258     ThrowReaderException(CorruptImageError,"UnableToReadImageData");
259   rla_info.next=ReadBlobMSBSignedLong(image);
260   /*
261     Initialize image structure.
262   */
263   image->alpha_trait=rla_info.number_matte_channels != 0 ? BlendPixelTrait :
264     UndefinedPixelTrait;
265   image->columns=(size_t) (rla_info.active_window.right-
266     rla_info.active_window.left+1);
267   image->rows=(size_t) (rla_info.active_window.top-
268     rla_info.active_window.bottom+1);
269   if (image_info->ping != MagickFalse)
270     {
271       (void) CloseBlob(image);
272       return(GetFirstImageInList(image));
273     }
274   status=SetImageExtent(image,image->columns,image->rows,exception);
275   if (status == MagickFalse)
276     return(DestroyImageList(image));
277   scanlines=(MagickOffsetType *) AcquireQuantumMemory(image->rows,
278     sizeof(*scanlines));
279   if (scanlines == (MagickOffsetType *) NULL)
280     ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
281   if (*rla_info.description != '\0')
282     (void) SetImageProperty(image,"comment",rla_info.description,exception);
283   /*
284     Read offsets to each scanline data.
285   */
286   for (i=0; i < (ssize_t) image->rows; i++)
287     scanlines[i]=(MagickOffsetType) ReadBlobMSBSignedLong(image);
288   if (EOFBlob(image) != MagickFalse)
289     {
290       scanlines=(MagickOffsetType *) RelinquishMagickMemory(scanlines);
291       ThrowReaderException(CorruptImageError,"ImproperImageHeader");
292     }
293   /*
294     Read image data.
295   */
296   for (y=0; y < (ssize_t) image->rows; y++)
297   {
298     offset=SeekBlob(image,scanlines[image->rows-y-1],SEEK_SET);
299     if (offset < 0)
300       {
301         scanlines=(MagickOffsetType *) RelinquishMagickMemory(scanlines);
302         ThrowReaderException(CorruptImageError,"ImproperImageHeader");
303       }
304     x=0;
305     for (channel=0; channel < (int) rla_info.number_channels; channel++)
306     {
307       length=ReadBlobMSBSignedShort(image);
308       while (length > 0)
309       {
310         byte=(unsigned char) ReadBlobByte(image);
311         runlength=byte;
312         if (byte > 127)
313           runlength=byte-256;
314         length--;
315         if (length == 0)
316           break;
317         if (runlength < 0)
318           {
319             while (runlength < 0)
320             {
321               q=GetAuthenticPixels(image,(ssize_t) (x % image->columns),y,1,1,
322                 exception);
323               if (q == (Quantum *) NULL)
324                 break;
325               byte=(unsigned char) ReadBlobByte(image);
326               length--;
327               switch (channel)
328               {
329                 case 0:
330                 {
331                   SetPixelRed(image,ScaleCharToQuantum(byte),q);
332                   break;
333                 }
334                 case 1:
335                 {
336                   SetPixelGreen(image,ScaleCharToQuantum(byte),q);
337                   break;
338                 }
339                 case 2:
340                 {
341                   SetPixelBlue(image,ScaleCharToQuantum(byte),q);
342                   break;
343                 }
344                 case 3:
345                 default:
346                 {
347                   SetPixelAlpha(image,ScaleCharToQuantum(byte),q);
348                   break;
349                 }
350               }
351               if (SyncAuthenticPixels(image,exception) == MagickFalse)
352                 break;
353               x++;
354               runlength++;
355             }
356             continue;
357           }
358         byte=(unsigned char) ReadBlobByte(image);
359         length--;
360         runlength++;
361         do
362         {
363           q=GetAuthenticPixels(image,(ssize_t) (x % image->columns),y,1,1,
364             exception);
365           if (q == (Quantum *) NULL)
366             break;
367           switch (channel)
368           {
369             case 0:
370             {
371               SetPixelRed(image,ScaleCharToQuantum(byte),q);
372               break;
373             }
374             case 1:
375             {
376               SetPixelGreen(image,ScaleCharToQuantum(byte),q);
377               break;
378             }
379             case 2:
380             {
381               SetPixelBlue(image,ScaleCharToQuantum(byte),q);
382               break;
383             }
384             case 3:
385             default:
386             {
387               SetPixelAlpha(image,ScaleCharToQuantum(byte),q);
388               break;
389             }
390           }
391           if (SyncAuthenticPixels(image,exception) == MagickFalse)
392             break;
393           x++;
394           runlength--;
395         }
396         while (runlength > 0);
397       }
398     }
399     if ((x/(ssize_t) rla_info.number_channels) > (ssize_t) image->columns)
400       {
401         scanlines=(MagickOffsetType *) RelinquishMagickMemory(scanlines);
402         ThrowReaderException(CorruptImageError,"CorruptImage");
403       }
404     if (EOFBlob(image) != MagickFalse)
405       break;
406     status=SetImageProgress(image,LoadImageTag,(MagickOffsetType) y,
407       image->rows);
408     if (status == MagickFalse)
409       break;
410   }
411   scanlines=(MagickOffsetType *) RelinquishMagickMemory(scanlines);
412   if (EOFBlob(image) != MagickFalse)
413     ThrowFileException(exception,CorruptImageError,"UnexpectedEndOfFile",
414       image->filename);
415   (void) CloseBlob(image);
416   return(GetFirstImageInList(image));
417 }
418 
419 /*
420 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
421 %                                                                             %
422 %                                                                             %
423 %                                                                             %
424 %   R e g i s t e r R L A I m a g e                                           %
425 %                                                                             %
426 %                                                                             %
427 %                                                                             %
428 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
429 %
430 %  RegisterRLAImage() adds attributes for the RLA image format to
431 %  the list of supported formats.  The attributes include the image format
432 %  tag, a method to read and/or write the format, whether the format
433 %  supports the saving of more than one frame to the same file or blob,
434 %  whether the format supports native in-memory I/O, and a brief
435 %  description of the format.
436 %
437 %  The format of the RegisterRLAImage method is:
438 %
439 %      size_t RegisterRLAImage(void)
440 %
441 */
RegisterRLAImage(void)442 ModuleExport size_t RegisterRLAImage(void)
443 {
444   MagickInfo
445     *entry;
446 
447   entry=AcquireMagickInfo("RLA","RLA","Alias/Wavefront image");
448   entry->decoder=(DecodeImageHandler *) ReadRLAImage;
449   entry->flags^=CoderAdjoinFlag;
450   entry->flags|=CoderDecoderSeekableStreamFlag;
451   (void) RegisterMagickInfo(entry);
452   return(MagickImageCoderSignature);
453 }
454 
455 /*
456 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
457 %                                                                             %
458 %                                                                             %
459 %                                                                             %
460 %   U n r e g i s t e r R L A I m a g e                                       %
461 %                                                                             %
462 %                                                                             %
463 %                                                                             %
464 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
465 %
466 %  UnregisterRLAImage() removes format registrations made by the
467 %  RLA module from the list of supported formats.
468 %
469 %  The format of the UnregisterRLAImage method is:
470 %
471 %      UnregisterRLAImage(void)
472 %
473 */
UnregisterRLAImage(void)474 ModuleExport void UnregisterRLAImage(void)
475 {
476   (void) UnregisterMagickInfo("RLA");
477 }
478