2012-01-06 23:49:06 +01:00
|
|
|
// Copyright 2011 Google Inc. All Rights Reserved.
|
2011-02-19 08:33:46 +01:00
|
|
|
//
|
2013-06-07 08:05:58 +02:00
|
|
|
// 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.
|
2011-02-19 08:33:46 +01:00
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// WebP encoder: internal header.
|
|
|
|
//
|
|
|
|
// Author: Skal (pascal.massimino@gmail.com)
|
|
|
|
|
|
|
|
#ifndef WEBP_ENC_VP8ENCI_H_
|
|
|
|
#define WEBP_ENC_VP8ENCI_H_
|
|
|
|
|
2012-01-06 23:38:34 +01:00
|
|
|
#include <string.h> // for memcpy()
|
2011-05-03 02:19:00 +02:00
|
|
|
#include "../webp/encode.h"
|
2011-09-02 23:30:08 +02:00
|
|
|
#include "../dsp/dsp.h"
|
2011-09-07 11:26:35 +02:00
|
|
|
#include "../utils/bit_writer.h"
|
2013-03-01 01:21:34 +01:00
|
|
|
#include "../utils/thread.h"
|
2011-02-19 08:33:46 +01:00
|
|
|
|
|
|
|
#if defined(__cplusplus) || defined(c_plusplus)
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2011-08-25 23:22:32 +02:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-19 08:33:46 +01:00
|
|
|
// Various defines and enums
|
|
|
|
|
2011-03-25 00:17:10 +01:00
|
|
|
// version numbers
|
|
|
|
#define ENC_MAJ_VERSION 0
|
2013-03-16 22:08:14 +01:00
|
|
|
#define ENC_MIN_VERSION 3
|
2013-06-12 23:58:32 +02:00
|
|
|
#define ENC_REV_VERSION 1
|
2011-03-25 00:17:10 +01:00
|
|
|
|
2011-02-19 08:33:46 +01:00
|
|
|
// intra prediction modes
|
|
|
|
enum { B_DC_PRED = 0, // 4x4 modes
|
|
|
|
B_TM_PRED = 1,
|
|
|
|
B_VE_PRED = 2,
|
|
|
|
B_HE_PRED = 3,
|
|
|
|
B_RD_PRED = 4,
|
|
|
|
B_VR_PRED = 5,
|
|
|
|
B_LD_PRED = 6,
|
|
|
|
B_VL_PRED = 7,
|
|
|
|
B_HD_PRED = 8,
|
|
|
|
B_HU_PRED = 9,
|
|
|
|
NUM_BMODES = B_HU_PRED + 1 - B_DC_PRED, // = 10
|
|
|
|
|
|
|
|
// Luma16 or UV modes
|
|
|
|
DC_PRED = B_DC_PRED, V_PRED = B_VE_PRED,
|
2013-02-26 14:22:44 +01:00
|
|
|
H_PRED = B_HE_PRED, TM_PRED = B_TM_PRED,
|
|
|
|
NUM_PRED_MODES = 4
|
2011-02-19 08:33:46 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
enum { NUM_MB_SEGMENTS = 4,
|
|
|
|
MAX_NUM_PARTITIONS = 8,
|
|
|
|
NUM_TYPES = 4, // 0: i16-AC, 1: i16-DC, 2:chroma-AC, 3:i4-AC
|
|
|
|
NUM_BANDS = 8,
|
|
|
|
NUM_CTX = 3,
|
|
|
|
NUM_PROBAS = 11,
|
2013-03-24 16:47:41 +01:00
|
|
|
MAX_LF_LEVELS = 64, // Maximum loop filter level
|
|
|
|
MAX_VARIABLE_LEVEL = 67, // last (inclusive) level with variable cost
|
|
|
|
MAX_LEVEL = 2047 // max level (note: max codable is 2047 + 67)
|
2011-02-19 08:33:46 +01:00
|
|
|
};
|
|
|
|
|
2013-02-26 11:20:59 +01:00
|
|
|
typedef enum { // Rate-distortion optimization levels
|
|
|
|
RD_OPT_NONE = 0, // no rd-opt
|
|
|
|
RD_OPT_BASIC = 1, // basic scoring (no trellis)
|
|
|
|
RD_OPT_TRELLIS = 2, // perform trellis-quant on the final decision only
|
|
|
|
RD_OPT_TRELLIS_ALL = 3 // trellis-quant for every scoring (much slower)
|
|
|
|
} VP8RDLevel;
|
|
|
|
|
2011-02-19 08:33:46 +01:00
|
|
|
// YUV-cache parameters. Cache is 16-pixels wide.
|
|
|
|
// The original or reconstructed samples can be accessed using VP8Scan[]
|
|
|
|
// The predicted blocks can be accessed using offsets to yuv_p_ and
|
|
|
|
// the arrays VP8*ModeOffsets[];
|
|
|
|
// +----+ YUV Samples area. See VP8Scan[] for accessing the blocks.
|
2013-08-31 23:38:11 +02:00
|
|
|
// Y_OFF |YYYY| <- original samples ('yuv_in_')
|
2011-02-19 08:33:46 +01:00
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// U_OFF |UUVV| V_OFF (=U_OFF + 8)
|
|
|
|
// |UUVV|
|
|
|
|
// +----+
|
|
|
|
// Y_OFF |YYYY| <- compressed/decoded samples ('yuv_out_')
|
|
|
|
// |YYYY| There are two buffers like this ('yuv_out_'/'yuv_out2_')
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// U_OFF |UUVV| V_OFF
|
|
|
|
// |UUVV|
|
|
|
|
// x2 (for yuv_out2_)
|
|
|
|
// +----+ Prediction area ('yuv_p_', size = PRED_SIZE)
|
|
|
|
// I16DC16 |YYYY| Intra16 predictions (16x16 block each)
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// I16TM16 |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// I16VE16 |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// I16HE16 |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// |YYYY|
|
|
|
|
// +----+ Chroma U/V predictions (16x8 block each)
|
|
|
|
// C8DC8 |UUVV|
|
|
|
|
// |UUVV|
|
|
|
|
// C8TM8 |UUVV|
|
|
|
|
// |UUVV|
|
|
|
|
// C8VE8 |UUVV|
|
|
|
|
// |UUVV|
|
|
|
|
// C8HE8 |UUVV|
|
|
|
|
// |UUVV|
|
|
|
|
// +----+ Intra 4x4 predictions (4x4 block each)
|
|
|
|
// |YYYY| I4DC4 I4TM4 I4VE4 I4HE4
|
|
|
|
// |YYYY| I4RD4 I4VR4 I4LD4 I4VL4
|
|
|
|
// |YY..| I4HD4 I4HU4 I4TMP
|
|
|
|
// +----+
|
|
|
|
#define BPS 16 // this is the common stride
|
|
|
|
#define Y_SIZE (BPS * 16)
|
|
|
|
#define UV_SIZE (BPS * 8)
|
|
|
|
#define YUV_SIZE (Y_SIZE + UV_SIZE)
|
|
|
|
#define PRED_SIZE (6 * 16 * BPS + 12 * BPS)
|
|
|
|
#define Y_OFF (0)
|
|
|
|
#define U_OFF (Y_SIZE)
|
|
|
|
#define V_OFF (U_OFF + 8)
|
|
|
|
#define ALIGN_CST 15
|
|
|
|
#define DO_ALIGN(PTR) ((uintptr_t)((PTR) + ALIGN_CST) & ~ALIGN_CST)
|
|
|
|
|
|
|
|
extern const int VP8Scan[16 + 4 + 4]; // in quant.c
|
|
|
|
extern const int VP8UVModeOffsets[4]; // in analyze.c
|
|
|
|
extern const int VP8I16ModeOffsets[4];
|
|
|
|
extern const int VP8I4ModeOffsets[NUM_BMODES];
|
|
|
|
|
|
|
|
// Layout of prediction blocks
|
|
|
|
// intra 16x16
|
|
|
|
#define I16DC16 (0 * 16 * BPS)
|
|
|
|
#define I16TM16 (1 * 16 * BPS)
|
|
|
|
#define I16VE16 (2 * 16 * BPS)
|
|
|
|
#define I16HE16 (3 * 16 * BPS)
|
|
|
|
// chroma 8x8, two U/V blocks side by side (hence: 16x8 each)
|
|
|
|
#define C8DC8 (4 * 16 * BPS)
|
|
|
|
#define C8TM8 (4 * 16 * BPS + 8 * BPS)
|
|
|
|
#define C8VE8 (5 * 16 * BPS)
|
|
|
|
#define C8HE8 (5 * 16 * BPS + 8 * BPS)
|
|
|
|
// intra 4x4
|
|
|
|
#define I4DC4 (6 * 16 * BPS + 0)
|
|
|
|
#define I4TM4 (6 * 16 * BPS + 4)
|
|
|
|
#define I4VE4 (6 * 16 * BPS + 8)
|
|
|
|
#define I4HE4 (6 * 16 * BPS + 12)
|
|
|
|
#define I4RD4 (6 * 16 * BPS + 4 * BPS + 0)
|
|
|
|
#define I4VR4 (6 * 16 * BPS + 4 * BPS + 4)
|
|
|
|
#define I4LD4 (6 * 16 * BPS + 4 * BPS + 8)
|
|
|
|
#define I4VL4 (6 * 16 * BPS + 4 * BPS + 12)
|
|
|
|
#define I4HD4 (6 * 16 * BPS + 8 * BPS + 0)
|
|
|
|
#define I4HU4 (6 * 16 * BPS + 8 * BPS + 4)
|
|
|
|
#define I4TMP (6 * 16 * BPS + 8 * BPS + 8)
|
|
|
|
|
|
|
|
typedef int64_t score_t; // type used for scores, rate, distortion
|
|
|
|
#define MAX_COST ((score_t)0x7fffffffffffffLL)
|
|
|
|
|
2011-03-24 01:33:05 +01:00
|
|
|
#define QFIX 17
|
|
|
|
#define BIAS(b) ((b) << (QFIX - 8))
|
|
|
|
// Fun fact: this is the _only_ line where we're actually being lossy and
|
|
|
|
// discarding bits.
|
2011-11-05 03:44:57 +01:00
|
|
|
static WEBP_INLINE int QUANTDIV(int n, int iQ, int B) {
|
2011-03-24 01:33:05 +01:00
|
|
|
return (n * iQ + B) >> QFIX;
|
|
|
|
}
|
|
|
|
|
2012-09-03 19:40:52 +02:00
|
|
|
// size of histogram used by CollectHistogram.
|
|
|
|
#define MAX_COEFF_THRESH 31
|
|
|
|
typedef struct VP8Histogram VP8Histogram;
|
|
|
|
struct VP8Histogram {
|
|
|
|
// TODO(skal): we only need to store the max_value and last_non_zero actually.
|
|
|
|
int distribution[MAX_COEFF_THRESH + 1];
|
|
|
|
};
|
|
|
|
|
2013-03-12 00:37:42 +01:00
|
|
|
// Uncomment the following to remove token-buffer code:
|
|
|
|
// #define DISABLE_TOKEN_BUFFER
|
|
|
|
|
2011-08-25 23:22:32 +02:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-19 08:33:46 +01:00
|
|
|
// Headers
|
|
|
|
|
2012-01-24 02:50:58 +01:00
|
|
|
typedef uint32_t proba_t; // 16b + 16b
|
2011-02-19 08:33:46 +01:00
|
|
|
typedef uint8_t ProbaArray[NUM_CTX][NUM_PROBAS];
|
2012-01-24 02:50:58 +01:00
|
|
|
typedef proba_t StatsArray[NUM_CTX][NUM_PROBAS];
|
2011-02-19 08:33:46 +01:00
|
|
|
typedef uint16_t CostArray[NUM_CTX][MAX_VARIABLE_LEVEL + 1];
|
|
|
|
typedef double LFStats[NUM_MB_SEGMENTS][MAX_LF_LEVELS]; // filter stats
|
|
|
|
|
|
|
|
typedef struct VP8Encoder VP8Encoder;
|
|
|
|
|
|
|
|
// segment features
|
|
|
|
typedef struct {
|
|
|
|
int num_segments_; // Actual number of segments. 1 segment only = unused.
|
|
|
|
int update_map_; // whether to update the segment map or not.
|
|
|
|
// must be 0 if there's only 1 segment.
|
|
|
|
int size_; // bit-cost for transmitting the segment map
|
|
|
|
} VP8SegmentHeader;
|
|
|
|
|
|
|
|
// Struct collecting all frame-persistent probabilities.
|
|
|
|
typedef struct {
|
|
|
|
uint8_t segments_[3]; // probabilities for segment tree
|
|
|
|
uint8_t skip_proba_; // final probability of being skipped.
|
|
|
|
ProbaArray coeffs_[NUM_TYPES][NUM_BANDS]; // 924 bytes
|
2012-01-24 02:50:58 +01:00
|
|
|
StatsArray stats_[NUM_TYPES][NUM_BANDS]; // 4224 bytes
|
2011-02-19 08:33:46 +01:00
|
|
|
CostArray level_cost_[NUM_TYPES][NUM_BANDS]; // 11.4k
|
2012-01-25 16:45:34 +01:00
|
|
|
int dirty_; // if true, need to call VP8CalculateLevelCosts()
|
2011-02-19 08:33:46 +01:00
|
|
|
int use_skip_proba_; // Note: we always use skip_proba for now.
|
2011-07-15 23:57:07 +02:00
|
|
|
int nb_skip_; // number of skipped blocks
|
2011-02-19 08:33:46 +01:00
|
|
|
} VP8Proba;
|
|
|
|
|
|
|
|
// Filter parameters. Not actually used in the code (we don't perform
|
|
|
|
// the in-loop filtering), but filled from user's config
|
|
|
|
typedef struct {
|
|
|
|
int simple_; // filtering type: 0=complex, 1=simple
|
|
|
|
int level_; // base filter level [0..63]
|
|
|
|
int sharpness_; // [0..7]
|
|
|
|
int i4x4_lf_delta_; // delta filter level for i4x4 relative to i16x16
|
|
|
|
} VP8FilterHeader;
|
|
|
|
|
2011-08-25 23:22:32 +02:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-19 08:33:46 +01:00
|
|
|
// Informations about the macroblocks.
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
// block type
|
2011-09-02 23:18:27 +02:00
|
|
|
unsigned int type_:2; // 0=i4x4, 1=i16x16
|
|
|
|
unsigned int uv_mode_:2;
|
|
|
|
unsigned int skip_:1;
|
|
|
|
unsigned int segment_:2;
|
2011-02-19 08:33:46 +01:00
|
|
|
uint8_t alpha_; // quantization-susceptibility
|
|
|
|
} VP8MBInfo;
|
|
|
|
|
2011-09-02 23:30:08 +02:00
|
|
|
typedef struct VP8Matrix {
|
2011-02-19 08:33:46 +01:00
|
|
|
uint16_t q_[16]; // quantizer steps
|
|
|
|
uint16_t iq_[16]; // reciprocals, fixed point.
|
|
|
|
uint16_t bias_[16]; // rounding bias
|
|
|
|
uint16_t zthresh_[16]; // value under which a coefficient is zeroed
|
|
|
|
uint16_t sharpen_[16]; // frequency boosters for slight sharpening
|
|
|
|
} VP8Matrix;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
VP8Matrix y1_, y2_, uv_; // quantization matrices
|
|
|
|
int alpha_; // quant-susceptibility, range [-127,127]. Zero is neutral.
|
|
|
|
// Lower values indicate a lower risk of blurriness.
|
|
|
|
int beta_; // filter-susceptibility, range [0,255].
|
|
|
|
int quant_; // final segment quantizer.
|
|
|
|
int fstrength_; // final in-loop filtering strength
|
2013-10-29 20:13:29 +01:00
|
|
|
int max_edge_; // max edge delta (for filtering strength)
|
|
|
|
int min_disto_; // minimum distortion required to trigger filtering record
|
2011-02-19 08:33:46 +01:00
|
|
|
// reactivities
|
|
|
|
int lambda_i16_, lambda_i4_, lambda_uv_;
|
|
|
|
int lambda_mode_, lambda_trellis_, tlambda_;
|
|
|
|
int lambda_trellis_i16_, lambda_trellis_i4_, lambda_trellis_uv_;
|
|
|
|
} VP8SegmentInfo;
|
|
|
|
|
|
|
|
// Handy transcient struct to accumulate score and info during RD-optimization
|
|
|
|
// and mode evaluation.
|
|
|
|
typedef struct {
|
2013-09-10 09:25:32 +02:00
|
|
|
score_t D, SD; // Distortion, spectral distortion
|
|
|
|
score_t H, R, score; // header bits, rate, score.
|
2011-02-19 08:33:46 +01:00
|
|
|
int16_t y_dc_levels[16]; // Quantized levels for luma-DC, luma-AC, chroma.
|
|
|
|
int16_t y_ac_levels[16][16];
|
|
|
|
int16_t uv_levels[4 + 4][16];
|
|
|
|
int mode_i16; // mode number for intra16 prediction
|
2012-01-27 12:17:41 +01:00
|
|
|
uint8_t modes_i4[16]; // mode numbers for intra4 predictions
|
2011-02-19 08:33:46 +01:00
|
|
|
int mode_uv; // mode number of chroma prediction
|
|
|
|
uint32_t nz; // non-zero blocks
|
|
|
|
} VP8ModeScore;
|
|
|
|
|
|
|
|
// Iterator structure to iterate through macroblocks, pointing to the
|
|
|
|
// right neighbouring data (samples, predictions, contexts, ...)
|
|
|
|
typedef struct {
|
|
|
|
int x_, y_; // current macroblock
|
|
|
|
int y_stride_, uv_stride_; // respective strides
|
2013-08-31 23:38:11 +02:00
|
|
|
uint8_t* yuv_in_; // input samples
|
|
|
|
uint8_t* yuv_out_; // output samples
|
|
|
|
uint8_t* yuv_out2_; // secondary buffer swapped with yuv_out_.
|
|
|
|
uint8_t* yuv_p_; // scratch buffer for prediction
|
2011-02-19 08:33:46 +01:00
|
|
|
VP8Encoder* enc_; // back-pointer
|
|
|
|
VP8MBInfo* mb_; // current macroblock
|
|
|
|
VP8BitWriter* bw_; // current bit-writer
|
|
|
|
uint8_t* preds_; // intra mode predictors (4x4 blocks)
|
|
|
|
uint32_t* nz_; // non-zero pattern
|
|
|
|
uint8_t i4_boundary_[37]; // 32+5 boundary samples needed by intra4x4
|
2011-06-21 00:38:14 +02:00
|
|
|
uint8_t* i4_top_; // pointer to the current top boundary sample
|
2011-02-19 08:33:46 +01:00
|
|
|
int i4_; // current intra4x4 mode being tested
|
|
|
|
int top_nz_[9]; // top-non-zero context.
|
|
|
|
int left_nz_[9]; // left-non-zero. left_nz[8] is independent.
|
|
|
|
uint64_t bit_count_[4][3]; // bit counters for coded levels.
|
|
|
|
uint64_t luma_bits_; // macroblock bit-cost for luma
|
|
|
|
uint64_t uv_bits_; // macroblock bit-cost for chroma
|
|
|
|
LFStats* lf_stats_; // filter stats (borrowed from enc_)
|
|
|
|
int do_trellis_; // if true, perform extra level optimisation
|
2013-07-27 02:08:37 +02:00
|
|
|
int count_down_; // number of mb still to be processed
|
2013-09-05 09:13:36 +02:00
|
|
|
int count_down0_; // starting counter value (for progress)
|
2011-12-01 11:24:50 +01:00
|
|
|
int percent0_; // saved initial progress percent
|
2013-07-27 02:08:37 +02:00
|
|
|
|
|
|
|
uint8_t* y_left_; // left luma samples (addressable from index -1 to 15).
|
|
|
|
uint8_t* u_left_; // left u samples (addressable from index -1 to 7)
|
|
|
|
uint8_t* v_left_; // left v samples (addressable from index -1 to 7)
|
2013-08-31 23:38:11 +02:00
|
|
|
|
|
|
|
uint8_t* y_top_; // top luma samples at position 'x_'
|
|
|
|
uint8_t* uv_top_; // top u/v samples at position 'x_', packed as 16 bytes
|
|
|
|
|
|
|
|
// memory for storing y/u/v_left_ and yuv_in_/out_*
|
|
|
|
uint8_t yuv_left_mem_[17 + 16 + 16 + 8 + ALIGN_CST]; // memory for *_left_
|
|
|
|
uint8_t yuv_mem_[3 * YUV_SIZE + PRED_SIZE + ALIGN_CST]; // memory for yuv_*
|
2011-02-19 08:33:46 +01:00
|
|
|
} VP8EncIterator;
|
|
|
|
|
|
|
|
// in iterator.c
|
2013-07-27 02:08:37 +02:00
|
|
|
// must be called first
|
2011-02-19 08:33:46 +01:00
|
|
|
void VP8IteratorInit(VP8Encoder* const enc, VP8EncIterator* const it);
|
2013-07-27 02:08:37 +02:00
|
|
|
// restart a scan
|
2011-02-19 08:33:46 +01:00
|
|
|
void VP8IteratorReset(VP8EncIterator* const it);
|
2013-07-27 02:08:37 +02:00
|
|
|
// reset iterator position to row 'y'
|
|
|
|
void VP8IteratorSetRow(VP8EncIterator* const it, int y);
|
2013-09-05 09:13:36 +02:00
|
|
|
// set count down (=number of iterations to go)
|
|
|
|
void VP8IteratorSetCountDown(VP8EncIterator* const it, int count_down);
|
|
|
|
// return true if iteration is finished
|
|
|
|
int VP8IteratorIsDone(const VP8EncIterator* const it);
|
|
|
|
// Import uncompressed samples from source.
|
|
|
|
// If tmp_32 is not NULL, import boundary samples too.
|
|
|
|
// tmp_32 is a 32-bytes scratch buffer that must be aligned in memory.
|
|
|
|
void VP8IteratorImport(VP8EncIterator* const it, uint8_t* tmp_32);
|
2011-02-19 08:33:46 +01:00
|
|
|
// export decimated samples
|
|
|
|
void VP8IteratorExport(const VP8EncIterator* const it);
|
2013-09-05 09:13:36 +02:00
|
|
|
// go to next macroblock. Returns false if not finished.
|
|
|
|
int VP8IteratorNext(VP8EncIterator* const it);
|
2013-09-06 11:11:16 +02:00
|
|
|
// save the yuv_out_ boundary values to top_/left_ arrays for next iterations.
|
|
|
|
void VP8IteratorSaveBoundary(VP8EncIterator* const it);
|
2011-12-01 11:24:50 +01:00
|
|
|
// Report progression based on macroblock rows. Return 0 for user-abort request.
|
|
|
|
int VP8IteratorProgress(const VP8EncIterator* const it,
|
|
|
|
int final_delta_percent);
|
2011-02-19 08:33:46 +01:00
|
|
|
// Intra4x4 iterations
|
|
|
|
void VP8IteratorStartI4(VP8EncIterator* const it);
|
|
|
|
// returns true if not done.
|
|
|
|
int VP8IteratorRotateI4(VP8EncIterator* const it,
|
|
|
|
const uint8_t* const yuv_out);
|
|
|
|
|
|
|
|
// Non-zero context setup/teardown
|
|
|
|
void VP8IteratorNzToBytes(VP8EncIterator* const it);
|
|
|
|
void VP8IteratorBytesToNz(VP8EncIterator* const it);
|
|
|
|
|
|
|
|
// Helper functions to set mode properties
|
|
|
|
void VP8SetIntra16Mode(const VP8EncIterator* const it, int mode);
|
2012-01-27 12:17:41 +01:00
|
|
|
void VP8SetIntra4Mode(const VP8EncIterator* const it, const uint8_t* modes);
|
2011-02-19 08:33:46 +01:00
|
|
|
void VP8SetIntraUVMode(const VP8EncIterator* const it, int mode);
|
|
|
|
void VP8SetSkip(const VP8EncIterator* const it, int skip);
|
|
|
|
void VP8SetSegment(const VP8EncIterator* const it, int segment);
|
|
|
|
|
2012-01-23 14:57:12 +01:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Paginated token buffer
|
|
|
|
|
2012-12-03 13:50:14 +01:00
|
|
|
typedef struct VP8Tokens VP8Tokens; // struct details in token.c
|
2012-01-24 01:56:33 +01:00
|
|
|
|
|
|
|
typedef struct {
|
2013-03-12 00:37:42 +01:00
|
|
|
#if !defined(DISABLE_TOKEN_BUFFER)
|
2012-12-03 13:50:14 +01:00
|
|
|
VP8Tokens* pages_; // first page
|
|
|
|
VP8Tokens** last_page_; // last page
|
|
|
|
uint16_t* tokens_; // set to (*last_page_)->tokens_
|
|
|
|
int left_; // how many free tokens left before the page is full.
|
2013-03-01 09:58:14 +01:00
|
|
|
#endif
|
2013-03-08 23:48:09 +01:00
|
|
|
int error_; // true in case of malloc error
|
2012-01-24 01:56:33 +01:00
|
|
|
} VP8TBuffer;
|
|
|
|
|
|
|
|
void VP8TBufferInit(VP8TBuffer* const b); // initialize an empty buffer
|
2013-03-01 09:58:14 +01:00
|
|
|
void VP8TBufferClear(VP8TBuffer* const b); // de-allocate pages memory
|
2012-12-03 13:50:14 +01:00
|
|
|
|
2013-03-12 00:37:42 +01:00
|
|
|
#if !defined(DISABLE_TOKEN_BUFFER)
|
2012-12-03 13:50:14 +01:00
|
|
|
|
2013-03-12 00:37:42 +01:00
|
|
|
// Finalizes bitstream when probabilities are known.
|
|
|
|
// Deletes the allocated token memory if final_pass is true.
|
|
|
|
int VP8EmitTokens(VP8TBuffer* const b, VP8BitWriter* const bw,
|
2012-12-03 13:50:14 +01:00
|
|
|
const uint8_t* const probas, int final_pass);
|
2013-03-12 00:37:42 +01:00
|
|
|
|
|
|
|
// record the coding of coefficients without knowing the probabilities yet
|
|
|
|
int VP8RecordCoeffTokens(int ctx, int coeff_type, int first, int last,
|
|
|
|
const int16_t* const coeffs,
|
|
|
|
VP8TBuffer* const tokens);
|
|
|
|
|
2013-09-11 10:08:49 +02:00
|
|
|
// Estimate the final coded size given a set of 'probas'.
|
|
|
|
size_t VP8EstimateTokenSize(VP8TBuffer* const b, const uint8_t* const probas);
|
|
|
|
|
2013-03-12 00:37:42 +01:00
|
|
|
// unused for now
|
2012-12-03 13:50:14 +01:00
|
|
|
void VP8TokenToStats(const VP8TBuffer* const b, proba_t* const stats);
|
2012-01-23 14:57:12 +01:00
|
|
|
|
2013-03-12 00:37:42 +01:00
|
|
|
#endif // !DISABLE_TOKEN_BUFFER
|
2012-01-23 14:57:12 +01:00
|
|
|
|
2011-08-25 23:22:32 +02:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-19 08:33:46 +01:00
|
|
|
// VP8Encoder
|
|
|
|
|
|
|
|
struct VP8Encoder {
|
|
|
|
const WebPConfig* config_; // user configuration and parameters
|
|
|
|
WebPPicture* pic_; // input / output picture
|
|
|
|
|
|
|
|
// headers
|
|
|
|
VP8FilterHeader filter_hdr_; // filtering information
|
|
|
|
VP8SegmentHeader segment_hdr_; // segment information
|
|
|
|
|
|
|
|
int profile_; // VP8's profile, deduced from Config.
|
|
|
|
|
|
|
|
// dimension, in macroblock units.
|
|
|
|
int mb_w_, mb_h_;
|
|
|
|
int preds_w_; // stride of the *preds_ prediction plane (=4*mb_w + 1)
|
|
|
|
|
|
|
|
// number of partitions (1, 2, 4 or 8 = MAX_NUM_PARTITIONS)
|
|
|
|
int num_parts_;
|
|
|
|
|
|
|
|
// per-partition boolean decoders.
|
|
|
|
VP8BitWriter bw_; // part0
|
|
|
|
VP8BitWriter parts_[MAX_NUM_PARTITIONS]; // token partitions
|
2013-03-12 00:37:42 +01:00
|
|
|
VP8TBuffer tokens_; // token buffer
|
2011-02-19 08:33:46 +01:00
|
|
|
|
2011-12-01 11:24:50 +01:00
|
|
|
int percent_; // for progress
|
|
|
|
|
EXPERIMENTAL: add support for alpha channel
This is a (minor) bitstream change: if the 'color_space' bit is set to '1'
(which is normally an undefined/invalid behaviour), we add extra data at the
end of partition #0 (so-called 'extensions')
Namely, we add the size of the extension data as 3 bytes (little-endian),
followed by a set of bits telling which extensions we're incorporating.
The data then _preceeds_ this trailing tags.
This is all experimental, and you'll need to have
'#define WEBP_EXPERIMENTAL_FEATURES' in webp/types.h to enable this code
(at your own risk! :))
Still, this hack produces almost-valid WebP file for decoders that don't
check this color_space bit. In particular, previous 'dwebp' (and for instance
Chrome) will recognize this files and decode them, but without the alpha
of course. Other decoder will just see random extra stuff at the end of
partition #0.
To experiment with the alpha-channel, you need to compile on Unix platform
and use PNGs for input/output.
If 'alpha.png' is a source with alpha channel, then you can try (on Unix):
cwebp alpha.png -o alpha.webp
dwebp alpha.webp -o test.png
cwebp now has a '-noalpha' flag to ignore any alpha information from the
source, if present.
More hacking and experimenting welcome!
Change-Id: I3c7b1fd8411c9e7a9f77690e898479ad85c52f3e
2011-04-26 01:58:04 +02:00
|
|
|
// transparency blob
|
2011-05-03 02:19:00 +02:00
|
|
|
int has_alpha_;
|
|
|
|
uint8_t* alpha_data_; // non-NULL if transparency is present
|
2011-12-01 07:44:15 +01:00
|
|
|
uint32_t alpha_data_size_;
|
2013-03-01 01:21:34 +01:00
|
|
|
WebPWorker alpha_worker_;
|
EXPERIMENTAL: add support for alpha channel
This is a (minor) bitstream change: if the 'color_space' bit is set to '1'
(which is normally an undefined/invalid behaviour), we add extra data at the
end of partition #0 (so-called 'extensions')
Namely, we add the size of the extension data as 3 bytes (little-endian),
followed by a set of bits telling which extensions we're incorporating.
The data then _preceeds_ this trailing tags.
This is all experimental, and you'll need to have
'#define WEBP_EXPERIMENTAL_FEATURES' in webp/types.h to enable this code
(at your own risk! :))
Still, this hack produces almost-valid WebP file for decoders that don't
check this color_space bit. In particular, previous 'dwebp' (and for instance
Chrome) will recognize this files and decode them, but without the alpha
of course. Other decoder will just see random extra stuff at the end of
partition #0.
To experiment with the alpha-channel, you need to compile on Unix platform
and use PNGs for input/output.
If 'alpha.png' is a source with alpha channel, then you can try (on Unix):
cwebp alpha.png -o alpha.webp
dwebp alpha.webp -o test.png
cwebp now has a '-noalpha' flag to ignore any alpha information from the
source, if present.
More hacking and experimenting welcome!
Change-Id: I3c7b1fd8411c9e7a9f77690e898479ad85c52f3e
2011-04-26 01:58:04 +02:00
|
|
|
|
2011-05-03 02:19:00 +02:00
|
|
|
// enhancement layer
|
|
|
|
int use_layer_;
|
|
|
|
VP8BitWriter layer_bw_;
|
|
|
|
uint8_t* layer_data_;
|
|
|
|
size_t layer_data_size_;
|
|
|
|
|
2011-02-19 08:33:46 +01:00
|
|
|
// quantization info (one set of DC/AC dequant factor per segment)
|
|
|
|
VP8SegmentInfo dqm_[NUM_MB_SEGMENTS];
|
|
|
|
int base_quant_; // nominal quantizer value. Only used
|
|
|
|
// for relative coding of segments' quant.
|
2013-02-05 19:40:18 +01:00
|
|
|
int alpha_; // global susceptibility (<=> complexity)
|
2011-02-19 08:33:46 +01:00
|
|
|
int uv_alpha_; // U/V quantization susceptibility
|
|
|
|
// global offset of quantizers, shared by all segments
|
|
|
|
int dq_y1_dc_;
|
|
|
|
int dq_y2_dc_, dq_y2_ac_;
|
|
|
|
int dq_uv_dc_, dq_uv_ac_;
|
|
|
|
|
|
|
|
// probabilities and statistics
|
|
|
|
VP8Proba proba_;
|
2012-07-25 01:15:36 +02:00
|
|
|
uint64_t sse_[4]; // sum of Y/U/V/A squared errors for all macroblocks
|
2011-02-19 08:33:46 +01:00
|
|
|
uint64_t sse_count_; // pixel count for the sse_[] stats
|
|
|
|
int coded_size_;
|
|
|
|
int residual_bytes_[3][4];
|
|
|
|
int block_count_[3];
|
|
|
|
|
|
|
|
// quality/speed settings
|
2013-02-26 11:20:59 +01:00
|
|
|
int method_; // 0=fastest, 6=best/slowest.
|
|
|
|
VP8RDLevel rd_opt_level_; // Deduced from method_.
|
|
|
|
int max_i4_header_bits_; // partition #0 safeness factor
|
2013-03-01 01:21:34 +01:00
|
|
|
int thread_level_; // derived from config->thread_level
|
2013-03-12 00:37:42 +01:00
|
|
|
int do_search_; // derived from config->target_XXX
|
|
|
|
int use_tokens_; // if true, use token buffer
|
2011-02-19 08:33:46 +01:00
|
|
|
|
|
|
|
// Memory
|
|
|
|
VP8MBInfo* mb_info_; // contextual macroblock infos (mb_w_ + 1)
|
|
|
|
uint8_t* preds_; // predictions modes: (4*mb_w+1) * (4*mb_h+1)
|
|
|
|
uint32_t* nz_; // non-zero bit context: mb_w+1
|
|
|
|
uint8_t *y_top_; // top luma samples.
|
|
|
|
uint8_t *uv_top_; // top u/v samples.
|
2013-07-27 02:08:37 +02:00
|
|
|
// U and V are packed into 16 bytes (8 U + 8 V)
|
2011-02-19 08:33:46 +01:00
|
|
|
LFStats *lf_stats_; // autofilter stats (if NULL, autofilter is off)
|
|
|
|
};
|
|
|
|
|
2011-08-25 23:22:32 +02:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-19 08:33:46 +01:00
|
|
|
// internal functions. Not public.
|
|
|
|
|
|
|
|
// in tree.c
|
|
|
|
extern const uint8_t VP8CoeffsProba0[NUM_TYPES][NUM_BANDS][NUM_CTX][NUM_PROBAS];
|
|
|
|
extern const uint8_t
|
|
|
|
VP8CoeffsUpdateProba[NUM_TYPES][NUM_BANDS][NUM_CTX][NUM_PROBAS];
|
|
|
|
// Reset the token probabilities to their initial (default) values
|
|
|
|
void VP8DefaultProbas(VP8Encoder* const enc);
|
|
|
|
// Write the token probabilities
|
|
|
|
void VP8WriteProbas(VP8BitWriter* const bw, const VP8Proba* const probas);
|
|
|
|
// Writes the partition #0 modes (that is: all intra modes)
|
|
|
|
void VP8CodeIntraModes(VP8Encoder* const enc);
|
|
|
|
|
|
|
|
// in syntax.c
|
|
|
|
// Generates the final bitstream by coding the partition0 and headers,
|
|
|
|
// and appending an assembly of all the pre-coded token partitions.
|
|
|
|
// Return true if everything is ok.
|
|
|
|
int VP8EncWrite(VP8Encoder* const enc);
|
2011-12-01 11:24:50 +01:00
|
|
|
// Release memory allocated for bit-writing in VP8EncLoop & seq.
|
|
|
|
void VP8EncFreeBitWriters(VP8Encoder* const enc);
|
2011-02-19 08:33:46 +01:00
|
|
|
|
|
|
|
// in frame.c
|
|
|
|
extern const uint8_t VP8EncBands[16 + 1];
|
2012-12-03 13:50:14 +01:00
|
|
|
extern const uint8_t VP8Cat3[];
|
|
|
|
extern const uint8_t VP8Cat4[];
|
|
|
|
extern const uint8_t VP8Cat5[];
|
|
|
|
extern const uint8_t VP8Cat6[];
|
|
|
|
|
2011-02-19 08:33:46 +01:00
|
|
|
// Form all the four Intra16x16 predictions in the yuv_p_ cache
|
|
|
|
void VP8MakeLuma16Preds(const VP8EncIterator* const it);
|
|
|
|
// Form all the four Chroma8x8 predictions in the yuv_p_ cache
|
|
|
|
void VP8MakeChroma8Preds(const VP8EncIterator* const it);
|
|
|
|
// Form all the ten Intra4x4 predictions in the yuv_p_ cache
|
|
|
|
// for the 4x4 block it->i4_
|
|
|
|
void VP8MakeIntra4Preds(const VP8EncIterator* const it);
|
|
|
|
// Rate calculation
|
|
|
|
int VP8GetCostLuma16(VP8EncIterator* const it, const VP8ModeScore* const rd);
|
|
|
|
int VP8GetCostLuma4(VP8EncIterator* const it, const int16_t levels[16]);
|
|
|
|
int VP8GetCostUV(VP8EncIterator* const it, const VP8ModeScore* const rd);
|
2013-03-01 09:58:14 +01:00
|
|
|
// Main coding calls
|
2011-02-19 08:33:46 +01:00
|
|
|
int VP8EncLoop(VP8Encoder* const enc);
|
2013-03-12 00:37:42 +01:00
|
|
|
int VP8EncTokenLoop(VP8Encoder* const enc);
|
2011-02-19 08:33:46 +01:00
|
|
|
|
2011-06-02 15:55:03 +02:00
|
|
|
// in webpenc.c
|
|
|
|
// Assign an error code to a picture. Return false for convenience.
|
2012-05-22 11:51:38 +02:00
|
|
|
int WebPEncodingSetError(const WebPPicture* const pic, WebPEncodingError error);
|
2012-05-29 08:02:02 +02:00
|
|
|
int WebPReportProgress(const WebPPicture* const pic,
|
|
|
|
int percent, int* const percent_store);
|
2011-09-02 23:30:08 +02:00
|
|
|
|
2011-02-19 08:33:46 +01:00
|
|
|
// in analysis.c
|
|
|
|
// Main analysis loop. Decides the segmentations and complexity.
|
|
|
|
// Assigns a first guess for Intra16 and uvmode_ prediction modes.
|
|
|
|
int VP8EncAnalyze(VP8Encoder* const enc);
|
|
|
|
|
|
|
|
// in quant.c
|
|
|
|
// Sets up segment's quantization values, base_quant_ and filter strengths.
|
|
|
|
void VP8SetSegmentParams(VP8Encoder* const enc, float quality);
|
|
|
|
// Pick best modes and fills the levels. Returns true if skipped.
|
2013-02-26 11:20:59 +01:00
|
|
|
int VP8Decimate(VP8EncIterator* const it, VP8ModeScore* const rd,
|
|
|
|
VP8RDLevel rd_opt);
|
2011-02-19 08:33:46 +01:00
|
|
|
|
EXPERIMENTAL: add support for alpha channel
This is a (minor) bitstream change: if the 'color_space' bit is set to '1'
(which is normally an undefined/invalid behaviour), we add extra data at the
end of partition #0 (so-called 'extensions')
Namely, we add the size of the extension data as 3 bytes (little-endian),
followed by a set of bits telling which extensions we're incorporating.
The data then _preceeds_ this trailing tags.
This is all experimental, and you'll need to have
'#define WEBP_EXPERIMENTAL_FEATURES' in webp/types.h to enable this code
(at your own risk! :))
Still, this hack produces almost-valid WebP file for decoders that don't
check this color_space bit. In particular, previous 'dwebp' (and for instance
Chrome) will recognize this files and decode them, but without the alpha
of course. Other decoder will just see random extra stuff at the end of
partition #0.
To experiment with the alpha-channel, you need to compile on Unix platform
and use PNGs for input/output.
If 'alpha.png' is a source with alpha channel, then you can try (on Unix):
cwebp alpha.png -o alpha.webp
dwebp alpha.webp -o test.png
cwebp now has a '-noalpha' flag to ignore any alpha information from the
source, if present.
More hacking and experimenting welcome!
Change-Id: I3c7b1fd8411c9e7a9f77690e898479ad85c52f3e
2011-04-26 01:58:04 +02:00
|
|
|
// in alpha.c
|
2012-07-25 01:15:36 +02:00
|
|
|
void VP8EncInitAlpha(VP8Encoder* const enc); // initialize alpha compression
|
2013-03-01 01:21:34 +01:00
|
|
|
int VP8EncStartAlpha(VP8Encoder* const enc); // start alpha coding process
|
2012-07-25 01:15:36 +02:00
|
|
|
int VP8EncFinishAlpha(VP8Encoder* const enc); // finalize compressed data
|
2013-03-01 01:21:34 +01:00
|
|
|
int VP8EncDeleteAlpha(VP8Encoder* const enc); // delete compressed data
|
2011-05-03 02:19:00 +02:00
|
|
|
|
|
|
|
// in layer.c
|
|
|
|
void VP8EncInitLayer(VP8Encoder* const enc); // init everything
|
|
|
|
void VP8EncCodeLayerBlock(VP8EncIterator* it); // code one more macroblock
|
|
|
|
int VP8EncFinishLayer(VP8Encoder* const enc); // finalize coding
|
|
|
|
void VP8EncDeleteLayer(VP8Encoder* enc); // reclaim memory
|
EXPERIMENTAL: add support for alpha channel
This is a (minor) bitstream change: if the 'color_space' bit is set to '1'
(which is normally an undefined/invalid behaviour), we add extra data at the
end of partition #0 (so-called 'extensions')
Namely, we add the size of the extension data as 3 bytes (little-endian),
followed by a set of bits telling which extensions we're incorporating.
The data then _preceeds_ this trailing tags.
This is all experimental, and you'll need to have
'#define WEBP_EXPERIMENTAL_FEATURES' in webp/types.h to enable this code
(at your own risk! :))
Still, this hack produces almost-valid WebP file for decoders that don't
check this color_space bit. In particular, previous 'dwebp' (and for instance
Chrome) will recognize this files and decode them, but without the alpha
of course. Other decoder will just see random extra stuff at the end of
partition #0.
To experiment with the alpha-channel, you need to compile on Unix platform
and use PNGs for input/output.
If 'alpha.png' is a source with alpha channel, then you can try (on Unix):
cwebp alpha.png -o alpha.webp
dwebp alpha.webp -o test.png
cwebp now has a '-noalpha' flag to ignore any alpha information from the
source, if present.
More hacking and experimenting welcome!
Change-Id: I3c7b1fd8411c9e7a9f77690e898479ad85c52f3e
2011-04-26 01:58:04 +02:00
|
|
|
|
2011-02-19 08:33:46 +01:00
|
|
|
// in filter.c
|
2012-01-20 16:20:56 +01:00
|
|
|
|
|
|
|
// SSIM utils
|
2012-01-28 02:39:47 +01:00
|
|
|
typedef struct {
|
|
|
|
double w, xm, ym, xxm, xym, yym;
|
|
|
|
} DistoStats;
|
2012-01-20 16:20:56 +01:00
|
|
|
void VP8SSIMAddStats(const DistoStats* const src, DistoStats* const dst);
|
|
|
|
void VP8SSIMAccumulatePlane(const uint8_t* src1, int stride1,
|
|
|
|
const uint8_t* src2, int stride2,
|
|
|
|
int W, int H, DistoStats* const stats);
|
|
|
|
double VP8SSIMGet(const DistoStats* const stats);
|
|
|
|
double VP8SSIMGetSquaredError(const DistoStats* const stats);
|
|
|
|
|
|
|
|
// autofilter
|
|
|
|
void VP8InitFilter(VP8EncIterator* const it);
|
|
|
|
void VP8StoreFilterStats(VP8EncIterator* const it);
|
|
|
|
void VP8AdjustFilterStrength(VP8EncIterator* const it);
|
2011-02-19 08:33:46 +01:00
|
|
|
|
2013-10-29 20:13:29 +01:00
|
|
|
// returns the approximate filtering strength needed to smooth a edge
|
|
|
|
// step of 'delta', given a sharpness parameter 'sharpness'.
|
|
|
|
int VP8FilterStrengthFromDelta(int sharpness, int delta);
|
|
|
|
|
2011-08-25 23:22:32 +02:00
|
|
|
//------------------------------------------------------------------------------
|
2011-02-19 08:33:46 +01:00
|
|
|
|
|
|
|
#if defined(__cplusplus) || defined(c_plusplus)
|
|
|
|
} // extern "C"
|
|
|
|
#endif
|
|
|
|
|
2011-09-07 11:26:35 +02:00
|
|
|
#endif /* WEBP_ENC_VP8ENCI_H_ */
|