1 // Copyright 2014 Google Inc. All Rights Reserved.
2 //
3 // Use of this source code is governed by a BSD-style license
4 // that can be found in the COPYING file in the root of the source
5 // tree. An additional intellectual property rights grant can be found
6 // in the file PATENTS. All contributing project authors may
7 // be found in the AUTHORS file in the root of the source tree.
8 // -----------------------------------------------------------------------------
9 //
10 //  AnimEncoder implementation.
11 //
12 
13 #include <assert.h>
14 #include <limits.h>
15 #include <math.h>    // for pow()
16 #include <stdio.h>
17 #include <stdlib.h>  // for abs()
18 
19 #include "src/mux/animi.h"
20 #include "src/utils/utils.h"
21 #include "src/webp/decode.h"
22 #include "src/webp/encode.h"
23 #include "src/webp/format_constants.h"
24 #include "src/webp/mux.h"
25 
26 #if defined(_MSC_VER) && _MSC_VER < 1900
27 #define snprintf _snprintf
28 #endif
29 
30 #define ERROR_STR_MAX_LENGTH 100
31 
32 //------------------------------------------------------------------------------
33 // Internal structs.
34 
35 // Stores frame rectangle dimensions.
36 typedef struct {
37   int x_offset_, y_offset_, width_, height_;
38 } FrameRectangle;
39 
40 // Used to store two candidates of encoded data for an animation frame. One of
41 // the two will be chosen later.
42 typedef struct {
43   WebPMuxFrameInfo sub_frame_;  // Encoded frame rectangle.
44   WebPMuxFrameInfo key_frame_;  // Encoded frame if it is a key-frame.
45   int is_key_frame_;            // True if 'key_frame' has been chosen.
46 } EncodedFrame;
47 
48 struct WebPAnimEncoder {
49   const int canvas_width_;                  // Canvas width.
50   const int canvas_height_;                 // Canvas height.
51   const WebPAnimEncoderOptions options_;    // Global encoding options.
52 
53   FrameRectangle prev_rect_;          // Previous WebP frame rectangle.
54   WebPConfig last_config_;            // Cached in case a re-encode is needed.
55   WebPConfig last_config_reversed_;   // If 'last_config_' uses lossless, then
56                                       // this config uses lossy and vice versa;
57                                       // only valid if 'options_.allow_mixed'
58                                       // is true.
59 
60   WebPPicture* curr_canvas_;          // Only pointer; we don't own memory.
61 
62   // Canvas buffers.
63   WebPPicture curr_canvas_copy_;      // Possibly modified current canvas.
64   int curr_canvas_copy_modified_;     // True if pixels in 'curr_canvas_copy_'
65                                       // differ from those in 'curr_canvas_'.
66 
67   WebPPicture prev_canvas_;           // Previous canvas.
68   WebPPicture prev_canvas_disposed_;  // Previous canvas disposed to background.
69 
70   // Encoded data.
71   EncodedFrame* encoded_frames_;      // Array of encoded frames.
72   size_t size_;             // Number of allocated frames.
73   size_t start_;            // Frame start index.
74   size_t count_;            // Number of valid frames.
75   size_t flush_count_;      // If >0, 'flush_count' frames starting from
76                             // 'start' are ready to be added to mux.
77 
78   // key-frame related.
79   int64_t best_delta_;      // min(canvas size - frame size) over the frames.
80                             // Can be negative in certain cases due to
81                             // transparent pixels in a frame.
82   int keyframe_;            // Index of selected key-frame relative to 'start_'.
83   int count_since_key_frame_;     // Frames seen since the last key-frame.
84 
85   int first_timestamp_;           // Timestamp of the first frame.
86   int prev_timestamp_;            // Timestamp of the last added frame.
87   int prev_candidate_undecided_;  // True if it's not yet decided if previous
88                                   // frame would be a sub-frame or a key-frame.
89 
90   // Misc.
91   int is_first_frame_;  // True if first frame is yet to be added/being added.
92   int got_null_frame_;  // True if WebPAnimEncoderAdd() has already been called
93                         // with a NULL frame.
94 
95   size_t in_frame_count_;   // Number of input frames processed so far.
96   size_t out_frame_count_;  // Number of frames added to mux so far. This may be
97                             // different from 'in_frame_count_' due to merging.
98 
99   WebPMux* mux_;        // Muxer to assemble the WebP bitstream.
100   char error_str_[ERROR_STR_MAX_LENGTH];  // Error string. Empty if no error.
101 };
102 
103 // -----------------------------------------------------------------------------
104 // Life of WebPAnimEncoder object.
105 
106 #define DELTA_INFINITY      (1ULL << 32)
107 #define KEYFRAME_NONE       (-1)
108 
109 // Reset the counters in the WebPAnimEncoder.
ResetCounters(WebPAnimEncoder * const enc)110 static void ResetCounters(WebPAnimEncoder* const enc) {
111   enc->start_ = 0;
112   enc->count_ = 0;
113   enc->flush_count_ = 0;
114   enc->best_delta_ = DELTA_INFINITY;
115   enc->keyframe_ = KEYFRAME_NONE;
116 }
117 
DisableKeyframes(WebPAnimEncoderOptions * const enc_options)118 static void DisableKeyframes(WebPAnimEncoderOptions* const enc_options) {
119   enc_options->kmax = INT_MAX;
120   enc_options->kmin = enc_options->kmax - 1;
121 }
122 
123 #define MAX_CACHED_FRAMES 30
124 
SanitizeEncoderOptions(WebPAnimEncoderOptions * const enc_options)125 static void SanitizeEncoderOptions(WebPAnimEncoderOptions* const enc_options) {
126   int print_warning = enc_options->verbose;
127 
128   if (enc_options->minimize_size) {
129     DisableKeyframes(enc_options);
130   }
131 
132   if (enc_options->kmax == 1) {  // All frames will be key-frames.
133     enc_options->kmin = 0;
134     enc_options->kmax = 0;
135     return;
136   } else if (enc_options->kmax <= 0) {
137     DisableKeyframes(enc_options);
138     print_warning = 0;
139   }
140 
141   if (enc_options->kmin >= enc_options->kmax) {
142     enc_options->kmin = enc_options->kmax - 1;
143     if (print_warning) {
144       fprintf(stderr, "WARNING: Setting kmin = %d, so that kmin < kmax.\n",
145               enc_options->kmin);
146     }
147   } else {
148     const int kmin_limit = enc_options->kmax / 2 + 1;
149     if (enc_options->kmin < kmin_limit && kmin_limit < enc_options->kmax) {
150       // This ensures that enc.keyframe + kmin >= kmax is always true. So, we
151       // can flush all the frames in the 'count_since_key_frame == kmax' case.
152       enc_options->kmin = kmin_limit;
153       if (print_warning) {
154         fprintf(stderr,
155                 "WARNING: Setting kmin = %d, so that kmin >= kmax / 2 + 1.\n",
156                 enc_options->kmin);
157       }
158     }
159   }
160   // Limit the max number of frames that are allocated.
161   if (enc_options->kmax - enc_options->kmin > MAX_CACHED_FRAMES) {
162     enc_options->kmin = enc_options->kmax - MAX_CACHED_FRAMES;
163     if (print_warning) {
164       fprintf(stderr,
165               "WARNING: Setting kmin = %d, so that kmax - kmin <= %d.\n",
166               enc_options->kmin, MAX_CACHED_FRAMES);
167     }
168   }
169   assert(enc_options->kmin < enc_options->kmax);
170 }
171 
172 #undef MAX_CACHED_FRAMES
173 
DefaultEncoderOptions(WebPAnimEncoderOptions * const enc_options)174 static void DefaultEncoderOptions(WebPAnimEncoderOptions* const enc_options) {
175   enc_options->anim_params.loop_count = 0;
176   enc_options->anim_params.bgcolor = 0xffffffff;  // White.
177   enc_options->minimize_size = 0;
178   DisableKeyframes(enc_options);
179   enc_options->allow_mixed = 0;
180   enc_options->verbose = 0;
181 }
182 
WebPAnimEncoderOptionsInitInternal(WebPAnimEncoderOptions * enc_options,int abi_version)183 int WebPAnimEncoderOptionsInitInternal(WebPAnimEncoderOptions* enc_options,
184                                        int abi_version) {
185   if (enc_options == NULL ||
186       WEBP_ABI_IS_INCOMPATIBLE(abi_version, WEBP_MUX_ABI_VERSION)) {
187     return 0;
188   }
189   DefaultEncoderOptions(enc_options);
190   return 1;
191 }
192 
193 // This starting value is more fit to WebPCleanupTransparentAreaLossless().
194 #define TRANSPARENT_COLOR   0x00000000
195 
ClearRectangle(WebPPicture * const picture,int left,int top,int width,int height)196 static void ClearRectangle(WebPPicture* const picture,
197                            int left, int top, int width, int height) {
198   int j;
199   for (j = top; j < top + height; ++j) {
200     uint32_t* const dst = picture->argb + j * picture->argb_stride;
201     int i;
202     for (i = left; i < left + width; ++i) {
203       dst[i] = TRANSPARENT_COLOR;
204     }
205   }
206 }
207 
WebPUtilClearPic(WebPPicture * const picture,const FrameRectangle * const rect)208 static void WebPUtilClearPic(WebPPicture* const picture,
209                              const FrameRectangle* const rect) {
210   if (rect != NULL) {
211     ClearRectangle(picture, rect->x_offset_, rect->y_offset_,
212                    rect->width_, rect->height_);
213   } else {
214     ClearRectangle(picture, 0, 0, picture->width, picture->height);
215   }
216 }
217 
MarkNoError(WebPAnimEncoder * const enc)218 static void MarkNoError(WebPAnimEncoder* const enc) {
219   enc->error_str_[0] = '\0';  // Empty string.
220 }
221 
MarkError(WebPAnimEncoder * const enc,const char * str)222 static void MarkError(WebPAnimEncoder* const enc, const char* str) {
223   if (snprintf(enc->error_str_, ERROR_STR_MAX_LENGTH, "%s.", str) < 0) {
224     assert(0);  // FIX ME!
225   }
226 }
227 
MarkError2(WebPAnimEncoder * const enc,const char * str,int error_code)228 static void MarkError2(WebPAnimEncoder* const enc,
229                        const char* str, int error_code) {
230   if (snprintf(enc->error_str_, ERROR_STR_MAX_LENGTH, "%s: %d.", str,
231                error_code) < 0) {
232     assert(0);  // FIX ME!
233   }
234 }
235 
WebPAnimEncoderNewInternal(int width,int height,const WebPAnimEncoderOptions * enc_options,int abi_version)236 WebPAnimEncoder* WebPAnimEncoderNewInternal(
237     int width, int height, const WebPAnimEncoderOptions* enc_options,
238     int abi_version) {
239   WebPAnimEncoder* enc;
240 
241   if (WEBP_ABI_IS_INCOMPATIBLE(abi_version, WEBP_MUX_ABI_VERSION)) {
242     return NULL;
243   }
244   if (width <= 0 || height <= 0 ||
245       (width * (uint64_t)height) >= MAX_IMAGE_AREA) {
246     return NULL;
247   }
248 
249   enc = (WebPAnimEncoder*)WebPSafeCalloc(1, sizeof(*enc));
250   if (enc == NULL) return NULL;
251   // sanity inits, so we can call WebPAnimEncoderDelete():
252   enc->encoded_frames_ = NULL;
253   enc->mux_ = NULL;
254   MarkNoError(enc);
255 
256   // Dimensions and options.
257   *(int*)&enc->canvas_width_ = width;
258   *(int*)&enc->canvas_height_ = height;
259   if (enc_options != NULL) {
260     *(WebPAnimEncoderOptions*)&enc->options_ = *enc_options;
261     SanitizeEncoderOptions((WebPAnimEncoderOptions*)&enc->options_);
262   } else {
263     DefaultEncoderOptions((WebPAnimEncoderOptions*)&enc->options_);
264   }
265 
266   // Canvas buffers.
267   if (!WebPPictureInit(&enc->curr_canvas_copy_) ||
268       !WebPPictureInit(&enc->prev_canvas_) ||
269       !WebPPictureInit(&enc->prev_canvas_disposed_)) {
270     goto Err;
271   }
272   enc->curr_canvas_copy_.width = width;
273   enc->curr_canvas_copy_.height = height;
274   enc->curr_canvas_copy_.use_argb = 1;
275   if (!WebPPictureAlloc(&enc->curr_canvas_copy_) ||
276       !WebPPictureCopy(&enc->curr_canvas_copy_, &enc->prev_canvas_) ||
277       !WebPPictureCopy(&enc->curr_canvas_copy_, &enc->prev_canvas_disposed_)) {
278     goto Err;
279   }
280   WebPUtilClearPic(&enc->prev_canvas_, NULL);
281   enc->curr_canvas_copy_modified_ = 1;
282 
283   // Encoded frames.
284   ResetCounters(enc);
285   // Note: one extra storage is for the previous frame.
286   enc->size_ = enc->options_.kmax - enc->options_.kmin + 1;
287   // We need space for at least 2 frames. But when kmin, kmax are both zero,
288   // enc->size_ will be 1. So we handle that special case below.
289   if (enc->size_ < 2) enc->size_ = 2;
290   enc->encoded_frames_ =
291       (EncodedFrame*)WebPSafeCalloc(enc->size_, sizeof(*enc->encoded_frames_));
292   if (enc->encoded_frames_ == NULL) goto Err;
293 
294   enc->mux_ = WebPMuxNew();
295   if (enc->mux_ == NULL) goto Err;
296 
297   enc->count_since_key_frame_ = 0;
298   enc->first_timestamp_ = 0;
299   enc->prev_timestamp_ = 0;
300   enc->prev_candidate_undecided_ = 0;
301   enc->is_first_frame_ = 1;
302   enc->got_null_frame_ = 0;
303 
304   return enc;  // All OK.
305 
306  Err:
307   WebPAnimEncoderDelete(enc);
308   return NULL;
309 }
310 
311 // Release the data contained by 'encoded_frame'.
FrameRelease(EncodedFrame * const encoded_frame)312 static void FrameRelease(EncodedFrame* const encoded_frame) {
313   if (encoded_frame != NULL) {
314     WebPDataClear(&encoded_frame->sub_frame_.bitstream);
315     WebPDataClear(&encoded_frame->key_frame_.bitstream);
316     memset(encoded_frame, 0, sizeof(*encoded_frame));
317   }
318 }
319 
WebPAnimEncoderDelete(WebPAnimEncoder * enc)320 void WebPAnimEncoderDelete(WebPAnimEncoder* enc) {
321   if (enc != NULL) {
322     WebPPictureFree(&enc->curr_canvas_copy_);
323     WebPPictureFree(&enc->prev_canvas_);
324     WebPPictureFree(&enc->prev_canvas_disposed_);
325     if (enc->encoded_frames_ != NULL) {
326       size_t i;
327       for (i = 0; i < enc->size_; ++i) {
328         FrameRelease(&enc->encoded_frames_[i]);
329       }
330       WebPSafeFree(enc->encoded_frames_);
331     }
332     WebPMuxDelete(enc->mux_);
333     WebPSafeFree(enc);
334   }
335 }
336 
337 // -----------------------------------------------------------------------------
338 // Frame addition.
339 
340 // Returns cached frame at the given 'position'.
GetFrame(const WebPAnimEncoder * const enc,size_t position)341 static EncodedFrame* GetFrame(const WebPAnimEncoder* const enc,
342                               size_t position) {
343   assert(enc->start_ + position < enc->size_);
344   return &enc->encoded_frames_[enc->start_ + position];
345 }
346 
347 typedef int (*ComparePixelsFunc)(const uint32_t*, int, const uint32_t*, int,
348                                  int, int);
349 
350 // Returns true if 'length' number of pixels in 'src' and 'dst' are equal,
351 // assuming the given step sizes between pixels.
352 // 'max_allowed_diff' is unused and only there to allow function pointer use.
ComparePixelsLossless(const uint32_t * src,int src_step,const uint32_t * dst,int dst_step,int length,int max_allowed_diff)353 static WEBP_INLINE int ComparePixelsLossless(const uint32_t* src, int src_step,
354                                              const uint32_t* dst, int dst_step,
355                                              int length, int max_allowed_diff) {
356   (void)max_allowed_diff;
357   assert(length > 0);
358   while (length-- > 0) {
359     if (*src != *dst) {
360       return 0;
361     }
362     src += src_step;
363     dst += dst_step;
364   }
365   return 1;
366 }
367 
368 // Helper to check if each channel in 'src' and 'dst' is at most off by
369 // 'max_allowed_diff'.
PixelsAreSimilar(uint32_t src,uint32_t dst,int max_allowed_diff)370 static WEBP_INLINE int PixelsAreSimilar(uint32_t src, uint32_t dst,
371                                         int max_allowed_diff) {
372   const int src_a = (src >> 24) & 0xff;
373   const int src_r = (src >> 16) & 0xff;
374   const int src_g = (src >> 8) & 0xff;
375   const int src_b = (src >> 0) & 0xff;
376   const int dst_a = (dst >> 24) & 0xff;
377   const int dst_r = (dst >> 16) & 0xff;
378   const int dst_g = (dst >> 8) & 0xff;
379   const int dst_b = (dst >> 0) & 0xff;
380 
381   return (src_a == dst_a) &&
382          (abs(src_r - dst_r) * dst_a <= (max_allowed_diff * 255)) &&
383          (abs(src_g - dst_g) * dst_a <= (max_allowed_diff * 255)) &&
384          (abs(src_b - dst_b) * dst_a <= (max_allowed_diff * 255));
385 }
386 
387 // Returns true if 'length' number of pixels in 'src' and 'dst' are within an
388 // error bound, assuming the given step sizes between pixels.
ComparePixelsLossy(const uint32_t * src,int src_step,const uint32_t * dst,int dst_step,int length,int max_allowed_diff)389 static WEBP_INLINE int ComparePixelsLossy(const uint32_t* src, int src_step,
390                                           const uint32_t* dst, int dst_step,
391                                           int length, int max_allowed_diff) {
392   assert(length > 0);
393   while (length-- > 0) {
394     if (!PixelsAreSimilar(*src, *dst, max_allowed_diff)) {
395       return 0;
396     }
397     src += src_step;
398     dst += dst_step;
399   }
400   return 1;
401 }
402 
IsEmptyRect(const FrameRectangle * const rect)403 static int IsEmptyRect(const FrameRectangle* const rect) {
404   return (rect->width_ == 0) || (rect->height_ == 0);
405 }
406 
QualityToMaxDiff(float quality)407 static int QualityToMaxDiff(float quality) {
408   const double val = pow(quality / 100., 0.5);
409   const double max_diff = 31 * (1 - val) + 1 * val;
410   return (int)(max_diff + 0.5);
411 }
412 
413 // Assumes that an initial valid guess of change rectangle 'rect' is passed.
MinimizeChangeRectangle(const WebPPicture * const src,const WebPPicture * const dst,FrameRectangle * const rect,int is_lossless,float quality)414 static void MinimizeChangeRectangle(const WebPPicture* const src,
415                                     const WebPPicture* const dst,
416                                     FrameRectangle* const rect,
417                                     int is_lossless, float quality) {
418   int i, j;
419   const ComparePixelsFunc compare_pixels =
420       is_lossless ? ComparePixelsLossless : ComparePixelsLossy;
421   const int max_allowed_diff_lossy = QualityToMaxDiff(quality);
422   const int max_allowed_diff = is_lossless ? 0 : max_allowed_diff_lossy;
423 
424   // Sanity checks.
425   assert(src->width == dst->width && src->height == dst->height);
426   assert(rect->x_offset_ + rect->width_ <= dst->width);
427   assert(rect->y_offset_ + rect->height_ <= dst->height);
428 
429   // Left boundary.
430   for (i = rect->x_offset_; i < rect->x_offset_ + rect->width_; ++i) {
431     const uint32_t* const src_argb =
432         &src->argb[rect->y_offset_ * src->argb_stride + i];
433     const uint32_t* const dst_argb =
434         &dst->argb[rect->y_offset_ * dst->argb_stride + i];
435     if (compare_pixels(src_argb, src->argb_stride, dst_argb, dst->argb_stride,
436                        rect->height_, max_allowed_diff)) {
437       --rect->width_;  // Redundant column.
438       ++rect->x_offset_;
439     } else {
440       break;
441     }
442   }
443   if (rect->width_ == 0) goto NoChange;
444 
445   // Right boundary.
446   for (i = rect->x_offset_ + rect->width_ - 1; i >= rect->x_offset_; --i) {
447     const uint32_t* const src_argb =
448         &src->argb[rect->y_offset_ * src->argb_stride + i];
449     const uint32_t* const dst_argb =
450         &dst->argb[rect->y_offset_ * dst->argb_stride + i];
451     if (compare_pixels(src_argb, src->argb_stride, dst_argb, dst->argb_stride,
452                        rect->height_, max_allowed_diff)) {
453       --rect->width_;  // Redundant column.
454     } else {
455       break;
456     }
457   }
458   if (rect->width_ == 0) goto NoChange;
459 
460   // Top boundary.
461   for (j = rect->y_offset_; j < rect->y_offset_ + rect->height_; ++j) {
462     const uint32_t* const src_argb =
463         &src->argb[j * src->argb_stride + rect->x_offset_];
464     const uint32_t* const dst_argb =
465         &dst->argb[j * dst->argb_stride + rect->x_offset_];
466     if (compare_pixels(src_argb, 1, dst_argb, 1, rect->width_,
467                        max_allowed_diff)) {
468       --rect->height_;  // Redundant row.
469       ++rect->y_offset_;
470     } else {
471       break;
472     }
473   }
474   if (rect->height_ == 0) goto NoChange;
475 
476   // Bottom boundary.
477   for (j = rect->y_offset_ + rect->height_ - 1; j >= rect->y_offset_; --j) {
478     const uint32_t* const src_argb =
479         &src->argb[j * src->argb_stride + rect->x_offset_];
480     const uint32_t* const dst_argb =
481         &dst->argb[j * dst->argb_stride + rect->x_offset_];
482     if (compare_pixels(src_argb, 1, dst_argb, 1, rect->width_,
483                        max_allowed_diff)) {
484       --rect->height_;  // Redundant row.
485     } else {
486       break;
487     }
488   }
489   if (rect->height_ == 0) goto NoChange;
490 
491   if (IsEmptyRect(rect)) {
492  NoChange:
493     rect->x_offset_ = 0;
494     rect->y_offset_ = 0;
495     rect->width_ = 0;
496     rect->height_ = 0;
497   }
498 }
499 
500 // Snap rectangle to even offsets (and adjust dimensions if needed).
SnapToEvenOffsets(FrameRectangle * const rect)501 static WEBP_INLINE void SnapToEvenOffsets(FrameRectangle* const rect) {
502   rect->width_ += (rect->x_offset_ & 1);
503   rect->height_ += (rect->y_offset_ & 1);
504   rect->x_offset_ &= ~1;
505   rect->y_offset_ &= ~1;
506 }
507 
508 typedef struct {
509   int should_try_;               // Should try this set of parameters.
510   int empty_rect_allowed_;       // Frame with empty rectangle can be skipped.
511   FrameRectangle rect_ll_;       // Frame rectangle for lossless compression.
512   WebPPicture sub_frame_ll_;     // Sub-frame pic for lossless compression.
513   FrameRectangle rect_lossy_;    // Frame rectangle for lossy compression.
514                                  // Could be smaller than rect_ll_ as pixels
515                                  // with small diffs can be ignored.
516   WebPPicture sub_frame_lossy_;  // Sub-frame pic for lossless compression.
517 } SubFrameParams;
518 
SubFrameParamsInit(SubFrameParams * const params,int should_try,int empty_rect_allowed)519 static int SubFrameParamsInit(SubFrameParams* const params,
520                               int should_try, int empty_rect_allowed) {
521   params->should_try_ = should_try;
522   params->empty_rect_allowed_ = empty_rect_allowed;
523   if (!WebPPictureInit(&params->sub_frame_ll_) ||
524       !WebPPictureInit(&params->sub_frame_lossy_)) {
525     return 0;
526   }
527   return 1;
528 }
529 
SubFrameParamsFree(SubFrameParams * const params)530 static void SubFrameParamsFree(SubFrameParams* const params) {
531   WebPPictureFree(&params->sub_frame_ll_);
532   WebPPictureFree(&params->sub_frame_lossy_);
533 }
534 
535 // Given previous and current canvas, picks the optimal rectangle for the
536 // current frame based on 'is_lossless' and other parameters. Assumes that the
537 // initial guess 'rect' is valid.
GetSubRect(const WebPPicture * const prev_canvas,const WebPPicture * const curr_canvas,int is_key_frame,int is_first_frame,int empty_rect_allowed,int is_lossless,float quality,FrameRectangle * const rect,WebPPicture * const sub_frame)538 static int GetSubRect(const WebPPicture* const prev_canvas,
539                       const WebPPicture* const curr_canvas, int is_key_frame,
540                       int is_first_frame, int empty_rect_allowed,
541                       int is_lossless, float quality,
542                       FrameRectangle* const rect,
543                       WebPPicture* const sub_frame) {
544   if (!is_key_frame || is_first_frame) {  // Optimize frame rectangle.
545     // Note: This behaves as expected for first frame, as 'prev_canvas' is
546     // initialized to a fully transparent canvas in the beginning.
547     MinimizeChangeRectangle(prev_canvas, curr_canvas, rect,
548                             is_lossless, quality);
549   }
550 
551   if (IsEmptyRect(rect)) {
552     if (empty_rect_allowed) {  // No need to get 'sub_frame'.
553       return 1;
554     } else {                   // Force a 1x1 rectangle.
555       rect->width_ = 1;
556       rect->height_ = 1;
557       assert(rect->x_offset_ == 0);
558       assert(rect->y_offset_ == 0);
559     }
560   }
561 
562   SnapToEvenOffsets(rect);
563   return WebPPictureView(curr_canvas, rect->x_offset_, rect->y_offset_,
564                          rect->width_, rect->height_, sub_frame);
565 }
566 
567 // Picks optimal frame rectangle for both lossless and lossy compression. The
568 // initial guess for frame rectangles will be the full canvas.
GetSubRects(const WebPPicture * const prev_canvas,const WebPPicture * const curr_canvas,int is_key_frame,int is_first_frame,float quality,SubFrameParams * const params)569 static int GetSubRects(const WebPPicture* const prev_canvas,
570                        const WebPPicture* const curr_canvas, int is_key_frame,
571                        int is_first_frame, float quality,
572                        SubFrameParams* const params) {
573   // Lossless frame rectangle.
574   params->rect_ll_.x_offset_ = 0;
575   params->rect_ll_.y_offset_ = 0;
576   params->rect_ll_.width_ = curr_canvas->width;
577   params->rect_ll_.height_ = curr_canvas->height;
578   if (!GetSubRect(prev_canvas, curr_canvas, is_key_frame, is_first_frame,
579                   params->empty_rect_allowed_, 1, quality,
580                   &params->rect_ll_, &params->sub_frame_ll_)) {
581     return 0;
582   }
583   // Lossy frame rectangle.
584   params->rect_lossy_ = params->rect_ll_;  // seed with lossless rect.
585   return GetSubRect(prev_canvas, curr_canvas, is_key_frame, is_first_frame,
586                     params->empty_rect_allowed_, 0, quality,
587                     &params->rect_lossy_, &params->sub_frame_lossy_);
588 }
589 
clip(int v,int min_v,int max_v)590 static WEBP_INLINE int clip(int v, int min_v, int max_v) {
591   return (v < min_v) ? min_v : (v > max_v) ? max_v : v;
592 }
593 
WebPAnimEncoderRefineRect(const WebPPicture * const prev_canvas,const WebPPicture * const curr_canvas,int is_lossless,float quality,int * const x_offset,int * const y_offset,int * const width,int * const height)594 int WebPAnimEncoderRefineRect(
595     const WebPPicture* const prev_canvas, const WebPPicture* const curr_canvas,
596     int is_lossless, float quality, int* const x_offset, int* const y_offset,
597     int* const width, int* const height) {
598   FrameRectangle rect;
599   const int right = clip(*x_offset + *width, 0, curr_canvas->width);
600   const int left = clip(*x_offset, 0, curr_canvas->width - 1);
601   const int bottom = clip(*y_offset + *height, 0, curr_canvas->height);
602   const int top = clip(*y_offset, 0, curr_canvas->height - 1);
603   if (prev_canvas == NULL || curr_canvas == NULL ||
604       prev_canvas->width != curr_canvas->width ||
605       prev_canvas->height != curr_canvas->height ||
606       !prev_canvas->use_argb || !curr_canvas->use_argb) {
607     return 0;
608   }
609   rect.x_offset_ = left;
610   rect.y_offset_ = top;
611   rect.width_ = clip(right - left, 0, curr_canvas->width - rect.x_offset_);
612   rect.height_ = clip(bottom - top, 0, curr_canvas->height - rect.y_offset_);
613   MinimizeChangeRectangle(prev_canvas, curr_canvas, &rect, is_lossless,
614                           quality);
615   SnapToEvenOffsets(&rect);
616   *x_offset = rect.x_offset_;
617   *y_offset = rect.y_offset_;
618   *width = rect.width_;
619   *height = rect.height_;
620   return 1;
621 }
622 
DisposeFrameRectangle(int dispose_method,const FrameRectangle * const rect,WebPPicture * const curr_canvas)623 static void DisposeFrameRectangle(int dispose_method,
624                                   const FrameRectangle* const rect,
625                                   WebPPicture* const curr_canvas) {
626   assert(rect != NULL);
627   if (dispose_method == WEBP_MUX_DISPOSE_BACKGROUND) {
628     WebPUtilClearPic(curr_canvas, rect);
629   }
630 }
631 
RectArea(const FrameRectangle * const rect)632 static uint32_t RectArea(const FrameRectangle* const rect) {
633   return (uint32_t)rect->width_ * rect->height_;
634 }
635 
IsLosslessBlendingPossible(const WebPPicture * const src,const WebPPicture * const dst,const FrameRectangle * const rect)636 static int IsLosslessBlendingPossible(const WebPPicture* const src,
637                                       const WebPPicture* const dst,
638                                       const FrameRectangle* const rect) {
639   int i, j;
640   assert(src->width == dst->width && src->height == dst->height);
641   assert(rect->x_offset_ + rect->width_ <= dst->width);
642   assert(rect->y_offset_ + rect->height_ <= dst->height);
643   for (j = rect->y_offset_; j < rect->y_offset_ + rect->height_; ++j) {
644     for (i = rect->x_offset_; i < rect->x_offset_ + rect->width_; ++i) {
645       const uint32_t src_pixel = src->argb[j * src->argb_stride + i];
646       const uint32_t dst_pixel = dst->argb[j * dst->argb_stride + i];
647       const uint32_t dst_alpha = dst_pixel >> 24;
648       if (dst_alpha != 0xff && src_pixel != dst_pixel) {
649         // In this case, if we use blending, we can't attain the desired
650         // 'dst_pixel' value for this pixel. So, blending is not possible.
651         return 0;
652       }
653     }
654   }
655   return 1;
656 }
657 
IsLossyBlendingPossible(const WebPPicture * const src,const WebPPicture * const dst,const FrameRectangle * const rect,float quality)658 static int IsLossyBlendingPossible(const WebPPicture* const src,
659                                    const WebPPicture* const dst,
660                                    const FrameRectangle* const rect,
661                                    float quality) {
662   const int max_allowed_diff_lossy = QualityToMaxDiff(quality);
663   int i, j;
664   assert(src->width == dst->width && src->height == dst->height);
665   assert(rect->x_offset_ + rect->width_ <= dst->width);
666   assert(rect->y_offset_ + rect->height_ <= dst->height);
667   for (j = rect->y_offset_; j < rect->y_offset_ + rect->height_; ++j) {
668     for (i = rect->x_offset_; i < rect->x_offset_ + rect->width_; ++i) {
669       const uint32_t src_pixel = src->argb[j * src->argb_stride + i];
670       const uint32_t dst_pixel = dst->argb[j * dst->argb_stride + i];
671       const uint32_t dst_alpha = dst_pixel >> 24;
672       if (dst_alpha != 0xff &&
673           !PixelsAreSimilar(src_pixel, dst_pixel, max_allowed_diff_lossy)) {
674         // In this case, if we use blending, we can't attain the desired
675         // 'dst_pixel' value for this pixel. So, blending is not possible.
676         return 0;
677       }
678     }
679   }
680   return 1;
681 }
682 
683 // For pixels in 'rect', replace those pixels in 'dst' that are same as 'src' by
684 // transparent pixels.
685 // Returns true if at least one pixel gets modified.
IncreaseTransparency(const WebPPicture * const src,const FrameRectangle * const rect,WebPPicture * const dst)686 static int IncreaseTransparency(const WebPPicture* const src,
687                                 const FrameRectangle* const rect,
688                                 WebPPicture* const dst) {
689   int i, j;
690   int modified = 0;
691   assert(src != NULL && dst != NULL && rect != NULL);
692   assert(src->width == dst->width && src->height == dst->height);
693   for (j = rect->y_offset_; j < rect->y_offset_ + rect->height_; ++j) {
694     const uint32_t* const psrc = src->argb + j * src->argb_stride;
695     uint32_t* const pdst = dst->argb + j * dst->argb_stride;
696     for (i = rect->x_offset_; i < rect->x_offset_ + rect->width_; ++i) {
697       if (psrc[i] == pdst[i] && pdst[i] != TRANSPARENT_COLOR) {
698         pdst[i] = TRANSPARENT_COLOR;
699         modified = 1;
700       }
701     }
702   }
703   return modified;
704 }
705 
706 #undef TRANSPARENT_COLOR
707 
708 // Replace similar blocks of pixels by a 'see-through' transparent block
709 // with uniform average color.
710 // Assumes lossy compression is being used.
711 // Returns true if at least one pixel gets modified.
FlattenSimilarBlocks(const WebPPicture * const src,const FrameRectangle * const rect,WebPPicture * const dst,float quality)712 static int FlattenSimilarBlocks(const WebPPicture* const src,
713                                 const FrameRectangle* const rect,
714                                 WebPPicture* const dst, float quality) {
715   const int max_allowed_diff_lossy = QualityToMaxDiff(quality);
716   int i, j;
717   int modified = 0;
718   const int block_size = 8;
719   const int y_start = (rect->y_offset_ + block_size) & ~(block_size - 1);
720   const int y_end = (rect->y_offset_ + rect->height_) & ~(block_size - 1);
721   const int x_start = (rect->x_offset_ + block_size) & ~(block_size - 1);
722   const int x_end = (rect->x_offset_ + rect->width_) & ~(block_size - 1);
723   assert(src != NULL && dst != NULL && rect != NULL);
724   assert(src->width == dst->width && src->height == dst->height);
725   assert((block_size & (block_size - 1)) == 0);  // must be a power of 2
726   // Iterate over each block and count similar pixels.
727   for (j = y_start; j < y_end; j += block_size) {
728     for (i = x_start; i < x_end; i += block_size) {
729       int cnt = 0;
730       int avg_r = 0, avg_g = 0, avg_b = 0;
731       int x, y;
732       const uint32_t* const psrc = src->argb + j * src->argb_stride + i;
733       uint32_t* const pdst = dst->argb + j * dst->argb_stride + i;
734       for (y = 0; y < block_size; ++y) {
735         for (x = 0; x < block_size; ++x) {
736           const uint32_t src_pixel = psrc[x + y * src->argb_stride];
737           const int alpha = src_pixel >> 24;
738           if (alpha == 0xff &&
739               PixelsAreSimilar(src_pixel, pdst[x + y * dst->argb_stride],
740                                max_allowed_diff_lossy)) {
741             ++cnt;
742             avg_r += (src_pixel >> 16) & 0xff;
743             avg_g += (src_pixel >> 8) & 0xff;
744             avg_b += (src_pixel >> 0) & 0xff;
745           }
746         }
747       }
748       // If we have a fully similar block, we replace it with an
749       // average transparent block. This compresses better in lossy mode.
750       if (cnt == block_size * block_size) {
751         const uint32_t color = (0x00          << 24) |
752                                ((avg_r / cnt) << 16) |
753                                ((avg_g / cnt) <<  8) |
754                                ((avg_b / cnt) <<  0);
755         for (y = 0; y < block_size; ++y) {
756           for (x = 0; x < block_size; ++x) {
757             pdst[x + y * dst->argb_stride] = color;
758           }
759         }
760         modified = 1;
761       }
762     }
763   }
764   return modified;
765 }
766 
EncodeFrame(const WebPConfig * const config,WebPPicture * const pic,WebPMemoryWriter * const memory)767 static int EncodeFrame(const WebPConfig* const config, WebPPicture* const pic,
768                        WebPMemoryWriter* const memory) {
769   pic->use_argb = 1;
770   pic->writer = WebPMemoryWrite;
771   pic->custom_ptr = memory;
772   if (!WebPEncode(config, pic)) {
773     return 0;
774   }
775   return 1;
776 }
777 
778 // Struct representing a candidate encoded frame including its metadata.
779 typedef struct {
780   WebPMemoryWriter  mem_;
781   WebPMuxFrameInfo  info_;
782   FrameRectangle    rect_;
783   int               evaluate_;  // True if this candidate should be evaluated.
784 } Candidate;
785 
786 // Generates a candidate encoded frame given a picture and metadata.
EncodeCandidate(WebPPicture * const sub_frame,const FrameRectangle * const rect,const WebPConfig * const encoder_config,int use_blending,Candidate * const candidate)787 static WebPEncodingError EncodeCandidate(WebPPicture* const sub_frame,
788                                          const FrameRectangle* const rect,
789                                          const WebPConfig* const encoder_config,
790                                          int use_blending,
791                                          Candidate* const candidate) {
792   WebPConfig config = *encoder_config;
793   WebPEncodingError error_code = VP8_ENC_OK;
794   assert(candidate != NULL);
795   memset(candidate, 0, sizeof(*candidate));
796 
797   // Set frame rect and info.
798   candidate->rect_ = *rect;
799   candidate->info_.id = WEBP_CHUNK_ANMF;
800   candidate->info_.x_offset = rect->x_offset_;
801   candidate->info_.y_offset = rect->y_offset_;
802   candidate->info_.dispose_method = WEBP_MUX_DISPOSE_NONE;  // Set later.
803   candidate->info_.blend_method =
804       use_blending ? WEBP_MUX_BLEND : WEBP_MUX_NO_BLEND;
805   candidate->info_.duration = 0;  // Set in next call to WebPAnimEncoderAdd().
806 
807   // Encode picture.
808   WebPMemoryWriterInit(&candidate->mem_);
809 
810   if (!config.lossless && use_blending) {
811     // Disable filtering to avoid blockiness in reconstructed frames at the
812     // time of decoding.
813     config.autofilter = 0;
814     config.filter_strength = 0;
815   }
816   if (!EncodeFrame(&config, sub_frame, &candidate->mem_)) {
817     error_code = sub_frame->error_code;
818     goto Err;
819   }
820 
821   candidate->evaluate_ = 1;
822   return error_code;
823 
824  Err:
825   WebPMemoryWriterClear(&candidate->mem_);
826   return error_code;
827 }
828 
CopyCurrentCanvas(WebPAnimEncoder * const enc)829 static void CopyCurrentCanvas(WebPAnimEncoder* const enc) {
830   if (enc->curr_canvas_copy_modified_) {
831     WebPCopyPixels(enc->curr_canvas_, &enc->curr_canvas_copy_);
832     enc->curr_canvas_copy_.progress_hook = enc->curr_canvas_->progress_hook;
833     enc->curr_canvas_copy_.user_data = enc->curr_canvas_->user_data;
834     enc->curr_canvas_copy_modified_ = 0;
835   }
836 }
837 
838 enum {
839   LL_DISP_NONE = 0,
840   LL_DISP_BG,
841   LOSSY_DISP_NONE,
842   LOSSY_DISP_BG,
843   CANDIDATE_COUNT
844 };
845 
846 #define MIN_COLORS_LOSSY     31  // Don't try lossy below this threshold.
847 #define MAX_COLORS_LOSSLESS 194  // Don't try lossless above this threshold.
848 
849 // Generates candidates for a given dispose method given pre-filled sub-frame
850 // 'params'.
GenerateCandidates(WebPAnimEncoder * const enc,Candidate candidates[CANDIDATE_COUNT],WebPMuxAnimDispose dispose_method,int is_lossless,int is_key_frame,SubFrameParams * const params,const WebPConfig * const config_ll,const WebPConfig * const config_lossy)851 static WebPEncodingError GenerateCandidates(
852     WebPAnimEncoder* const enc, Candidate candidates[CANDIDATE_COUNT],
853     WebPMuxAnimDispose dispose_method, int is_lossless, int is_key_frame,
854     SubFrameParams* const params,
855     const WebPConfig* const config_ll, const WebPConfig* const config_lossy) {
856   WebPEncodingError error_code = VP8_ENC_OK;
857   const int is_dispose_none = (dispose_method == WEBP_MUX_DISPOSE_NONE);
858   Candidate* const candidate_ll =
859       is_dispose_none ? &candidates[LL_DISP_NONE] : &candidates[LL_DISP_BG];
860   Candidate* const candidate_lossy = is_dispose_none
861                                      ? &candidates[LOSSY_DISP_NONE]
862                                      : &candidates[LOSSY_DISP_BG];
863   WebPPicture* const curr_canvas = &enc->curr_canvas_copy_;
864   const WebPPicture* const prev_canvas =
865       is_dispose_none ? &enc->prev_canvas_ : &enc->prev_canvas_disposed_;
866   int use_blending_ll, use_blending_lossy;
867   int evaluate_ll, evaluate_lossy;
868 
869   CopyCurrentCanvas(enc);
870   use_blending_ll =
871       !is_key_frame &&
872       IsLosslessBlendingPossible(prev_canvas, curr_canvas, &params->rect_ll_);
873   use_blending_lossy =
874       !is_key_frame &&
875       IsLossyBlendingPossible(prev_canvas, curr_canvas, &params->rect_lossy_,
876                               config_lossy->quality);
877 
878   // Pick candidates to be tried.
879   if (!enc->options_.allow_mixed) {
880     evaluate_ll = is_lossless;
881     evaluate_lossy = !is_lossless;
882   } else if (enc->options_.minimize_size) {
883     evaluate_ll = 1;
884     evaluate_lossy = 1;
885   } else {  // Use a heuristic for trying lossless and/or lossy compression.
886     const int num_colors = WebPGetColorPalette(&params->sub_frame_ll_, NULL);
887     evaluate_ll = (num_colors < MAX_COLORS_LOSSLESS);
888     evaluate_lossy = (num_colors >= MIN_COLORS_LOSSY);
889   }
890 
891   // Generate candidates.
892   if (evaluate_ll) {
893     CopyCurrentCanvas(enc);
894     if (use_blending_ll) {
895       enc->curr_canvas_copy_modified_ =
896           IncreaseTransparency(prev_canvas, &params->rect_ll_, curr_canvas);
897     }
898     error_code = EncodeCandidate(&params->sub_frame_ll_, &params->rect_ll_,
899                                  config_ll, use_blending_ll, candidate_ll);
900     if (error_code != VP8_ENC_OK) return error_code;
901   }
902   if (evaluate_lossy) {
903     CopyCurrentCanvas(enc);
904     if (use_blending_lossy) {
905       enc->curr_canvas_copy_modified_ =
906           FlattenSimilarBlocks(prev_canvas, &params->rect_lossy_, curr_canvas,
907                                config_lossy->quality);
908     }
909     error_code =
910         EncodeCandidate(&params->sub_frame_lossy_, &params->rect_lossy_,
911                         config_lossy, use_blending_lossy, candidate_lossy);
912     if (error_code != VP8_ENC_OK) return error_code;
913     enc->curr_canvas_copy_modified_ = 1;
914   }
915   return error_code;
916 }
917 
918 #undef MIN_COLORS_LOSSY
919 #undef MAX_COLORS_LOSSLESS
920 
GetEncodedData(const WebPMemoryWriter * const memory,WebPData * const encoded_data)921 static void GetEncodedData(const WebPMemoryWriter* const memory,
922                            WebPData* const encoded_data) {
923   encoded_data->bytes = memory->mem;
924   encoded_data->size  = memory->size;
925 }
926 
927 // Sets dispose method of the previous frame to be 'dispose_method'.
SetPreviousDisposeMethod(WebPAnimEncoder * const enc,WebPMuxAnimDispose dispose_method)928 static void SetPreviousDisposeMethod(WebPAnimEncoder* const enc,
929                                      WebPMuxAnimDispose dispose_method) {
930   const size_t position = enc->count_ - 2;
931   EncodedFrame* const prev_enc_frame = GetFrame(enc, position);
932   assert(enc->count_ >= 2);  // As current and previous frames are in enc.
933 
934   if (enc->prev_candidate_undecided_) {
935     assert(dispose_method == WEBP_MUX_DISPOSE_NONE);
936     prev_enc_frame->sub_frame_.dispose_method = dispose_method;
937     prev_enc_frame->key_frame_.dispose_method = dispose_method;
938   } else {
939     WebPMuxFrameInfo* const prev_info = prev_enc_frame->is_key_frame_
940                                         ? &prev_enc_frame->key_frame_
941                                         : &prev_enc_frame->sub_frame_;
942     prev_info->dispose_method = dispose_method;
943   }
944 }
945 
IncreasePreviousDuration(WebPAnimEncoder * const enc,int duration)946 static int IncreasePreviousDuration(WebPAnimEncoder* const enc, int duration) {
947   const size_t position = enc->count_ - 1;
948   EncodedFrame* const prev_enc_frame = GetFrame(enc, position);
949   int new_duration;
950 
951   assert(enc->count_ >= 1);
952   assert(prev_enc_frame->sub_frame_.duration ==
953          prev_enc_frame->key_frame_.duration);
954   assert(prev_enc_frame->sub_frame_.duration ==
955          (prev_enc_frame->sub_frame_.duration & (MAX_DURATION - 1)));
956   assert(duration == (duration & (MAX_DURATION - 1)));
957 
958   new_duration = prev_enc_frame->sub_frame_.duration + duration;
959   if (new_duration >= MAX_DURATION) {  // Special case.
960     // Separate out previous frame from earlier merged frames to avoid overflow.
961     // We add a 1x1 transparent frame for the previous frame, with blending on.
962     const FrameRectangle rect = { 0, 0, 1, 1 };
963     const uint8_t lossless_1x1_bytes[] = {
964       0x52, 0x49, 0x46, 0x46, 0x14, 0x00, 0x00, 0x00, 0x57, 0x45, 0x42, 0x50,
965       0x56, 0x50, 0x38, 0x4c, 0x08, 0x00, 0x00, 0x00, 0x2f, 0x00, 0x00, 0x00,
966       0x10, 0x88, 0x88, 0x08
967     };
968     const WebPData lossless_1x1 = {
969         lossless_1x1_bytes, sizeof(lossless_1x1_bytes)
970     };
971     const uint8_t lossy_1x1_bytes[] = {
972       0x52, 0x49, 0x46, 0x46, 0x40, 0x00, 0x00, 0x00, 0x57, 0x45, 0x42, 0x50,
973       0x56, 0x50, 0x38, 0x58, 0x0a, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00,
974       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x41, 0x4c, 0x50, 0x48, 0x02, 0x00,
975       0x00, 0x00, 0x00, 0x00, 0x56, 0x50, 0x38, 0x20, 0x18, 0x00, 0x00, 0x00,
976       0x30, 0x01, 0x00, 0x9d, 0x01, 0x2a, 0x01, 0x00, 0x01, 0x00, 0x02, 0x00,
977       0x34, 0x25, 0xa4, 0x00, 0x03, 0x70, 0x00, 0xfe, 0xfb, 0xfd, 0x50, 0x00
978     };
979     const WebPData lossy_1x1 = { lossy_1x1_bytes, sizeof(lossy_1x1_bytes) };
980     const int can_use_lossless =
981         (enc->last_config_.lossless || enc->options_.allow_mixed);
982     EncodedFrame* const curr_enc_frame = GetFrame(enc, enc->count_);
983     curr_enc_frame->is_key_frame_ = 0;
984     curr_enc_frame->sub_frame_.id = WEBP_CHUNK_ANMF;
985     curr_enc_frame->sub_frame_.x_offset = 0;
986     curr_enc_frame->sub_frame_.y_offset = 0;
987     curr_enc_frame->sub_frame_.dispose_method = WEBP_MUX_DISPOSE_NONE;
988     curr_enc_frame->sub_frame_.blend_method = WEBP_MUX_BLEND;
989     curr_enc_frame->sub_frame_.duration = duration;
990     if (!WebPDataCopy(can_use_lossless ? &lossless_1x1 : &lossy_1x1,
991                       &curr_enc_frame->sub_frame_.bitstream)) {
992       return 0;
993     }
994     ++enc->count_;
995     ++enc->count_since_key_frame_;
996     enc->flush_count_ = enc->count_ - 1;
997     enc->prev_candidate_undecided_ = 0;
998     enc->prev_rect_ = rect;
999   } else {                           // Regular case.
1000     // Increase duration of the previous frame by 'duration'.
1001     prev_enc_frame->sub_frame_.duration = new_duration;
1002     prev_enc_frame->key_frame_.duration = new_duration;
1003   }
1004   return 1;
1005 }
1006 
1007 // Pick the candidate encoded frame with smallest size and release other
1008 // candidates.
1009 // TODO(later): Perhaps a rough SSIM/PSNR produced by the encoder should
1010 // also be a criteria, in addition to sizes.
PickBestCandidate(WebPAnimEncoder * const enc,Candidate * const candidates,int is_key_frame,EncodedFrame * const encoded_frame)1011 static void PickBestCandidate(WebPAnimEncoder* const enc,
1012                               Candidate* const candidates, int is_key_frame,
1013                               EncodedFrame* const encoded_frame) {
1014   int i;
1015   int best_idx = -1;
1016   size_t best_size = ~0;
1017   for (i = 0; i < CANDIDATE_COUNT; ++i) {
1018     if (candidates[i].evaluate_) {
1019       const size_t candidate_size = candidates[i].mem_.size;
1020       if (candidate_size < best_size) {
1021         best_idx = i;
1022         best_size = candidate_size;
1023       }
1024     }
1025   }
1026   assert(best_idx != -1);
1027   for (i = 0; i < CANDIDATE_COUNT; ++i) {
1028     if (candidates[i].evaluate_) {
1029       if (i == best_idx) {
1030         WebPMuxFrameInfo* const dst = is_key_frame
1031                                       ? &encoded_frame->key_frame_
1032                                       : &encoded_frame->sub_frame_;
1033         *dst = candidates[i].info_;
1034         GetEncodedData(&candidates[i].mem_, &dst->bitstream);
1035         if (!is_key_frame) {
1036           // Note: Previous dispose method only matters for non-keyframes.
1037           // Also, we don't want to modify previous dispose method that was
1038           // selected when a non key-frame was assumed.
1039           const WebPMuxAnimDispose prev_dispose_method =
1040               (best_idx == LL_DISP_NONE || best_idx == LOSSY_DISP_NONE)
1041                   ? WEBP_MUX_DISPOSE_NONE
1042                   : WEBP_MUX_DISPOSE_BACKGROUND;
1043           SetPreviousDisposeMethod(enc, prev_dispose_method);
1044         }
1045         enc->prev_rect_ = candidates[i].rect_;  // save for next frame.
1046       } else {
1047         WebPMemoryWriterClear(&candidates[i].mem_);
1048         candidates[i].evaluate_ = 0;
1049       }
1050     }
1051   }
1052 }
1053 
1054 // Depending on the configuration, tries different compressions
1055 // (lossy/lossless), dispose methods, blending methods etc to encode the current
1056 // frame and outputs the best one in 'encoded_frame'.
1057 // 'frame_skipped' will be set to true if this frame should actually be skipped.
SetFrame(WebPAnimEncoder * const enc,const WebPConfig * const config,int is_key_frame,EncodedFrame * const encoded_frame,int * const frame_skipped)1058 static WebPEncodingError SetFrame(WebPAnimEncoder* const enc,
1059                                   const WebPConfig* const config,
1060                                   int is_key_frame,
1061                                   EncodedFrame* const encoded_frame,
1062                                   int* const frame_skipped) {
1063   int i;
1064   WebPEncodingError error_code = VP8_ENC_OK;
1065   const WebPPicture* const curr_canvas = &enc->curr_canvas_copy_;
1066   const WebPPicture* const prev_canvas = &enc->prev_canvas_;
1067   Candidate candidates[CANDIDATE_COUNT];
1068   const int is_lossless = config->lossless;
1069   const int consider_lossless = is_lossless || enc->options_.allow_mixed;
1070   const int consider_lossy = !is_lossless || enc->options_.allow_mixed;
1071   const int is_first_frame = enc->is_first_frame_;
1072 
1073   // First frame cannot be skipped as there is no 'previous frame' to merge it
1074   // to. So, empty rectangle is not allowed for the first frame.
1075   const int empty_rect_allowed_none = !is_first_frame;
1076 
1077   // Even if there is exact pixel match between 'disposed previous canvas' and
1078   // 'current canvas', we can't skip current frame, as there may not be exact
1079   // pixel match between 'previous canvas' and 'current canvas'. So, we don't
1080   // allow empty rectangle in this case.
1081   const int empty_rect_allowed_bg = 0;
1082 
1083   // If current frame is a key-frame, dispose method of previous frame doesn't
1084   // matter, so we don't try dispose to background.
1085   // Also, if key-frame insertion is on, and previous frame could be picked as
1086   // either a sub-frame or a key-frame, then we can't be sure about what frame
1087   // rectangle would be disposed. In that case too, we don't try dispose to
1088   // background.
1089   const int dispose_bg_possible =
1090       !is_key_frame && !enc->prev_candidate_undecided_;
1091 
1092   SubFrameParams dispose_none_params;
1093   SubFrameParams dispose_bg_params;
1094 
1095   WebPConfig config_ll = *config;
1096   WebPConfig config_lossy = *config;
1097   config_ll.lossless = 1;
1098   config_lossy.lossless = 0;
1099   enc->last_config_ = *config;
1100   enc->last_config_reversed_ = config->lossless ? config_lossy : config_ll;
1101   *frame_skipped = 0;
1102 
1103   if (!SubFrameParamsInit(&dispose_none_params, 1, empty_rect_allowed_none) ||
1104       !SubFrameParamsInit(&dispose_bg_params, 0, empty_rect_allowed_bg)) {
1105     return VP8_ENC_ERROR_INVALID_CONFIGURATION;
1106   }
1107 
1108   memset(candidates, 0, sizeof(candidates));
1109 
1110   // Change-rectangle assuming previous frame was DISPOSE_NONE.
1111   if (!GetSubRects(prev_canvas, curr_canvas, is_key_frame, is_first_frame,
1112                    config_lossy.quality, &dispose_none_params)) {
1113     error_code = VP8_ENC_ERROR_INVALID_CONFIGURATION;
1114     goto Err;
1115   }
1116 
1117   if ((consider_lossless && IsEmptyRect(&dispose_none_params.rect_ll_)) ||
1118       (consider_lossy && IsEmptyRect(&dispose_none_params.rect_lossy_))) {
1119     // Don't encode the frame at all. Instead, the duration of the previous
1120     // frame will be increased later.
1121     assert(empty_rect_allowed_none);
1122     *frame_skipped = 1;
1123     goto End;
1124   }
1125 
1126   if (dispose_bg_possible) {
1127     // Change-rectangle assuming previous frame was DISPOSE_BACKGROUND.
1128     WebPPicture* const prev_canvas_disposed = &enc->prev_canvas_disposed_;
1129     WebPCopyPixels(prev_canvas, prev_canvas_disposed);
1130     DisposeFrameRectangle(WEBP_MUX_DISPOSE_BACKGROUND, &enc->prev_rect_,
1131                           prev_canvas_disposed);
1132 
1133     if (!GetSubRects(prev_canvas_disposed, curr_canvas, is_key_frame,
1134                      is_first_frame, config_lossy.quality,
1135                      &dispose_bg_params)) {
1136       error_code = VP8_ENC_ERROR_INVALID_CONFIGURATION;
1137       goto Err;
1138     }
1139     assert(!IsEmptyRect(&dispose_bg_params.rect_ll_));
1140     assert(!IsEmptyRect(&dispose_bg_params.rect_lossy_));
1141 
1142     if (enc->options_.minimize_size) {  // Try both dispose methods.
1143       dispose_bg_params.should_try_ = 1;
1144       dispose_none_params.should_try_ = 1;
1145     } else if ((is_lossless &&
1146                 RectArea(&dispose_bg_params.rect_ll_) <
1147                     RectArea(&dispose_none_params.rect_ll_)) ||
1148                (!is_lossless &&
1149                 RectArea(&dispose_bg_params.rect_lossy_) <
1150                     RectArea(&dispose_none_params.rect_lossy_))) {
1151       dispose_bg_params.should_try_ = 1;  // Pick DISPOSE_BACKGROUND.
1152       dispose_none_params.should_try_ = 0;
1153     }
1154   }
1155 
1156   if (dispose_none_params.should_try_) {
1157     error_code = GenerateCandidates(
1158         enc, candidates, WEBP_MUX_DISPOSE_NONE, is_lossless, is_key_frame,
1159         &dispose_none_params, &config_ll, &config_lossy);
1160     if (error_code != VP8_ENC_OK) goto Err;
1161   }
1162 
1163   if (dispose_bg_params.should_try_) {
1164     assert(!enc->is_first_frame_);
1165     assert(dispose_bg_possible);
1166     error_code = GenerateCandidates(
1167         enc, candidates, WEBP_MUX_DISPOSE_BACKGROUND, is_lossless, is_key_frame,
1168         &dispose_bg_params, &config_ll, &config_lossy);
1169     if (error_code != VP8_ENC_OK) goto Err;
1170   }
1171 
1172   PickBestCandidate(enc, candidates, is_key_frame, encoded_frame);
1173 
1174   goto End;
1175 
1176  Err:
1177   for (i = 0; i < CANDIDATE_COUNT; ++i) {
1178     if (candidates[i].evaluate_) {
1179       WebPMemoryWriterClear(&candidates[i].mem_);
1180     }
1181   }
1182 
1183  End:
1184   SubFrameParamsFree(&dispose_none_params);
1185   SubFrameParamsFree(&dispose_bg_params);
1186   return error_code;
1187 }
1188 
1189 // Calculate the penalty incurred if we encode given frame as a key frame
1190 // instead of a sub-frame.
KeyFramePenalty(const EncodedFrame * const encoded_frame)1191 static int64_t KeyFramePenalty(const EncodedFrame* const encoded_frame) {
1192   return ((int64_t)encoded_frame->key_frame_.bitstream.size -
1193           encoded_frame->sub_frame_.bitstream.size);
1194 }
1195 
CacheFrame(WebPAnimEncoder * const enc,const WebPConfig * const config)1196 static int CacheFrame(WebPAnimEncoder* const enc,
1197                       const WebPConfig* const config) {
1198   int ok = 0;
1199   int frame_skipped = 0;
1200   WebPEncodingError error_code = VP8_ENC_OK;
1201   const size_t position = enc->count_;
1202   EncodedFrame* const encoded_frame = GetFrame(enc, position);
1203 
1204   ++enc->count_;
1205 
1206   if (enc->is_first_frame_) {  // Add this as a key-frame.
1207     error_code = SetFrame(enc, config, 1, encoded_frame, &frame_skipped);
1208     if (error_code != VP8_ENC_OK) goto End;
1209     assert(frame_skipped == 0);  // First frame can't be skipped, even if empty.
1210     assert(position == 0 && enc->count_ == 1);
1211     encoded_frame->is_key_frame_ = 1;
1212     enc->flush_count_ = 0;
1213     enc->count_since_key_frame_ = 0;
1214     enc->prev_candidate_undecided_ = 0;
1215   } else {
1216     ++enc->count_since_key_frame_;
1217     if (enc->count_since_key_frame_ <= enc->options_.kmin) {
1218       // Add this as a frame rectangle.
1219       error_code = SetFrame(enc, config, 0, encoded_frame, &frame_skipped);
1220       if (error_code != VP8_ENC_OK) goto End;
1221       if (frame_skipped) goto Skip;
1222       encoded_frame->is_key_frame_ = 0;
1223       enc->flush_count_ = enc->count_ - 1;
1224       enc->prev_candidate_undecided_ = 0;
1225     } else {
1226       int64_t curr_delta;
1227       FrameRectangle prev_rect_key, prev_rect_sub;
1228 
1229       // Add this as a frame rectangle to enc.
1230       error_code = SetFrame(enc, config, 0, encoded_frame, &frame_skipped);
1231       if (error_code != VP8_ENC_OK) goto End;
1232       if (frame_skipped) goto Skip;
1233       prev_rect_sub = enc->prev_rect_;
1234 
1235 
1236       // Add this as a key-frame to enc, too.
1237       error_code = SetFrame(enc, config, 1, encoded_frame, &frame_skipped);
1238       if (error_code != VP8_ENC_OK) goto End;
1239       assert(frame_skipped == 0);  // Key-frame cannot be an empty rectangle.
1240       prev_rect_key = enc->prev_rect_;
1241 
1242       // Analyze size difference of the two variants.
1243       curr_delta = KeyFramePenalty(encoded_frame);
1244       if (curr_delta <= enc->best_delta_) {  // Pick this as the key-frame.
1245         if (enc->keyframe_ != KEYFRAME_NONE) {
1246           EncodedFrame* const old_keyframe = GetFrame(enc, enc->keyframe_);
1247           assert(old_keyframe->is_key_frame_);
1248           old_keyframe->is_key_frame_ = 0;
1249         }
1250         encoded_frame->is_key_frame_ = 1;
1251         enc->prev_candidate_undecided_ = 1;
1252         enc->keyframe_ = (int)position;
1253         enc->best_delta_ = curr_delta;
1254         enc->flush_count_ = enc->count_ - 1;  // We can flush previous frames.
1255       } else {
1256         encoded_frame->is_key_frame_ = 0;
1257         enc->prev_candidate_undecided_ = 0;
1258       }
1259       // Note: We need '>=' below because when kmin and kmax are both zero,
1260       // count_since_key_frame will always be > kmax.
1261       if (enc->count_since_key_frame_ >= enc->options_.kmax) {
1262         enc->flush_count_ = enc->count_ - 1;
1263         enc->count_since_key_frame_ = 0;
1264         enc->keyframe_ = KEYFRAME_NONE;
1265         enc->best_delta_ = DELTA_INFINITY;
1266       }
1267       if (!enc->prev_candidate_undecided_) {
1268         enc->prev_rect_ =
1269             encoded_frame->is_key_frame_ ? prev_rect_key : prev_rect_sub;
1270       }
1271     }
1272   }
1273 
1274   // Update previous to previous and previous canvases for next call.
1275   WebPCopyPixels(enc->curr_canvas_, &enc->prev_canvas_);
1276   enc->is_first_frame_ = 0;
1277 
1278  Skip:
1279   ok = 1;
1280   ++enc->in_frame_count_;
1281 
1282  End:
1283   if (!ok || frame_skipped) {
1284     FrameRelease(encoded_frame);
1285     // We reset some counters, as the frame addition failed/was skipped.
1286     --enc->count_;
1287     if (!enc->is_first_frame_) --enc->count_since_key_frame_;
1288     if (!ok) {
1289       MarkError2(enc, "ERROR adding frame. WebPEncodingError", error_code);
1290     }
1291   }
1292   enc->curr_canvas_->error_code = error_code;   // report error_code
1293   assert(ok || error_code != VP8_ENC_OK);
1294   return ok;
1295 }
1296 
FlushFrames(WebPAnimEncoder * const enc)1297 static int FlushFrames(WebPAnimEncoder* const enc) {
1298   while (enc->flush_count_ > 0) {
1299     WebPMuxError err;
1300     EncodedFrame* const curr = GetFrame(enc, 0);
1301     const WebPMuxFrameInfo* const info =
1302         curr->is_key_frame_ ? &curr->key_frame_ : &curr->sub_frame_;
1303     assert(enc->mux_ != NULL);
1304     err = WebPMuxPushFrame(enc->mux_, info, 1);
1305     if (err != WEBP_MUX_OK) {
1306       MarkError2(enc, "ERROR adding frame. WebPMuxError", err);
1307       return 0;
1308     }
1309     if (enc->options_.verbose) {
1310       fprintf(stderr, "INFO: Added frame. offset:%d,%d dispose:%d blend:%d\n",
1311               info->x_offset, info->y_offset, info->dispose_method,
1312               info->blend_method);
1313     }
1314     ++enc->out_frame_count_;
1315     FrameRelease(curr);
1316     ++enc->start_;
1317     --enc->flush_count_;
1318     --enc->count_;
1319     if (enc->keyframe_ != KEYFRAME_NONE) --enc->keyframe_;
1320   }
1321 
1322   if (enc->count_ == 1 && enc->start_ != 0) {
1323     // Move enc->start to index 0.
1324     const int enc_start_tmp = (int)enc->start_;
1325     EncodedFrame temp = enc->encoded_frames_[0];
1326     enc->encoded_frames_[0] = enc->encoded_frames_[enc_start_tmp];
1327     enc->encoded_frames_[enc_start_tmp] = temp;
1328     FrameRelease(&enc->encoded_frames_[enc_start_tmp]);
1329     enc->start_ = 0;
1330   }
1331   return 1;
1332 }
1333 
1334 #undef DELTA_INFINITY
1335 #undef KEYFRAME_NONE
1336 
WebPAnimEncoderAdd(WebPAnimEncoder * enc,WebPPicture * frame,int timestamp,const WebPConfig * encoder_config)1337 int WebPAnimEncoderAdd(WebPAnimEncoder* enc, WebPPicture* frame, int timestamp,
1338                        const WebPConfig* encoder_config) {
1339   WebPConfig config;
1340   int ok;
1341 
1342   if (enc == NULL) {
1343     return 0;
1344   }
1345   MarkNoError(enc);
1346 
1347   if (!enc->is_first_frame_) {
1348     // Make sure timestamps are non-decreasing (integer wrap-around is OK).
1349     const uint32_t prev_frame_duration =
1350         (uint32_t)timestamp - enc->prev_timestamp_;
1351     if (prev_frame_duration >= MAX_DURATION) {
1352       if (frame != NULL) {
1353         frame->error_code = VP8_ENC_ERROR_INVALID_CONFIGURATION;
1354       }
1355       MarkError(enc, "ERROR adding frame: timestamps must be non-decreasing");
1356       return 0;
1357     }
1358     if (!IncreasePreviousDuration(enc, (int)prev_frame_duration)) {
1359       return 0;
1360     }
1361   } else {
1362     enc->first_timestamp_ = timestamp;
1363   }
1364 
1365   if (frame == NULL) {  // Special: last call.
1366     enc->got_null_frame_ = 1;
1367     enc->prev_timestamp_ = timestamp;
1368     return 1;
1369   }
1370 
1371   if (frame->width != enc->canvas_width_ ||
1372       frame->height != enc->canvas_height_) {
1373     frame->error_code = VP8_ENC_ERROR_INVALID_CONFIGURATION;
1374     MarkError(enc, "ERROR adding frame: Invalid frame dimensions");
1375     return 0;
1376   }
1377 
1378   if (!frame->use_argb) {  // Convert frame from YUV(A) to ARGB.
1379     if (enc->options_.verbose) {
1380       fprintf(stderr, "WARNING: Converting frame from YUV(A) to ARGB format; "
1381               "this incurs a small loss.\n");
1382     }
1383     if (!WebPPictureYUVAToARGB(frame)) {
1384       MarkError(enc, "ERROR converting frame from YUV(A) to ARGB");
1385       return 0;
1386     }
1387   }
1388 
1389   if (encoder_config != NULL) {
1390     if (!WebPValidateConfig(encoder_config)) {
1391       MarkError(enc, "ERROR adding frame: Invalid WebPConfig");
1392       return 0;
1393     }
1394     config = *encoder_config;
1395   } else {
1396     WebPConfigInit(&config);
1397     config.lossless = 1;
1398   }
1399   assert(enc->curr_canvas_ == NULL);
1400   enc->curr_canvas_ = frame;  // Store reference.
1401   assert(enc->curr_canvas_copy_modified_ == 1);
1402   CopyCurrentCanvas(enc);
1403 
1404   ok = CacheFrame(enc, &config) && FlushFrames(enc);
1405 
1406   enc->curr_canvas_ = NULL;
1407   enc->curr_canvas_copy_modified_ = 1;
1408   if (ok) {
1409     enc->prev_timestamp_ = timestamp;
1410   }
1411   return ok;
1412 }
1413 
1414 // -----------------------------------------------------------------------------
1415 // Bitstream assembly.
1416 
DecodeFrameOntoCanvas(const WebPMuxFrameInfo * const frame,WebPPicture * const canvas)1417 static int DecodeFrameOntoCanvas(const WebPMuxFrameInfo* const frame,
1418                                  WebPPicture* const canvas) {
1419   const WebPData* const image = &frame->bitstream;
1420   WebPPicture sub_image;
1421   WebPDecoderConfig config;
1422   WebPInitDecoderConfig(&config);
1423   WebPUtilClearPic(canvas, NULL);
1424   if (WebPGetFeatures(image->bytes, image->size, &config.input) !=
1425       VP8_STATUS_OK) {
1426     return 0;
1427   }
1428   if (!WebPPictureView(canvas, frame->x_offset, frame->y_offset,
1429                        config.input.width, config.input.height, &sub_image)) {
1430     return 0;
1431   }
1432   config.output.is_external_memory = 1;
1433   config.output.colorspace = MODE_BGRA;
1434   config.output.u.RGBA.rgba = (uint8_t*)sub_image.argb;
1435   config.output.u.RGBA.stride = sub_image.argb_stride * 4;
1436   config.output.u.RGBA.size = config.output.u.RGBA.stride * sub_image.height;
1437 
1438   if (WebPDecode(image->bytes, image->size, &config) != VP8_STATUS_OK) {
1439     return 0;
1440   }
1441   return 1;
1442 }
1443 
FrameToFullCanvas(WebPAnimEncoder * const enc,const WebPMuxFrameInfo * const frame,WebPData * const full_image)1444 static int FrameToFullCanvas(WebPAnimEncoder* const enc,
1445                              const WebPMuxFrameInfo* const frame,
1446                              WebPData* const full_image) {
1447   WebPPicture* const canvas_buf = &enc->curr_canvas_copy_;
1448   WebPMemoryWriter mem1, mem2;
1449   WebPMemoryWriterInit(&mem1);
1450   WebPMemoryWriterInit(&mem2);
1451 
1452   if (!DecodeFrameOntoCanvas(frame, canvas_buf)) goto Err;
1453   if (!EncodeFrame(&enc->last_config_, canvas_buf, &mem1)) goto Err;
1454   GetEncodedData(&mem1, full_image);
1455 
1456   if (enc->options_.allow_mixed) {
1457     if (!EncodeFrame(&enc->last_config_reversed_, canvas_buf, &mem2)) goto Err;
1458     if (mem2.size < mem1.size) {
1459       GetEncodedData(&mem2, full_image);
1460       WebPMemoryWriterClear(&mem1);
1461     } else {
1462       WebPMemoryWriterClear(&mem2);
1463     }
1464   }
1465   return 1;
1466 
1467  Err:
1468   WebPMemoryWriterClear(&mem1);
1469   WebPMemoryWriterClear(&mem2);
1470   return 0;
1471 }
1472 
1473 // Convert a single-frame animation to a non-animated image if appropriate.
1474 // TODO(urvang): Can we pick one of the two heuristically (based on frame
1475 // rectangle and/or presence of alpha)?
OptimizeSingleFrame(WebPAnimEncoder * const enc,WebPData * const webp_data)1476 static WebPMuxError OptimizeSingleFrame(WebPAnimEncoder* const enc,
1477                                         WebPData* const webp_data) {
1478   WebPMuxError err = WEBP_MUX_OK;
1479   int canvas_width, canvas_height;
1480   WebPMuxFrameInfo frame;
1481   WebPData full_image;
1482   WebPData webp_data2;
1483   WebPMux* const mux = WebPMuxCreate(webp_data, 0);
1484   if (mux == NULL) return WEBP_MUX_BAD_DATA;
1485   assert(enc->out_frame_count_ == 1);
1486   WebPDataInit(&frame.bitstream);
1487   WebPDataInit(&full_image);
1488   WebPDataInit(&webp_data2);
1489 
1490   err = WebPMuxGetFrame(mux, 1, &frame);
1491   if (err != WEBP_MUX_OK) goto End;
1492   if (frame.id != WEBP_CHUNK_ANMF) goto End;  // Non-animation: nothing to do.
1493   err = WebPMuxGetCanvasSize(mux, &canvas_width, &canvas_height);
1494   if (err != WEBP_MUX_OK) goto End;
1495   if (!FrameToFullCanvas(enc, &frame, &full_image)) {
1496     err = WEBP_MUX_BAD_DATA;
1497     goto End;
1498   }
1499   err = WebPMuxSetImage(mux, &full_image, 1);
1500   if (err != WEBP_MUX_OK) goto End;
1501   err = WebPMuxAssemble(mux, &webp_data2);
1502   if (err != WEBP_MUX_OK) goto End;
1503 
1504   if (webp_data2.size < webp_data->size) {  // Pick 'webp_data2' if smaller.
1505     WebPDataClear(webp_data);
1506     *webp_data = webp_data2;
1507     WebPDataInit(&webp_data2);
1508   }
1509 
1510  End:
1511   WebPDataClear(&frame.bitstream);
1512   WebPDataClear(&full_image);
1513   WebPMuxDelete(mux);
1514   WebPDataClear(&webp_data2);
1515   return err;
1516 }
1517 
WebPAnimEncoderAssemble(WebPAnimEncoder * enc,WebPData * webp_data)1518 int WebPAnimEncoderAssemble(WebPAnimEncoder* enc, WebPData* webp_data) {
1519   WebPMux* mux;
1520   WebPMuxError err;
1521 
1522   if (enc == NULL) {
1523     return 0;
1524   }
1525   MarkNoError(enc);
1526 
1527   if (webp_data == NULL) {
1528     MarkError(enc, "ERROR assembling: NULL input");
1529     return 0;
1530   }
1531 
1532   if (enc->in_frame_count_ == 0) {
1533     MarkError(enc, "ERROR: No frames to assemble");
1534     return 0;
1535   }
1536 
1537   if (!enc->got_null_frame_ && enc->in_frame_count_ > 1 && enc->count_ > 0) {
1538     // set duration of the last frame to be avg of durations of previous frames.
1539     const double delta_time =
1540         (uint32_t)enc->prev_timestamp_ - enc->first_timestamp_;
1541     const int average_duration = (int)(delta_time / (enc->in_frame_count_ - 1));
1542     if (!IncreasePreviousDuration(enc, average_duration)) {
1543       return 0;
1544     }
1545   }
1546 
1547   // Flush any remaining frames.
1548   enc->flush_count_ = enc->count_;
1549   if (!FlushFrames(enc)) {
1550     return 0;
1551   }
1552 
1553   // Set definitive canvas size.
1554   mux = enc->mux_;
1555   err = WebPMuxSetCanvasSize(mux, enc->canvas_width_, enc->canvas_height_);
1556   if (err != WEBP_MUX_OK) goto Err;
1557 
1558   err = WebPMuxSetAnimationParams(mux, &enc->options_.anim_params);
1559   if (err != WEBP_MUX_OK) goto Err;
1560 
1561   // Assemble into a WebP bitstream.
1562   err = WebPMuxAssemble(mux, webp_data);
1563   if (err != WEBP_MUX_OK) goto Err;
1564 
1565   if (enc->out_frame_count_ == 1) {
1566     err = OptimizeSingleFrame(enc, webp_data);
1567     if (err != WEBP_MUX_OK) goto Err;
1568   }
1569   return 1;
1570 
1571  Err:
1572   MarkError2(enc, "ERROR assembling WebP", err);
1573   return 0;
1574 }
1575 
WebPAnimEncoderGetError(WebPAnimEncoder * enc)1576 const char* WebPAnimEncoderGetError(WebPAnimEncoder* enc) {
1577   if (enc == NULL) return NULL;
1578   return enc->error_str_;
1579 }
1580 
1581 // -----------------------------------------------------------------------------
1582