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