]> granicus.if.org Git - imagemagick/blob - coders/rla.c
0884d06a49ff2bc5ff9985a5e658e8fc9e4407a4
[imagemagick] / coders / rla.c
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-2014 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 %    http://www.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 \f
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 \f
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 */
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],
117       red_primary[24],
118       green_primary[24],
119       blue_primary[24],
120       white_point[24];
121
122     ssize_t
123       job_number;
124
125     char
126       name[128],
127       description[128],
128       program[64],
129       machine[32],
130       user[32],
131       date[20],
132       aspect[24],
133       aspect_ratio[8],
134       chan[32];
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],
152       space[36];
153
154     ssize_t
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
172   register ssize_t
173     i,
174     x;
175
176   register Quantum
177     *q;
178
179   ssize_t
180     count,
181     *scanlines,
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 == MagickSignature);
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 == MagickSignature);
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   rla_info.window.left=(short) ReadBlobMSBShort(image);
208   rla_info.window.right=(short) ReadBlobMSBShort(image);
209   rla_info.window.bottom=(short) ReadBlobMSBShort(image);
210   rla_info.window.top=(short) ReadBlobMSBShort(image);
211   rla_info.active_window.left=(short) ReadBlobMSBShort(image);
212   rla_info.active_window.right=(short) ReadBlobMSBShort(image);
213   rla_info.active_window.bottom=(short) ReadBlobMSBShort(image);
214   rla_info.active_window.top=(short) ReadBlobMSBShort(image);
215   rla_info.frame=(short) ReadBlobMSBShort(image);
216   rla_info.storage_type=(short) ReadBlobMSBShort(image);
217   rla_info.number_channels=(short) ReadBlobMSBShort(image);
218   rla_info.number_matte_channels=(short) ReadBlobMSBShort(image);
219   if (rla_info.number_channels == 0)
220     rla_info.number_channels=3;
221   rla_info.number_channels+=rla_info.number_matte_channels;
222   rla_info.number_auxiliary_channels=(short) ReadBlobMSBShort(image);
223   rla_info.revision=(short) ReadBlobMSBShort(image);
224   count=ReadBlob(image,16,(unsigned char *) rla_info.gamma);
225   count=ReadBlob(image,24,(unsigned char *) rla_info.red_primary);
226   count=ReadBlob(image,24,(unsigned char *) rla_info.green_primary);
227   count=ReadBlob(image,24,(unsigned char *) rla_info.blue_primary);
228   count=ReadBlob(image,24,(unsigned char *) rla_info.white_point);
229   rla_info.job_number=(int) ReadBlobMSBLong(image);
230   count=ReadBlob(image,128,(unsigned char *) rla_info.name);
231   (void) ResetMagickMemory(value,0,sizeof(rla_info.description));
232   count=ReadBlob(image,128,(unsigned char *) rla_info.description);
233   count=ReadBlob(image,64,(unsigned char *) rla_info.program);
234   count=ReadBlob(image,32,(unsigned char *) rla_info.machine);
235   count=ReadBlob(image,32,(unsigned char *) rla_info.user);
236   count=ReadBlob(image,20,(unsigned char *) rla_info.date);
237   count=ReadBlob(image,24,(unsigned char *) rla_info.aspect);
238   count=ReadBlob(image,8,(unsigned char *) rla_info.aspect_ratio);
239   count=ReadBlob(image,32,(unsigned char *) rla_info.chan);
240   rla_info.field=(short) ReadBlobMSBShort(image);
241   count=ReadBlob(image,12,(unsigned char *) rla_info.time);
242   count=ReadBlob(image,32,(unsigned char *) rla_info.filter);
243   rla_info.bits_per_channel=(short) ReadBlobMSBShort(image);
244   rla_info.matte_type=(short) ReadBlobMSBShort(image);
245   rla_info.matte_bits=(short) ReadBlobMSBShort(image);
246   rla_info.auxiliary_type=(short) ReadBlobMSBShort(image);
247   rla_info.auxiliary_bits=(short) ReadBlobMSBShort(image);
248   count=ReadBlob(image,32,(unsigned char *) rla_info.auxiliary);
249   count=ReadBlob(image,36,(unsigned char *) rla_info.space);
250   if ((size_t) count != 36)
251     ThrowReaderException(CorruptImageError,"UnableToReadImageData");
252   rla_info.next=(int) ReadBlobMSBLong(image);
253   /*
254     Initialize image structure.
255   */
256   image->alpha_trait=rla_info.number_matte_channels != 0 ? BlendPixelTrait : 
257     UndefinedPixelTrait;
258   image->columns=1UL*rla_info.active_window.right-rla_info.active_window.left+1;
259   image->rows=1UL*rla_info.active_window.top-rla_info.active_window.bottom+1;
260   if (image_info->ping != MagickFalse)
261     {
262       (void) CloseBlob(image);
263       return(GetFirstImageInList(image));
264     }
265   scanlines=(ssize_t *) AcquireQuantumMemory(image->rows,sizeof(*scanlines));
266   if (scanlines == (ssize_t *) NULL)
267     ThrowReaderException(ResourceLimitError,"MemoryAllocationFailed");
268   if (*rla_info.description != '\0')
269     (void) SetImageProperty(image,"comment",rla_info.description,exception);
270   /*
271     Read offsets to each scanline data.
272   */
273   for (i=0; i < (ssize_t) image->rows; i++)
274     scanlines[i]=(int) ReadBlobMSBLong(image);
275   /*
276     Read image data.
277   */
278   x=0;
279   for (y=0; y < (ssize_t) image->rows; y++)
280   {
281     offset=SeekBlob(image,scanlines[image->rows-y-1],SEEK_SET);
282     if (offset < 0)
283       ThrowReaderException(CorruptImageError,"ImproperImageHeader");
284     for (channel=0; channel < (int) rla_info.number_channels; channel++)
285     {
286       length=(int) ReadBlobMSBShort(image);
287       while (length > 0)
288       {
289         byte=(unsigned char) ReadBlobByte(image);
290         runlength=byte;
291         if (byte > 127)
292           runlength=byte-256;
293         length--;
294         if (length == 0)
295           break;
296         if (runlength < 0)
297           {
298             while (runlength < 0)
299             {
300               q=GetAuthenticPixels(image,(ssize_t) (x % image->columns),
301                 (ssize_t) (y % image->rows),1,1,exception);
302               if (q == (Quantum *) NULL)
303                 break;
304               byte=(unsigned char) ReadBlobByte(image);
305               length--;
306               switch (channel)
307               {
308                 case 0:
309                 {
310                   SetPixelRed(image,ScaleCharToQuantum(byte),q);
311                   break;
312                 }
313                 case 1:
314                 {
315                   SetPixelGreen(image,ScaleCharToQuantum(byte),q);
316                   break;
317                 }
318                 case 2:
319                 {
320                   SetPixelBlue(image,ScaleCharToQuantum(byte),q);
321                   break;
322                 }
323                 case 3:
324                 default:
325                 {
326                   SetPixelAlpha(image,ScaleCharToQuantum(byte),q);
327                   break;
328                 }
329               }
330               if (SyncAuthenticPixels(image,exception) == MagickFalse)
331                 break;
332               x++;
333               runlength++;
334             }
335             continue;
336           }
337         byte=(unsigned char) ReadBlobByte(image);
338         length--;
339         runlength++;
340         do
341         {
342           q=GetAuthenticPixels(image,(ssize_t) (x % image->columns),
343             (ssize_t) (y % image->rows),1,1,exception);
344           if (q == (Quantum *) NULL)
345             break;
346           switch (channel)
347           {
348             case 0:
349             {
350               SetPixelRed(image,ScaleCharToQuantum(byte),q);
351               break;
352             }
353             case 1:
354             {
355               SetPixelGreen(image,ScaleCharToQuantum(byte),q);
356               break;
357             }
358             case 2:
359             {
360               SetPixelBlue(image,ScaleCharToQuantum(byte),q);
361               break;
362             }
363             case 3:
364             default:
365             {
366               SetPixelAlpha(image,ScaleCharToQuantum(byte),q);
367               break;
368             }
369           }
370           if (SyncAuthenticPixels(image,exception) == MagickFalse)
371             break;
372           x++;
373           runlength--;
374         }
375         while (runlength > 0);
376       }
377     }
378     status=SetImageProgress(image,LoadImageTag,(MagickOffsetType) y,
379       image->rows);
380     if (status == MagickFalse)
381       break;
382   }
383   if (EOFBlob(image) != MagickFalse)
384     ThrowFileException(exception,CorruptImageError,"UnexpectedEndOfFile",
385       image->filename);
386   (void) CloseBlob(image);
387   return(GetFirstImageInList(image));
388 }
389 \f
390 /*
391 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
392 %                                                                             %
393 %                                                                             %
394 %                                                                             %
395 %   R e g i s t e r R L A I m a g e                                           %
396 %                                                                             %
397 %                                                                             %
398 %                                                                             %
399 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
400 %
401 %  RegisterRLAImage() adds attributes for the RLA image format to
402 %  the list of supported formats.  The attributes include the image format
403 %  tag, a method to read and/or write the format, whether the format
404 %  supports the saving of more than one frame to the same file or blob,
405 %  whether the format supports native in-memory I/O, and a brief
406 %  description of the format.
407 %
408 %  The format of the RegisterRLAImage method is:
409 %
410 %      size_t RegisterRLAImage(void)
411 %
412 */
413 ModuleExport size_t RegisterRLAImage(void)
414 {
415   MagickInfo
416     *entry;
417
418   entry=SetMagickInfo("RLA");
419   entry->decoder=(DecodeImageHandler *) ReadRLAImage;
420   entry->adjoin=MagickFalse;
421   entry->seekable_stream=MagickTrue;
422   entry->description=ConstantString("Alias/Wavefront image");
423   entry->module=ConstantString("RLA");
424   (void) RegisterMagickInfo(entry);
425   return(MagickImageCoderSignature);
426 }
427 \f
428 /*
429 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
430 %                                                                             %
431 %                                                                             %
432 %                                                                             %
433 %   U n r e g i s t e r R L A I m a g e                                       %
434 %                                                                             %
435 %                                                                             %
436 %                                                                             %
437 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
438 %
439 %  UnregisterRLAImage() removes format registrations made by the
440 %  RLA module from the list of supported formats.
441 %
442 %  The format of the UnregisterRLAImage method is:
443 %
444 %      UnregisterRLAImage(void)
445 %
446 */
447 ModuleExport void UnregisterRLAImage(void)
448 {
449   (void) UnregisterMagickInfo("RLA");
450 }