blob: d37d4898576d525676ce1d02e3a9abcd1f3c094a [file] [log] [blame]
cristyb1860752011-03-14 00:27:46 +00001/*
2%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3% %
4% %
5% %
6% W W EEEEE BBBB PPPP %
7% W W E B B P P %
8% W W W EEE BBBB PPPP %
9% WW WW E B B P %
10% W W EEEEE BBBB P %
11% %
12% %
13% Read/Write WebP Image Format %
14% %
15% Software Design %
Cristycb635602017-08-02 06:59:15 -040016% Cristy %
cristyb1860752011-03-14 00:27:46 +000017% March 2011 %
18% %
19% %
Cristyd8420112021-01-01 14:52:00 -050020% Copyright 1999-2021 ImageMagick Studio LLC, a non-profit organization %
cristyb1860752011-03-14 00:27:46 +000021% 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% %
Cristy83d74de2018-10-13 10:17:25 -040026% https://imagemagick.org/script/license.php %
cristyb1860752011-03-14 00:27:46 +000027% %
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*/
cristy4c08aed2011-07-01 19:47:50 +000042#include "MagickCore/studio.h"
cristy76ce6e12013-04-05 14:33:38 +000043#include "MagickCore/artifact.h"
cristy4c08aed2011-07-01 19:47:50 +000044#include "MagickCore/blob.h"
45#include "MagickCore/blob-private.h"
46#include "MagickCore/client.h"
cristye65bb192013-08-01 11:22:06 +000047#include "MagickCore/colorspace-private.h"
cristy4c08aed2011-07-01 19:47:50 +000048#include "MagickCore/display.h"
49#include "MagickCore/exception.h"
50#include "MagickCore/exception-private.h"
51#include "MagickCore/image.h"
52#include "MagickCore/image-private.h"
Cristy75f47302020-08-02 10:08:22 -040053#include "MagickCore/layer.h"
cristy4c08aed2011-07-01 19:47:50 +000054#include "MagickCore/list.h"
55#include "MagickCore/magick.h"
56#include "MagickCore/monitor.h"
57#include "MagickCore/monitor-private.h"
58#include "MagickCore/memory_.h"
59#include "MagickCore/option.h"
60#include "MagickCore/pixel-accessor.h"
Cristy692b6782017-11-19 12:35:18 -050061#include "MagickCore/profile.h"
Cristy46d4fb52019-09-29 20:32:20 -040062#include "MagickCore/property.h"
cristy4c08aed2011-07-01 19:47:50 +000063#include "MagickCore/quantum-private.h"
64#include "MagickCore/static.h"
65#include "MagickCore/string_.h"
cristye71e0892013-02-18 13:13:53 +000066#include "MagickCore/string-private.h"
cristy4c08aed2011-07-01 19:47:50 +000067#include "MagickCore/module.h"
68#include "MagickCore/utility.h"
69#include "MagickCore/xwindow.h"
70#include "MagickCore/xwindow-private.h"
cristy644040e2011-03-14 17:31:59 +000071#if defined(MAGICKCORE_WEBP_DELEGATE)
72#include <webp/decode.h>
73#include <webp/encode.h>
Cristy692b6782017-11-19 12:35:18 -050074#if defined(MAGICKCORE_WEBPMUX_DELEGATE)
75#include <webp/mux.h>
Cristy46d4fb52019-09-29 20:32:20 -040076#include <webp/demux.h>
Cristy692b6782017-11-19 12:35:18 -050077#endif
cristy644040e2011-03-14 17:31:59 +000078#endif
cristyb1860752011-03-14 00:27:46 +000079
80/*
81 Forward declarations.
82*/
83#if defined(MAGICKCORE_WEBP_DELEGATE)
84static MagickBooleanType
cristy018f07f2011-09-04 21:15:19 +000085 WriteWEBPImage(const ImageInfo *,Image *,ExceptionInfo *);
cristyb1860752011-03-14 00:27:46 +000086#endif
87
cristydffa79e2013-02-20 00:57:26 +000088/*
89%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
90% %
91% %
92% %
93% I s W E B P %
94% %
95% %
96% %
97%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
98%
99% IsWEBP() returns MagickTrue if the image format type, identified by the
100% magick string, is WebP.
101%
102% The format of the IsWEBP method is:
103%
104% MagickBooleanType IsWEBP(const unsigned char *magick,const size_t length)
105%
106% A description of each parameter follows:
107%
108% o magick: compare image format pattern against these bytes.
109%
110% o length: Specifies the length of the magick string.
111%
112*/
113static MagickBooleanType IsWEBP(const unsigned char *magick,const size_t length)
114{
115 if (length < 12)
116 return(MagickFalse);
117 if (LocaleNCompare((const char *) magick+8,"WEBP",4) == 0)
118 return(MagickTrue);
119 return(MagickFalse);
120}
121
cristyb1860752011-03-14 00:27:46 +0000122#if defined(MAGICKCORE_WEBP_DELEGATE)
123/*
124%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
125% %
126% %
127% %
128% R e a d W E B P I m a g e %
129% %
130% %
131% %
132%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
133%
134% ReadWEBPImage() reads an image in the WebP image format.
135%
136% The format of the ReadWEBPImage method is:
137%
138% Image *ReadWEBPImage(const ImageInfo *image_info,
139% ExceptionInfo *exception)
140%
141% A description of each parameter follows:
142%
143% o image_info: the image info.
144%
145% o exception: return any errors or warnings in this structure.
146%
147*/
cristy8f25aa82013-02-23 02:14:37 +0000148
dirk05d2ff72015-11-18 23:13:43 +0100149static inline uint32_t ReadWebPLSBWord(
150 const unsigned char *magick_restrict data)
cristy8f25aa82013-02-23 02:14:37 +0000151{
Cristyf2dc1dd2020-12-28 13:59:26 -0500152 const unsigned char
cristy0b7a0332013-02-23 02:28:08 +0000153 *p;
cristy8f25aa82013-02-23 02:14:37 +0000154
Cristyf2dc1dd2020-12-28 13:59:26 -0500155 uint32_t
cristy0b7a0332013-02-23 02:28:08 +0000156 value;
157
158 p=data;
159 value=(uint32_t) (*p++);
160 value|=((uint32_t) (*p++)) << 8;
161 value|=((uint32_t) (*p++)) << 16;
162 value|=((uint32_t) (*p++)) << 24;
163 return(value);
cristy8f25aa82013-02-23 02:14:37 +0000164}
165
166static MagickBooleanType IsWEBPImageLossless(const unsigned char *stream,
167 const size_t length)
168{
cristy0b7a0332013-02-23 02:28:08 +0000169#define VP8_CHUNK_INDEX 15
cristy8f25aa82013-02-23 02:14:37 +0000170#define LOSSLESS_FLAG 'L'
171#define EXTENDED_HEADER 'X'
172#define VP8_CHUNK_HEADER "VP8"
173#define VP8_CHUNK_HEADER_SIZE 3
cristy0b7a0332013-02-23 02:28:08 +0000174#define RIFF_HEADER_SIZE 12
cristy8f25aa82013-02-23 02:14:37 +0000175#define VP8X_CHUNK_SIZE 10
176#define TAG_SIZE 4
177#define CHUNK_SIZE_BYTES 4
178#define CHUNK_HEADER_SIZE 8
cristya230a612014-01-05 20:51:47 +0000179#define MAX_CHUNK_PAYLOAD (~0U-CHUNK_HEADER_SIZE-1)
cristy8f25aa82013-02-23 02:14:37 +0000180
Cristy27e15b72019-11-17 16:30:19 -0500181 size_t
cristy8f25aa82013-02-23 02:14:37 +0000182 offset;
183
184 /*
185 Read simple header.
186 */
Cristy361ed682018-03-27 20:39:27 -0400187 if (length <= VP8_CHUNK_INDEX)
188 return(MagickFalse);
cristy8f25aa82013-02-23 02:14:37 +0000189 if (stream[VP8_CHUNK_INDEX] != EXTENDED_HEADER)
cristyef632a52013-12-03 11:06:33 +0000190 return(stream[VP8_CHUNK_INDEX] == LOSSLESS_FLAG ? MagickTrue : MagickFalse);
cristy8f25aa82013-02-23 02:14:37 +0000191 /*
192 Read extended header.
193 */
194 offset=RIFF_HEADER_SIZE+TAG_SIZE+CHUNK_SIZE_BYTES+VP8X_CHUNK_SIZE;
Alex Gaynore0ed80f2019-11-17 10:29:04 -0500195 while (offset <= (length-TAG_SIZE-TAG_SIZE-4))
cristy8f25aa82013-02-23 02:14:37 +0000196 {
197 uint32_t
198 chunk_size,
199 chunk_size_pad;
200
cristyef978292013-02-23 02:32:06 +0000201 chunk_size=ReadWebPLSBWord(stream+offset+TAG_SIZE);
cristy8f25aa82013-02-23 02:14:37 +0000202 if (chunk_size > MAX_CHUNK_PAYLOAD)
203 break;
204 chunk_size_pad=(CHUNK_HEADER_SIZE+chunk_size+1) & ~1;
cristy5def2032013-06-30 17:44:08 +0000205 if (memcmp(stream+offset,VP8_CHUNK_HEADER,VP8_CHUNK_HEADER_SIZE) == 0)
cristyef978292013-02-23 02:32:06 +0000206 return(*(stream+offset+VP8_CHUNK_HEADER_SIZE) == LOSSLESS_FLAG ?
cristy8f25aa82013-02-23 02:14:37 +0000207 MagickTrue : MagickFalse);
208 offset+=chunk_size_pad;
209 }
210 return(MagickFalse);
211}
212
Cristy9dd05872019-10-12 10:31:46 -0400213static int FillBasicWEBPInfo(Image *image,const uint8_t *stream,size_t length,
214 WebPDecoderConfig *configure)
215{
Cristy46d4fb52019-09-29 20:32:20 -0400216 WebPBitstreamFeatures
217 *magick_restrict features = &configure->input;
cristyffa663d2011-03-14 00:58:51 +0000218
cristyef632a52013-12-03 11:06:33 +0000219 int
220 webp_status;
221
Cristy46d4fb52019-09-29 20:32:20 -0400222 webp_status=WebPGetFeatures(stream,length,features);
223
224 if (webp_status != VP8_STATUS_OK)
Dirk Lemstraabb91792019-10-26 13:23:52 +0200225 return(webp_status);
Cristy46d4fb52019-09-29 20:32:20 -0400226
227 image->columns=(size_t) features->width;
228 image->rows=(size_t) features->height;
229 image->depth=8;
230 image->alpha_trait=features->has_alpha != 0 ? BlendPixelTrait :
231 UndefinedPixelTrait;
232
Dirk Lemstraabb91792019-10-26 13:23:52 +0200233 return(webp_status);
Cristy46d4fb52019-09-29 20:32:20 -0400234}
235
Cristy9dd05872019-10-12 10:31:46 -0400236static int ReadSingleWEBPImage(Image *image,const uint8_t *stream,
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900237 size_t length,WebPDecoderConfig *configure,ExceptionInfo *exception,
238 MagickBooleanType is_first)
Cristy9dd05872019-10-12 10:31:46 -0400239{
Cristy46d4fb52019-09-29 20:32:20 -0400240 int
241 webp_status;
cristyffa663d2011-03-14 00:58:51 +0000242
Cristyf2dc1dd2020-12-28 13:59:26 -0500243 unsigned char
cristy644040e2011-03-14 17:31:59 +0000244 *p;
245
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900246 size_t
247 canvas_width,
248 canvas_height,
249 image_width,
250 image_height;
251
cristy644040e2011-03-14 17:31:59 +0000252 ssize_t
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900253 x_offset,
254 y_offset,
cristy644040e2011-03-14 17:31:59 +0000255 y;
256
cristyaf29b622013-02-18 15:06:48 +0000257 WebPDecBuffer
Dirk Lemstra89d15bb2020-12-28 09:56:41 +0100258 *magick_restrict webp_image;
cristyaf29b622013-02-18 15:06:48 +0000259
Cristy46d4fb52019-09-29 20:32:20 -0400260 MagickBooleanType
261 status;
cristy644040e2011-03-14 17:31:59 +0000262
Dirk Lemstra89d15bb2020-12-28 09:56:41 +0100263 webp_image=&configure->output;
jace.k (김준성)c835fda2019-12-18 08:38:52 +0900264 if (is_first)
265 {
266 canvas_width=image->columns;
267 canvas_height=image->rows;
268 x_offset=image->page.x;
269 y_offset=image->page.y;
270 image->page.x=0;
271 image->page.y=0;
272 }
273 else
274 {
Elliott Hughes5d41fba2021-04-12 16:36:42 -0700275 canvas_width=0;
276 canvas_height=0;
jace.k (김준성)c835fda2019-12-18 08:38:52 +0900277 x_offset=0;
278 y_offset=0;
279 }
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900280 webp_status=FillBasicWEBPInfo(image,stream,length,configure);
281 image_width=image->columns;
282 image_height=image->rows;
283 if (is_first)
284 {
285 image->columns=canvas_width;
286 image->rows=canvas_height;
287 }
288
289 if (webp_status != VP8_STATUS_OK)
Dirk Lemstraabb91792019-10-26 13:23:52 +0200290 return(webp_status);
Cristy46d4fb52019-09-29 20:32:20 -0400291
292 if (IsWEBPImageLossless(stream,length) != MagickFalse)
293 image->quality=100;
294
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900295 webp_status=WebPDecode(stream,length,configure);
296 if (webp_status != VP8_STATUS_OK)
Dirk Lemstraabb91792019-10-26 13:23:52 +0200297 return(webp_status);
Cristy46d4fb52019-09-29 20:32:20 -0400298
cristyb22ab412014-04-06 15:09:41 +0000299 p=(unsigned char *) webp_image->u.RGBA.rgba;
cristy644040e2011-03-14 17:31:59 +0000300 for (y=0; y < (ssize_t) image->rows; y++)
301 {
Cristyf2dc1dd2020-12-28 13:59:26 -0500302 Quantum
cristyaf29b622013-02-18 15:06:48 +0000303 *q;
304
Cristyf2dc1dd2020-12-28 13:59:26 -0500305 ssize_t
cristyaf29b622013-02-18 15:06:48 +0000306 x;
307
cristy644040e2011-03-14 17:31:59 +0000308 q=QueueAuthenticPixels(image,0,y,image->columns,1,exception);
cristyacd2ed22011-08-30 01:44:23 +0000309 if (q == (Quantum *) NULL)
cristy644040e2011-03-14 17:31:59 +0000310 break;
311 for (x=0; x < (ssize_t) image->columns; x++)
312 {
Cristyfcaccf42020-12-22 16:20:58 -0500313 if ((x >= x_offset && x < (ssize_t) (x_offset+image_width)) &&
314 (y >= y_offset && y < (ssize_t) (y_offset+image_height)))
jace.k (김준성)c835fda2019-12-18 08:38:52 +0900315 {
316 SetPixelRed(image,ScaleCharToQuantum(*p++),q);
317 SetPixelGreen(image,ScaleCharToQuantum(*p++),q);
318 SetPixelBlue(image,ScaleCharToQuantum(*p++),q);
319 SetPixelAlpha(image,ScaleCharToQuantum(*p++),q);
320 }
321 else
322 {
323 SetPixelRed(image,0,q);
324 SetPixelGreen(image,0,q);
325 SetPixelBlue(image,0,q);
326 SetPixelAlpha(image,0,q);
327 }
cristyed231572011-07-14 02:18:59 +0000328 q+=GetPixelChannels(image);
cristy644040e2011-03-14 17:31:59 +0000329 }
330 if (SyncAuthenticPixels(image,exception) == MagickFalse)
331 break;
332 status=SetImageProgress(image,LoadImageTag,(MagickOffsetType) y,
333 image->rows);
334 if (status == MagickFalse)
335 break;
336 }
cristyaf29b622013-02-18 15:06:48 +0000337 WebPFreeDecBuffer(webp_image);
Cristy692b6782017-11-19 12:35:18 -0500338#if defined(MAGICKCORE_WEBPMUX_DELEGATE)
339 {
340 StringInfo
341 *profile;
342
343 uint32_t
344 webp_flags = 0;
345
346 WebPData
Cristy5e87ab22018-06-24 16:26:06 -0400347 chunk,
Dirk Lemstraabb91792019-10-26 13:23:52 +0200348 content;
Cristy692b6782017-11-19 12:35:18 -0500349
350 WebPMux
351 *mux;
352
353 /*
Cristy5fa0d442020-09-22 14:50:00 -0400354 Extract any profiles:
355 https://developers.google.com/speed/webp/docs/container-api.
Cristy692b6782017-11-19 12:35:18 -0500356 */
Dirk Lemstraabb91792019-10-26 13:23:52 +0200357 content.bytes=stream;
358 content.size=length;
Cristy692b6782017-11-19 12:35:18 -0500359 mux=WebPMuxCreate(&content,0);
Cristy5e87ab22018-06-24 16:26:06 -0400360 (void) memset(&chunk,0,sizeof(chunk));
Cristy692b6782017-11-19 12:35:18 -0500361 WebPMuxGetFeatures(mux,&webp_flags);
362 if (webp_flags & ICCP_FLAG)
363 {
364 WebPMuxGetChunk(mux,"ICCP",&chunk);
365 profile=BlobToStringInfo(chunk.bytes,chunk.size);
366 if (profile != (StringInfo *) NULL)
367 {
368 SetImageProfile(image,"ICC",profile,exception);
369 profile=DestroyStringInfo(profile);
370 }
371 }
372 if (webp_flags & EXIF_FLAG)
373 {
374 WebPMuxGetChunk(mux,"EXIF",&chunk);
375 profile=BlobToStringInfo(chunk.bytes,chunk.size);
376 if (profile != (StringInfo *) NULL)
377 {
378 SetImageProfile(image,"EXIF",profile,exception);
379 profile=DestroyStringInfo(profile);
380 }
381 }
382 if (webp_flags & XMP_FLAG)
383 {
384 WebPMuxGetChunk(mux,"XMP",&chunk);
385 profile=BlobToStringInfo(chunk.bytes,chunk.size);
386 if (profile != (StringInfo *) NULL)
387 {
388 SetImageProfile(image,"XMP",profile,exception);
389 profile=DestroyStringInfo(profile);
390 }
391 }
392 WebPMuxDelete(mux);
393 }
394#endif
Dirk Lemstraabb91792019-10-26 13:23:52 +0200395 return(webp_status);
Cristy46d4fb52019-09-29 20:32:20 -0400396}
397
398#if defined(MAGICKCORE_WEBPMUX_DELEGATE)
Cristy7caee072019-11-17 10:54:58 -0500399static int ReadAnimatedWEBPImage(const ImageInfo *image_info,Image *image,
400 uint8_t *stream,size_t length,WebPDecoderConfig *configure,
Dirk Lemstraabb91792019-10-26 13:23:52 +0200401 ExceptionInfo *exception)
402{
403 Image
404 *original_image;
Cristy46d4fb52019-09-29 20:32:20 -0400405
406 int
Dirk Lemstraabb91792019-10-26 13:23:52 +0200407 image_count,
408 webp_status;
Cristy46d4fb52019-09-29 20:32:20 -0400409
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900410 size_t
411 canvas_width,
412 canvas_height;
413
Dirk Lemstraabb91792019-10-26 13:23:52 +0200414 WebPData
415 data;
Cristy46d4fb52019-09-29 20:32:20 -0400416
Dirk Lemstraabb91792019-10-26 13:23:52 +0200417 WebPDemuxer
418 *demux;
Cristy46d4fb52019-09-29 20:32:20 -0400419
Dirk Lemstraabb91792019-10-26 13:23:52 +0200420 WebPIterator
421 iter;
422
423 image_count=0;
424 webp_status=0;
425 original_image=image;
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900426 webp_status=FillBasicWEBPInfo(image,stream,length,configure);
427 canvas_width=image->columns;
428 canvas_height=image->rows;
Dirk Lemstraabb91792019-10-26 13:23:52 +0200429 data.bytes=stream;
430 data.size=length;
Cristy0e421ed2020-12-01 00:58:17 +0000431 {
432 WebPMux
433 *mux;
434
435 WebPMuxAnimParams
436 params;
437
438 WebPMuxError
439 status;
440
441 mux=WebPMuxCreate(&data,0);
442 status=WebPMuxGetAnimationParams(mux,&params);
443 if (status >= 0)
444 image->iterations=params.loop_count;
445 WebPMuxDelete(mux);
446 }
Dirk Lemstraabb91792019-10-26 13:23:52 +0200447 demux=WebPDemux(&data);
Cristyfab704b2020-08-03 18:46:38 -0400448 if (WebPDemuxGetFrame(demux,1,&iter))
449 {
450 do
451 {
452 if (image_count != 0)
453 {
454 AcquireNextImage(image_info,image,exception);
455 if (GetNextImageInList(image) == (Image *) NULL)
456 break;
457 image=SyncNextImageInList(image);
458 CloneImageProperties(image,original_image);
459 image->page.x=iter.x_offset;
460 image->page.y=iter.y_offset;
461 webp_status=ReadSingleWEBPImage(image,iter.fragment.bytes,
462 iter.fragment.size,configure,exception,MagickFalse);
463 }
464 else
465 {
466 image->page.x=iter.x_offset;
467 image->page.y=iter.y_offset;
468 webp_status=ReadSingleWEBPImage(image,iter.fragment.bytes,
469 iter.fragment.size,configure,exception,MagickTrue);
470 }
471 if (webp_status != VP8_STATUS_OK)
472 break;
473 image->page.width=canvas_width;
474 image->page.height=canvas_height;
475 image->ticks_per_second=100;
476 image->delay=iter.duration/10;
477 image->dispose=NoneDispose;
478 if (iter.dispose_method == WEBP_MUX_DISPOSE_BACKGROUND)
479 image->dispose=BackgroundDispose;
480 (void) SetImageProperty(image,"webp:mux-blend",
481 "AtopPreviousAlphaBlend",exception);
482 if (iter.blend_method == WEBP_MUX_BLEND)
483 (void) SetImageProperty(image,"webp:mux-blend",
484 "AtopBackgroundAlphaBlend",exception);
485 image_count++;
486 } while (WebPDemuxNextFrame(&iter));
487 WebPDemuxReleaseIterator(&iter);
488 }
Cristy46d4fb52019-09-29 20:32:20 -0400489 WebPDemuxDelete(demux);
Dirk Lemstraabb91792019-10-26 13:23:52 +0200490 return(webp_status);
Cristy46d4fb52019-09-29 20:32:20 -0400491}
492#endif
493
494static Image *ReadWEBPImage(const ImageInfo *image_info,
495 ExceptionInfo *exception)
496{
497#define ThrowWEBPException(severity,tag) \
498{ \
499 if (stream != (unsigned char *) NULL) \
500 stream=(unsigned char*) RelinquishMagickMemory(stream); \
501 if (webp_image != (WebPDecBuffer *) NULL) \
502 WebPFreeDecBuffer(webp_image); \
503 ThrowReaderException(severity,tag); \
504}
505
506 Image
507 *image;
508
509 int
510 webp_status;
511
512 MagickBooleanType
513 status;
514
515 size_t
516 length;
517
518 ssize_t
519 count;
520
521 unsigned char
522 header[12],
523 *stream;
524
525 WebPDecoderConfig
526 configure;
527
528 WebPDecBuffer
529 *magick_restrict webp_image = &configure.output;
530
531 /*
532 Open image file.
533 */
534 assert(image_info != (const ImageInfo *) NULL);
535 assert(image_info->signature == MagickCoreSignature);
536 if (image_info->debug != MagickFalse)
537 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
538 image_info->filename);
539 assert(exception != (ExceptionInfo *) NULL);
540 assert(exception->signature == MagickCoreSignature);
541 image=AcquireImage(image_info,exception);
542 status=OpenBlob(image_info,image,ReadBinaryBlobMode,exception);
543 if (status == MagickFalse)
544 {
545 image=DestroyImageList(image);
546 return((Image *) NULL);
547 }
548 stream=(unsigned char *) NULL;
549 if (WebPInitDecoderConfig(&configure) == 0)
550 ThrowReaderException(ResourceLimitError,"UnableToDecodeImageFile");
551 webp_image->colorspace=MODE_RGBA;
552 count=ReadBlob(image,12,header);
553 if (count != 12)
554 ThrowWEBPException(CorruptImageError,"InsufficientImageDataInFile");
555 status=IsWEBP(header,count);
556 if (status == MagickFalse)
557 ThrowWEBPException(CorruptImageError,"CorruptImage");
558 length=(size_t) (ReadWebPLSBWord(header+4)+8);
559 if (length < 12)
560 ThrowWEBPException(CorruptImageError,"CorruptImage");
561 if (length > GetBlobSize(image))
562 ThrowWEBPException(CorruptImageError,"InsufficientImageDataInFile");
563 stream=(unsigned char *) AcquireQuantumMemory(length,sizeof(*stream));
564 if (stream == (unsigned char *) NULL)
565 ThrowWEBPException(ResourceLimitError,"MemoryAllocationFailed");
566 (void) memcpy(stream,header,12);
567 count=ReadBlob(image,length-12,stream+12);
568 if (count != (ssize_t) (length-12))
569 ThrowWEBPException(CorruptImageError,"InsufficientImageDataInFile");
570
571 webp_status=FillBasicWEBPInfo(image,stream,length,&configure);
572 if (webp_status == VP8_STATUS_OK) {
Cristy46d4fb52019-09-29 20:32:20 -0400573 if (configure.input.has_animation) {
574#if defined(MAGICKCORE_WEBPMUX_DELEGATE)
Cristy7caee072019-11-17 10:54:58 -0500575 webp_status=ReadAnimatedWEBPImage(image_info,image,stream,length,
576 &configure,exception);
Cristy46d4fb52019-09-29 20:32:20 -0400577#else
Alex Gaynord58edd32019-09-30 07:41:01 -0400578 webp_status=VP8_STATUS_UNSUPPORTED_FEATURE;
Cristy46d4fb52019-09-29 20:32:20 -0400579#endif
580 } else {
jace.k (김준성)7b743df2019-12-17 14:57:46 +0900581 webp_status=ReadSingleWEBPImage(image,stream,length,&configure,exception,MagickFalse);
Cristy46d4fb52019-09-29 20:32:20 -0400582 }
583 }
584
585 if (webp_status != VP8_STATUS_OK)
586 switch (webp_status)
587 {
588 case VP8_STATUS_OUT_OF_MEMORY:
589 {
590 ThrowWEBPException(ResourceLimitError,"MemoryAllocationFailed");
591 break;
592 }
593 case VP8_STATUS_INVALID_PARAM:
594 {
595 ThrowWEBPException(CorruptImageError,"invalid parameter");
596 break;
597 }
598 case VP8_STATUS_BITSTREAM_ERROR:
599 {
600 ThrowWEBPException(CorruptImageError,"CorruptImage");
601 break;
602 }
603 case VP8_STATUS_UNSUPPORTED_FEATURE:
604 {
605 ThrowWEBPException(CoderError,"DataEncodingSchemeIsNotSupported");
606 break;
607 }
608 case VP8_STATUS_SUSPENDED:
609 {
610 ThrowWEBPException(CorruptImageError,"decoder suspended");
611 break;
612 }
613 case VP8_STATUS_USER_ABORT:
614 {
615 ThrowWEBPException(CorruptImageError,"user abort");
616 break;
617 }
618 case VP8_STATUS_NOT_ENOUGH_DATA:
619 {
620 ThrowWEBPException(CorruptImageError,"InsufficientImageDataInFile");
621 break;
622 }
623 default:
624 ThrowWEBPException(CorruptImageError,"CorruptImage");
625 }
626
cristye71e0892013-02-18 13:13:53 +0000627 stream=(unsigned char*) RelinquishMagickMemory(stream);
zhangjife757a22017-02-17 18:34:45 +0800628 (void) CloseBlob(image);
cristyffa663d2011-03-14 00:58:51 +0000629 return(image);
cristyb1860752011-03-14 00:27:46 +0000630}
631#endif
632
633/*
634%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
635% %
636% %
637% %
638% R e g i s t e r W E B P I m a g e %
639% %
640% %
641% %
642%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
643%
644% RegisterWEBPImage() adds attributes for the WebP image format to
645% the list of supported formats. The attributes include the image format
646% tag, a method to read and/or write the format, whether the format
647% supports the saving of more than one frame to the same file or blob,
648% whether the format supports native in-memory I/O, and a brief
649% description of the format.
650%
651% The format of the RegisterWEBPImage method is:
652%
653% size_t RegisterWEBPImage(void)
654%
655*/
656ModuleExport size_t RegisterWEBPImage(void)
657{
cristyaf29b622013-02-18 15:06:48 +0000658 char
cristy151b66d2015-04-15 10:50:31 +0000659 version[MagickPathExtent];
cristyaf29b622013-02-18 15:06:48 +0000660
cristyb1860752011-03-14 00:27:46 +0000661 MagickInfo
662 *entry;
663
cristyaf29b622013-02-18 15:06:48 +0000664 *version='\0';
dirk06b627a2015-04-06 18:59:17 +0000665 entry=AcquireMagickInfo("WEBP","WEBP","WebP Image Format");
cristyb1860752011-03-14 00:27:46 +0000666#if defined(MAGICKCORE_WEBP_DELEGATE)
667 entry->decoder=(DecodeImageHandler *) ReadWEBPImage;
668 entry->encoder=(EncodeImageHandler *) WriteWEBPImage;
cristy151b66d2015-04-15 10:50:31 +0000669 (void) FormatLocaleString(version,MagickPathExtent,"libwebp %d.%d.%d [%04X]",
よや5ce163f2019-02-10 22:42:15 +0900670 (WebPGetEncoderVersion() >> 16) & 0xff,
671 (WebPGetEncoderVersion() >> 8) & 0xff,
672 (WebPGetEncoderVersion() >> 0) & 0xff,WEBP_ENCODER_ABI_VERSION);
cristyb1860752011-03-14 00:27:46 +0000673#endif
dirk8648daf2015-09-14 22:26:43 +0200674 entry->mime_type=ConstantString("image/webp");
Cristycb635602017-08-02 06:59:15 -0400675 entry->flags|=CoderDecoderSeekableStreamFlag;
Dirk Lemstra47f757a2020-01-10 14:58:40 +0100676#if !defined(MAGICKCORE_WEBPMUX_DELEGATE)
677 entry->flags^=CoderAdjoinFlag;
Dirk Lemstra2424fb22020-01-10 14:55:54 +0100678#endif
cristydffa79e2013-02-20 00:57:26 +0000679 entry->magick=(IsImageFormatHandler *) IsWEBP;
cristyaf29b622013-02-18 15:06:48 +0000680 if (*version != '\0')
681 entry->version=ConstantString(version);
cristyb1860752011-03-14 00:27:46 +0000682 (void) RegisterMagickInfo(entry);
683 return(MagickImageCoderSignature);
684}
685
686/*
687%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
688% %
689% %
690% %
691% U n r e g i s t e r W E B P I m a g e %
692% %
693% %
694% %
695%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
696%
697% UnregisterWEBPImage() removes format registrations made by the WebP module
698% from the list of supported formats.
699%
700% The format of the UnregisterWEBPImage method is:
701%
702% UnregisterWEBPImage(void)
703%
704*/
705ModuleExport void UnregisterWEBPImage(void)
706{
707 (void) UnregisterMagickInfo("WEBP");
708}
709#if defined(MAGICKCORE_WEBP_DELEGATE)
710
711/*
712%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
713% %
714% %
715% %
716% W r i t e W E B P I m a g e %
717% %
718% %
719% %
720%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
721%
722% WriteWEBPImage() writes an image in the WebP image format.
723%
724% The format of the WriteWEBPImage method is:
725%
726% MagickBooleanType WriteWEBPImage(const ImageInfo *image_info,
727% Image *image)
728%
729% A description of each parameter follows.
730%
731% o image_info: the image info.
732%
733% o image: The image.
734%
735*/
cristy644040e2011-03-14 17:31:59 +0000736
よや5ce163f2019-02-10 22:42:15 +0900737#if WEBP_ENCODER_ABI_VERSION >= 0x0100
cristyb22ab412014-04-06 15:09:41 +0000738static int WebPEncodeProgress(int percent,const WebPPicture* picture)
739{
cristy88bf7c12014-04-06 15:21:42 +0000740#define EncodeImageTag "Encode/Image"
741
cristyb22ab412014-04-06 15:09:41 +0000742 Image
743 *image;
744
745 MagickBooleanType
746 status;
747
Cristycf4ad302017-12-23 08:32:30 -0500748 image=(Image *) picture->user_data;
cristy88bf7c12014-04-06 15:21:42 +0000749 status=SetImageProgress(image,EncodeImageTag,percent-1,100);
cristyb22ab412014-04-06 15:09:41 +0000750 return(status == MagickFalse ? 0 : 1);
751}
752#endif
753
tomwei717e76432020-11-26 23:34:47 +0800754static const char * WebPErrorCodeMessage(WebPEncodingError error_code)
755{
756 switch (error_code)
757 {
758 case VP8_ENC_OK:
759 return NULL;
760 case VP8_ENC_ERROR_OUT_OF_MEMORY:
761 return "out of memory";
762 case VP8_ENC_ERROR_BITSTREAM_OUT_OF_MEMORY:
763 return "bitstream out of memory";
764 case VP8_ENC_ERROR_NULL_PARAMETER:
765 return "NULL parameter";
766 case VP8_ENC_ERROR_INVALID_CONFIGURATION:
767 return "invalid configuration";
768 case VP8_ENC_ERROR_BAD_DIMENSION:
769 return "bad dimension";
770 case VP8_ENC_ERROR_PARTITION0_OVERFLOW:
771 return "partition 0 overflow (> 512K)";
772 case VP8_ENC_ERROR_PARTITION_OVERFLOW:
773 return "partition overflow (> 16M)";
774 case VP8_ENC_ERROR_BAD_WRITE:
775 return "bad write";
776 case VP8_ENC_ERROR_FILE_TOO_BIG:
777 return "file too big (> 4GB)";
778#if WEBP_ENCODER_ABI_VERSION >= 0x0100
779 case VP8_ENC_ERROR_USER_ABORT:
780 return "user abort";
781 case VP8_ENC_ERROR_LAST:
782 return "error last";
783#endif
784 }
785 return "unknown exception";
786}
787
788static MagickBooleanType WriteSingleWEBPPicture(const ImageInfo *image_info,
789 Image *image,WebPConfig *configure,WebPPicture *picture,ExceptionInfo *exception)
Cristy46d4fb52019-09-29 20:32:20 -0400790{
791 MagickBooleanType
Dirk Lemstraacc06822020-12-28 10:26:17 +0100792 status;
Cristy46d4fb52019-09-29 20:32:20 -0400793
Cristyf2dc1dd2020-12-28 13:59:26 -0500794 uint32_t
Cristy46d4fb52019-09-29 20:32:20 -0400795 *magick_restrict q;
796
797 ssize_t
798 y;
799
tomwei717e76432020-11-26 23:34:47 +0800800 MemoryInfo
801 *pixel_info;
802
Cristy46d4fb52019-09-29 20:32:20 -0400803#if WEBP_ENCODER_ABI_VERSION >= 0x0100
804 picture->progress_hook=WebPEncodeProgress;
805 picture->user_data=(void *) image;
806#endif
807 picture->width=(int) image->columns;
808 picture->height=(int) image->rows;
809 picture->argb_stride=(int) image->columns;
810 picture->use_argb=1;
811
812 /*
813 Allocate memory for pixels.
814 */
815 (void) TransformImageColorspace(image,sRGBColorspace,exception);
tomwei717e76432020-11-26 23:34:47 +0800816 pixel_info=AcquireVirtualMemory(image->columns,image->rows*
Cristy46d4fb52019-09-29 20:32:20 -0400817 sizeof(*(picture->argb)));
818
tomwei717e76432020-11-26 23:34:47 +0800819 if (pixel_info == (MemoryInfo *) NULL)
Cristy46d4fb52019-09-29 20:32:20 -0400820 ThrowWriterException(ResourceLimitError,"MemoryAllocationFailed");
tomwei717e76432020-11-26 23:34:47 +0800821 picture->argb=(uint32_t *) GetVirtualMemoryBlob(pixel_info);
Cristy46d4fb52019-09-29 20:32:20 -0400822 /*
823 Convert image to WebP raster pixels.
824 */
Dirk Lemstraacc06822020-12-28 10:26:17 +0100825 status=MagickFalse;
Cristy46d4fb52019-09-29 20:32:20 -0400826 q=picture->argb;
827 for (y=0; y < (ssize_t) image->rows; y++)
828 {
Cristyf2dc1dd2020-12-28 13:59:26 -0500829 const Quantum
Cristy46d4fb52019-09-29 20:32:20 -0400830 *magick_restrict p;
831
Cristyf2dc1dd2020-12-28 13:59:26 -0500832 ssize_t
Cristy46d4fb52019-09-29 20:32:20 -0400833 x;
834
835 p=GetVirtualPixels(image,0,y,image->columns,1,exception);
836 if (p == (const Quantum *) NULL)
837 break;
838 for (x=0; x < (ssize_t) image->columns; x++)
839 {
840 *q++=(uint32_t) (image->alpha_trait != UndefinedPixelTrait ? (uint32_t)
841 ScaleQuantumToChar(GetPixelAlpha(image,p)) << 24 : 0xff000000) |
842 ((uint32_t) ScaleQuantumToChar(GetPixelRed(image,p)) << 16) |
843 ((uint32_t) ScaleQuantumToChar(GetPixelGreen(image,p)) << 8) |
844 ((uint32_t) ScaleQuantumToChar(GetPixelBlue(image,p)));
845 p+=GetPixelChannels(image);
846 }
847 status=SetImageProgress(image,SaveImageTag,(MagickOffsetType) y,
848 image->rows);
849 if (status == MagickFalse)
850 break;
851 }
tomwei717e76432020-11-26 23:34:47 +0800852
Dirk Lemstraacc06822020-12-28 10:26:17 +0100853 if (status != MagickFalse)
854 status=(MagickBooleanType) WebPEncode(configure, picture);
tomwei717e76432020-11-26 23:34:47 +0800855
856 RelinquishVirtualMemory(pixel_info);
857
Dirk Lemstraacc06822020-12-28 10:26:17 +0100858 if (status == MagickFalse)
tomwei717e76432020-11-26 23:34:47 +0800859 (void) ThrowMagickException(exception,GetMagickModule(),CorruptImageError,
Dirk Lemstraacc06822020-12-28 10:26:17 +0100860 WebPErrorCodeMessage(picture->error_code),"`%s'",
861 image->filename);
tomwei717e76432020-11-26 23:34:47 +0800862
Dirk Lemstraacc06822020-12-28 10:26:17 +0100863 return(status);
tomwei717e76432020-11-26 23:34:47 +0800864}
865
866static MagickBooleanType WriteSingleWEBPImage(const ImageInfo *image_info,
Dirk Lemstrad2197622020-12-28 15:57:43 +0100867 Image *image,WebPConfig *configure,WebPMemoryWriter *writer,
Dirk Lemstraacc06822020-12-28 10:26:17 +0100868 ExceptionInfo *exception)
tomwei717e76432020-11-26 23:34:47 +0800869{
870 MagickBooleanType
Dirk Lemstraacc06822020-12-28 10:26:17 +0100871 status;
tomwei717e76432020-11-26 23:34:47 +0800872
873 WebPPicture
874 picture;
875
Dirk Lemstraacc06822020-12-28 10:26:17 +0100876 if (WebPPictureInit(&picture) == 0)
tomwei717e76432020-11-26 23:34:47 +0800877 ThrowWriterException(ResourceLimitError,"UnableToEncodeImageFile");
878
tomwei717e76432020-11-26 23:34:47 +0800879 picture.writer=WebPMemoryWrite;
Dirk Lemstrad2197622020-12-28 15:57:43 +0100880 picture.custom_ptr=writer;
tomwei717e76432020-11-26 23:34:47 +0800881
Dirk Lemstraacc06822020-12-28 10:26:17 +0100882 status=WriteSingleWEBPPicture(image_info,image,configure,&picture,exception);
tomwei717e76432020-11-26 23:34:47 +0800883 WebPPictureFree(&picture);
884
Dirk Lemstraacc06822020-12-28 10:26:17 +0100885 return(status);
Cristy46d4fb52019-09-29 20:32:20 -0400886}
887
888#if defined(MAGICKCORE_WEBPMUX_DELEGATE)
889static MagickBooleanType WriteAnimatedWEBPImage(const ImageInfo *image_info,
Dirk Lemstraacc06822020-12-28 10:26:17 +0100890 Image *image,WebPConfig *configure,WebPData *webp_data,
891 ExceptionInfo *exception)
Cristy46d4fb52019-09-29 20:32:20 -0400892{
Cristy46d4fb52019-09-29 20:32:20 -0400893 Image
Dirk Lemstraabb91792019-10-26 13:23:52 +0200894 *first_image;
895
Cristy7caee072019-11-17 10:54:58 -0500896 size_t
Dirk Lemstraacc06822020-12-28 10:26:17 +0100897 effective_delta,
898 frame_timestamp;
Cristy7caee072019-11-17 10:54:58 -0500899
Dirk Lemstraabb91792019-10-26 13:23:52 +0200900 WebPAnimEncoder
901 *enc;
902
903 WebPAnimEncoderOptions
904 enc_options;
905
Dirk Lemstraabb91792019-10-26 13:23:52 +0200906 WebPPicture
907 picture;
Cristy46d4fb52019-09-29 20:32:20 -0400908
tomwei72d2e0142020-12-27 23:52:46 +0800909 MagickBooleanType
Dirk Lemstraacc06822020-12-28 10:26:17 +0100910 status;
tomwei72d2e0142020-12-27 23:52:46 +0800911
912 first_image=CoalesceImages(image,exception);
tomwei72d2e0142020-12-27 23:52:46 +0800913 if (first_image == (Image *) NULL)
914 return(MagickFalse);
Dirk Lemstraacc06822020-12-28 10:26:17 +0100915 image=first_image;
916 effective_delta=0;
917 frame_timestamp=0;
tomwei72d2e0142020-12-27 23:52:46 +0800918
Dirk Lemstraacc06822020-12-28 10:26:17 +0100919 (void) WebPAnimEncoderOptionsInit(&enc_options);
Cristy46d4fb52019-09-29 20:32:20 -0400920 if (image_info->verbose)
Dirk Lemstraacc06822020-12-28 10:26:17 +0100921 enc_options.verbose=1;
Dirk Lemstraabb91792019-10-26 13:23:52 +0200922 enc=WebPAnimEncoderNew((int) image->page.width,(int) image->page.height,
923 &enc_options);
Cristy46d4fb52019-09-29 20:32:20 -0400924
Dirk Lemstraacc06822020-12-28 10:26:17 +0100925 status=MagickTrue;
Cristy6f84d892019-10-29 21:32:11 -0400926 while (image != NULL)
Dirk Lemstraabb91792019-10-26 13:23:52 +0200927 {
Dirk Lemstraacc06822020-12-28 10:26:17 +0100928 status=(MagickBooleanType) WebPPictureInit(&picture);
929 if (status == MagickFalse)
tomwei72d2e0142020-12-27 23:52:46 +0800930 {
931 (void) ThrowMagickException(exception,GetMagickModule(),
932 ResourceLimitError,"UnableToEncodeImageFile","`%s'",image->filename);
933 break;
934 }
Cristy46d4fb52019-09-29 20:32:20 -0400935
Dirk Lemstraacc06822020-12-28 10:26:17 +0100936 status=WriteSingleWEBPPicture(image_info,image,configure,&picture,
937 exception);
938 if (status == MagickFalse)
tomwei72d2e0142020-12-27 23:52:46 +0800939 break;
940
Dirk Lemstraacc06822020-12-28 10:26:17 +0100941 status=(MagickBooleanType) WebPAnimEncoderAdd(enc,&picture,
942 (int) frame_timestamp,configure);
tomwei717e76432020-11-26 23:34:47 +0800943 WebPPictureFree(&picture);
Dirk Lemstraacc06822020-12-28 10:26:17 +0100944 if (status == MagickFalse)
tomwei72d2e0142020-12-27 23:52:46 +0800945 {
946 (void) ThrowMagickException(exception,GetMagickModule(),
947 CoderError,WebPAnimEncoderGetError(enc),"`%s'",image->filename);
948 break;
949 }
Cristy46d4fb52019-09-29 20:32:20 -0400950
Elliott Hughes5d41fba2021-04-12 16:36:42 -0700951 effective_delta=image->delay*1000*PerceptibleReciprocal(
952 image->ticks_per_second);
Cristy46d4fb52019-09-29 20:32:20 -0400953 if (effective_delta < 10)
Dirk Lemstraacc06822020-12-28 10:26:17 +0100954 effective_delta=100; /* Consistent with gif2webp */
Cristy46d4fb52019-09-29 20:32:20 -0400955 frame_timestamp+=effective_delta;
956
tomwei72d2e0142020-12-27 23:52:46 +0800957 image=GetNextImageInList(image);
Cristy46d4fb52019-09-29 20:32:20 -0400958 }
tomwei717e76432020-11-26 23:34:47 +0800959
Dirk Lemstraacc06822020-12-28 10:26:17 +0100960 if (status != MagickFalse)
tomwei72d2e0142020-12-27 23:52:46 +0800961 {
962 // add last null frame and assemble picture.
Dirk Lemstraacc06822020-12-28 10:26:17 +0100963 status=(MagickBooleanType) WebPAnimEncoderAdd(enc,(WebPPicture *) NULL,
964 (int) frame_timestamp,configure);
965 if (status != MagickFalse)
966 status=(MagickBooleanType) WebPAnimEncoderAssemble(enc,webp_data);
tomwei72d2e0142020-12-27 23:52:46 +0800967
Dirk Lemstraacc06822020-12-28 10:26:17 +0100968 if (status == MagickFalse)
tomwei72d2e0142020-12-27 23:52:46 +0800969 (void) ThrowMagickException(exception,GetMagickModule(),
Dirk Lemstraacc06822020-12-28 10:26:17 +0100970 CoderError,WebPAnimEncoderGetError(enc),"`%s'",
971 first_image->filename);
tomwei72d2e0142020-12-27 23:52:46 +0800972 }
tomwei717e76432020-11-26 23:34:47 +0800973
Cristy46d4fb52019-09-29 20:32:20 -0400974 WebPAnimEncoderDelete(enc);
975 DestroyImageList(first_image);
tomwei72d2e0142020-12-27 23:52:46 +0800976 return(status);
Cristy46d4fb52019-09-29 20:32:20 -0400977}
tomwei717e76432020-11-26 23:34:47 +0800978
Dirk Lemstraacc06822020-12-28 10:26:17 +0100979static MagickBooleanType WriteWEBPImageProfile(Image *image,
980 WebPData *webp_data,ExceptionInfo *exception)
tomwei717e76432020-11-26 23:34:47 +0800981{
982 const StringInfo
983 *icc_profile,
984 *exif_profile,
985 *xmp_profile;
986
987 WebPData
988 chunk;
989
990 WebPMux
991 *mux;
992
993 WebPMuxError
Elliott Hughes5d41fba2021-04-12 16:36:42 -0700994 mux_error;
995
996 WebPMuxAnimParams
997 new_params;
tomwei717e76432020-11-26 23:34:47 +0800998
999 icc_profile=GetImageProfile(image,"ICC");
1000 exif_profile=GetImageProfile(image,"EXIF");
1001 xmp_profile=GetImageProfile(image,"XMP");
1002
1003 if (icc_profile == (StringInfo *) NULL
1004 && exif_profile == (StringInfo *) NULL
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001005 && xmp_profile == (StringInfo *) NULL
1006 && image->iterations == 0)
Dirk Lemstraacc06822020-12-28 10:26:17 +01001007 return(MagickTrue);
tomwei717e76432020-11-26 23:34:47 +08001008
1009 mux=WebPMuxCreate(webp_data, 1);
1010 WebPDataClear(webp_data);
1011
1012 if (mux == NULL)
1013 (void) ThrowMagickException(exception,GetMagickModule(),
1014 ResourceLimitError,"UnableToEncodeImageFile","`%s'",image->filename);
1015
1016 // Clean up returned data
1017 memset(webp_data, 0, sizeof(*webp_data));
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001018 mux_error=WEBP_MUX_OK;
1019 if (image->iterations > 0)
1020 {
1021 mux_error=WebPMuxGetAnimationParams(mux, &new_params);
1022 /*
1023 If there is only 1 frame webp_data will be created by WriteSingleWEBPImage
1024 and WebPMuxGetAnimationParams will return WEBP_MUX_NOT_FOUND
1025 */
1026 if (mux_error == WEBP_MUX_NOT_FOUND)
1027 mux_error=WEBP_MUX_OK;
1028 else
1029 if (mux_error == WEBP_MUX_OK)
1030 {
1031 new_params.loop_count=MagickMin((int) image->iterations,65535);
1032 mux_error=WebPMuxSetAnimationParams(mux, &new_params);
1033 }
1034 }
1035 if ((icc_profile != (StringInfo *) NULL) && (mux_error == WEBP_MUX_OK))
tomwei717e76432020-11-26 23:34:47 +08001036 {
1037 chunk.bytes=GetStringInfoDatum(icc_profile);
1038 chunk.size=GetStringInfoLength(icc_profile);
1039 mux_error=WebPMuxSetChunk(mux,"ICCP",&chunk,0);
1040 }
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001041 if ((exif_profile != (StringInfo *) NULL) && (mux_error == WEBP_MUX_OK))
tomwei717e76432020-11-26 23:34:47 +08001042 {
1043 chunk.bytes=GetStringInfoDatum(exif_profile);
1044 chunk.size=GetStringInfoLength(exif_profile);
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001045 if ((chunk.size >= 6) &&
1046 (chunk.bytes[0] == 'E') && (chunk.bytes[1] == 'x') &&
1047 (chunk.bytes[2] == 'i') && (chunk.bytes[3] == 'f') &&
1048 (chunk.bytes[4] == '\0') && (chunk.bytes[5] == '\0'))
1049 {
1050 chunk.bytes=GetStringInfoDatum(exif_profile)+6;
1051 chunk.size-=6;
1052 }
tomwei717e76432020-11-26 23:34:47 +08001053 mux_error=WebPMuxSetChunk(mux,"EXIF",&chunk,0);
1054 }
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001055 if ((xmp_profile != (StringInfo *) NULL) && (mux_error == WEBP_MUX_OK))
tomwei717e76432020-11-26 23:34:47 +08001056 {
1057 chunk.bytes=GetStringInfoDatum(xmp_profile);
1058 chunk.size=GetStringInfoLength(xmp_profile);
1059 mux_error=WebPMuxSetChunk(mux,"XMP",&chunk,0);
1060 }
1061 if (mux_error == WEBP_MUX_OK)
1062 mux_error=WebPMuxAssemble(mux,webp_data);
1063 WebPMuxDelete(mux);
1064
1065 if (mux_error != WEBP_MUX_OK)
1066 (void) ThrowMagickException(exception,GetMagickModule(),
1067 ResourceLimitError,"UnableToEncodeImageFile","`%s'",image->filename);
Dirk Lemstraacc06822020-12-28 10:26:17 +01001068 return(MagickTrue);
tomwei717e76432020-11-26 23:34:47 +08001069}
Cristy46d4fb52019-09-29 20:32:20 -04001070#endif
1071
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001072static inline void SetBooleanOption(const ImageInfo *image_info,
1073 const char *option,int *setting)
1074{
1075 const char
1076 *value;
1077
1078 value=GetImageOption(image_info,option);
1079 if (value != (char *) NULL)
1080 *setting=(int) ParseCommandOption(MagickBooleanOptions,MagickFalse,value);
1081}
1082
Dirk Lemstra4e056d62020-12-30 11:09:32 +01001083static inline void SetIntegerOption(const ImageInfo *image_info,
1084 const char *option,int *setting)
1085{
1086 const char
1087 *value;
1088
1089 value=GetImageOption(image_info,option);
1090 if (value != (const char *) NULL)
1091 *setting=StringToInteger(value);
1092}
1093
cristyb1860752011-03-14 00:27:46 +00001094static MagickBooleanType WriteWEBPImage(const ImageInfo *image_info,
Cristy46d4fb52019-09-29 20:32:20 -04001095 Image *image,ExceptionInfo * exception)
cristyb1860752011-03-14 00:27:46 +00001096{
cristye71e0892013-02-18 13:13:53 +00001097 const char
1098 *value;
1099
cristyffa663d2011-03-14 00:58:51 +00001100 MagickBooleanType
1101 status;
1102
cristy644040e2011-03-14 17:31:59 +00001103 WebPConfig
1104 configure;
1105
Dirk Lemstrad2197622020-12-28 15:57:43 +01001106 WebPMemoryWriter
1107 writer;
Cristy46d4fb52019-09-29 20:32:20 -04001108
cristyffa663d2011-03-14 00:58:51 +00001109 /*
1110 Open output image file.
1111 */
1112 assert(image_info != (const ImageInfo *) NULL);
cristye1c94d92015-06-28 12:16:33 +00001113 assert(image_info->signature == MagickCoreSignature);
cristyffa663d2011-03-14 00:58:51 +00001114 assert(image != (Image *) NULL);
cristye1c94d92015-06-28 12:16:33 +00001115 assert(image->signature == MagickCoreSignature);
cristyffa663d2011-03-14 00:58:51 +00001116 if (image->debug != MagickFalse)
1117 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
cristy57790302013-02-18 18:19:08 +00001118 if ((image->columns > 16383UL) || (image->rows > 16383UL))
cristye4d08fd2012-04-26 17:29:08 +00001119 ThrowWriterException(ImageError,"WidthOrHeightExceedsLimit");
cristyc82a27b2011-10-21 01:07:16 +00001120 status=OpenBlob(image_info,image,WriteBinaryBlobMode,exception);
cristyffa663d2011-03-14 00:58:51 +00001121 if (status == MagickFalse)
1122 return(status);
Cristy46d4fb52019-09-29 20:32:20 -04001123 if (WebPConfigInit(&configure) == 0)
1124 ThrowWriterException(ResourceLimitError,"UnableToEncodeImageFile");
cristy35c63c72012-10-23 14:40:54 +00001125 if (image->quality != UndefinedCompressionQuality)
1126 configure.quality=(float) image->quality;
cristy7ab1ba82014-11-13 21:37:21 +00001127 if (image->quality >= 100)
1128 configure.lossless=1;
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001129 SetBooleanOption(image_info,"webp:lossless",&configure.lossless);
cristy092ec8d2013-04-26 13:46:22 +00001130 value=GetImageOption(image_info,"webp:image-hint");
cristye71e0892013-02-18 13:13:53 +00001131 if (value != (char *) NULL)
cristyd6f2cc42013-02-18 17:24:19 +00001132 {
cristy9bb4d7a2014-04-20 01:41:32 +00001133 if (LocaleCompare(value,"default") == 0)
1134 configure.image_hint=WEBP_HINT_DEFAULT;
cristy547fe7f2013-02-18 17:27:13 +00001135 if (LocaleCompare(value,"photo") == 0)
cristyd6f2cc42013-02-18 17:24:19 +00001136 configure.image_hint=WEBP_HINT_PHOTO;
cristy547fe7f2013-02-18 17:27:13 +00001137 if (LocaleCompare(value,"picture") == 0)
cristyd6f2cc42013-02-18 17:24:19 +00001138 configure.image_hint=WEBP_HINT_PICTURE;
よや5ce163f2019-02-10 22:42:15 +09001139#if WEBP_ENCODER_ABI_VERSION >= 0x0200
cristy9bb4d7a2014-04-20 01:41:32 +00001140 if (LocaleCompare(value,"graph") == 0)
1141 configure.image_hint=WEBP_HINT_GRAPH;
1142#endif
cristyd6f2cc42013-02-18 17:24:19 +00001143 }
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001144 SetBooleanOption(image_info,"webp:auto-filter",&configure.autofilter);
Dirk Lemstra4e056d62020-12-30 11:09:32 +01001145 value=GetImageOption(image_info,"webp:target-psnr");
cristye71e0892013-02-18 13:13:53 +00001146 if (value != (char *) NULL)
Dirk Lemstra4e056d62020-12-30 11:09:32 +01001147 configure.target_PSNR=(float) StringToDouble(value,(char **) NULL);
1148 SetIntegerOption(image_info,"webp:alpha-compression",
1149 &configure.alpha_compression);
1150 SetIntegerOption(image_info,"webp:alpha-filtering",
1151 &configure.alpha_filtering);
1152 SetIntegerOption(image_info,"webp:alpha-quality",
1153 &configure.alpha_quality);
1154 SetIntegerOption(image_info,"webp:filter-strength",
1155 &configure.filter_strength);
1156 SetIntegerOption(image_info,"webp:filter-sharpness",
1157 &configure.filter_sharpness);
1158 SetIntegerOption(image_info,"webp:filter-type",&configure.filter_type);
1159 SetIntegerOption(image_info,"webp:method",&configure.method);
1160 SetIntegerOption(image_info,"webp:partitions",&configure.partitions);
1161 SetIntegerOption(image_info,"webp:partition-limit",
1162 &configure.partition_limit);
1163 SetIntegerOption(image_info,"webp:pass",&configure.pass);
1164 SetIntegerOption(image_info,"webp:preprocessing",&configure.preprocessing);
1165 SetIntegerOption(image_info,"webp:segments",&configure.segments);
1166 SetIntegerOption(image_info,"webp:show-compressed",
1167 &configure.show_compressed);
1168 SetIntegerOption(image_info,"webp:sns-strength",&configure.sns_strength);
1169 SetIntegerOption(image_info,"webp:target-size",&configure.target_size);
よや5ce163f2019-02-10 22:42:15 +09001170#if WEBP_ENCODER_ABI_VERSION >= 0x0201
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001171 SetBooleanOption(image_info,"webp:emulate-jpeg-size",
1172 &configure.emulate_jpeg_size);
1173 SetBooleanOption(image_info,"webp:low-memory",&configure.low_memory);
Dirk Lemstra4e056d62020-12-30 11:09:32 +01001174 SetIntegerOption(image_info,"webp:thread-level",&configure.thread_level);
cristy9bb4d7a2014-04-20 01:41:32 +00001175#endif
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001176#if WEBP_ENCODER_ABI_VERSION >= 0x0209
1177 SetBooleanOption(image_info,"webp:exact",&configure.exact);
1178#endif
よや5ce163f2019-02-10 22:42:15 +09001179#if WEBP_ENCODER_ABI_VERSION >= 0x020e
Dirk Lemstra0b770a92020-12-30 11:14:36 +01001180 SetIntegerOption(image_info,"webp:near-lossless",&configure.near_lossless);
Elliott Hughes5d41fba2021-04-12 16:36:42 -07001181 SetBooleanOption(image_info,"webp:use-sharp-yuv",&configure.use_sharp_yuv);
Cristyaa5bcab2019-01-31 07:38:50 -05001182#endif
cristy644040e2011-03-14 17:31:59 +00001183 if (WebPValidateConfig(&configure) == 0)
1184 ThrowWriterException(ResourceLimitError,"UnableToEncodeImageFile");
Cristy46d4fb52019-09-29 20:32:20 -04001185#if defined(MAGICKCORE_WEBPMUX_DELEGATE)
Dirk Lemstrad2197622020-12-28 15:57:43 +01001186 {
1187 WebPData
1188 webp_data;
Cristy46d4fb52019-09-29 20:32:20 -04001189
Dirk Lemstrad2197622020-12-28 15:57:43 +01001190 memset(&webp_data,0,sizeof(webp_data));
1191 if ((image_info->adjoin != MagickFalse) &&
1192 (GetPreviousImageInList(image) == (Image *) NULL) &&
1193 (GetNextImageInList(image) != (Image *) NULL))
1194 status=WriteAnimatedWEBPImage(image_info,image,&configure,&webp_data,
1195 exception);
1196 else
1197 {
1198 WebPMemoryWriterInit(&writer);
1199 status=WriteSingleWEBPImage(image_info,image,&configure,&writer,
1200 exception);
1201 if (status == MagickFalse)
1202 WebPMemoryWriterClear(&writer);
1203 else
1204 {
1205 webp_data.bytes=writer.mem;
1206 webp_data.size=writer.size;
1207 }
1208 }
1209 if (status != MagickFalse)
1210 status=WriteWEBPImageProfile(image,&webp_data,exception);
1211 if (status != MagickFalse)
1212 (void) WriteBlob(image,webp_data.size,webp_data.bytes);
1213 WebPDataClear(&webp_data);
1214 }
1215#else
1216 WebPMemoryWriterInit(&writer);
1217 status=WriteSingleWEBPImage(image_info,image,&configure,&writer,
1218 exception);
Dirk Lemstraacc06822020-12-28 10:26:17 +01001219 if (status != MagickFalse)
Dirk Lemstrad2197622020-12-28 15:57:43 +01001220 (void) WriteBlob(image,writer.size,writer.mem);
1221 WebPMemoryWriterClear(&writer);
1222#endif
cristy644040e2011-03-14 17:31:59 +00001223 (void) CloseBlob(image);
tomwei717e76432020-11-26 23:34:47 +08001224 return(status);
cristyb1860752011-03-14 00:27:46 +00001225}
1226#endif