1 /*
2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3 % %
4 % %
5 % %
6 % IIIII N N L IIIII N N EEEEE %
7 % I NN N L I NN N E %
8 % I N N N L I N N N EEE %
9 % I N NN L I N NN E %
10 % IIIII N N LLLLL IIIII N N EEEEE %
11 % %
12 % %
13 % Read Inline Images %
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/client.h"
46 #include "MagickCore/constitute.h"
47 #include "MagickCore/display.h"
48 #include "MagickCore/exception.h"
49 #include "MagickCore/exception-private.h"
50 #include "MagickCore/image.h"
51 #include "MagickCore/image-private.h"
52 #include "MagickCore/list.h"
53 #include "MagickCore/magick.h"
54 #include "MagickCore/memory_.h"
55 #include "MagickCore/option.h"
56 #include "MagickCore/quantum-private.h"
57 #include "MagickCore/static.h"
58 #include "MagickCore/string_.h"
59 #include "MagickCore/module.h"
60 #include "MagickCore/utility.h"
61 #include "MagickCore/xwindow.h"
62 #include "MagickCore/xwindow-private.h"
63
64 /*
65 Forward declarations.
66 */
67 static MagickBooleanType
68 WriteINLINEImage(const ImageInfo *,Image *,ExceptionInfo *);
69
70 /*
71 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
72 % %
73 % %
74 % %
75 % R e a d I N L I N E I m a g e %
76 % %
77 % %
78 % %
79 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
80 %
81 % ReadINLINEImage() reads base64-encoded inlines images.
82 %
83 % The format of the ReadINLINEImage method is:
84 %
85 % Image *ReadINLINEImage(const ImageInfo *image_info,
86 % ExceptionInfo *exception)
87 %
88 % A description of each parameter follows:
89 %
90 % o image_info: the image info.
91 %
92 % o exception: return any errors or warnings in this structure.
93 %
94 */
ReadINLINEImage(const ImageInfo * image_info,ExceptionInfo * exception)95 static Image *ReadINLINEImage(const ImageInfo *image_info,
96 ExceptionInfo *exception)
97 {
98 Image
99 *image;
100
101 MagickBooleanType
102 status;
103
104 size_t
105 i;
106
107 size_t
108 quantum;
109
110 ssize_t
111 count;
112
113 unsigned char
114 *inline_image;
115
116 /*
117 Open image file.
118 */
119 assert(image_info != (const ImageInfo *) NULL);
120 assert(image_info->signature == MagickCoreSignature);
121 if (image_info->debug != MagickFalse)
122 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
123 image_info->filename);
124 assert(exception != (ExceptionInfo *) NULL);
125 assert(exception->signature == MagickCoreSignature);
126 if (LocaleNCompare(image_info->filename,"data:",5) == 0)
127 {
128 char
129 *filename;
130
131 Image
132 *data_image;
133
134 filename=AcquireString("data:");
135 (void) ConcatenateMagickString(filename,image_info->filename,
136 MagickPathExtent);
137 data_image=ReadInlineImage(image_info,filename,exception);
138 filename=DestroyString(filename);
139 return(data_image);
140 }
141 image=AcquireImage(image_info,exception);
142 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
143 if (status == MagickFalse)
144 {
145 image=DestroyImageList(image);
146 return((Image *) NULL);
147 }
148 quantum=MagickMin((size_t) GetBlobSize(image),MagickMaxBufferExtent);
149 if (quantum == 0)
150 quantum=MagickMaxBufferExtent;
151 inline_image=(unsigned char *) AcquireQuantumMemory(quantum,
152 sizeof(*inline_image));
153 count=0;
154 for (i=0; inline_image != (unsigned char *) NULL; i+=count)
155 {
156 count=(ssize_t) ReadBlob(image,quantum,inline_image+i);
157 if (count <= 0)
158 {
159 count=0;
160 if (errno != EINTR)
161 break;
162 }
163 if (~((size_t) i) < (quantum+1))
164 {
165 inline_image=(unsigned char *) RelinquishMagickMemory(inline_image);
166 break;
167 }
168 inline_image=(unsigned char *) ResizeQuantumMemory(inline_image,i+count+
169 quantum+1,sizeof(*inline_image));
170 }
171 if (inline_image == (unsigned char *) NULL)
172 {
173 (void) ThrowMagickException(exception,GetMagickModule(),
174 ResourceLimitError,"MemoryAllocationFailed","`%s'",image->filename);
175 return((Image *) NULL);
176 }
177 inline_image[i+count]='\0';
178 image=DestroyImageList(image);
179 image=ReadInlineImage(image_info,(char *) inline_image,exception);
180 inline_image=(unsigned char *) RelinquishMagickMemory(inline_image);
181 return(image);
182 }
183
184 /*
185 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
186 % %
187 % %
188 % %
189 % R e g i s t e r I N L I N E I m a g e %
190 % %
191 % %
192 % %
193 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
194 %
195 % RegisterINLINEImage() adds attributes for the INLINE image format to
196 % the list of supported formats. The attributes include the image format
197 % tag, a method to read and/or write the format, whether the format
198 % supports the saving of more than one frame to the same file or blob,
199 % whether the format supports native in-memory I/O, and a brief
200 % description of the format.
201 %
202 % The format of the RegisterINLINEImage method is:
203 %
204 % size_t RegisterINLINEImage(void)
205 %
206 */
RegisterINLINEImage(void)207 ModuleExport size_t RegisterINLINEImage(void)
208 {
209 MagickInfo
210 *entry;
211
212 entry=AcquireMagickInfo("INLINE","DATA","Base64-encoded inline images");
213 entry->decoder=(DecodeImageHandler *) ReadINLINEImage;
214 entry->encoder=(EncodeImageHandler *) WriteINLINEImage;
215 entry->format_type=ImplicitFormatType;
216 (void) RegisterMagickInfo(entry);
217 entry=AcquireMagickInfo("INLINE","INLINE","Base64-encoded inline images");
218 entry->decoder=(DecodeImageHandler *) ReadINLINEImage;
219 entry->encoder=(EncodeImageHandler *) WriteINLINEImage;
220 entry->format_type=ImplicitFormatType;
221 (void) RegisterMagickInfo(entry);
222 return(MagickImageCoderSignature);
223 }
224
225 /*
226 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
227 % %
228 % %
229 % %
230 % U n r e g i s t e r I N L I N E I m a g e %
231 % %
232 % %
233 % %
234 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
235 %
236 % UnregisterINLINEImage() removes format registrations made by the
237 % INLINE module from the list of supported formats.
238 %
239 % The format of the UnregisterINLINEImage method is:
240 %
241 % UnregisterINLINEImage(void)
242 %
243 */
UnregisterINLINEImage(void)244 ModuleExport void UnregisterINLINEImage(void)
245 {
246 (void) UnregisterMagickInfo("INLINE");
247 (void) UnregisterMagickInfo("DATA");
248 }
249
250 /*
251 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
252 % %
253 % %
254 % %
255 % W r i t e I N L I N E I m a g e %
256 % %
257 % %
258 % %
259 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
260 %
261 % WriteINLINEImage() writes an image to a file in INLINE format (Base64).
262 %
263 % The format of the WriteINLINEImage method is:
264 %
265 % MagickBooleanType WriteINLINEImage(const ImageInfo *image_info,
266 % Image *image,ExceptionInfo *exception)
267 %
268 % A description of each parameter follows.
269 %
270 % o image_info: the image info.
271 %
272 % o image: The image.
273 %
274 % o exception: return any errors or warnings in this structure.
275 %
276 */
WriteINLINEImage(const ImageInfo * image_info,Image * image,ExceptionInfo * exception)277 static MagickBooleanType WriteINLINEImage(const ImageInfo *image_info,
278 Image *image,ExceptionInfo *exception)
279 {
280 char
281 *base64,
282 message[MagickPathExtent];
283
284 const MagickInfo
285 *magick_info;
286
287 Image
288 *write_image;
289
290 ImageInfo
291 *write_info;
292
293 MagickBooleanType
294 status;
295
296 size_t
297 blob_length,
298 encode_length;
299
300 unsigned char
301 *blob;
302
303 /*
304 Convert image to base64-encoding.
305 */
306 assert(image_info != (const ImageInfo *) NULL);
307 assert(image_info->signature == MagickCoreSignature);
308 assert(image != (Image *) NULL);
309 assert(image->signature == MagickCoreSignature);
310 if (image->debug != MagickFalse)
311 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
312 write_info=CloneImageInfo(image_info);
313 (void) SetImageInfo(write_info,1,exception);
314 if (LocaleCompare(write_info->magick,"INLINE") == 0)
315 (void) CopyMagickString(write_info->magick,image->magick,MagickPathExtent);
316 magick_info=GetMagickInfo(write_info->magick,exception);
317 if ((magick_info == (const MagickInfo *) NULL) ||
318 (GetMagickMimeType(magick_info) == (const char *) NULL))
319 {
320 write_info=DestroyImageInfo(write_info);
321 ThrowWriterException(CorruptImageError,"ImageTypeNotSupported");
322 }
323 (void) CopyMagickString(image->filename,write_info->filename,
324 MagickPathExtent);
325 blob_length=2048;
326 write_image=CloneImage(image,0,0,MagickTrue,exception);
327 if (write_image == (Image *) NULL)
328 {
329 write_info=DestroyImageInfo(write_info);
330 return(MagickTrue);
331 }
332 blob=(unsigned char *) ImageToBlob(write_info,write_image,&blob_length,
333 exception);
334 write_image=DestroyImage(write_image);
335 write_info=DestroyImageInfo(write_info);
336 if (blob == (unsigned char *) NULL)
337 return(MagickFalse);
338 encode_length=0;
339 base64=Base64Encode(blob,blob_length,&encode_length);
340 blob=(unsigned char *) RelinquishMagickMemory(blob);
341 if (base64 == (char *) NULL)
342 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
343 /*
344 Write base64-encoded image.
345 */
346 status=OpenBlob(image_info,image,WriteBinaryBlobMode,exception);
347 if (status == MagickFalse)
348 {
349 base64=DestroyString(base64);
350 return(status);
351 }
352 (void) FormatLocaleString(message,MagickPathExtent,"data:%s;base64,",
353 GetMagickMimeType(magick_info));
354 (void) WriteBlobString(image,message);
355 (void) WriteBlobString(image,base64);
356 base64=DestroyString(base64);
357 return(MagickTrue);
358 }
359