2013-09-12 22:41:09 +02:00
|
|
|
|
// Copyright 2013 Google Inc. All Rights Reserved.
|
|
|
|
|
//
|
|
|
|
|
// Use of this source code is governed by a BSD-style license
|
|
|
|
|
// that can be found in the COPYING file in the root of the source
|
|
|
|
|
// tree. An additional intellectual property rights grant can be found
|
|
|
|
|
// in the file PATENTS. All contributing project authors may
|
|
|
|
|
// be found in the AUTHORS file in the root of the source tree.
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
//
|
|
|
|
|
// Helper structs and methods for gif2webp tool.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
|
|
#include "webp/encode.h"
|
|
|
|
|
#include "./gif2webp_util.h"
|
|
|
|
|
|
|
|
|
|
#define DELTA_INFINITY 1ULL << 32
|
|
|
|
|
#define KEYFRAME_NONE -1
|
|
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------
|
2013-10-14 23:39:46 +02:00
|
|
|
|
// Helper utilities.
|
2013-09-12 22:41:09 +02:00
|
|
|
|
|
|
|
|
|
static void ClearRectangle(WebPPicture* const picture,
|
|
|
|
|
int left, int top, int width, int height) {
|
|
|
|
|
int j;
|
|
|
|
|
for (j = top; j < top + height; ++j) {
|
|
|
|
|
uint32_t* const dst = picture->argb + j * picture->argb_stride;
|
|
|
|
|
int i;
|
|
|
|
|
for (i = left; i < left + width; ++i) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
dst[i] = WEBP_UTIL_TRANSPARENT_COLOR;
|
2013-09-12 22:41:09 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void WebPUtilClearPic(WebPPicture* const picture,
|
|
|
|
|
const WebPFrameRect* const rect) {
|
|
|
|
|
if (rect != NULL) {
|
|
|
|
|
ClearRectangle(picture, rect->x_offset, rect->y_offset,
|
|
|
|
|
rect->width, rect->height);
|
|
|
|
|
} else {
|
|
|
|
|
ClearRectangle(picture, 0, 0, picture->width, picture->height);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: Also used in picture.c. Move to a common location?
|
|
|
|
|
// Copy width x height pixels from 'src' to 'dst' honoring the strides.
|
|
|
|
|
static void CopyPlane(const uint8_t* src, int src_stride,
|
|
|
|
|
uint8_t* dst, int dst_stride, int width, int height) {
|
|
|
|
|
while (height-- > 0) {
|
|
|
|
|
memcpy(dst, src, width);
|
|
|
|
|
src += src_stride;
|
|
|
|
|
dst += dst_stride;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-10-14 23:39:46 +02:00
|
|
|
|
// Copy pixels from 'src' to 'dst' honoring strides. 'src' and 'dst' are assumed
|
|
|
|
|
// to be already allocated.
|
|
|
|
|
static void CopyPixels(const WebPPicture* const src, WebPPicture* const dst) {
|
2013-09-12 22:41:09 +02:00
|
|
|
|
assert(src->width == dst->width && src->height == dst->height);
|
|
|
|
|
CopyPlane((uint8_t*)src->argb, 4 * src->argb_stride, (uint8_t*)dst->argb,
|
|
|
|
|
4 * dst->argb_stride, 4 * src->width, src->height);
|
|
|
|
|
}
|
|
|
|
|
|
2013-10-14 23:39:46 +02:00
|
|
|
|
// Given 'src' picture and its frame rectangle 'rect', blend it into 'dst'.
|
|
|
|
|
static void BlendPixels(const WebPPicture* const src,
|
|
|
|
|
const WebPFrameRect* const rect,
|
|
|
|
|
WebPPicture* const dst) {
|
2013-09-12 22:41:09 +02:00
|
|
|
|
int j;
|
|
|
|
|
assert(src->width == dst->width && src->height == dst->height);
|
|
|
|
|
for (j = rect->y_offset; j < rect->y_offset + rect->height; ++j) {
|
|
|
|
|
int i;
|
|
|
|
|
for (i = rect->x_offset; i < rect->x_offset + rect->width; ++i) {
|
|
|
|
|
const uint32_t src_pixel = src->argb[j * src->argb_stride + i];
|
|
|
|
|
const int src_alpha = src_pixel >> 24;
|
|
|
|
|
if (src_alpha != 0) {
|
|
|
|
|
dst->argb[j * dst->argb_stride + i] = src_pixel;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// Returns true if 'length' number of pixels in 'src' and 'dst' are identical,
|
|
|
|
|
// assuming the given step sizes between pixels.
|
|
|
|
|
static WEBP_INLINE int ComparePixels(const uint32_t* src, int src_step,
|
|
|
|
|
const uint32_t* dst, int dst_step,
|
|
|
|
|
int length) {
|
|
|
|
|
assert(length > 0);
|
|
|
|
|
while (length-- > 0) {
|
|
|
|
|
if (*src != *dst) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
src += src_step;
|
|
|
|
|
dst += dst_step;
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Assumes that an initial valid guess of change rectangle 'rect' is passed.
|
|
|
|
|
static void MinimizeChangeRectangle(const WebPPicture* const src,
|
|
|
|
|
const WebPPicture* const dst,
|
|
|
|
|
WebPFrameRect* const rect) {
|
|
|
|
|
int i, j;
|
|
|
|
|
// Sanity checks.
|
|
|
|
|
assert(src->width == dst->width && src->height == dst->height);
|
|
|
|
|
assert(rect->x_offset + rect->width <= dst->width);
|
|
|
|
|
assert(rect->y_offset + rect->height <= dst->height);
|
|
|
|
|
|
|
|
|
|
// Left boundary.
|
|
|
|
|
for (i = rect->x_offset; i < rect->x_offset + rect->width; ++i) {
|
|
|
|
|
const uint32_t* const src_argb =
|
|
|
|
|
&src->argb[rect->y_offset * src->argb_stride + i];
|
|
|
|
|
const uint32_t* const dst_argb =
|
|
|
|
|
&dst->argb[rect->y_offset * dst->argb_stride + i];
|
|
|
|
|
if (ComparePixels(src_argb, src->argb_stride, dst_argb, dst->argb_stride,
|
|
|
|
|
rect->height)) {
|
|
|
|
|
--rect->width; // Redundant column.
|
|
|
|
|
++rect->x_offset;
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (rect->width == 0) goto End;
|
|
|
|
|
|
|
|
|
|
// Right boundary.
|
|
|
|
|
for (i = rect->x_offset + rect->width - 1; i >= rect->x_offset; --i) {
|
|
|
|
|
const uint32_t* const src_argb =
|
|
|
|
|
&src->argb[rect->y_offset * src->argb_stride + i];
|
|
|
|
|
const uint32_t* const dst_argb =
|
|
|
|
|
&dst->argb[rect->y_offset * dst->argb_stride + i];
|
|
|
|
|
if (ComparePixels(src_argb, src->argb_stride, dst_argb, dst->argb_stride,
|
|
|
|
|
rect->height)) {
|
|
|
|
|
--rect->width; // Redundant column.
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (rect->width == 0) goto End;
|
|
|
|
|
|
|
|
|
|
// Top boundary.
|
|
|
|
|
for (j = rect->y_offset; j < rect->y_offset + rect->height; ++j) {
|
|
|
|
|
const uint32_t* const src_argb =
|
|
|
|
|
&src->argb[j * src->argb_stride + rect->x_offset];
|
|
|
|
|
const uint32_t* const dst_argb =
|
|
|
|
|
&dst->argb[j * dst->argb_stride + rect->x_offset];
|
|
|
|
|
if (ComparePixels(src_argb, 1, dst_argb, 1, rect->width)) {
|
|
|
|
|
--rect->height; // Redundant row.
|
|
|
|
|
++rect->y_offset;
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (rect->height == 0) goto End;
|
|
|
|
|
|
|
|
|
|
// Bottom boundary.
|
|
|
|
|
for (j = rect->y_offset + rect->height - 1; j >= rect->y_offset; --j) {
|
|
|
|
|
const uint32_t* const src_argb =
|
|
|
|
|
&src->argb[j * src->argb_stride + rect->x_offset];
|
|
|
|
|
const uint32_t* const dst_argb =
|
|
|
|
|
&dst->argb[j * dst->argb_stride + rect->x_offset];
|
|
|
|
|
if (ComparePixels(src_argb, 1, dst_argb, 1, rect->width)) {
|
|
|
|
|
--rect->height; // Redundant row.
|
|
|
|
|
} else {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (rect->height == 0) goto End;
|
|
|
|
|
|
|
|
|
|
if (rect->width == 0 || rect->height == 0) {
|
|
|
|
|
End:
|
|
|
|
|
// TODO(later): This rare case can happen for a bad GIF. In such a case, the
|
|
|
|
|
// frame should not be encoded at all and the duration of prev frame should
|
|
|
|
|
// be increased instead. For now, we just create a 1x1 frame at zero offset.
|
|
|
|
|
rect->x_offset = 0;
|
|
|
|
|
rect->y_offset = 0;
|
|
|
|
|
rect->width = 1;
|
|
|
|
|
rect->height = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// For pixels in 'rect', replace those pixels in 'dst' that are same as 'src' by
|
|
|
|
|
// transparent pixels.
|
|
|
|
|
static void IncreaseTransparency(const WebPPicture* const src,
|
|
|
|
|
const WebPFrameRect* const rect,
|
|
|
|
|
WebPPicture* const dst) {
|
2013-10-08 15:04:52 +02:00
|
|
|
|
int i, j;
|
|
|
|
|
assert(src != NULL && dst != NULL && rect != NULL);
|
2013-10-01 01:48:39 +02:00
|
|
|
|
assert(src->width == dst->width && src->height == dst->height);
|
|
|
|
|
for (j = rect->y_offset; j < rect->y_offset + rect->height; ++j) {
|
2014-10-23 00:43:26 +02:00
|
|
|
|
const uint32_t* const psrc = src->argb + j * src->argb_stride;
|
|
|
|
|
uint32_t* const pdst = dst->argb + j * dst->argb_stride;
|
2013-10-01 01:48:39 +02:00
|
|
|
|
for (i = rect->x_offset; i < rect->x_offset + rect->width; ++i) {
|
2014-10-23 00:43:26 +02:00
|
|
|
|
if (psrc[i] == pdst[i]) {
|
|
|
|
|
pdst[i] = WEBP_UTIL_TRANSPARENT_COLOR;
|
2013-10-01 01:48:39 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-10-14 23:39:46 +02:00
|
|
|
|
// Replace similar blocks of pixels by a 'see-through' transparent block
|
|
|
|
|
// with uniform average color.
|
|
|
|
|
static void FlattenSimilarBlocks(const WebPPicture* const src,
|
|
|
|
|
const WebPFrameRect* const rect,
|
|
|
|
|
WebPPicture* const dst) {
|
2013-10-08 15:04:52 +02:00
|
|
|
|
int i, j;
|
|
|
|
|
const int block_size = 8;
|
|
|
|
|
const int y_start = (rect->y_offset + block_size) & ~(block_size - 1);
|
|
|
|
|
const int y_end = (rect->y_offset + rect->height) & ~(block_size - 1);
|
|
|
|
|
const int x_start = (rect->x_offset + block_size) & ~(block_size - 1);
|
|
|
|
|
const int x_end = (rect->x_offset + rect->width) & ~(block_size - 1);
|
|
|
|
|
assert(src != NULL && dst != NULL && rect != NULL);
|
|
|
|
|
assert(src->width == dst->width && src->height == dst->height);
|
|
|
|
|
assert((block_size & (block_size - 1)) == 0); // must be a power of 2
|
|
|
|
|
// Iterate over each block and count similar pixels.
|
|
|
|
|
for (j = y_start; j < y_end; j += block_size) {
|
|
|
|
|
for (i = x_start; i < x_end; i += block_size) {
|
|
|
|
|
int cnt = 0;
|
|
|
|
|
int avg_r = 0, avg_g = 0, avg_b = 0;
|
|
|
|
|
int x, y;
|
|
|
|
|
const uint32_t* const psrc = src->argb + j * src->argb_stride + i;
|
|
|
|
|
uint32_t* const pdst = dst->argb + j * dst->argb_stride + i;
|
|
|
|
|
for (y = 0; y < block_size; ++y) {
|
|
|
|
|
for (x = 0; x < block_size; ++x) {
|
|
|
|
|
const uint32_t src_pixel = psrc[x + y * src->argb_stride];
|
|
|
|
|
const int alpha = src_pixel >> 24;
|
|
|
|
|
if (alpha == 0xff &&
|
|
|
|
|
src_pixel == pdst[x + y * dst->argb_stride]) {
|
|
|
|
|
++cnt;
|
|
|
|
|
avg_r += (src_pixel >> 16) & 0xff;
|
|
|
|
|
avg_g += (src_pixel >> 8) & 0xff;
|
|
|
|
|
avg_b += (src_pixel >> 0) & 0xff;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// If we have a fully similar block, we replace it with an
|
|
|
|
|
// average transparent block. This compresses better in lossy mode.
|
|
|
|
|
if (cnt == block_size * block_size) {
|
|
|
|
|
const uint32_t color = (0x00 << 24) |
|
|
|
|
|
((avg_r / cnt) << 16) |
|
|
|
|
|
((avg_g / cnt) << 8) |
|
|
|
|
|
((avg_b / cnt) << 0);
|
|
|
|
|
for (y = 0; y < block_size; ++y) {
|
|
|
|
|
for (x = 0; x < block_size; ++x) {
|
|
|
|
|
pdst[x + y * dst->argb_stride] = color;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-12 22:41:09 +02:00
|
|
|
|
//------------------------------------------------------------------------------
|
2013-10-14 23:39:46 +02:00
|
|
|
|
// Encoded frame.
|
|
|
|
|
|
|
|
|
|
// Used to store two candidates of encoded data for an animation frame. One of
|
|
|
|
|
// the two will be chosen later.
|
|
|
|
|
typedef struct {
|
|
|
|
|
WebPMuxFrameInfo sub_frame; // Encoded frame rectangle.
|
|
|
|
|
WebPMuxFrameInfo key_frame; // Encoded frame if it was converted to keyframe.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
int is_key_frame; // True if 'key_frame' has been chosen.
|
2013-10-14 23:39:46 +02:00
|
|
|
|
} EncodedFrame;
|
|
|
|
|
|
|
|
|
|
// Release the data contained by 'encoded_frame'.
|
|
|
|
|
static void FrameRelease(EncodedFrame* const encoded_frame) {
|
2013-12-16 22:31:45 +01:00
|
|
|
|
if (encoded_frame != NULL) {
|
|
|
|
|
WebPDataClear(&encoded_frame->sub_frame.bitstream);
|
|
|
|
|
WebPDataClear(&encoded_frame->key_frame.bitstream);
|
|
|
|
|
memset(encoded_frame, 0, sizeof(*encoded_frame));
|
|
|
|
|
}
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
// Frame cache.
|
|
|
|
|
|
|
|
|
|
// Used to store encoded frames that haven't been output yet.
|
|
|
|
|
struct WebPFrameCache {
|
|
|
|
|
EncodedFrame* encoded_frames; // Array of encoded frames.
|
|
|
|
|
size_t size; // Number of allocated data elements.
|
|
|
|
|
size_t start; // Start index.
|
|
|
|
|
size_t count; // Number of valid data elements.
|
|
|
|
|
int flush_count; // If >0, ‘flush_count’ frames starting from
|
|
|
|
|
// 'start' are ready to be added to mux.
|
|
|
|
|
int64_t best_delta; // min(canvas size - frame size) over the frames.
|
|
|
|
|
// Can be negative in certain cases due to
|
|
|
|
|
// transparent pixels in a frame.
|
|
|
|
|
int keyframe; // Index of selected keyframe relative to 'start'.
|
|
|
|
|
|
|
|
|
|
size_t kmin; // Min distance between key frames.
|
|
|
|
|
size_t kmax; // Max distance between key frames.
|
|
|
|
|
size_t count_since_key_frame; // Frames seen since the last key frame.
|
2013-11-18 03:04:07 +01:00
|
|
|
|
int allow_mixed; // If true, each frame can be lossy or lossless.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
|
|
|
|
|
WebPFrameRect prev_orig_rect; // Previous input (e.g. GIF) frame rectangle.
|
|
|
|
|
WebPFrameRect prev_webp_rect; // Previous WebP frame rectangle.
|
|
|
|
|
WebPMuxAnimDispose prev_orig_dispose; // Previous input dispose method.
|
|
|
|
|
int prev_candidate_undecided; // True if sub-frame vs keyframe decision
|
|
|
|
|
// hasn't been made for the previous frame yet.
|
|
|
|
|
|
|
|
|
|
WebPPicture prev_canvas; // Previous canvas (NOT disposed).
|
|
|
|
|
WebPPicture curr_canvas; // Current canvas (NOT disposed).
|
|
|
|
|
WebPPicture prev_canvas_disposed; // Previous canvas disposed to background.
|
|
|
|
|
WebPPicture curr_canvas_tmp; // Temporary storage for current canvas.
|
2013-10-14 23:39:46 +02:00
|
|
|
|
int is_first_frame; // True if no frames have been added to the cache
|
|
|
|
|
// since WebPFrameCacheNew().
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Reset the counters in the cache struct. Doesn't touch 'cache->encoded_frames'
|
|
|
|
|
// and 'cache->size'.
|
|
|
|
|
static void CacheReset(WebPFrameCache* const cache) {
|
|
|
|
|
cache->start = 0;
|
|
|
|
|
cache->count = 0;
|
|
|
|
|
cache->flush_count = 0;
|
|
|
|
|
cache->best_delta = DELTA_INFINITY;
|
|
|
|
|
cache->keyframe = KEYFRAME_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
WebPFrameCache* WebPFrameCacheNew(int width, int height,
|
2013-11-18 03:04:07 +01:00
|
|
|
|
size_t kmin, size_t kmax, int allow_mixed) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
WebPFrameCache* cache = (WebPFrameCache*)malloc(sizeof(*cache));
|
|
|
|
|
if (cache == NULL) return NULL;
|
|
|
|
|
CacheReset(cache);
|
2013-12-16 22:31:45 +01:00
|
|
|
|
// sanity init, so we can call WebPFrameCacheDelete():
|
|
|
|
|
cache->encoded_frames = NULL;
|
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
cache->prev_candidate_undecided = 0;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
cache->is_first_frame = 1;
|
|
|
|
|
|
|
|
|
|
// Picture buffers.
|
|
|
|
|
if (!WebPPictureInit(&cache->prev_canvas) ||
|
2014-10-23 00:43:26 +02:00
|
|
|
|
!WebPPictureInit(&cache->curr_canvas) ||
|
|
|
|
|
!WebPPictureInit(&cache->prev_canvas_disposed) ||
|
|
|
|
|
!WebPPictureInit(&cache->curr_canvas_tmp)) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
cache->prev_canvas.width = width;
|
|
|
|
|
cache->prev_canvas.height = height;
|
|
|
|
|
cache->prev_canvas.use_argb = 1;
|
|
|
|
|
if (!WebPPictureAlloc(&cache->prev_canvas) ||
|
2014-10-23 00:43:26 +02:00
|
|
|
|
!WebPPictureCopy(&cache->prev_canvas, &cache->curr_canvas) ||
|
|
|
|
|
!WebPPictureCopy(&cache->prev_canvas, &cache->prev_canvas_disposed) ||
|
|
|
|
|
!WebPPictureCopy(&cache->prev_canvas, &cache->curr_canvas_tmp)) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
goto Err;
|
|
|
|
|
}
|
|
|
|
|
WebPUtilClearPic(&cache->prev_canvas, NULL);
|
|
|
|
|
|
|
|
|
|
// Cache data.
|
2013-11-18 03:04:07 +01:00
|
|
|
|
cache->allow_mixed = allow_mixed;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
cache->kmin = kmin;
|
|
|
|
|
cache->kmax = kmax;
|
|
|
|
|
cache->count_since_key_frame = 0;
|
|
|
|
|
assert(kmax > kmin);
|
2014-10-23 00:43:26 +02:00
|
|
|
|
cache->size = kmax - kmin + 1; // One extra storage for previous frame.
|
2013-10-14 23:39:46 +02:00
|
|
|
|
cache->encoded_frames =
|
|
|
|
|
(EncodedFrame*)calloc(cache->size, sizeof(*cache->encoded_frames));
|
|
|
|
|
if (cache->encoded_frames == NULL) goto Err;
|
|
|
|
|
|
|
|
|
|
return cache; // All OK.
|
|
|
|
|
|
|
|
|
|
Err:
|
|
|
|
|
WebPFrameCacheDelete(cache);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void WebPFrameCacheDelete(WebPFrameCache* const cache) {
|
|
|
|
|
if (cache != NULL) {
|
2013-12-16 22:31:45 +01:00
|
|
|
|
if (cache->encoded_frames != NULL) {
|
|
|
|
|
size_t i;
|
|
|
|
|
for (i = 0; i < cache->size; ++i) {
|
|
|
|
|
FrameRelease(&cache->encoded_frames[i]);
|
|
|
|
|
}
|
|
|
|
|
free(cache->encoded_frames);
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
|
|
|
|
WebPPictureFree(&cache->prev_canvas);
|
|
|
|
|
WebPPictureFree(&cache->curr_canvas);
|
2014-10-23 00:43:26 +02:00
|
|
|
|
WebPPictureFree(&cache->prev_canvas_disposed);
|
|
|
|
|
WebPPictureFree(&cache->curr_canvas_tmp);
|
2013-10-14 23:39:46 +02:00
|
|
|
|
free(cache);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int EncodeFrame(const WebPConfig* const config, WebPPicture* const pic,
|
2013-11-18 03:04:07 +01:00
|
|
|
|
WebPMemoryWriter* const memory) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
pic->use_argb = 1;
|
|
|
|
|
pic->writer = WebPMemoryWrite;
|
2013-11-18 03:04:07 +01:00
|
|
|
|
pic->custom_ptr = memory;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
if (!WebPEncode(config, pic)) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-18 03:04:07 +01:00
|
|
|
|
static void GetEncodedData(const WebPMemoryWriter* const memory,
|
|
|
|
|
WebPData* const encoded_data) {
|
|
|
|
|
encoded_data->bytes = memory->mem;
|
|
|
|
|
encoded_data->size = memory->size;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
|
|
|
|
|
2013-11-18 03:04:07 +01:00
|
|
|
|
#define MIN_COLORS_LOSSY 31 // Don't try lossy below this threshold.
|
|
|
|
|
#define MAX_COLORS_LOSSLESS 194 // Don't try lossless above this threshold.
|
|
|
|
|
#define MAX_COLOR_COUNT 256 // Power of 2 greater than MAX_COLORS_LOSSLESS.
|
|
|
|
|
#define HASH_SIZE (MAX_COLOR_COUNT * 4)
|
|
|
|
|
#define HASH_RIGHT_SHIFT 22 // 32 - log2(HASH_SIZE).
|
|
|
|
|
|
|
|
|
|
// TODO(urvang): Also used in enc/vp8l.c. Move to utils.
|
|
|
|
|
// If the number of colors in the 'pic' is at least MAX_COLOR_COUNT, return
|
|
|
|
|
// MAX_COLOR_COUNT. Otherwise, return the exact number of colors in the 'pic'.
|
|
|
|
|
static int GetColorCount(const WebPPicture* const pic) {
|
|
|
|
|
int x, y;
|
|
|
|
|
int num_colors = 0;
|
|
|
|
|
uint8_t in_use[HASH_SIZE] = { 0 };
|
|
|
|
|
uint32_t colors[HASH_SIZE];
|
|
|
|
|
static const uint32_t kHashMul = 0x1e35a7bd;
|
|
|
|
|
const uint32_t* argb = pic->argb;
|
|
|
|
|
const int width = pic->width;
|
|
|
|
|
const int height = pic->height;
|
|
|
|
|
uint32_t last_pix = ~argb[0]; // so we're sure that last_pix != argb[0]
|
|
|
|
|
|
|
|
|
|
for (y = 0; y < height; ++y) {
|
|
|
|
|
for (x = 0; x < width; ++x) {
|
|
|
|
|
int key;
|
|
|
|
|
if (argb[x] == last_pix) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
last_pix = argb[x];
|
|
|
|
|
key = (kHashMul * last_pix) >> HASH_RIGHT_SHIFT;
|
|
|
|
|
while (1) {
|
|
|
|
|
if (!in_use[key]) {
|
|
|
|
|
colors[key] = last_pix;
|
|
|
|
|
in_use[key] = 1;
|
|
|
|
|
++num_colors;
|
|
|
|
|
if (num_colors >= MAX_COLOR_COUNT) {
|
|
|
|
|
return MAX_COLOR_COUNT; // Exact count not needed.
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
} else if (colors[key] == last_pix) {
|
|
|
|
|
break; // The color is already there.
|
|
|
|
|
} else {
|
|
|
|
|
// Some other color sits here, so do linear conflict resolution.
|
|
|
|
|
++key;
|
|
|
|
|
key &= (HASH_SIZE - 1); // Key mask.
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
argb += pic->argb_stride;
|
|
|
|
|
}
|
|
|
|
|
return num_colors;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
|
|
|
|
|
2013-11-18 03:04:07 +01:00
|
|
|
|
#undef MAX_COLOR_COUNT
|
|
|
|
|
#undef HASH_SIZE
|
|
|
|
|
#undef HASH_RIGHT_SHIFT
|
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
static void DisposeFullFrame(WebPMuxAnimDispose dispose_method,
|
|
|
|
|
WebPPicture* const frame) {
|
|
|
|
|
if (dispose_method == WEBP_MUX_DISPOSE_BACKGROUND) {
|
|
|
|
|
WebPUtilClearPic(frame, NULL);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void DisposeFrameRectangle(WebPMuxAnimDispose dispose_method,
|
|
|
|
|
const WebPFrameRect* const gif_rect,
|
|
|
|
|
WebPPicture* const frame) {
|
|
|
|
|
if (dispose_method == WEBP_MUX_DISPOSE_BACKGROUND) {
|
|
|
|
|
WebPUtilClearPic(frame, gif_rect);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Snap rectangle to even offsets (and adjust dimensions if needed).
|
|
|
|
|
static WEBP_INLINE void SnapToEvenOffsets(WebPFrameRect* const rect) {
|
|
|
|
|
rect->width += (rect->x_offset & 1);
|
|
|
|
|
rect->height += (rect->y_offset & 1);
|
|
|
|
|
rect->x_offset &= ~1;
|
|
|
|
|
rect->y_offset &= ~1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Given previous and current canvas, picks the optimal rectangle for the
|
|
|
|
|
// current frame.
|
|
|
|
|
// The initial guess for 'rect' will be 'orig_rect' if is non-NULL, otherwise
|
|
|
|
|
// the initial guess will be the full canvas.
|
|
|
|
|
static int GetSubRect(const WebPPicture* const prev_canvas,
|
|
|
|
|
const WebPPicture* const curr_canvas,
|
|
|
|
|
const WebPFrameRect* const orig_rect, int is_key_frame,
|
|
|
|
|
WebPFrameRect* const rect, WebPPicture* const sub_frame) {
|
|
|
|
|
if (orig_rect != NULL) {
|
|
|
|
|
*rect = *orig_rect;
|
|
|
|
|
} else {
|
|
|
|
|
rect->x_offset = 0;
|
|
|
|
|
rect->y_offset = 0;
|
|
|
|
|
rect->width = curr_canvas->width;
|
|
|
|
|
rect->height = curr_canvas->height;
|
|
|
|
|
}
|
|
|
|
|
if (!is_key_frame) { // Optimize frame rectangle.
|
|
|
|
|
MinimizeChangeRectangle(prev_canvas, curr_canvas, rect);
|
|
|
|
|
}
|
|
|
|
|
SnapToEvenOffsets(rect);
|
|
|
|
|
|
|
|
|
|
return WebPPictureView(curr_canvas, rect->x_offset, rect->y_offset,
|
|
|
|
|
rect->width, rect->height, sub_frame);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int IsBlendingPossible(const WebPPicture* const src,
|
|
|
|
|
const WebPPicture* const dst,
|
|
|
|
|
const WebPFrameRect* const rect) {
|
|
|
|
|
int i, j;
|
|
|
|
|
assert(src->width == dst->width && src->height == dst->height);
|
|
|
|
|
assert(rect->x_offset + rect->width <= dst->width);
|
|
|
|
|
assert(rect->y_offset + rect->height <= dst->height);
|
|
|
|
|
for (j = rect->y_offset; j < rect->y_offset + rect->height; ++j) {
|
|
|
|
|
for (i = rect->x_offset; i < rect->x_offset + rect->width; ++i) {
|
|
|
|
|
const uint32_t src_pixel = src->argb[j * src->argb_stride + i];
|
|
|
|
|
const uint32_t dst_pixel = dst->argb[j * dst->argb_stride + i];
|
|
|
|
|
const uint32_t dst_alpha = dst_pixel >> 24;
|
|
|
|
|
if (dst_alpha != 0xff && src_pixel != dst_pixel) {
|
|
|
|
|
// In this case, if we use blending, we can't attain the desired
|
|
|
|
|
// 'dst_pixel' value for this pixel. So, blending is not possible.
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int RectArea(const WebPFrameRect* const rect) {
|
|
|
|
|
return rect->width * rect->height;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Struct representing a candidate encoded frame including its metadata.
|
|
|
|
|
typedef struct {
|
|
|
|
|
WebPMemoryWriter mem;
|
|
|
|
|
WebPMuxFrameInfo info;
|
|
|
|
|
WebPFrameRect rect;
|
|
|
|
|
int evaluate; // True if this candidate should be evaluated.
|
|
|
|
|
} Candidate;
|
|
|
|
|
|
|
|
|
|
// Generates a candidate encoded frame given a picture and metadata.
|
|
|
|
|
static WebPEncodingError EncodeCandidate(WebPPicture* const sub_frame,
|
|
|
|
|
const WebPFrameRect* const rect,
|
|
|
|
|
const WebPMuxFrameInfo* const info,
|
|
|
|
|
const WebPConfig* const config,
|
|
|
|
|
int use_blending,
|
|
|
|
|
Candidate* const candidate) {
|
2014-06-11 23:26:47 +02:00
|
|
|
|
WebPEncodingError error_code = VP8_ENC_OK;
|
2014-10-23 00:43:26 +02:00
|
|
|
|
assert(candidate != NULL);
|
|
|
|
|
memset(candidate, 0, sizeof(*candidate));
|
|
|
|
|
|
|
|
|
|
// Set frame rect and info.
|
|
|
|
|
candidate->rect = *rect;
|
|
|
|
|
candidate->info = *info;
|
|
|
|
|
candidate->info.x_offset = rect->x_offset;
|
|
|
|
|
candidate->info.y_offset = rect->y_offset;
|
|
|
|
|
candidate->info.blend_method =
|
|
|
|
|
use_blending ? WEBP_MUX_BLEND : WEBP_MUX_NO_BLEND;
|
|
|
|
|
|
|
|
|
|
// Encode picture.
|
|
|
|
|
WebPMemoryWriterInit(&candidate->mem);
|
|
|
|
|
|
|
|
|
|
if (!EncodeFrame(config, sub_frame, &candidate->mem)) {
|
|
|
|
|
error_code = sub_frame->error_code;
|
|
|
|
|
goto Err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
candidate->evaluate = 1;
|
|
|
|
|
return error_code;
|
|
|
|
|
|
|
|
|
|
Err:
|
|
|
|
|
#if WEBP_ENCODER_ABI_VERSION > 0x0203
|
|
|
|
|
WebPMemoryWriterClear(&candidate->mem);
|
|
|
|
|
#else
|
|
|
|
|
free(candidate->mem.mem);
|
|
|
|
|
memset(&candidate->mem, 0, sizeof(candidate->mem));
|
|
|
|
|
#endif
|
|
|
|
|
return error_code;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Returns cached frame at given 'position' index.
|
|
|
|
|
static EncodedFrame* CacheGetFrame(const WebPFrameCache* const cache,
|
|
|
|
|
size_t position) {
|
|
|
|
|
assert(cache->start + position < cache->size);
|
|
|
|
|
return &cache->encoded_frames[cache->start + position];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Sets dispose method of the previous frame to be 'dispose_method'.
|
|
|
|
|
static void SetPreviousDisposeMethod(WebPFrameCache* const cache,
|
|
|
|
|
WebPMuxAnimDispose dispose_method) {
|
|
|
|
|
const size_t position = cache->count - 2;
|
|
|
|
|
EncodedFrame* const prev_enc_frame = CacheGetFrame(cache, position);
|
|
|
|
|
assert(cache->count >= 2); // As current and previous frames are in cache.
|
|
|
|
|
|
|
|
|
|
if (cache->prev_candidate_undecided) {
|
|
|
|
|
assert(dispose_method == WEBP_MUX_DISPOSE_NONE);
|
|
|
|
|
prev_enc_frame->sub_frame.dispose_method = dispose_method;
|
|
|
|
|
prev_enc_frame->key_frame.dispose_method = dispose_method;
|
|
|
|
|
} else {
|
|
|
|
|
WebPMuxFrameInfo* const prev_info = prev_enc_frame->is_key_frame
|
|
|
|
|
? &prev_enc_frame->key_frame
|
|
|
|
|
: &prev_enc_frame->sub_frame;
|
|
|
|
|
prev_info->dispose_method = dispose_method;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
LL_DISP_NONE = 0,
|
|
|
|
|
LL_DISP_BG,
|
|
|
|
|
LOSSY_DISP_NONE,
|
|
|
|
|
LOSSY_DISP_BG,
|
|
|
|
|
CANDIDATE_COUNT
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Generates candidates for a given dispose method given pre-filled 'rect'
|
|
|
|
|
// and 'sub_frame'.
|
|
|
|
|
static WebPEncodingError GenerateCandidates(
|
|
|
|
|
WebPFrameCache* const cache, Candidate candidates[CANDIDATE_COUNT],
|
|
|
|
|
WebPMuxAnimDispose dispose_method, int is_lossless, int is_key_frame,
|
|
|
|
|
const WebPFrameRect* const rect, WebPPicture* sub_frame,
|
|
|
|
|
const WebPMuxFrameInfo* const info,
|
|
|
|
|
const WebPConfig* const config_ll, const WebPConfig* const config_lossy) {
|
|
|
|
|
WebPEncodingError error_code = VP8_ENC_OK;
|
|
|
|
|
const int is_dispose_none = (dispose_method == WEBP_MUX_DISPOSE_NONE);
|
|
|
|
|
Candidate* const candidate_ll =
|
|
|
|
|
is_dispose_none ? &candidates[LL_DISP_NONE] : &candidates[LL_DISP_BG];
|
|
|
|
|
Candidate* const candidate_lossy = is_dispose_none
|
|
|
|
|
? &candidates[LOSSY_DISP_NONE]
|
|
|
|
|
: &candidates[LOSSY_DISP_BG];
|
|
|
|
|
const WebPPicture* const prev_canvas =
|
|
|
|
|
is_dispose_none ? &cache->prev_canvas : &cache->prev_canvas_disposed;
|
|
|
|
|
WebPPicture* const curr_canvas = &cache->curr_canvas;
|
|
|
|
|
WebPPicture* const curr_canvas_tmp = &cache->curr_canvas_tmp;
|
|
|
|
|
const int use_blending =
|
|
|
|
|
!is_key_frame &&
|
|
|
|
|
IsBlendingPossible(prev_canvas, curr_canvas, rect);
|
|
|
|
|
int curr_canvas_saved = 0; // If 'curr_canvas' is saved in 'curr_canvas_tmp'.
|
|
|
|
|
|
|
|
|
|
// Pick candidates to be tried.
|
|
|
|
|
if (!cache->allow_mixed) {
|
|
|
|
|
candidate_ll->evaluate = is_lossless;
|
|
|
|
|
candidate_lossy->evaluate = !is_lossless;
|
2013-11-18 03:04:07 +01:00
|
|
|
|
} else { // Use a heuristic for trying lossless and/or lossy compression.
|
|
|
|
|
const int num_colors = GetColorCount(sub_frame);
|
2014-10-23 00:43:26 +02:00
|
|
|
|
candidate_ll->evaluate = (num_colors < MAX_COLORS_LOSSLESS);
|
|
|
|
|
candidate_lossy->evaluate = (num_colors >= MIN_COLORS_LOSSY);
|
2013-11-18 03:04:07 +01:00
|
|
|
|
}
|
2013-10-14 23:39:46 +02:00
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// Generate candidates.
|
|
|
|
|
if (candidate_ll->evaluate) {
|
|
|
|
|
if (use_blending) {
|
|
|
|
|
CopyPixels(curr_canvas, curr_canvas_tmp); // save
|
|
|
|
|
curr_canvas_saved = 1;
|
|
|
|
|
IncreaseTransparency(prev_canvas, rect, curr_canvas);
|
|
|
|
|
}
|
|
|
|
|
error_code = EncodeCandidate(sub_frame, rect, info, config_ll, use_blending,
|
|
|
|
|
candidate_ll);
|
|
|
|
|
if (error_code != VP8_ENC_OK) return error_code;
|
|
|
|
|
if (use_blending) {
|
|
|
|
|
CopyPixels(curr_canvas_tmp, curr_canvas); // restore
|
2013-11-18 03:04:07 +01:00
|
|
|
|
}
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
2014-10-23 00:43:26 +02:00
|
|
|
|
if (candidate_lossy->evaluate) {
|
2013-11-18 03:04:07 +01:00
|
|
|
|
if (!is_key_frame) {
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// For lossy compression of a frame, it's better to:
|
|
|
|
|
// * Replace transparent pixels of 'curr' with actual RGB values,
|
|
|
|
|
// whenever possible, and
|
|
|
|
|
// * Replace similar blocks of pixels by a transparent block.
|
|
|
|
|
if (!curr_canvas_saved) { // save if not already done so.
|
|
|
|
|
CopyPixels(curr_canvas, curr_canvas_tmp);
|
|
|
|
|
}
|
|
|
|
|
FlattenSimilarBlocks(prev_canvas, rect, curr_canvas);
|
2013-11-18 03:04:07 +01:00
|
|
|
|
}
|
2014-10-23 00:43:26 +02:00
|
|
|
|
error_code = EncodeCandidate(sub_frame, rect, info, config_lossy,
|
|
|
|
|
use_blending, candidate_lossy);
|
|
|
|
|
if (error_code != VP8_ENC_OK) return error_code;
|
|
|
|
|
if (!is_key_frame) {
|
|
|
|
|
CopyPixels(curr_canvas_tmp, curr_canvas); // restore
|
2013-11-18 03:04:07 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-10-23 00:43:26 +02:00
|
|
|
|
return error_code;
|
|
|
|
|
}
|
2013-11-18 03:04:07 +01:00
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// Pick the candidate encoded frame with smallest size and release other
|
|
|
|
|
// candidates.
|
|
|
|
|
// TODO(later): Perhaps a rough SSIM/PSNR produced by the encoder should
|
|
|
|
|
// also be a criteria, in addition to sizes.
|
|
|
|
|
static void PickBestCandidate(WebPFrameCache* const cache,
|
|
|
|
|
Candidate* const candidates, int is_key_frame,
|
|
|
|
|
EncodedFrame* const encoded_frame) {
|
|
|
|
|
int i;
|
|
|
|
|
int best_idx = -1;
|
|
|
|
|
size_t best_size = ~0;
|
|
|
|
|
for (i = 0; i < CANDIDATE_COUNT; ++i) {
|
|
|
|
|
if (candidates[i].evaluate) {
|
|
|
|
|
const size_t candidate_size = candidates[i].mem.size;
|
|
|
|
|
if (candidate_size < best_size) {
|
|
|
|
|
best_idx = i;
|
|
|
|
|
best_size = candidate_size;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert(best_idx != -1);
|
|
|
|
|
for (i = 0; i < CANDIDATE_COUNT; ++i) {
|
|
|
|
|
if (candidates[i].evaluate) {
|
|
|
|
|
if (i == best_idx) {
|
|
|
|
|
WebPMuxFrameInfo* const dst = is_key_frame
|
|
|
|
|
? &encoded_frame->key_frame
|
|
|
|
|
: &encoded_frame->sub_frame;
|
|
|
|
|
*dst = candidates[i].info;
|
|
|
|
|
GetEncodedData(&candidates[i].mem, &dst->bitstream);
|
|
|
|
|
if (!is_key_frame) {
|
|
|
|
|
// Note: Previous dispose method only matters for non-keyframes.
|
|
|
|
|
// Also, we don't want to modify previous dispose method that was
|
|
|
|
|
// selected when a non key-frame was assumed.
|
|
|
|
|
const WebPMuxAnimDispose prev_dispose_method =
|
|
|
|
|
(best_idx == LL_DISP_NONE || best_idx == LOSSY_DISP_NONE)
|
|
|
|
|
? WEBP_MUX_DISPOSE_NONE
|
|
|
|
|
: WEBP_MUX_DISPOSE_BACKGROUND;
|
|
|
|
|
SetPreviousDisposeMethod(cache, prev_dispose_method);
|
|
|
|
|
}
|
|
|
|
|
cache->prev_webp_rect = candidates[i].rect; // save for next frame.
|
|
|
|
|
} else {
|
2014-10-13 13:46:13 +02:00
|
|
|
|
#if WEBP_ENCODER_ABI_VERSION > 0x0203
|
2014-10-23 00:43:26 +02:00
|
|
|
|
WebPMemoryWriterClear(&candidates[i].mem);
|
2014-07-23 05:24:59 +02:00
|
|
|
|
#else
|
2014-10-23 00:43:26 +02:00
|
|
|
|
free(candidates[i].mem.mem);
|
|
|
|
|
memset(&candidates[i].mem, 0, sizeof(candidates[i].mem));
|
2014-07-23 05:24:59 +02:00
|
|
|
|
#endif
|
2014-10-23 00:43:26 +02:00
|
|
|
|
candidates[i].evaluate = 0;
|
|
|
|
|
}
|
2013-11-18 03:04:07 +01:00
|
|
|
|
}
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
2014-10-23 00:43:26 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Depending on the configuration, tries different compressions
|
|
|
|
|
// (lossy/lossless), dispose methods, blending methods etc to encode the current
|
|
|
|
|
// frame and outputs the best one in 'encoded_frame'.
|
|
|
|
|
static WebPEncodingError SetFrame(WebPFrameCache* const cache,
|
|
|
|
|
const WebPConfig* const config,
|
|
|
|
|
const WebPMuxFrameInfo* const info,
|
|
|
|
|
const WebPFrameRect* const orig_rect,
|
|
|
|
|
int is_key_frame,
|
|
|
|
|
EncodedFrame* const encoded_frame) {
|
|
|
|
|
int i;
|
|
|
|
|
WebPEncodingError error_code = VP8_ENC_OK;
|
|
|
|
|
const WebPPicture* const prev_canvas = &cache->prev_canvas;
|
|
|
|
|
WebPPicture* const prev_canvas_disposed = &cache->prev_canvas_disposed;
|
|
|
|
|
WebPPicture* const curr_canvas = &cache->curr_canvas;
|
|
|
|
|
Candidate candidates[CANDIDATE_COUNT];
|
|
|
|
|
const int is_lossless = config->lossless;
|
|
|
|
|
|
|
|
|
|
int try_dispose_none = 1; // Default.
|
|
|
|
|
WebPFrameRect rect_none;
|
|
|
|
|
WebPPicture sub_frame_none;
|
|
|
|
|
|
|
|
|
|
// If current frame is a key-frame, dispose method of previous frame doesn't
|
|
|
|
|
// matter, so we don't try dispose to background.
|
|
|
|
|
// Also, if keyframe insertion is on, and previous frame could be picked as
|
|
|
|
|
// either a sub-frame or a keyframe, then we can't be sure about what frame
|
|
|
|
|
// rectangle would be disposed. In that case too, we don't try dispose to
|
|
|
|
|
// background.
|
|
|
|
|
const int dispose_bg_possible =
|
|
|
|
|
!is_key_frame && !cache->prev_candidate_undecided;
|
|
|
|
|
int try_dispose_bg = 0; // Default.
|
|
|
|
|
WebPFrameRect rect_bg;
|
|
|
|
|
WebPPicture sub_frame_bg;
|
|
|
|
|
|
|
|
|
|
WebPConfig config_ll = *config;
|
|
|
|
|
WebPConfig config_lossy = *config;
|
|
|
|
|
config_ll.lossless = 1;
|
|
|
|
|
config_lossy.lossless = 0;
|
|
|
|
|
|
|
|
|
|
if (!WebPPictureInit(&sub_frame_none) || !WebPPictureInit(&sub_frame_bg)) {
|
|
|
|
|
return VP8_ENC_ERROR_INVALID_CONFIGURATION;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < CANDIDATE_COUNT; ++i) {
|
|
|
|
|
candidates[i].evaluate = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Change-rectangle assuming previous frame was DISPOSE_NONE.
|
|
|
|
|
GetSubRect(prev_canvas, curr_canvas, orig_rect, is_key_frame,
|
|
|
|
|
&rect_none, &sub_frame_none);
|
|
|
|
|
|
|
|
|
|
if (dispose_bg_possible) {
|
|
|
|
|
// Change-rectangle assuming previous frame was DISPOSE_BACKGROUND.
|
|
|
|
|
CopyPixels(prev_canvas, prev_canvas_disposed);
|
|
|
|
|
DisposeFrameRectangle(WEBP_MUX_DISPOSE_BACKGROUND, &cache->prev_webp_rect,
|
|
|
|
|
prev_canvas_disposed);
|
|
|
|
|
GetSubRect(prev_canvas_disposed, curr_canvas, orig_rect, is_key_frame,
|
|
|
|
|
&rect_bg, &sub_frame_bg);
|
|
|
|
|
|
|
|
|
|
if (RectArea(&rect_bg) < RectArea(&rect_none)) {
|
|
|
|
|
try_dispose_bg = 1; // Pick DISPOSE_BACKGROUND.
|
|
|
|
|
try_dispose_none = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (try_dispose_none) {
|
|
|
|
|
error_code = GenerateCandidates(
|
|
|
|
|
cache, candidates, WEBP_MUX_DISPOSE_NONE, is_lossless, is_key_frame,
|
|
|
|
|
&rect_none, &sub_frame_none, info, &config_ll, &config_lossy);
|
|
|
|
|
if (error_code != VP8_ENC_OK) goto Err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (try_dispose_bg) {
|
|
|
|
|
assert(!cache->is_first_frame);
|
|
|
|
|
assert(dispose_bg_possible);
|
|
|
|
|
error_code = GenerateCandidates(
|
|
|
|
|
cache, candidates, WEBP_MUX_DISPOSE_BACKGROUND, is_lossless,
|
|
|
|
|
is_key_frame, &rect_bg, &sub_frame_bg, info, &config_ll, &config_lossy);
|
|
|
|
|
if (error_code != VP8_ENC_OK) goto Err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PickBestCandidate(cache, candidates, is_key_frame, encoded_frame);
|
|
|
|
|
|
|
|
|
|
goto End;
|
2013-11-18 03:04:07 +01:00
|
|
|
|
|
|
|
|
|
Err:
|
2014-10-23 00:43:26 +02:00
|
|
|
|
for (i = 0; i < CANDIDATE_COUNT; ++i) {
|
|
|
|
|
if (candidates[i].evaluate) {
|
2014-10-13 13:46:13 +02:00
|
|
|
|
#if WEBP_ENCODER_ABI_VERSION > 0x0203
|
2014-10-23 00:43:26 +02:00
|
|
|
|
WebPMemoryWriterClear(&candidates[i].mem);
|
2014-07-23 05:24:59 +02:00
|
|
|
|
#else
|
2014-10-23 00:43:26 +02:00
|
|
|
|
free(candidates[i].mem.mem);
|
|
|
|
|
memset(&candidates[i].mem, 0, sizeof(candidates[i].mem));
|
2014-07-23 05:24:59 +02:00
|
|
|
|
#endif
|
2014-10-23 00:43:26 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
End:
|
|
|
|
|
WebPPictureFree(&sub_frame_none);
|
|
|
|
|
WebPPictureFree(&sub_frame_bg);
|
2014-06-11 23:26:47 +02:00
|
|
|
|
return error_code;
|
2013-11-18 03:04:07 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#undef MIN_COLORS_LOSSY
|
|
|
|
|
#undef MAX_COLORS_LOSSLESS
|
|
|
|
|
|
|
|
|
|
// Calculate the penalty incurred if we encode given frame as a key frame
|
|
|
|
|
// instead of a sub-frame.
|
|
|
|
|
static int64_t KeyFramePenalty(const EncodedFrame* const encoded_frame) {
|
|
|
|
|
return ((int64_t)encoded_frame->key_frame.bitstream.size -
|
|
|
|
|
encoded_frame->sub_frame.bitstream.size);
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int WebPFrameCacheAddFrame(WebPFrameCache* const cache,
|
|
|
|
|
const WebPConfig* const config,
|
2014-06-12 08:07:45 +02:00
|
|
|
|
const WebPFrameRect* const orig_rect_ptr,
|
2013-10-14 23:39:46 +02:00
|
|
|
|
WebPPicture* const frame,
|
|
|
|
|
WebPMuxFrameInfo* const info) {
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// Initialize.
|
2013-10-14 23:39:46 +02:00
|
|
|
|
int ok = 0;
|
2014-06-11 23:26:47 +02:00
|
|
|
|
WebPEncodingError error_code = VP8_ENC_OK;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
WebPPicture* const prev_canvas = &cache->prev_canvas;
|
2014-10-23 00:43:26 +02:00
|
|
|
|
WebPPicture* const curr_canvas = &cache->curr_canvas;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
const size_t position = cache->count;
|
|
|
|
|
EncodedFrame* const encoded_frame = CacheGetFrame(cache, position);
|
2014-06-12 08:07:45 +02:00
|
|
|
|
WebPFrameRect orig_rect;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
assert(position < cache->size);
|
|
|
|
|
|
2014-06-11 23:26:47 +02:00
|
|
|
|
if (frame == NULL || info == NULL) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// As we are encoding (part of) 'curr_canvas', and not 'frame' directly, make
|
|
|
|
|
// sure the progress is still reported back.
|
|
|
|
|
curr_canvas->progress_hook = frame->progress_hook;
|
|
|
|
|
curr_canvas->user_data = frame->user_data;
|
|
|
|
|
curr_canvas->stats = frame->stats;
|
|
|
|
|
|
2014-06-12 08:07:45 +02:00
|
|
|
|
if (orig_rect_ptr == NULL) {
|
|
|
|
|
orig_rect.width = frame->width;
|
|
|
|
|
orig_rect.height = frame->height;
|
|
|
|
|
orig_rect.x_offset = 0;
|
|
|
|
|
orig_rect.y_offset = 0;
|
2014-06-11 23:26:47 +02:00
|
|
|
|
} else {
|
2014-06-12 08:07:45 +02:00
|
|
|
|
orig_rect = *orig_rect_ptr;
|
2014-06-11 23:26:47 +02:00
|
|
|
|
}
|
2013-10-14 23:39:46 +02:00
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// Main frame addition.
|
2013-10-14 23:39:46 +02:00
|
|
|
|
++cache->count;
|
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
if (cache->is_first_frame) {
|
|
|
|
|
// 'curr_canvas' is same as 'frame'.
|
|
|
|
|
CopyPixels(frame, curr_canvas);
|
2013-10-14 23:39:46 +02:00
|
|
|
|
// Add this as a key frame.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// Note: we use original rectangle as-is for the first frame.
|
|
|
|
|
error_code = SetFrame(cache, config, info, &orig_rect, 1, encoded_frame);
|
2014-06-11 23:26:47 +02:00
|
|
|
|
if (error_code != VP8_ENC_OK) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
goto End;
|
|
|
|
|
}
|
2014-10-23 00:43:26 +02:00
|
|
|
|
assert(position == 0 && cache->count == 1);
|
|
|
|
|
encoded_frame->is_key_frame = 1;
|
|
|
|
|
cache->flush_count = 0;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
cache->count_since_key_frame = 0;
|
2014-10-23 00:43:26 +02:00
|
|
|
|
cache->prev_candidate_undecided = 0;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
} else {
|
2014-10-23 00:43:26 +02:00
|
|
|
|
// Store previous canvas.
|
|
|
|
|
CopyPixels(curr_canvas, prev_canvas);
|
|
|
|
|
// Create curr_canvas:
|
|
|
|
|
// * Start with disposed previous canvas.
|
|
|
|
|
// * Then blend 'frame' onto it.
|
|
|
|
|
DisposeFrameRectangle(cache->prev_orig_dispose, &cache->prev_orig_rect,
|
|
|
|
|
curr_canvas);
|
|
|
|
|
BlendPixels(frame, &orig_rect, curr_canvas);
|
|
|
|
|
|
2013-10-14 23:39:46 +02:00
|
|
|
|
++cache->count_since_key_frame;
|
|
|
|
|
if (cache->count_since_key_frame <= cache->kmin) {
|
|
|
|
|
// Add this as a frame rectangle.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
error_code = SetFrame(cache, config, info, NULL, 0, encoded_frame);
|
2014-06-11 23:26:47 +02:00
|
|
|
|
if (error_code != VP8_ENC_OK) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
goto End;
|
|
|
|
|
}
|
2014-10-23 00:43:26 +02:00
|
|
|
|
encoded_frame->is_key_frame = 0;
|
|
|
|
|
cache->flush_count = cache->count - 1;
|
|
|
|
|
cache->prev_candidate_undecided = 0;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
} else {
|
|
|
|
|
WebPMuxFrameInfo full_image_info;
|
|
|
|
|
int64_t curr_delta;
|
|
|
|
|
|
|
|
|
|
// Add frame rectangle to cache.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
error_code = SetFrame(cache, config, info, NULL, 0, encoded_frame);
|
2014-06-11 23:26:47 +02:00
|
|
|
|
if (error_code != VP8_ENC_OK) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
goto End;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add key frame to cache, too.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
full_image_info = *info;
|
|
|
|
|
full_image_info.x_offset = 0;
|
|
|
|
|
full_image_info.y_offset = 0;
|
|
|
|
|
error_code =
|
|
|
|
|
SetFrame(cache, config, &full_image_info, NULL, 1, encoded_frame);
|
2014-06-11 23:26:47 +02:00
|
|
|
|
if (error_code != VP8_ENC_OK) goto End;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
|
|
|
|
|
// Analyze size difference of the two variants.
|
|
|
|
|
curr_delta = KeyFramePenalty(encoded_frame);
|
|
|
|
|
if (curr_delta <= cache->best_delta) { // Pick this as keyframe.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
if (cache->keyframe != KEYFRAME_NONE) {
|
|
|
|
|
EncodedFrame* const old_keyframe =
|
|
|
|
|
CacheGetFrame(cache, cache->keyframe);
|
|
|
|
|
assert(old_keyframe->is_key_frame);
|
|
|
|
|
old_keyframe->is_key_frame = 0;
|
|
|
|
|
}
|
|
|
|
|
encoded_frame->is_key_frame = 1;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
cache->keyframe = position;
|
|
|
|
|
cache->best_delta = curr_delta;
|
|
|
|
|
cache->flush_count = cache->count - 1; // We can flush previous frames.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
} else {
|
|
|
|
|
encoded_frame->is_key_frame = 0;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
|
|
|
|
if (cache->count_since_key_frame == cache->kmax) {
|
2014-10-23 00:43:26 +02:00
|
|
|
|
cache->flush_count = cache->count - 1;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
cache->count_since_key_frame = 0;
|
|
|
|
|
}
|
2014-10-23 00:43:26 +02:00
|
|
|
|
cache->prev_candidate_undecided = 1;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
DisposeFullFrame(info->dispose_method, frame);
|
2013-10-14 23:39:46 +02:00
|
|
|
|
|
|
|
|
|
cache->is_first_frame = 0;
|
2014-10-23 00:43:26 +02:00
|
|
|
|
cache->prev_orig_dispose = info->dispose_method;
|
|
|
|
|
cache->prev_orig_rect = orig_rect;
|
2013-10-14 23:39:46 +02:00
|
|
|
|
ok = 1;
|
|
|
|
|
|
|
|
|
|
End:
|
2013-11-18 03:04:07 +01:00
|
|
|
|
if (!ok) {
|
|
|
|
|
FrameRelease(encoded_frame);
|
|
|
|
|
--cache->count; // We reset the count, as the frame addition failed.
|
|
|
|
|
}
|
2014-06-11 23:26:47 +02:00
|
|
|
|
frame->error_code = error_code; // report error_code
|
|
|
|
|
assert(ok || error_code != VP8_ENC_OK);
|
2013-10-14 23:39:46 +02:00
|
|
|
|
return ok;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
WebPMuxError WebPFrameCacheFlush(WebPFrameCache* const cache, int verbose,
|
|
|
|
|
WebPMux* const mux) {
|
|
|
|
|
while (cache->flush_count > 0) {
|
|
|
|
|
WebPMuxFrameInfo* info;
|
|
|
|
|
WebPMuxError err;
|
|
|
|
|
EncodedFrame* const curr = CacheGetFrame(cache, 0);
|
|
|
|
|
// Pick frame or full canvas.
|
2014-10-23 00:43:26 +02:00
|
|
|
|
if (curr->is_key_frame) {
|
2013-10-14 23:39:46 +02:00
|
|
|
|
info = &curr->key_frame;
|
2014-10-23 00:43:26 +02:00
|
|
|
|
if (cache->keyframe == 0) {
|
|
|
|
|
cache->keyframe = KEYFRAME_NONE;
|
|
|
|
|
cache->best_delta = DELTA_INFINITY;
|
|
|
|
|
}
|
2013-10-14 23:39:46 +02:00
|
|
|
|
} else {
|
|
|
|
|
info = &curr->sub_frame;
|
|
|
|
|
}
|
|
|
|
|
// Add to mux.
|
|
|
|
|
err = WebPMuxPushFrame(mux, info, 1);
|
|
|
|
|
if (err != WEBP_MUX_OK) return err;
|
|
|
|
|
if (verbose) {
|
|
|
|
|
printf("Added frame. offset:%d,%d duration:%d dispose:%d blend:%d\n",
|
|
|
|
|
info->x_offset, info->y_offset, info->duration,
|
|
|
|
|
info->dispose_method, info->blend_method);
|
|
|
|
|
}
|
|
|
|
|
FrameRelease(curr);
|
|
|
|
|
++cache->start;
|
|
|
|
|
--cache->flush_count;
|
|
|
|
|
--cache->count;
|
|
|
|
|
if (cache->keyframe != KEYFRAME_NONE) --cache->keyframe;
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-23 00:43:26 +02:00
|
|
|
|
if (cache->count == 1 && cache->start != 0) {
|
|
|
|
|
// Move cache->start to index 0.
|
|
|
|
|
const int cache_start_tmp = (int)cache->start;
|
|
|
|
|
EncodedFrame temp = cache->encoded_frames[0];
|
|
|
|
|
cache->encoded_frames[0] = cache->encoded_frames[cache_start_tmp];
|
|
|
|
|
cache->encoded_frames[cache_start_tmp] = temp;
|
|
|
|
|
FrameRelease(&cache->encoded_frames[cache_start_tmp]);
|
|
|
|
|
cache->start = 0;
|
|
|
|
|
}
|
2013-10-14 23:39:46 +02:00
|
|
|
|
return WEBP_MUX_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
WebPMuxError WebPFrameCacheFlushAll(WebPFrameCache* const cache, int verbose,
|
|
|
|
|
WebPMux* const mux) {
|
|
|
|
|
cache->flush_count = cache->count; // Force flushing of all frames.
|
|
|
|
|
return WebPFrameCacheFlush(cache, verbose, mux);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------
|