1 /*
2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3 %                                                                             %
4 %                                                                             %
5 %                                                                             %
6 %                         CCCC   AAA   L      SSSSS                           %
7 %                        C      A   A  L      SS                              %
8 %                        C      AAAAA  L       SSS                            %
9 %                        C      A   A  L         SS                           %
10 %                         CCCC  A   A  LLLLL  SSSSS                           %
11 %                                                                             %
12 %                                                                             %
13 %                 Read/Write CALS Raster Group 1 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 % The CALS raster format is a standard developed by the Computer Aided
37 % Acquisition and Logistics Support (CALS) office of the United States
38 % Department of Defense to standardize graphics data interchange for
39 % electronic publishing, especially in the areas of technical graphics,
40 % CAD/CAM, and image processing applications.
41 %
42 */
43 
44 /*
45   Include declarations.
46 */
47 #include "MagickCore/studio.h"
48 #include "MagickCore/blob.h"
49 #include "MagickCore/blob-private.h"
50 #include "MagickCore/cache.h"
51 #include "MagickCore/colorspace.h"
52 #include "MagickCore/constitute.h"
53 #include "MagickCore/exception.h"
54 #include "MagickCore/exception-private.h"
55 #include "MagickCore/geometry.h"
56 #include "MagickCore/image.h"
57 #include "MagickCore/image-private.h"
58 #include "MagickCore/list.h"
59 #include "MagickCore/magick.h"
60 #include "MagickCore/memory_.h"
61 #include "MagickCore/monitor.h"
62 #include "MagickCore/monitor-private.h"
63 #include "MagickCore/option.h"
64 #include "MagickCore/quantum-private.h"
65 #include "MagickCore/resource_.h"
66 #include "MagickCore/static.h"
67 #include "MagickCore/string_.h"
68 #include "MagickCore/module.h"
69 
70 #if defined(MAGICKCORE_TIFF_DELEGATE)
71 /*
72  Forward declarations.
73 */
74 static MagickBooleanType
75   WriteCALSImage(const ImageInfo *,Image *,ExceptionInfo *);
76 #endif
77 
78 /*
79 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
80 %                                                                             %
81 %                                                                             %
82 %                                                                             %
83 %   I s C A L S                                                               %
84 %                                                                             %
85 %                                                                             %
86 %                                                                             %
87 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
88 %
89 %  IsCALS() returns MagickTrue if the image format type, identified by the
90 %  magick string, is CALS Raster Group 1.
91 %
92 %  The format of the IsCALS method is:
93 %
94 %      MagickBooleanType IsCALS(const unsigned char *magick,const size_t length)
95 %
96 %  A description of each parameter follows:
97 %
98 %    o magick: compare image format pattern against these bytes.
99 %
100 %    o length: Specifies the length of the magick string.
101 %
102 */
IsCALS(const unsigned char * magick,const size_t length)103 static MagickBooleanType IsCALS(const unsigned char *magick,const size_t length)
104 {
105   if (length < 128)
106     return(MagickFalse);
107   if (LocaleNCompare((const char *) magick,"version: MIL-STD-1840",21) == 0)
108     return(MagickTrue);
109   if (LocaleNCompare((const char *) magick,"srcdocid:",9) == 0)
110     return(MagickTrue);
111   if (LocaleNCompare((const char *) magick,"rorient:",8) == 0)
112     return(MagickTrue);
113   return(MagickFalse);
114 }
115 
116 /*
117 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
118 %                                                                             %
119 %                                                                             %
120 %                                                                             %
121 %   R e a d C A L S I m a g e                                                 %
122 %                                                                             %
123 %                                                                             %
124 %                                                                             %
125 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
126 %
127 %  ReadCALSImage() reads an CALS Raster Group 1 image format image file and
128 %  returns it.  It allocates the memory necessary for the new Image structure
129 %  and returns a pointer to the new image.
130 %
131 %  The format of the ReadCALSImage method is:
132 %
133 %      Image *ReadCALSImage(const ImageInfo *image_info,
134 %        ExceptionInfo *exception)
135 %
136 %  A description of each parameter follows:
137 %
138 %    o image_info: the image info.
139 %
140 %    o exception: return any errors or warnings in this structure.
141 %
142 */
ReadCALSImage(const ImageInfo * image_info,ExceptionInfo * exception)143 static Image *ReadCALSImage(const ImageInfo *image_info,
144   ExceptionInfo *exception)
145 {
146   char
147     filename[MagickPathExtent],
148     header[MagickPathExtent],
149     message[MagickPathExtent];
150 
151   FILE
152     *file;
153 
154   Image
155     *image;
156 
157   ImageInfo
158     *read_info;
159 
160   int
161     c,
162     unique_file;
163 
164   MagickBooleanType
165     status;
166 
167   ssize_t
168     i;
169 
170   unsigned long
171     density,
172     direction,
173     height,
174     orientation,
175     pel_path,
176     type,
177     width;
178 
179   /*
180     Open image file.
181   */
182   assert(image_info != (const ImageInfo *) NULL);
183   assert(image_info->signature == MagickCoreSignature);
184   if (image_info->debug != MagickFalse)
185     (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
186       image_info->filename);
187   assert(exception != (ExceptionInfo *) NULL);
188   assert(exception->signature == MagickCoreSignature);
189   image=AcquireImage(image_info,exception);
190   status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
191   if (status == MagickFalse)
192     {
193       image=DestroyImageList(image);
194       return((Image *) NULL);
195     }
196   /*
197     Read CALS header.
198   */
199   (void) memset(header,0,sizeof(header));
200   density=0;
201   direction=0;
202   orientation=1;
203   pel_path=0;
204   type=1;
205   width=0;
206   height=0;
207   for (i=0; i < 16; i++)
208   {
209     if (ReadBlob(image,128,(unsigned char *) header) != 128)
210       break;
211     switch (*header)
212     {
213       case 'R':
214       case 'r':
215       {
216         if (LocaleNCompare(header,"rdensty:",8) == 0)
217           {
218             (void) sscanf(header+8,"%lu",&density);
219             break;
220           }
221         if (LocaleNCompare(header,"rpelcnt:",8) == 0)
222           {
223             (void) sscanf(header+8,"%lu,%lu",&width,&height);
224             break;
225           }
226         if (LocaleNCompare(header,"rorient:",8) == 0)
227           {
228             (void) sscanf(header+8,"%lu,%lu",&pel_path,&direction);
229             if (pel_path == 90)
230               orientation=5;
231             else
232               if (pel_path == 180)
233                 orientation=3;
234               else
235                 if (pel_path == 270)
236                   orientation=7;
237             if (direction == 90)
238               orientation++;
239             break;
240           }
241         if (LocaleNCompare(header,"rtype:",6) == 0)
242           {
243             (void) sscanf(header+6,"%lu",&type);
244             break;
245           }
246         break;
247       }
248     }
249   }
250   /*
251     Read CALS pixels.
252   */
253   file=(FILE *) NULL;
254   unique_file=AcquireUniqueFileResource(filename);
255   if (unique_file != -1)
256     file=fdopen(unique_file,"wb");
257   if ((unique_file == -1) || (file == (FILE *) NULL))
258     ThrowImageException(FileOpenError,"UnableToCreateTemporaryFile");
259   while ((c=ReadBlobByte(image)) != EOF)
260     if (fputc(c,file) != c)
261       break;
262   (void) fclose(file);
263   (void) CloseBlob(image);
264   image=DestroyImage(image);
265   read_info=CloneImageInfo(image_info);
266   SetImageInfoBlob(read_info,(void *) NULL,0);
267   (void) FormatLocaleString(read_info->filename,MagickPathExtent,"group4:%s",
268     filename);
269   (void) FormatLocaleString(message,MagickPathExtent,"%lux%lu",width,height);
270   (void) CloneString(&read_info->size,message);
271   (void) FormatLocaleString(message,MagickPathExtent,"%lu",density);
272   (void) CloneString(&read_info->density,message);
273   read_info->orientation=(OrientationType) orientation;
274   image=ReadImage(read_info,exception);
275   if (image != (Image *) NULL)
276     {
277       (void) CopyMagickString(image->filename,image_info->filename,
278         MagickPathExtent);
279       (void) CopyMagickString(image->magick_filename,image_info->filename,
280         MagickPathExtent);
281       (void) CopyMagickString(image->magick,"CALS",MagickPathExtent);
282     }
283   read_info=DestroyImageInfo(read_info);
284   (void) RelinquishUniqueFileResource(filename);
285   return(image);
286 }
287 
288 /*
289 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
290 %                                                                             %
291 %                                                                             %
292 %                                                                             %
293 %   R e g i s t e r C A L S I m a g e                                         %
294 %                                                                             %
295 %                                                                             %
296 %                                                                             %
297 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
298 %
299 %  RegisterCALSImage() adds attributes for the CALS Raster Group 1 image file
300 %  image format to the list of supported formats.  The attributes include the
301 %  image format tag, a method to read and/or write the format, whether the
302 %  format supports the saving of more than one frame to the same file or blob,
303 %  whether the format supports native in-memory I/O, and a brief description
304 %  of the format.
305 %
306 %  The format of the RegisterCALSImage method is:
307 %
308 %      size_t RegisterCALSImage(void)
309 %
310 */
RegisterCALSImage(void)311 ModuleExport size_t RegisterCALSImage(void)
312 {
313 #define CALSDescription  "Continuous Acquisition and Life-cycle Support Type 1"
314 #define CALSNote  "Specified in MIL-R-28002 and MIL-PRF-28002"
315 
316   MagickInfo
317     *entry;
318 
319   entry=AcquireMagickInfo("CALS","CAL",CALSDescription);
320   entry->decoder=(DecodeImageHandler *) ReadCALSImage;
321 #if defined(MAGICKCORE_TIFF_DELEGATE)
322   entry->encoder=(EncodeImageHandler *) WriteCALSImage;
323 #endif
324   entry->flags^=CoderAdjoinFlag;
325   entry->magick=(IsImageFormatHandler *) IsCALS;
326   entry->note=ConstantString(CALSNote);
327   (void) RegisterMagickInfo(entry);
328   entry=AcquireMagickInfo("CALS","CALS",CALSDescription);
329   entry->decoder=(DecodeImageHandler *) ReadCALSImage;
330 #if defined(MAGICKCORE_TIFF_DELEGATE)
331   entry->encoder=(EncodeImageHandler *) WriteCALSImage;
332 #endif
333   entry->flags^=CoderAdjoinFlag;
334   entry->magick=(IsImageFormatHandler *) IsCALS;
335   entry->note=ConstantString(CALSNote);
336   (void) RegisterMagickInfo(entry);
337   return(MagickImageCoderSignature);
338 }
339 
340 /*
341 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
342 %                                                                             %
343 %                                                                             %
344 %                                                                             %
345 %   U n r e g i s t e r C A L S I m a g e                                     %
346 %                                                                             %
347 %                                                                             %
348 %                                                                             %
349 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
350 %
351 %  UnregisterCALSImage() removes format registrations made by the
352 %  CALS module from the list of supported formats.
353 %
354 %  The format of the UnregisterCALSImage method is:
355 %
356 %      UnregisterCALSImage(void)
357 %
358 */
UnregisterCALSImage(void)359 ModuleExport void UnregisterCALSImage(void)
360 {
361   (void) UnregisterMagickInfo("CAL");
362   (void) UnregisterMagickInfo("CALS");
363 }
364 
365 #if defined(MAGICKCORE_TIFF_DELEGATE)
366 /*
367 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
368 %                                                                             %
369 %                                                                             %
370 %                                                                             %
371 %   W r i t e C A L S I m a g e                                               %
372 %                                                                             %
373 %                                                                             %
374 %                                                                             %
375 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
376 %
377 %  WriteCALSImage() writes an image to a file in CALS Raster Group 1 image
378 %  format.
379 %
380 %  The format of the WriteCALSImage method is:
381 %
382 %      MagickBooleanType WriteCALSImage(const ImageInfo *image_info,
383 %        Image *image,ExceptionInfo *exception)
384 %
385 %  A description of each parameter follows.
386 %
387 %    o image_info: the image info.
388 %
389 %    o image:  The image.
390 %
391 %    o exception: return any errors or warnings in this structure.
392 %
393 */
394 
WriteCALSRecord(Image * image,const char * data)395 static ssize_t WriteCALSRecord(Image *image,const char *data)
396 {
397   char
398     pad[128];
399 
400   const char
401     *p;
402 
403   ssize_t
404     i;
405 
406   ssize_t
407     count;
408 
409   i=0;
410   count=0;
411   if (data != (const char *) NULL)
412     {
413       p=data;
414       for (i=0; (i < 128) && (p[i] != '\0'); i++);
415       count=WriteBlob(image,(size_t) i,(const unsigned char *) data);
416     }
417   if (i < 128)
418     {
419       i=128-i;
420       (void) memset(pad,' ',(size_t) i);
421       count=WriteBlob(image,(size_t) i,(const unsigned char *) pad);
422     }
423   return(count);
424 }
425 
WriteCALSImage(const ImageInfo * image_info,Image * image,ExceptionInfo * exception)426 static MagickBooleanType WriteCALSImage(const ImageInfo *image_info,
427   Image *image,ExceptionInfo *exception)
428 {
429   char
430     header[129];
431 
432   Image
433     *group4_image;
434 
435   ImageInfo
436     *write_info;
437 
438   MagickBooleanType
439     status;
440 
441   ssize_t
442     i;
443 
444   size_t
445     density,
446     length,
447     orient_x,
448     orient_y;
449 
450   ssize_t
451     count;
452 
453   unsigned char
454     *group4;
455 
456   /*
457     Open output image file.
458   */
459   assert(image_info != (const ImageInfo *) NULL);
460   assert(image_info->signature == MagickCoreSignature);
461   assert(image != (Image *) NULL);
462   assert(image->signature == MagickCoreSignature);
463   if (image->debug != MagickFalse)
464     (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
465   assert(exception != (ExceptionInfo *) NULL);
466   assert(exception->signature == MagickCoreSignature);
467   status=OpenBlob(image_info,image,WriteBinaryBlobMode,exception);
468   if (status == MagickFalse)
469     return(status);
470   /*
471     Create standard CALS header.
472   */
473   count=WriteCALSRecord(image,"srcdocid: NONE");
474   (void) count;
475   count=WriteCALSRecord(image,"dstdocid: NONE");
476   count=WriteCALSRecord(image,"txtfilid: NONE");
477   count=WriteCALSRecord(image,"figid: NONE");
478   count=WriteCALSRecord(image,"srcgph: NONE");
479   count=WriteCALSRecord(image,"doccls: NONE");
480   count=WriteCALSRecord(image,"rtype: 1");
481   orient_x=0;
482   orient_y=0;
483   switch (image->orientation)
484   {
485     case TopRightOrientation:
486     {
487       orient_x=180;
488       orient_y=270;
489       break;
490     }
491     case BottomRightOrientation:
492     {
493       orient_x=180;
494       orient_y=90;
495       break;
496     }
497     case BottomLeftOrientation:
498     {
499       orient_y=90;
500       break;
501     }
502     case LeftTopOrientation:
503     {
504       orient_x=270;
505       break;
506     }
507     case RightTopOrientation:
508     {
509       orient_x=270;
510       orient_y=180;
511       break;
512     }
513     case RightBottomOrientation:
514     {
515       orient_x=90;
516       orient_y=180;
517       break;
518     }
519     case LeftBottomOrientation:
520     {
521       orient_x=90;
522       break;
523     }
524     default:
525     {
526       orient_y=270;
527       break;
528     }
529   }
530   (void) FormatLocaleString(header,sizeof(header),"rorient: %03ld,%03ld",
531     (long) orient_x,(long) orient_y);
532   count=WriteCALSRecord(image,header);
533   (void) FormatLocaleString(header,sizeof(header),"rpelcnt: %06lu,%06lu",
534     (unsigned long) image->columns,(unsigned long) image->rows);
535   count=WriteCALSRecord(image,header);
536   density=200;
537   if (image_info->density != (char *) NULL)
538     {
539       GeometryInfo
540         geometry_info;
541 
542       (void) ParseGeometry(image_info->density,&geometry_info);
543       density=(size_t) floor(geometry_info.rho+0.5);
544     }
545   (void) FormatLocaleString(header,sizeof(header),"rdensty: %04lu",
546     (unsigned long) density);
547   count=WriteCALSRecord(image,header);
548   count=WriteCALSRecord(image,"notes: NONE");
549   (void) memset(header,' ',128);
550   for (i=0; i < 5; i++)
551     (void) WriteBlob(image,128,(unsigned char *) header);
552   /*
553     Write CALS pixels.
554   */
555   write_info=CloneImageInfo(image_info);
556   (void) CopyMagickString(write_info->filename,"GROUP4:",MagickPathExtent);
557   (void) CopyMagickString(write_info->magick,"GROUP4",MagickPathExtent);
558   group4_image=CloneImage(image,0,0,MagickTrue,exception);
559   if (group4_image == (Image *) NULL)
560     {
561       write_info=DestroyImageInfo(write_info);
562       (void) CloseBlob(image);
563       return(MagickFalse);
564     }
565   group4=(unsigned char *) ImageToBlob(write_info,group4_image,&length,
566     exception);
567   group4_image=DestroyImage(group4_image);
568   if (group4 == (unsigned char *) NULL)
569     {
570       write_info=DestroyImageInfo(write_info);
571       (void) CloseBlob(image);
572       return(MagickFalse);
573     }
574   write_info=DestroyImageInfo(write_info);
575   if (WriteBlob(image,length,group4) != (ssize_t) length)
576     status=MagickFalse;
577   group4=(unsigned char *) RelinquishMagickMemory(group4);
578   (void) CloseBlob(image);
579   return(status);
580 }
581 #endif
582