2010-05-18 17:58:33 +02:00
|
|
|
/*
|
2010-09-09 14:16:39 +02:00
|
|
|
* Copyright (c) 2010 The WebM project authors. All Rights Reserved.
|
2010-05-18 17:58:33 +02:00
|
|
|
*
|
2010-06-18 18:39:21 +02:00
|
|
|
* Use of this source code is governed by a BSD-style license
|
2010-06-04 22:19:40 +02:00
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
|
* tree. An additional intellectual property rights grant can be found
|
2010-06-18 18:39:21 +02:00
|
|
|
* in the file PATENTS. All contributing project authors may
|
2010-06-04 22:19:40 +02:00
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
2010-05-18 17:58:33 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <assert.h>
|
2012-11-27 22:59:17 +01:00
|
|
|
#include "vp9/common/vp9_pragmas.h"
|
|
|
|
|
2012-11-28 19:41:40 +01:00
|
|
|
#include "vp9/encoder/vp9_tokenize.h"
|
|
|
|
#include "vp9/encoder/vp9_treewriter.h"
|
|
|
|
#include "vp9/encoder/vp9_onyx_int.h"
|
|
|
|
#include "vp9/encoder/vp9_modecosts.h"
|
|
|
|
#include "vp9/encoder/vp9_encodeintra.h"
|
2012-11-27 22:59:17 +01:00
|
|
|
#include "vp9/common/vp9_entropymode.h"
|
|
|
|
#include "vp9/common/vp9_reconinter.h"
|
|
|
|
#include "vp9/common/vp9_reconintra.h"
|
|
|
|
#include "vp9/common/vp9_findnearmv.h"
|
|
|
|
#include "vp9/common/vp9_quant_common.h"
|
2012-11-28 19:41:40 +01:00
|
|
|
#include "vp9/encoder/vp9_encodemb.h"
|
|
|
|
#include "vp9/encoder/vp9_quantize.h"
|
|
|
|
#include "vp9/encoder/vp9_variance.h"
|
|
|
|
#include "vp9/encoder/vp9_mcomp.h"
|
|
|
|
#include "vp9/encoder/vp9_rdopt.h"
|
|
|
|
#include "vp9/encoder/vp9_ratectrl.h"
|
2010-05-18 17:58:33 +02:00
|
|
|
#include "vpx_mem/vpx_mem.h"
|
2012-11-27 22:59:17 +01:00
|
|
|
#include "vp9/common/vp9_systemdependent.h"
|
|
|
|
#include "vp9/encoder/vp9_encodemv.h"
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-27 22:59:17 +01:00
|
|
|
#include "vp9/common/vp9_seg_common.h"
|
|
|
|
#include "vp9/common/vp9_pred_common.h"
|
|
|
|
#include "vp9/common/vp9_entropy.h"
|
2012-11-09 02:09:30 +01:00
|
|
|
#include "vp9_rtcd.h"
|
2012-11-27 22:59:17 +01:00
|
|
|
#include "vp9/common/vp9_mvref_common.h"
|
2013-01-06 03:20:25 +01:00
|
|
|
#include "vp9/common/vp9_common.h"
|
2012-08-24 16:44:01 +02:00
|
|
|
|
2010-05-18 17:58:33 +02:00
|
|
|
#define MAXF(a,b) (((a) > (b)) ? (a) : (b))
|
|
|
|
|
2012-04-18 22:51:58 +02:00
|
|
|
#define INVALID_MV 0x80008000
|
|
|
|
|
2012-07-18 22:43:01 +02:00
|
|
|
/* Factor to weigh the rate for switchable interp filters */
|
|
|
|
#define SWITCHABLE_INTERP_RATE_FACTOR 1
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
static const int auto_speed_thresh[17] = {
|
|
|
|
1000,
|
|
|
|
200,
|
|
|
|
150,
|
|
|
|
130,
|
|
|
|
150,
|
|
|
|
125,
|
|
|
|
120,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
115,
|
|
|
|
105
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
2012-10-31 01:12:12 +01:00
|
|
|
const MODE_DEFINITION vp9_mode_order[MAX_MODES] = {
|
2012-11-07 15:50:25 +01:00
|
|
|
{ZEROMV, LAST_FRAME, NONE},
|
|
|
|
{DC_PRED, INTRA_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{NEARESTMV, LAST_FRAME, NONE},
|
|
|
|
{NEARMV, LAST_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{ZEROMV, GOLDEN_FRAME, NONE},
|
|
|
|
{NEARESTMV, GOLDEN_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{ZEROMV, ALTREF_FRAME, NONE},
|
|
|
|
{NEARESTMV, ALTREF_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{NEARMV, GOLDEN_FRAME, NONE},
|
|
|
|
{NEARMV, ALTREF_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{V_PRED, INTRA_FRAME, NONE},
|
|
|
|
{H_PRED, INTRA_FRAME, NONE},
|
|
|
|
{D45_PRED, INTRA_FRAME, NONE},
|
|
|
|
{D135_PRED, INTRA_FRAME, NONE},
|
|
|
|
{D117_PRED, INTRA_FRAME, NONE},
|
|
|
|
{D153_PRED, INTRA_FRAME, NONE},
|
|
|
|
{D27_PRED, INTRA_FRAME, NONE},
|
|
|
|
{D63_PRED, INTRA_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{TM_PRED, INTRA_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{NEWMV, LAST_FRAME, NONE},
|
|
|
|
{NEWMV, GOLDEN_FRAME, NONE},
|
|
|
|
{NEWMV, ALTREF_FRAME, NONE},
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
{SPLITMV, LAST_FRAME, NONE},
|
|
|
|
{SPLITMV, GOLDEN_FRAME, NONE},
|
|
|
|
{SPLITMV, ALTREF_FRAME, NONE},
|
Dual 16x16 inter prediction.
This patch introduces the concept of dual inter16x16 prediction. A
16x16 inter-predicted macroblock can use 2 references instead of 1,
where both references use the same mvmode (new, near/est, zero). In the
case of newmv, this means that two MVs are coded instead of one. The
frame can be encoded in 3 ways: all MBs single-prediction, all MBs dual
prediction, or per-MB single/dual prediction selection ("hybrid"), in
which case a single bit is coded per-MB to indicate whether the MB uses
single or dual inter prediction.
In the future, we can (maybe?) get further gains by mixing this with
Adrian's 32x32 work, per-segment dual prediction settings, or adding
support for dual splitmv/8x8mv inter prediction.
Gain (on derf-set, CQ mode) is ~2.8% (SSIM) or ~3.6% (glb PSNR). Most
gain is at medium/high bitrates, but there's minor gains at low bitrates
also. Output was confirmed to match between encoder and decoder.
Note for optimization people: this patch introduces a 2nd version of
16x16/8x8 sixtap/bilin functions, which does an avg instead of a
store. They may want to look and make sure this is implemented to
their satisfaction so we can optimize it best in the future.
Change-ID: I59dc84b07cbb3ccf073ac0f756d03d294cb19281
2011-12-06 20:53:02 +01:00
|
|
|
|
2013-04-11 21:16:35 +02:00
|
|
|
{I4X4_PRED, INTRA_FRAME, NONE},
|
2013-05-01 01:13:20 +02:00
|
|
|
#if !CONFIG_SB8X8
|
2012-11-07 15:50:25 +01:00
|
|
|
{I8X8_PRED, INTRA_FRAME, NONE},
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif
|
Dual 16x16 inter prediction.
This patch introduces the concept of dual inter16x16 prediction. A
16x16 inter-predicted macroblock can use 2 references instead of 1,
where both references use the same mvmode (new, near/est, zero). In the
case of newmv, this means that two MVs are coded instead of one. The
frame can be encoded in 3 ways: all MBs single-prediction, all MBs dual
prediction, or per-MB single/dual prediction selection ("hybrid"), in
which case a single bit is coded per-MB to indicate whether the MB uses
single or dual inter prediction.
In the future, we can (maybe?) get further gains by mixing this with
Adrian's 32x32 work, per-segment dual prediction settings, or adding
support for dual splitmv/8x8mv inter prediction.
Gain (on derf-set, CQ mode) is ~2.8% (SSIM) or ~3.6% (glb PSNR). Most
gain is at medium/high bitrates, but there's minor gains at low bitrates
also. Output was confirmed to match between encoder and decoder.
Note for optimization people: this patch introduces a 2nd version of
16x16/8x8 sixtap/bilin functions, which does an avg instead of a
store. They may want to look and make sure this is implemented to
their satisfaction so we can optimize it best in the future.
Change-ID: I59dc84b07cbb3ccf073ac0f756d03d294cb19281
2011-12-06 20:53:02 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* compound prediction modes */
|
|
|
|
{ZEROMV, LAST_FRAME, GOLDEN_FRAME},
|
|
|
|
{NEARESTMV, LAST_FRAME, GOLDEN_FRAME},
|
|
|
|
{NEARMV, LAST_FRAME, GOLDEN_FRAME},
|
Dual 16x16 inter prediction.
This patch introduces the concept of dual inter16x16 prediction. A
16x16 inter-predicted macroblock can use 2 references instead of 1,
where both references use the same mvmode (new, near/est, zero). In the
case of newmv, this means that two MVs are coded instead of one. The
frame can be encoded in 3 ways: all MBs single-prediction, all MBs dual
prediction, or per-MB single/dual prediction selection ("hybrid"), in
which case a single bit is coded per-MB to indicate whether the MB uses
single or dual inter prediction.
In the future, we can (maybe?) get further gains by mixing this with
Adrian's 32x32 work, per-segment dual prediction settings, or adding
support for dual splitmv/8x8mv inter prediction.
Gain (on derf-set, CQ mode) is ~2.8% (SSIM) or ~3.6% (glb PSNR). Most
gain is at medium/high bitrates, but there's minor gains at low bitrates
also. Output was confirmed to match between encoder and decoder.
Note for optimization people: this patch introduces a 2nd version of
16x16/8x8 sixtap/bilin functions, which does an avg instead of a
store. They may want to look and make sure this is implemented to
their satisfaction so we can optimize it best in the future.
Change-ID: I59dc84b07cbb3ccf073ac0f756d03d294cb19281
2011-12-06 20:53:02 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
{ZEROMV, ALTREF_FRAME, LAST_FRAME},
|
|
|
|
{NEARESTMV, ALTREF_FRAME, LAST_FRAME},
|
|
|
|
{NEARMV, ALTREF_FRAME, LAST_FRAME},
|
Dual 16x16 inter prediction.
This patch introduces the concept of dual inter16x16 prediction. A
16x16 inter-predicted macroblock can use 2 references instead of 1,
where both references use the same mvmode (new, near/est, zero). In the
case of newmv, this means that two MVs are coded instead of one. The
frame can be encoded in 3 ways: all MBs single-prediction, all MBs dual
prediction, or per-MB single/dual prediction selection ("hybrid"), in
which case a single bit is coded per-MB to indicate whether the MB uses
single or dual inter prediction.
In the future, we can (maybe?) get further gains by mixing this with
Adrian's 32x32 work, per-segment dual prediction settings, or adding
support for dual splitmv/8x8mv inter prediction.
Gain (on derf-set, CQ mode) is ~2.8% (SSIM) or ~3.6% (glb PSNR). Most
gain is at medium/high bitrates, but there's minor gains at low bitrates
also. Output was confirmed to match between encoder and decoder.
Note for optimization people: this patch introduces a 2nd version of
16x16/8x8 sixtap/bilin functions, which does an avg instead of a
store. They may want to look and make sure this is implemented to
their satisfaction so we can optimize it best in the future.
Change-ID: I59dc84b07cbb3ccf073ac0f756d03d294cb19281
2011-12-06 20:53:02 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
{ZEROMV, GOLDEN_FRAME, ALTREF_FRAME},
|
|
|
|
{NEARESTMV, GOLDEN_FRAME, ALTREF_FRAME},
|
|
|
|
{NEARMV, GOLDEN_FRAME, ALTREF_FRAME},
|
Dual 16x16 inter prediction.
This patch introduces the concept of dual inter16x16 prediction. A
16x16 inter-predicted macroblock can use 2 references instead of 1,
where both references use the same mvmode (new, near/est, zero). In the
case of newmv, this means that two MVs are coded instead of one. The
frame can be encoded in 3 ways: all MBs single-prediction, all MBs dual
prediction, or per-MB single/dual prediction selection ("hybrid"), in
which case a single bit is coded per-MB to indicate whether the MB uses
single or dual inter prediction.
In the future, we can (maybe?) get further gains by mixing this with
Adrian's 32x32 work, per-segment dual prediction settings, or adding
support for dual splitmv/8x8mv inter prediction.
Gain (on derf-set, CQ mode) is ~2.8% (SSIM) or ~3.6% (glb PSNR). Most
gain is at medium/high bitrates, but there's minor gains at low bitrates
also. Output was confirmed to match between encoder and decoder.
Note for optimization people: this patch introduces a 2nd version of
16x16/8x8 sixtap/bilin functions, which does an avg instead of a
store. They may want to look and make sure this is implemented to
their satisfaction so we can optimize it best in the future.
Change-ID: I59dc84b07cbb3ccf073ac0f756d03d294cb19281
2011-12-06 20:53:02 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
{NEWMV, LAST_FRAME, GOLDEN_FRAME},
|
|
|
|
{NEWMV, ALTREF_FRAME, LAST_FRAME },
|
|
|
|
{NEWMV, GOLDEN_FRAME, ALTREF_FRAME},
|
2012-04-18 22:51:58 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
{SPLITMV, LAST_FRAME, GOLDEN_FRAME},
|
|
|
|
{SPLITMV, ALTREF_FRAME, LAST_FRAME },
|
2012-11-07 15:50:25 +01:00
|
|
|
{SPLITMV, GOLDEN_FRAME, ALTREF_FRAME},
|
|
|
|
|
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
/* compound inter-intra prediction */
|
|
|
|
{ZEROMV, LAST_FRAME, INTRA_FRAME},
|
|
|
|
{NEARESTMV, LAST_FRAME, INTRA_FRAME},
|
|
|
|
{NEARMV, LAST_FRAME, INTRA_FRAME},
|
|
|
|
{NEWMV, LAST_FRAME, INTRA_FRAME},
|
|
|
|
|
|
|
|
{ZEROMV, GOLDEN_FRAME, INTRA_FRAME},
|
|
|
|
{NEARESTMV, GOLDEN_FRAME, INTRA_FRAME},
|
|
|
|
{NEARMV, GOLDEN_FRAME, INTRA_FRAME},
|
|
|
|
{NEWMV, GOLDEN_FRAME, INTRA_FRAME},
|
|
|
|
|
|
|
|
{ZEROMV, ALTREF_FRAME, INTRA_FRAME},
|
|
|
|
{NEARESTMV, ALTREF_FRAME, INTRA_FRAME},
|
|
|
|
{NEARMV, ALTREF_FRAME, INTRA_FRAME},
|
|
|
|
{NEWMV, ALTREF_FRAME, INTRA_FRAME},
|
|
|
|
#endif
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
2012-12-08 01:09:59 +01:00
|
|
|
static void fill_token_costs(vp9_coeff_count *c,
|
|
|
|
vp9_coeff_probs *p,
|
2013-03-26 23:23:30 +01:00
|
|
|
TX_SIZE tx_size) {
|
2013-02-19 22:36:38 +01:00
|
|
|
int i, j, k, l;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-03-26 23:23:30 +01:00
|
|
|
for (i = 0; i < BLOCK_TYPES; i++)
|
2013-02-19 22:36:38 +01:00
|
|
|
for (j = 0; j < REF_TYPES; j++)
|
|
|
|
for (k = 0; k < COEF_BANDS; k++)
|
|
|
|
for (l = 0; l < PREV_COEF_CONTEXTS; l++) {
|
2013-02-20 19:16:24 +01:00
|
|
|
vp9_cost_tokens_skip((int *)(c[i][j][k][l]),
|
|
|
|
p[i][j][k][l],
|
|
|
|
vp9_coef_tree);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-30 22:25:33 +01:00
|
|
|
static int rd_iifactor[32] = { 4, 4, 3, 2, 1, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, };
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2010-12-02 00:50:14 +01:00
|
|
|
// 3* dc_qlookup[Q]*dc_qlookup[Q];
|
2011-11-23 12:32:20 +01:00
|
|
|
|
2010-12-24 04:59:12 +01:00
|
|
|
/* values are now correlated to quantizer */
|
2011-11-21 16:45:10 +01:00
|
|
|
static int sad_per_bit16lut[QINDEX_RANGE];
|
|
|
|
static int sad_per_bit4lut[QINDEX_RANGE];
|
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
void vp9_init_me_luts() {
|
2012-07-14 00:21:29 +02:00
|
|
|
int i;
|
|
|
|
|
|
|
|
// Initialize the sad lut tables using a formulaic calculation for now
|
|
|
|
// This is to make it easier to resolve the impact of experimental changes
|
|
|
|
// to the quantizer tables.
|
|
|
|
for (i = 0; i < QINDEX_RANGE; i++) {
|
|
|
|
sad_per_bit16lut[i] =
|
2012-10-30 20:58:42 +01:00
|
|
|
(int)((0.0418 * vp9_convert_qindex_to_q(i)) + 2.4107);
|
|
|
|
sad_per_bit4lut[i] = (int)((0.063 * vp9_convert_qindex_to_q(i)) + 2.742);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2011-11-21 16:45:10 +01:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-30 02:04:33 +01:00
|
|
|
static int compute_rd_mult(int qindex) {
|
2013-04-26 21:30:20 +02:00
|
|
|
const int q = vp9_dc_quant(qindex, 0);
|
2013-03-07 16:02:57 +01:00
|
|
|
return (11 * q * q) >> 2;
|
2011-11-23 12:32:20 +01:00
|
|
|
}
|
|
|
|
|
2013-03-02 02:39:31 +01:00
|
|
|
void vp9_initialize_me_consts(VP9_COMP *cpi, int qindex) {
|
|
|
|
cpi->mb.sadperbit16 = sad_per_bit16lut[qindex];
|
|
|
|
cpi->mb.sadperbit4 = sad_per_bit4lut[qindex];
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2010-12-02 00:50:14 +01:00
|
|
|
|
2013-03-02 02:39:31 +01:00
|
|
|
void vp9_initialize_rd_consts(VP9_COMP *cpi, int qindex) {
|
2012-08-07 00:03:04 +02:00
|
|
|
int q, i;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_clear_system_state(); // __asm emms;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// Further tests required to see if optimum is different
|
|
|
|
// for key frames, golden frames and arf frames.
|
|
|
|
// if (cpi->common.refresh_golden_frame ||
|
|
|
|
// cpi->common.refresh_alt_ref_frame)
|
2013-03-02 02:39:31 +01:00
|
|
|
qindex = (qindex < 0) ? 0 : ((qindex > MAXQ) ? MAXQ : qindex);
|
2011-11-23 12:32:20 +01:00
|
|
|
|
2013-03-02 02:39:31 +01:00
|
|
|
cpi->RDMULT = compute_rd_mult(qindex);
|
2012-07-14 00:21:29 +02:00
|
|
|
if (cpi->pass == 2 && (cpi->common.frame_type != KEY_FRAME)) {
|
|
|
|
if (cpi->twopass.next_iiratio > 31)
|
|
|
|
cpi->RDMULT += (cpi->RDMULT * rd_iifactor[31]) >> 4;
|
|
|
|
else
|
|
|
|
cpi->RDMULT +=
|
2013-03-02 02:39:31 +01:00
|
|
|
(cpi->RDMULT * rd_iifactor[cpi->twopass.next_iiratio]) >> 4;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2013-03-07 16:02:57 +01:00
|
|
|
cpi->mb.errorperbit = cpi->RDMULT >> 6;
|
2012-07-14 00:21:29 +02:00
|
|
|
cpi->mb.errorperbit += (cpi->mb.errorperbit == 0);
|
2011-01-26 07:24:22 +01:00
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_set_speed_features(cpi);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-03-02 02:39:31 +01:00
|
|
|
q = (int)pow(vp9_dc_quant(qindex, 0) >> 2, 1.25);
|
|
|
|
q <<= 2;
|
2012-07-14 00:21:29 +02:00
|
|
|
if (q < 8)
|
|
|
|
q = 8;
|
2010-12-02 00:50:14 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (cpi->RDMULT > 1000) {
|
|
|
|
cpi->RDDIV = 1;
|
|
|
|
cpi->RDMULT /= 100;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
for (i = 0; i < MAX_MODES; i++) {
|
|
|
|
if (cpi->sf.thresh_mult[i] < INT_MAX) {
|
|
|
|
cpi->rd_threshes[i] = cpi->sf.thresh_mult[i] * q / 100;
|
|
|
|
} else {
|
|
|
|
cpi->rd_threshes[i] = INT_MAX;
|
|
|
|
}
|
|
|
|
cpi->rd_baseline_thresh[i] = cpi->rd_threshes[i];
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
|
|
|
cpi->RDDIV = 100;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
for (i = 0; i < MAX_MODES; i++) {
|
|
|
|
if (cpi->sf.thresh_mult[i] < (INT_MAX / q)) {
|
|
|
|
cpi->rd_threshes[i] = cpi->sf.thresh_mult[i] * q;
|
|
|
|
} else {
|
|
|
|
cpi->rd_threshes[i] = INT_MAX;
|
|
|
|
}
|
|
|
|
cpi->rd_baseline_thresh[i] = cpi->rd_threshes[i];
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-08 01:09:59 +01:00
|
|
|
fill_token_costs(cpi->mb.token_costs[TX_4X4],
|
2013-03-26 23:23:30 +01:00
|
|
|
cpi->common.fc.coef_probs_4x4, TX_4X4);
|
2012-12-08 01:09:59 +01:00
|
|
|
fill_token_costs(cpi->mb.token_costs[TX_8X8],
|
2013-03-26 23:23:30 +01:00
|
|
|
cpi->common.fc.coef_probs_8x8, TX_8X8);
|
2012-12-08 01:09:59 +01:00
|
|
|
fill_token_costs(cpi->mb.token_costs[TX_16X16],
|
2013-03-26 23:23:30 +01:00
|
|
|
cpi->common.fc.coef_probs_16x16, TX_16X16);
|
2012-12-08 01:09:59 +01:00
|
|
|
fill_token_costs(cpi->mb.token_costs[TX_32X32],
|
2013-03-26 23:23:30 +01:00
|
|
|
cpi->common.fc.coef_probs_32x32, TX_32X32);
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
|
2013-04-23 19:12:18 +02:00
|
|
|
for (i = 0; i < NUM_PARTITION_CONTEXTS; i++)
|
2013-04-17 18:25:06 +02:00
|
|
|
vp9_cost_tokens(cpi->mb.partition_cost[i],
|
|
|
|
cpi->common.fc.partition_prob[i],
|
|
|
|
vp9_partition_tree);
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/*rough estimate for costing*/
|
|
|
|
cpi->common.kf_ymode_probs_index = cpi->common.base_qindex >> 4;
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_init_mode_costs(cpi);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
if (cpi->common.frame_type != KEY_FRAME) {
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_build_nmv_cost_table(
|
2012-07-26 22:42:07 +02:00
|
|
|
cpi->mb.nmvjointcost,
|
|
|
|
cpi->mb.e_mbd.allow_high_precision_mv ?
|
|
|
|
cpi->mb.nmvcost_hp : cpi->mb.nmvcost,
|
|
|
|
&cpi->common.fc.nmvc,
|
|
|
|
cpi->mb.e_mbd.allow_high_precision_mv, 1, 1);
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2012-12-19 00:31:19 +01:00
|
|
|
int vp9_block_error_c(int16_t *coeff, int16_t *dqcoeff, int block_size) {
|
2012-08-07 00:03:04 +02:00
|
|
|
int i, error = 0;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-08-07 00:03:04 +02:00
|
|
|
for (i = 0; i < block_size; i++) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int this_diff = coeff[i] - dqcoeff[i];
|
|
|
|
error += this_diff * this_diff;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return error;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-02-20 19:16:24 +01:00
|
|
|
static INLINE int cost_coeffs(VP9_COMMON *const cm, MACROBLOCK *mb,
|
2013-03-04 23:12:17 +01:00
|
|
|
int ib, PLANE_TYPE type,
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT *A,
|
|
|
|
ENTROPY_CONTEXT *L,
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_SIZE tx_size,
|
|
|
|
int y_blocks) {
|
2013-02-27 19:00:24 +01:00
|
|
|
MACROBLOCKD *const xd = &mb->e_mbd;
|
2013-03-04 23:12:17 +01:00
|
|
|
MB_MODE_INFO *mbmi = &xd->mode_info_context->mbmi;
|
|
|
|
int pt;
|
2013-02-15 19:15:42 +01:00
|
|
|
int c = 0;
|
2013-03-27 00:46:09 +01:00
|
|
|
int cost = 0, pad;
|
|
|
|
const int *scan, *nb;
|
2013-04-04 21:03:27 +02:00
|
|
|
const struct plane_block_idx pb_idx = plane_block_idx(y_blocks, ib);
|
|
|
|
const int eob = xd->plane[pb_idx.plane].eobs[pb_idx.block];
|
2013-04-02 23:50:40 +02:00
|
|
|
const int16_t *qcoeff_ptr = BLOCK_OFFSET(xd->plane[pb_idx.plane].qcoeff,
|
|
|
|
pb_idx.block, 16);
|
2013-03-04 23:12:17 +01:00
|
|
|
const int ref = mbmi->ref_frame != INTRA_FRAME;
|
2013-02-07 02:16:36 +01:00
|
|
|
unsigned int (*token_costs)[PREV_COEF_CONTEXTS][MAX_ENTROPY_TOKENS] =
|
2013-02-19 22:36:38 +01:00
|
|
|
mb->token_costs[tx_size][type][ref];
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT above_ec, left_ec;
|
2013-03-28 18:42:23 +01:00
|
|
|
TX_TYPE tx_type = DCT_DCT;
|
2012-08-07 03:29:59 +02:00
|
|
|
|
2013-03-28 18:42:23 +01:00
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
int last_nz_pos[3] = {-1, -1, -1}; // Encoder only
|
|
|
|
int is_eoo_list[3] = {0, 0, 0};
|
|
|
|
int is_eoo_negative[3] = {0, 0, 0};
|
|
|
|
int is_last_zero[3] = {0, 0, 0};
|
|
|
|
int o, rc, skip_coef_val;
|
|
|
|
vp9_zpc_probs *zpc_probs;
|
|
|
|
uint8_t token_cache_full[1024];
|
2013-03-26 23:23:30 +01:00
|
|
|
#endif
|
2013-02-20 19:16:24 +01:00
|
|
|
const int segment_id = xd->mode_info_context->mbmi.segment_id;
|
2013-03-27 00:46:09 +01:00
|
|
|
vp9_prob (*coef_probs)[REF_TYPES][COEF_BANDS][PREV_COEF_CONTEXTS]
|
|
|
|
[ENTROPY_NODES];
|
|
|
|
int seg_eob, default_eob;
|
|
|
|
uint8_t token_cache[1024];
|
2013-02-20 19:16:24 +01:00
|
|
|
|
2013-03-28 18:42:23 +01:00
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
vpx_memset(token_cache, UNKNOWN_TOKEN, sizeof(token_cache));
|
|
|
|
#endif
|
|
|
|
|
2013-02-20 19:16:24 +01:00
|
|
|
// Check for consistency of tx_size with mode info
|
2013-04-04 21:03:27 +02:00
|
|
|
assert((!type && !pb_idx.plane) || (type && pb_idx.plane));
|
2013-02-20 19:16:24 +01:00
|
|
|
if (type == PLANE_TYPE_Y_WITH_DC) {
|
|
|
|
assert(xd->mode_info_context->mbmi.txfm_size == tx_size);
|
|
|
|
} else {
|
|
|
|
TX_SIZE tx_size_uv = get_uv_tx_size(xd);
|
|
|
|
assert(tx_size == tx_size_uv);
|
|
|
|
}
|
|
|
|
|
2012-09-10 07:42:35 +02:00
|
|
|
switch (tx_size) {
|
2013-03-06 00:18:06 +01:00
|
|
|
case TX_4X4: {
|
2013-03-28 18:42:23 +01:00
|
|
|
tx_type = (type == PLANE_TYPE_Y_WITH_DC) ?
|
|
|
|
get_tx_type_4x4(xd, ib) : DCT_DCT;
|
2013-04-29 19:37:25 +02:00
|
|
|
above_ec = A[0] != 0;
|
|
|
|
left_ec = L[0] != 0;
|
2013-03-26 23:23:30 +01:00
|
|
|
coef_probs = cm->fc.coef_probs_4x4;
|
2013-03-27 00:46:09 +01:00
|
|
|
seg_eob = 16;
|
2013-04-22 23:08:41 +02:00
|
|
|
scan = get_scan_4x4(tx_type);
|
2013-03-28 18:42:23 +01:00
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
zpc_probs = &cm->fc.zpc_probs_4x4;
|
|
|
|
#endif
|
2012-08-06 21:15:24 +02:00
|
|
|
break;
|
2013-03-06 00:18:06 +01:00
|
|
|
}
|
2013-03-25 20:30:00 +01:00
|
|
|
case TX_8X8: {
|
|
|
|
const BLOCK_SIZE_TYPE sb_type = xd->mode_info_context->mbmi.sb_type;
|
2013-04-24 18:18:16 +02:00
|
|
|
const int sz = 1 + b_width_log2(sb_type);
|
2013-04-11 01:50:01 +02:00
|
|
|
const int x = ib & ((1 << sz) - 1), y = ib - x;
|
2013-03-28 18:42:23 +01:00
|
|
|
TX_TYPE tx_type = (type == PLANE_TYPE_Y_WITH_DC) ?
|
|
|
|
get_tx_type_8x8(xd, y + (x >> 1)) : DCT_DCT;
|
2013-04-29 19:37:25 +02:00
|
|
|
above_ec = (A[0] + A[1]) != 0;
|
|
|
|
left_ec = (L[0] + L[1]) != 0;
|
2013-04-22 23:08:41 +02:00
|
|
|
scan = get_scan_8x8(tx_type);
|
2013-03-26 23:23:30 +01:00
|
|
|
coef_probs = cm->fc.coef_probs_8x8;
|
2013-03-27 00:46:09 +01:00
|
|
|
seg_eob = 64;
|
2013-03-28 18:42:23 +01:00
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
zpc_probs = &cm->fc.zpc_probs_8x8;
|
|
|
|
#endif
|
2012-08-06 21:15:24 +02:00
|
|
|
break;
|
2013-03-25 20:30:00 +01:00
|
|
|
}
|
|
|
|
case TX_16X16: {
|
|
|
|
const BLOCK_SIZE_TYPE sb_type = xd->mode_info_context->mbmi.sb_type;
|
2013-04-24 18:18:16 +02:00
|
|
|
const int sz = 2 + b_width_log2(sb_type);
|
2013-04-11 01:50:01 +02:00
|
|
|
const int x = ib & ((1 << sz) - 1), y = ib - x;
|
2013-03-28 18:42:23 +01:00
|
|
|
TX_TYPE tx_type = (type == PLANE_TYPE_Y_WITH_DC) ?
|
|
|
|
get_tx_type_16x16(xd, y + (x >> 2)) : DCT_DCT;
|
2013-04-22 23:08:41 +02:00
|
|
|
scan = get_scan_16x16(tx_type);
|
2013-03-26 23:23:30 +01:00
|
|
|
coef_probs = cm->fc.coef_probs_16x16;
|
2013-03-27 00:46:09 +01:00
|
|
|
seg_eob = 256;
|
2013-04-29 19:37:25 +02:00
|
|
|
above_ec = (A[0] + A[1] + A[2] + A[3]) != 0;
|
|
|
|
left_ec = (L[0] + L[1] + L[2] + L[3]) != 0;
|
2013-03-28 18:42:23 +01:00
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
zpc_probs = &cm->fc.zpc_probs_16x16;
|
|
|
|
#endif
|
2012-08-06 21:15:24 +02:00
|
|
|
break;
|
2013-03-25 20:30:00 +01:00
|
|
|
}
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
case TX_32X32:
|
|
|
|
scan = vp9_default_zig_zag1d_32x32;
|
2013-03-26 23:23:30 +01:00
|
|
|
coef_probs = cm->fc.coef_probs_32x32;
|
2013-03-27 00:46:09 +01:00
|
|
|
seg_eob = 1024;
|
2013-04-29 19:37:25 +02:00
|
|
|
above_ec = (A[0] + A[1] + A[2] + A[3] + A[4] + A[5] + A[6] + A[7]) != 0;
|
|
|
|
left_ec = (L[0] + L[1] + L[2] + L[3] + L[4] + L[5] + L[6] + L[7]) != 0;
|
|
|
|
|
2013-03-28 18:42:23 +01:00
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
zpc_probs = &cm->fc.zpc_probs_32x32;
|
|
|
|
#endif
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
break;
|
2012-08-06 21:15:24 +02:00
|
|
|
default:
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
abort();
|
2012-08-06 21:15:24 +02:00
|
|
|
break;
|
|
|
|
}
|
2013-04-04 21:03:27 +02:00
|
|
|
assert(eob <= seg_eob);
|
2012-08-06 20:22:28 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
pt = combine_entropy_contexts(above_ec, left_ec);
|
2013-03-27 00:46:09 +01:00
|
|
|
nb = vp9_get_coef_neighbors_handle(scan, &pad);
|
|
|
|
default_eob = seg_eob;
|
2013-01-03 18:00:30 +01:00
|
|
|
|
2013-04-22 19:58:49 +02:00
|
|
|
if (vp9_segfeature_active(xd, segment_id, SEG_LVL_SKIP))
|
|
|
|
seg_eob = 0;
|
2013-01-03 18:00:30 +01:00
|
|
|
|
2013-04-04 20:07:19 +02:00
|
|
|
/* sanity check to ensure that we do not have spurious non-zero q values */
|
|
|
|
if (eob < seg_eob)
|
|
|
|
assert(qcoeff_ptr[scan[eob]] == 0);
|
|
|
|
|
2013-03-28 18:42:23 +01:00
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
vpx_memset(token_cache_full, ZERO_TOKEN, sizeof(token_cache_full));
|
|
|
|
for (c = 0; c < eob; ++c) {
|
|
|
|
rc = scan[c];
|
|
|
|
token_cache_full[rc] = vp9_dct_value_tokens_ptr[qcoeff_ptr[rc]].token;
|
|
|
|
o = vp9_get_orientation(rc, tx_size);
|
|
|
|
if (qcoeff_ptr[rc] != 0)
|
|
|
|
last_nz_pos[o] = c;
|
|
|
|
}
|
|
|
|
#endif
|
2013-02-19 22:36:38 +01:00
|
|
|
{
|
2013-03-28 18:42:23 +01:00
|
|
|
for (c = 0; c < eob; c++) {
|
2012-09-10 07:42:35 +02:00
|
|
|
int v = qcoeff_ptr[scan[c]];
|
2013-04-19 01:18:08 +02:00
|
|
|
int t = vp9_dct_value_tokens_ptr[v].token;
|
2013-03-28 18:42:23 +01:00
|
|
|
int band = get_coef_band(scan, tx_size, c);
|
|
|
|
if (c)
|
|
|
|
pt = vp9_get_coef_context(scan, nb, pad, token_cache, c, default_eob);
|
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
rc = scan[c];
|
|
|
|
o = vp9_get_orientation(rc, tx_size);
|
|
|
|
skip_coef_val = (token_cache[rc] == ZERO_TOKEN || is_eoo_list[o]);
|
|
|
|
if (!skip_coef_val) {
|
|
|
|
cost += token_costs[band][pt][t] + vp9_dct_value_cost_ptr[v];
|
|
|
|
} else {
|
|
|
|
assert(v == 0);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
cost += token_costs[band][pt][t] + vp9_dct_value_cost_ptr[v];
|
|
|
|
#endif
|
2013-04-22 19:58:49 +02:00
|
|
|
if (!c || token_cache[scan[c - 1]])
|
|
|
|
cost += vp9_cost_bit(coef_probs[type][ref][band][pt][0], 1);
|
2013-03-28 18:42:23 +01:00
|
|
|
token_cache[scan[c]] = t;
|
|
|
|
#if CONFIG_CODE_ZEROGROUP
|
|
|
|
if (t == ZERO_TOKEN && !skip_coef_val) {
|
|
|
|
int eoo = 0, use_eoo;
|
|
|
|
#if USE_ZPC_EOORIENT == 1
|
|
|
|
use_eoo = vp9_use_eoo(c, seg_eob, scan, tx_size,
|
|
|
|
is_last_zero, is_eoo_list);
|
|
|
|
#else
|
|
|
|
use_eoo = 0;
|
|
|
|
#endif
|
|
|
|
if (use_eoo) {
|
|
|
|
eoo = vp9_is_eoo(c, eob, scan, tx_size, qcoeff_ptr, last_nz_pos);
|
|
|
|
if (eoo && is_eoo_negative[o]) eoo = 0;
|
|
|
|
if (eoo) {
|
|
|
|
int c_;
|
|
|
|
int savings = 0;
|
|
|
|
int zsaved = 0;
|
|
|
|
savings = vp9_cost_bit((*zpc_probs)[ref]
|
|
|
|
[coef_to_zpc_band(band)]
|
|
|
|
[coef_to_zpc_ptok(pt)][0], 1) -
|
|
|
|
vp9_cost_bit((*zpc_probs)[ref]
|
|
|
|
[coef_to_zpc_band(band)]
|
|
|
|
[coef_to_zpc_ptok(pt)][0], 0);
|
|
|
|
for (c_ = c + 1; c_ < eob; ++c_) {
|
|
|
|
if (o == vp9_get_orientation(scan[c_], tx_size)) {
|
|
|
|
int pt_ = vp9_get_coef_context(scan, nb, pad,
|
|
|
|
token_cache_full, c_,
|
|
|
|
default_eob);
|
|
|
|
int band_ = get_coef_band(scan, tx_size, c_);
|
|
|
|
assert(token_cache_full[scan[c_]] == ZERO_TOKEN);
|
|
|
|
if (!c_ || token_cache_full[scan[c_ - 1]])
|
|
|
|
savings += vp9_cost_bit(
|
|
|
|
coef_probs[type][ref][band_][pt_][0], 1);
|
|
|
|
savings += vp9_cost_bit(
|
|
|
|
coef_probs[type][ref][band_][pt_][1], 0);
|
|
|
|
zsaved++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (savings < 0) {
|
|
|
|
// if (zsaved < ZPC_ZEROSSAVED_EOO) {
|
|
|
|
eoo = 0;
|
|
|
|
is_eoo_negative[o] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (use_eoo) {
|
|
|
|
cost += vp9_cost_bit((*zpc_probs)[ref]
|
|
|
|
[coef_to_zpc_band(band)]
|
|
|
|
[coef_to_zpc_ptok(pt)][0], !eoo);
|
|
|
|
if (eoo) {
|
|
|
|
assert(is_eoo_list[o] == 0);
|
|
|
|
is_eoo_list[o] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
is_last_zero[o] = (t == ZERO_TOKEN);
|
2013-03-27 00:46:09 +01:00
|
|
|
#endif
|
2012-09-10 07:42:35 +02:00
|
|
|
}
|
2013-04-22 19:58:49 +02:00
|
|
|
if (c < seg_eob) {
|
|
|
|
if (c)
|
|
|
|
pt = vp9_get_coef_context(scan, nb, pad, token_cache, c, default_eob);
|
|
|
|
cost += mb->token_costs[tx_size][type][ref]
|
|
|
|
[get_coef_band(scan, tx_size, c)]
|
|
|
|
[pt][DCT_EOB_TOKEN];
|
|
|
|
}
|
2012-08-06 20:22:28 +02:00
|
|
|
}
|
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
// is eob first coefficient;
|
|
|
|
for (pt = 0; pt < (1 << tx_size); pt++) {
|
|
|
|
A[pt] = L[pt] = c > 0;
|
2013-02-26 02:01:30 +01:00
|
|
|
}
|
2013-04-29 19:37:25 +02:00
|
|
|
|
2012-08-06 20:22:28 +02:00
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
|
2012-11-08 20:03:00 +01:00
|
|
|
static void choose_txfm_size_from_rd(VP9_COMP *cpi, MACROBLOCK *x,
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
int (*r)[2], int *rate,
|
|
|
|
int *d, int *distortion,
|
|
|
|
int *s, int *skip,
|
|
|
|
int64_t txfm_cache[NB_TXFM_MODES],
|
|
|
|
TX_SIZE max_txfm_size) {
|
2012-11-08 20:03:00 +01:00
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
MB_MODE_INFO *const mbmi = &xd->mode_info_context->mbmi;
|
2013-04-09 19:54:19 +02:00
|
|
|
vp9_prob skip_prob = vp9_get_pred_prob(cm, xd, PRED_MBSKIP);
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
int64_t rd[TX_SIZE_MAX_SB][2];
|
|
|
|
int n, m;
|
2013-04-09 19:54:19 +02:00
|
|
|
int s0, s1;
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
|
|
|
|
for (n = TX_4X4; n <= max_txfm_size; n++) {
|
|
|
|
r[n][1] = r[n][0];
|
|
|
|
for (m = 0; m <= n - (n == max_txfm_size); m++) {
|
|
|
|
if (m == n)
|
|
|
|
r[n][1] += vp9_cost_zero(cm->prob_tx[m]);
|
|
|
|
else
|
|
|
|
r[n][1] += vp9_cost_one(cm->prob_tx[m]);
|
|
|
|
}
|
|
|
|
}
|
2012-11-08 20:03:00 +01:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
assert(skip_prob > 0);
|
|
|
|
s0 = vp9_cost_bit(skip_prob, 0);
|
|
|
|
s1 = vp9_cost_bit(skip_prob, 1);
|
2012-11-08 20:03:00 +01:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
for (n = TX_4X4; n <= max_txfm_size; n++) {
|
|
|
|
if (s[n]) {
|
|
|
|
rd[n][0] = rd[n][1] = RDCOST(x->rdmult, x->rddiv, s1, d[n]);
|
|
|
|
} else {
|
|
|
|
rd[n][0] = RDCOST(x->rdmult, x->rddiv, r[n][0] + s0, d[n]);
|
|
|
|
rd[n][1] = RDCOST(x->rdmult, x->rddiv, r[n][1] + s0, d[n]);
|
2012-10-09 18:18:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
if (max_txfm_size == TX_32X32 &&
|
|
|
|
(cm->txfm_mode == ALLOW_32X32 ||
|
|
|
|
(cm->txfm_mode == TX_MODE_SELECT &&
|
|
|
|
rd[TX_32X32][1] < rd[TX_16X16][1] && rd[TX_32X32][1] < rd[TX_8X8][1] &&
|
|
|
|
rd[TX_32X32][1] < rd[TX_4X4][1]))) {
|
|
|
|
mbmi->txfm_size = TX_32X32;
|
2013-05-01 01:13:20 +02:00
|
|
|
} else if (max_txfm_size >= TX_16X16 &&
|
|
|
|
(cm->txfm_mode == ALLOW_16X16 ||
|
|
|
|
cm->txfm_mode == ALLOW_32X32 ||
|
|
|
|
(cm->txfm_mode == TX_MODE_SELECT &&
|
|
|
|
rd[TX_16X16][1] < rd[TX_8X8][1] &&
|
|
|
|
rd[TX_16X16][1] < rd[TX_4X4][1]))) {
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = TX_16X16;
|
2012-11-08 20:03:00 +01:00
|
|
|
} else if (cm->txfm_mode == ALLOW_8X8 ||
|
2013-05-01 01:13:20 +02:00
|
|
|
cm->txfm_mode == ALLOW_16X16 ||
|
|
|
|
cm->txfm_mode == ALLOW_32X32 ||
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
(cm->txfm_mode == TX_MODE_SELECT && rd[TX_8X8][1] < rd[TX_4X4][1])) {
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = TX_8X8;
|
|
|
|
} else {
|
|
|
|
mbmi->txfm_size = TX_4X4;
|
|
|
|
}
|
|
|
|
|
2012-11-08 20:03:00 +01:00
|
|
|
*distortion = d[mbmi->txfm_size];
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
*rate = r[mbmi->txfm_size][cm->txfm_mode == TX_MODE_SELECT];
|
2012-11-08 20:03:00 +01:00
|
|
|
*skip = s[mbmi->txfm_size];
|
|
|
|
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
txfm_cache[ONLY_4X4] = rd[TX_4X4][0];
|
|
|
|
txfm_cache[ALLOW_8X8] = rd[TX_8X8][0];
|
2013-05-01 01:13:20 +02:00
|
|
|
txfm_cache[ALLOW_16X16] = rd[MIN(max_txfm_size, TX_16X16)][0];
|
|
|
|
txfm_cache[ALLOW_32X32] = rd[MIN(max_txfm_size, TX_32X32)][0];
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
if (max_txfm_size == TX_32X32 &&
|
|
|
|
rd[TX_32X32][1] < rd[TX_16X16][1] && rd[TX_32X32][1] < rd[TX_8X8][1] &&
|
|
|
|
rd[TX_32X32][1] < rd[TX_4X4][1])
|
|
|
|
txfm_cache[TX_MODE_SELECT] = rd[TX_32X32][1];
|
2013-05-01 01:13:20 +02:00
|
|
|
else if (max_txfm_size >= TX_16X16 &&
|
|
|
|
rd[TX_16X16][1] < rd[TX_8X8][1] && rd[TX_16X16][1] < rd[TX_4X4][1])
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
txfm_cache[TX_MODE_SELECT] = rd[TX_16X16][1];
|
2012-10-09 18:18:21 +02:00
|
|
|
else
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
txfm_cache[TX_MODE_SELECT] = rd[TX_4X4][1] < rd[TX_8X8][1] ?
|
|
|
|
rd[TX_4X4][1] : rd[TX_8X8][1];
|
2012-11-08 20:03:00 +01:00
|
|
|
}
|
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
static int block_error(int16_t *coeff, int16_t *dqcoeff,
|
|
|
|
int block_size, int shift) {
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
int i;
|
|
|
|
int64_t error = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < block_size; i++) {
|
2013-04-23 18:51:09 +02:00
|
|
|
int this_diff = coeff[i] - dqcoeff[i];
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
error += this_diff * this_diff;
|
|
|
|
}
|
2013-03-06 00:18:06 +01:00
|
|
|
error >>= shift;
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
|
2013-01-18 18:44:23 +01:00
|
|
|
return error > INT_MAX ? INT_MAX : (int)error;
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
}
|
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
static int block_error_sby(MACROBLOCK *x, int block_size, int shift) {
|
|
|
|
return block_error(x->plane[0].coeff, x->e_mbd.plane[0].dqcoeff,
|
|
|
|
block_size, shift);
|
|
|
|
}
|
2013-04-02 23:50:40 +02:00
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
static int block_error_sbuv(MACROBLOCK *x, BLOCK_SIZE_TYPE bsize, int shift) {
|
|
|
|
const int bwl = b_width_log2(bsize), bhl = b_height_log2(bsize);
|
|
|
|
int64_t sum = 0;
|
|
|
|
int plane;
|
2013-04-02 23:50:40 +02:00
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
for (plane = 1; plane < MAX_MB_PLANE; plane++) {
|
|
|
|
const int subsampling = x->e_mbd.plane[plane].subsampling_x +
|
|
|
|
x->e_mbd.plane[plane].subsampling_y;
|
|
|
|
sum += block_error(x->plane[plane].coeff, x->e_mbd.plane[plane].dqcoeff,
|
|
|
|
16 << (bwl + bhl - subsampling), 0);
|
|
|
|
}
|
|
|
|
sum >>= shift;
|
|
|
|
return sum > INT_MAX ? INT_MAX : (int)sum;
|
2013-04-02 23:50:40 +02:00
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rdcost_sby_4x4(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize), bw = 1 << bwl;
|
|
|
|
const int bh = 1 << b_height_log2(bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
int cost = 0, b;
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT t_above[16], t_left[16];
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(&t_above, xd->plane[0].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * bw);
|
|
|
|
vpx_memcpy(&t_left, xd->plane[0].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * bh);
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b & (bw - 1), y_idx = b >> bwl;
|
2013-03-06 00:18:06 +01:00
|
|
|
cost += cost_coeffs(cm, x, b, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx, t_left + y_idx,
|
|
|
|
TX_4X4, bw * bh);
|
2013-04-11 00:55:59 +02:00
|
|
|
}
|
2013-03-06 00:18:06 +01:00
|
|
|
|
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void super_block_yrd_4x4(VP9_COMMON *const cm, MACROBLOCK *x,
|
2013-04-11 00:55:59 +02:00
|
|
|
int *rate, int *distortion, int *skippable,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize), bhl = b_height_log2(bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_4X4;
|
2013-04-11 00:55:59 +02:00
|
|
|
vp9_transform_sby_4x4(x, bsize);
|
|
|
|
vp9_quantize_sby_4x4(x, bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
*distortion = block_error_sby(x, 16 << (bwl + bhl), 2);
|
2013-04-11 00:55:59 +02:00
|
|
|
*rate = rdcost_sby_4x4(cm, x, bsize);
|
2013-04-11 20:14:31 +02:00
|
|
|
*skippable = vp9_sby_is_skippable(xd, bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rdcost_sby_8x8(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize) - 1, bw = 1 << bwl;
|
|
|
|
const int bh = 1 << (b_height_log2(bsize) - 1);
|
2013-03-06 00:18:06 +01:00
|
|
|
int cost = 0, b;
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT t_above[16], t_left[16];
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(&t_above, xd->plane[0].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 2 * bw);
|
|
|
|
vpx_memcpy(&t_left, xd->plane[0].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 2 * bh);
|
2013-04-11 00:55:59 +02:00
|
|
|
|
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b & (bw - 1), y_idx = b >> bwl;
|
|
|
|
cost += cost_coeffs(cm, x, b * 4, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx * 2, t_left + y_idx * 2,
|
|
|
|
TX_8X8, 4 * bw * bh);
|
2013-04-11 00:55:59 +02:00
|
|
|
}
|
2013-03-06 00:18:06 +01:00
|
|
|
|
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void super_block_yrd_8x8(VP9_COMMON *const cm, MACROBLOCK *x,
|
2013-04-11 00:55:59 +02:00
|
|
|
int *rate, int *distortion, int *skippable,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-23 18:51:09 +02:00
|
|
|
const int bwl = b_width_log2(bsize), bhl = b_height_log2(bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_8X8;
|
2013-04-11 00:55:59 +02:00
|
|
|
vp9_transform_sby_8x8(x, bsize);
|
|
|
|
vp9_quantize_sby_8x8(x, bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
*distortion = block_error_sby(x, 16 << (bhl + bwl), 2);
|
2013-04-11 00:55:59 +02:00
|
|
|
*rate = rdcost_sby_8x8(cm, x, bsize);
|
2013-04-11 20:14:31 +02:00
|
|
|
*skippable = vp9_sby_is_skippable(xd, bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rdcost_sby_16x16(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize) - 2, bw = 1 << bwl;
|
|
|
|
const int bh = 1 << (b_height_log2(bsize) - 2);
|
2013-03-06 00:18:06 +01:00
|
|
|
int cost = 0, b;
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT t_above[16], t_left[16];
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(&t_above, xd->plane[0].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 4 * bw);
|
|
|
|
vpx_memcpy(&t_left, xd->plane[0].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 4 * bh);
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b & (bw - 1), y_idx = b >> bwl;
|
|
|
|
cost += cost_coeffs(cm, x, b * 16, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx * 4, t_left + y_idx * 4,
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_16X16, bw * bh * 16);
|
|
|
|
}
|
2013-03-06 00:18:06 +01:00
|
|
|
|
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void super_block_yrd_16x16(VP9_COMMON *const cm, MACROBLOCK *x,
|
2013-04-11 00:55:59 +02:00
|
|
|
int *rate, int *distortion, int *skippable,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-23 18:51:09 +02:00
|
|
|
const int bwl = b_width_log2(bsize), bhl = b_height_log2(bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_16X16;
|
2013-04-11 00:55:59 +02:00
|
|
|
vp9_transform_sby_16x16(x, bsize);
|
|
|
|
vp9_quantize_sby_16x16(x, bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
*distortion = block_error_sby(x, 16 << (bwl + bhl), 2);
|
2013-04-11 00:55:59 +02:00
|
|
|
*rate = rdcost_sby_16x16(cm, x, bsize);
|
2013-04-11 20:14:31 +02:00
|
|
|
*skippable = vp9_sby_is_skippable(xd, bsize);
|
2013-03-06 00:18:06 +01:00
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rdcost_sby_32x32(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize) - 3, bw = 1 << bwl;
|
|
|
|
const int bh = 1 << (b_height_log2(bsize) - 3);
|
2013-04-11 00:55:59 +02:00
|
|
|
int cost = 0, b;
|
2013-03-06 00:18:06 +01:00
|
|
|
MACROBLOCKD * const xd = &x->e_mbd;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT t_above[16], t_left[16];
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(&t_above, xd->plane[0].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 8 * bw);
|
|
|
|
vpx_memcpy(&t_left, xd->plane[0].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 8 * bh);
|
2013-04-11 00:55:59 +02:00
|
|
|
|
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b & (bw - 1), y_idx = b >> bwl;
|
|
|
|
cost += cost_coeffs(cm, x, b * 64, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx * 8, t_left + y_idx * 8,
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_32X32, bw * bh * 64);
|
|
|
|
}
|
2013-03-06 00:18:06 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
return cost;
|
2013-03-06 00:18:06 +01:00
|
|
|
}
|
|
|
|
|
2013-02-20 19:16:24 +01:00
|
|
|
static void super_block_yrd_32x32(VP9_COMMON *const cm, MACROBLOCK *x,
|
2013-04-11 00:55:59 +02:00
|
|
|
int *rate, int *distortion, int *skippable,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-23 18:51:09 +02:00
|
|
|
const int bwl = b_width_log2(bsize), bhl = b_height_log2(bsize);
|
2013-03-04 23:12:17 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
|
2013-03-06 00:18:06 +01:00
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_32X32;
|
2013-04-11 00:55:59 +02:00
|
|
|
vp9_transform_sby_32x32(x, bsize);
|
|
|
|
vp9_quantize_sby_32x32(x, bsize);
|
2013-02-11 13:35:28 +01:00
|
|
|
|
2013-04-23 18:51:09 +02:00
|
|
|
*distortion = block_error_sby(x, 16 << (bwl + bhl), 0);
|
2013-04-11 00:55:59 +02:00
|
|
|
*rate = rdcost_sby_32x32(cm, x, bsize);
|
2013-04-11 20:14:31 +02:00
|
|
|
*skippable = vp9_sby_is_skippable(xd, bsize);
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
}
|
|
|
|
|
2012-11-08 20:03:00 +01:00
|
|
|
static void super_block_yrd(VP9_COMP *cpi,
|
|
|
|
MACROBLOCK *x, int *rate, int *distortion,
|
2013-04-11 00:55:59 +02:00
|
|
|
int *skip, BLOCK_SIZE_TYPE bs,
|
2012-11-08 20:03:00 +01:00
|
|
|
int64_t txfm_cache[NB_TXFM_MODES]) {
|
2013-03-06 00:18:06 +01:00
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
|
|
|
int r[TX_SIZE_MAX_SB][2], d[TX_SIZE_MAX_SB], s[TX_SIZE_MAX_SB];
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-23 17:26:10 +02:00
|
|
|
vp9_subtract_sby(x, bs);
|
2013-01-06 03:20:25 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
if (bs >= BLOCK_SIZE_SB32X32)
|
|
|
|
super_block_yrd_32x32(cm, x, &r[TX_32X32][0], &d[TX_32X32], &s[TX_32X32],
|
|
|
|
bs);
|
2013-05-01 01:13:20 +02:00
|
|
|
if (bs >= BLOCK_SIZE_MB16X16)
|
|
|
|
super_block_yrd_16x16(cm, x, &r[TX_16X16][0], &d[TX_16X16], &s[TX_16X16],
|
|
|
|
bs);
|
2013-04-11 00:55:59 +02:00
|
|
|
super_block_yrd_8x8(cm, x, &r[TX_8X8][0], &d[TX_8X8], &s[TX_8X8], bs);
|
|
|
|
super_block_yrd_4x4(cm, x, &r[TX_4X4][0], &d[TX_4X4], &s[TX_4X4], bs);
|
2013-01-06 03:20:25 +01:00
|
|
|
|
|
|
|
choose_txfm_size_from_rd(cpi, x, r, rate, d, distortion, s, skip, txfm_cache,
|
2013-05-01 01:13:20 +02:00
|
|
|
TX_32X32 - (bs < BLOCK_SIZE_SB32X32)
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
- (bs < BLOCK_SIZE_MB16X16)
|
|
|
|
#endif
|
|
|
|
);
|
2013-01-06 03:20:25 +01:00
|
|
|
}
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-02 23:50:40 +02:00
|
|
|
static int64_t rd_pick_intra4x4block(VP9_COMP *cpi, MACROBLOCK *x, int ib,
|
|
|
|
B_PREDICTION_MODE *best_mode,
|
2012-08-07 01:21:23 +02:00
|
|
|
int *bmode_costs,
|
|
|
|
ENTROPY_CONTEXT *a, ENTROPY_CONTEXT *l,
|
|
|
|
int *bestrate, int *bestratey,
|
|
|
|
int *bestdistortion) {
|
2012-07-14 00:21:29 +02:00
|
|
|
B_PREDICTION_MODE mode;
|
2012-10-16 01:41:41 +02:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_rd = INT64_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
int rate = 0;
|
|
|
|
int distortion;
|
2013-02-20 19:16:24 +01:00
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
2013-04-24 01:22:47 +02:00
|
|
|
const int src_stride = x->plane[0].src.stride;
|
|
|
|
uint8_t* const src =
|
2013-05-01 01:13:20 +02:00
|
|
|
raster_block_offset_uint8(xd,
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
BLOCK_SIZE_SB8X8,
|
|
|
|
#else
|
|
|
|
BLOCK_SIZE_MB16X16,
|
|
|
|
#endif
|
|
|
|
0, ib,
|
2013-04-24 01:22:47 +02:00
|
|
|
x->plane[0].src.buf, src_stride);
|
2013-04-23 17:26:10 +02:00
|
|
|
int16_t* const src_diff =
|
2013-05-01 01:13:20 +02:00
|
|
|
raster_block_offset_int16(xd,
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
BLOCK_SIZE_SB8X8,
|
|
|
|
#else
|
|
|
|
BLOCK_SIZE_MB16X16,
|
|
|
|
#endif
|
|
|
|
0, ib,
|
2013-04-23 17:26:10 +02:00
|
|
|
x->plane[0].src_diff);
|
2013-04-25 00:01:35 +02:00
|
|
|
int16_t* const diff =
|
2013-05-01 01:13:20 +02:00
|
|
|
raster_block_offset_int16(xd,
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
BLOCK_SIZE_SB8X8,
|
|
|
|
#else
|
|
|
|
BLOCK_SIZE_MB16X16,
|
|
|
|
#endif
|
|
|
|
0, ib,
|
2013-04-25 00:01:35 +02:00
|
|
|
xd->plane[0].diff);
|
2013-04-23 23:38:52 +02:00
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff, ib, 16);
|
2013-04-25 20:15:38 +02:00
|
|
|
uint8_t* const dst =
|
2013-05-01 01:13:20 +02:00
|
|
|
raster_block_offset_uint8(xd,
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
BLOCK_SIZE_SB8X8,
|
|
|
|
#else
|
|
|
|
BLOCK_SIZE_MB16X16,
|
|
|
|
#endif
|
|
|
|
0, ib,
|
2013-04-25 20:15:38 +02:00
|
|
|
xd->plane[0].dst.buf, xd->plane[0].dst.stride);
|
2012-07-14 00:21:29 +02:00
|
|
|
ENTROPY_CONTEXT ta = *a, tempa = *a;
|
|
|
|
ENTROPY_CONTEXT tl = *l, templ = *l;
|
2012-10-16 01:41:41 +02:00
|
|
|
TX_TYPE tx_type = DCT_DCT;
|
|
|
|
TX_TYPE best_tx_type = DCT_DCT;
|
2012-07-14 00:21:29 +02:00
|
|
|
/*
|
|
|
|
* The predictor buffer is a 2d buffer with a stride of 16. Create
|
|
|
|
* a temp buffer that meets the stride requirements, but we are only
|
|
|
|
* interested in the left 4x4 block
|
|
|
|
* */
|
2012-12-19 00:31:19 +01:00
|
|
|
DECLARE_ALIGNED_ARRAY(16, int16_t, best_dqcoeff, 16);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
assert(ib < (16 >> (2 * CONFIG_SB8X8)));
|
2012-10-09 22:19:15 +02:00
|
|
|
#if CONFIG_NEWBINTRAMODES
|
2013-04-26 16:19:43 +02:00
|
|
|
xd->mode_info_context->bmi[ib].as_mode.context =
|
2013-04-25 20:15:38 +02:00
|
|
|
vp9_find_bpred_context(xd, ib, dst, xd->plane[0].dst.stride);
|
2012-10-09 22:19:15 +02:00
|
|
|
#endif
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_4X4;
|
2012-10-09 22:19:15 +02:00
|
|
|
for (mode = B_DC_PRED; mode < LEFT4X4; mode++) {
|
2013-01-14 23:37:53 +01:00
|
|
|
int64_t this_rd;
|
|
|
|
int ratey;
|
2012-10-09 22:19:15 +02:00
|
|
|
|
|
|
|
#if CONFIG_NEWBINTRAMODES
|
2013-01-14 23:37:53 +01:00
|
|
|
if (xd->frame_type == KEY_FRAME) {
|
|
|
|
if (mode == B_CONTEXT_PRED) continue;
|
|
|
|
} else {
|
|
|
|
if (mode >= B_CONTEXT_PRED - CONTEXT_PRED_REPLACEMENTS &&
|
|
|
|
mode < B_CONTEXT_PRED)
|
|
|
|
continue;
|
|
|
|
}
|
2012-10-09 22:19:15 +02:00
|
|
|
#endif
|
2011-02-14 19:32:58 +01:00
|
|
|
|
2013-04-26 16:19:43 +02:00
|
|
|
xd->mode_info_context->bmi[ib].as_mode.first = mode;
|
2012-10-09 22:19:15 +02:00
|
|
|
#if CONFIG_NEWBINTRAMODES
|
2013-01-14 23:37:53 +01:00
|
|
|
rate = bmode_costs[
|
|
|
|
mode == B_CONTEXT_PRED ? mode - CONTEXT_PRED_REPLACEMENTS : mode];
|
2012-10-09 22:19:15 +02:00
|
|
|
#else
|
2013-01-14 23:37:53 +01:00
|
|
|
rate = bmode_costs[mode];
|
2012-10-09 22:19:15 +02:00
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
vp9_intra4x4_predict(xd, ib, mode, dst, xd->plane[0].dst.stride);
|
2013-05-01 01:13:20 +02:00
|
|
|
vp9_subtract_block(4, 4, src_diff, 16 >> CONFIG_SB8X8,
|
2013-04-24 01:22:47 +02:00
|
|
|
src, src_stride,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst, xd->plane[0].dst.stride);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-26 16:19:43 +02:00
|
|
|
xd->mode_info_context->bmi[ib].as_mode.first = mode;
|
2013-04-24 01:22:47 +02:00
|
|
|
tx_type = get_tx_type_4x4(xd, ib);
|
2013-01-14 23:37:53 +01:00
|
|
|
if (tx_type != DCT_DCT) {
|
2013-05-01 01:13:20 +02:00
|
|
|
vp9_short_fht4x4(src_diff, coeff, 16 >> CONFIG_SB8X8, tx_type);
|
2013-04-25 23:44:44 +02:00
|
|
|
x->quantize_b_4x4(x, ib, tx_type, 16);
|
2013-01-14 23:37:53 +01:00
|
|
|
} else {
|
2013-05-01 01:13:20 +02:00
|
|
|
x->fwd_txm4x4(src_diff, coeff, 32 >> CONFIG_SB8X8);
|
2013-04-25 23:44:44 +02:00
|
|
|
x->quantize_b_4x4(x, ib, tx_type, 16);
|
2013-01-14 23:37:53 +01:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
tempa = ta;
|
|
|
|
templ = tl;
|
2012-06-25 21:26:09 +02:00
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
ratey = cost_coeffs(cm, x, ib,
|
2013-04-04 21:03:27 +02:00
|
|
|
PLANE_TYPE_Y_WITH_DC, &tempa, &templ, TX_4X4, 16);
|
2013-01-14 23:37:53 +01:00
|
|
|
rate += ratey;
|
2013-04-23 23:38:52 +02:00
|
|
|
distortion = vp9_block_error(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, ib, 16),
|
|
|
|
16) >> 2;
|
2012-06-25 21:26:09 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
this_rd = RDCOST(x->rdmult, x->rddiv, rate, distortion);
|
2012-06-25 21:26:09 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
if (this_rd < best_rd) {
|
|
|
|
*bestrate = rate;
|
|
|
|
*bestratey = ratey;
|
|
|
|
*bestdistortion = distortion;
|
|
|
|
best_rd = this_rd;
|
|
|
|
*best_mode = mode;
|
|
|
|
best_tx_type = tx_type;
|
|
|
|
*a = tempa;
|
|
|
|
*l = templ;
|
2013-04-02 23:50:40 +02:00
|
|
|
vpx_memcpy(best_dqcoeff, BLOCK_OFFSET(xd->plane[0].dqcoeff, ib, 16), 32);
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2013-04-26 16:19:43 +02:00
|
|
|
xd->mode_info_context->bmi[ib].as_mode.first =
|
|
|
|
(B_PREDICTION_MODE)(*best_mode);
|
2011-02-14 19:32:58 +01:00
|
|
|
|
2012-06-25 21:26:09 +02:00
|
|
|
// inverse transform
|
2012-10-16 01:41:41 +02:00
|
|
|
if (best_tx_type != DCT_DCT)
|
2013-05-01 01:13:20 +02:00
|
|
|
vp9_short_iht4x4(best_dqcoeff, diff, 16 >> CONFIG_SB8X8, best_tx_type);
|
2012-08-07 01:21:23 +02:00
|
|
|
else
|
2013-05-01 01:13:20 +02:00
|
|
|
xd->inv_txm4x4(best_dqcoeff, diff, 32 >> CONFIG_SB8X8);
|
2012-06-25 21:26:09 +02:00
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
vp9_intra4x4_predict(xd, ib, *best_mode,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst, xd->plane[0].dst.stride);
|
|
|
|
vp9_recon_b(dst, diff,
|
|
|
|
dst, xd->plane[0].dst.stride);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return best_rd;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
static int64_t rd_pick_intra4x4mby_modes(VP9_COMP *cpi, MACROBLOCK *mb,
|
|
|
|
int *Rate, int *rate_y,
|
2013-01-25 01:28:53 +01:00
|
|
|
int *Distortion, int64_t best_rd) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int i;
|
|
|
|
MACROBLOCKD *const xd = &mb->e_mbd;
|
2013-04-11 21:16:35 +02:00
|
|
|
int cost = mb->mbmode_cost[xd->frame_type][I4X4_PRED];
|
2012-07-14 00:21:29 +02:00
|
|
|
int distortion = 0;
|
|
|
|
int tot_rate_y = 0;
|
|
|
|
int64_t total_rd = 0;
|
2013-05-01 01:13:20 +02:00
|
|
|
ENTROPY_CONTEXT t_above[4 >> CONFIG_SB8X8], t_left[4 >> CONFIG_SB8X8];
|
2012-07-14 00:21:29 +02:00
|
|
|
int *bmode_costs;
|
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, xd->plane[0].above_context, sizeof(t_above));
|
|
|
|
vpx_memcpy(t_left, xd->plane[0].left_context, sizeof(t_left));
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-04-11 21:16:35 +02:00
|
|
|
xd->mode_info_context->mbmi.mode = I4X4_PRED;
|
2012-07-14 00:21:29 +02:00
|
|
|
bmode_costs = mb->inter_bmode_costs;
|
2011-02-14 19:32:58 +01:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
for (i = 0; i < (16 >> (2 * CONFIG_SB8X8)); i++) {
|
|
|
|
const int x_idx = i & (3 >> CONFIG_SB8X8), y_idx = i >> (2 >> CONFIG_SB8X8);
|
2012-07-14 00:21:29 +02:00
|
|
|
MODE_INFO *const mic = xd->mode_info_context;
|
|
|
|
const int mis = xd->mode_info_stride;
|
|
|
|
B_PREDICTION_MODE UNINITIALIZED_IS_SAFE(best_mode);
|
|
|
|
int UNINITIALIZED_IS_SAFE(r), UNINITIALIZED_IS_SAFE(ry), UNINITIALIZED_IS_SAFE(d);
|
2013-04-25 20:15:38 +02:00
|
|
|
#if CONFIG_NEWBINTRAMODES
|
|
|
|
uint8_t* const dst =
|
2013-05-01 01:13:20 +02:00
|
|
|
raster_block_offset_uint8(xd,
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
BLOCK_SIZE_SB8X8,
|
|
|
|
#else
|
|
|
|
BLOCK_SIZE_MB16X16,
|
|
|
|
#endif
|
|
|
|
0, i,
|
2013-04-25 20:15:38 +02:00
|
|
|
xd->plane[0].dst.buf,
|
|
|
|
xd->plane[0].dst.stride);
|
|
|
|
#endif
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-08-10 15:12:43 +02:00
|
|
|
if (xd->frame_type == KEY_FRAME) {
|
2012-07-14 00:21:29 +02:00
|
|
|
const B_PREDICTION_MODE A = above_block_mode(mic, i, mis);
|
|
|
|
const B_PREDICTION_MODE L = left_block_mode(mic, i);
|
2011-02-14 19:32:58 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
bmode_costs = mb->bmode_costs[A][L];
|
|
|
|
}
|
2012-10-09 22:19:15 +02:00
|
|
|
#if CONFIG_NEWBINTRAMODES
|
2013-04-25 20:15:38 +02:00
|
|
|
mic->bmi[i].as_mode.context = vp9_find_bpred_context(xd, i, dst,
|
|
|
|
xd->plane[0].dst.stride);
|
2012-10-09 22:19:15 +02:00
|
|
|
#endif
|
2011-02-14 19:32:58 +01:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
total_rd += rd_pick_intra4x4block(cpi, mb, i, &best_mode, bmode_costs,
|
|
|
|
t_above + x_idx, t_left + y_idx,
|
|
|
|
&r, &ry, &d);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
cost += r;
|
|
|
|
distortion += d;
|
|
|
|
tot_rate_y += ry;
|
2011-05-24 19:24:52 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
mic->bmi[i].as_mode.first = best_mode;
|
2011-02-08 22:50:43 +01:00
|
|
|
|
2012-10-09 22:19:15 +02:00
|
|
|
#if 0 // CONFIG_NEWBINTRAMODES
|
|
|
|
printf("%d %d\n", mic->bmi[i].as_mode.first, mic->bmi[i].as_mode.context);
|
|
|
|
#endif
|
|
|
|
|
2012-08-02 19:07:33 +02:00
|
|
|
if (total_rd >= best_rd)
|
2012-07-14 00:21:29 +02:00
|
|
|
break;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-08-02 19:07:33 +02:00
|
|
|
if (total_rd >= best_rd)
|
2013-01-14 20:49:30 +01:00
|
|
|
return INT64_MAX;
|
2011-02-08 22:50:43 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
*Rate = cost;
|
2012-11-05 21:50:16 +01:00
|
|
|
*rate_y = tot_rate_y;
|
2012-07-14 00:21:29 +02:00
|
|
|
*Distortion = distortion;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return RDCOST(mb->rdmult, mb->rddiv, cost, distortion);
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2011-06-08 18:05:05 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int64_t rd_pick_intra_sby_mode(VP9_COMP *cpi, MACROBLOCK *x,
|
|
|
|
int *rate, int *rate_tokenonly,
|
|
|
|
int *distortion, int *skippable,
|
|
|
|
BLOCK_SIZE_TYPE bsize,
|
2012-11-08 20:03:00 +01:00
|
|
|
int64_t txfm_cache[NB_TXFM_MODES]) {
|
2012-08-20 23:43:34 +02:00
|
|
|
MB_PREDICTION_MODE mode;
|
|
|
|
MB_PREDICTION_MODE UNINITIALIZED_IS_SAFE(mode_selected);
|
|
|
|
int this_rate, this_rate_tokenonly;
|
2012-08-29 19:43:20 +02:00
|
|
|
int this_distortion, s;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_rd = INT64_MAX, this_rd;
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_SIZE UNINITIALIZED_IS_SAFE(best_tx);
|
|
|
|
int i;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++)
|
|
|
|
txfm_cache[i] = INT64_MAX;
|
2013-01-06 03:20:25 +01:00
|
|
|
|
|
|
|
/* Y Search for 32x32 intra prediction mode */
|
|
|
|
for (mode = DC_PRED; mode <= TM_PRED; mode++) {
|
2013-04-11 00:55:59 +02:00
|
|
|
int64_t local_txfm_cache[NB_TXFM_MODES];
|
|
|
|
|
2013-01-06 03:20:25 +01:00
|
|
|
x->e_mbd.mode_info_context->mbmi.mode = mode;
|
2013-04-15 18:31:27 +02:00
|
|
|
vp9_build_intra_predictors_sby_s(&x->e_mbd, bsize);
|
2013-01-06 03:20:25 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
super_block_yrd(cpi, x, &this_rate_tokenonly, &this_distortion, &s,
|
|
|
|
bsize, local_txfm_cache);
|
|
|
|
this_rate = this_rate_tokenonly + x->mbmode_cost[x->e_mbd.frame_type][mode];
|
2013-01-06 03:20:25 +01:00
|
|
|
this_rd = RDCOST(x->rdmult, x->rddiv, this_rate, this_distortion);
|
|
|
|
|
|
|
|
if (this_rd < best_rd) {
|
|
|
|
mode_selected = mode;
|
|
|
|
best_rd = this_rd;
|
2013-04-11 00:55:59 +02:00
|
|
|
best_tx = x->e_mbd.mode_info_context->mbmi.txfm_size;
|
2013-01-06 03:20:25 +01:00
|
|
|
*rate = this_rate;
|
|
|
|
*rate_tokenonly = this_rate_tokenonly;
|
|
|
|
*distortion = this_distortion;
|
|
|
|
*skippable = s;
|
|
|
|
}
|
2012-10-09 18:18:21 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
int64_t adj_rd = this_rd + local_txfm_cache[i] -
|
2013-04-11 00:55:59 +02:00
|
|
|
local_txfm_cache[cpi->common.txfm_mode];
|
2013-01-14 23:37:53 +01:00
|
|
|
if (adj_rd < txfm_cache[i]) {
|
|
|
|
txfm_cache[i] = adj_rd;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
x->e_mbd.mode_info_context->mbmi.mode = mode_selected;
|
|
|
|
x->e_mbd.mode_info_context->mbmi.txfm_size = best_tx;
|
2012-08-29 20:25:38 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return best_rd;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-06-25 21:26:09 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#if !CONFIG_SB8X8
|
2012-10-31 01:53:32 +01:00
|
|
|
static int64_t rd_pick_intra8x8block(VP9_COMP *cpi, MACROBLOCK *x, int ib,
|
2012-08-07 01:21:23 +02:00
|
|
|
B_PREDICTION_MODE *best_mode,
|
|
|
|
int *mode_costs,
|
|
|
|
ENTROPY_CONTEXT *a, ENTROPY_CONTEXT *l,
|
|
|
|
int *bestrate, int *bestratey,
|
|
|
|
int *bestdistortion) {
|
2013-02-20 19:16:24 +01:00
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
2012-07-14 00:21:29 +02:00
|
|
|
MB_PREDICTION_MODE mode;
|
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_rd = INT64_MAX;
|
2012-11-16 00:14:38 +01:00
|
|
|
int distortion = 0, rate = 0;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT ta[2], tl[2], ta_temp[2], tl_temp[2];
|
2012-08-01 19:18:25 +02:00
|
|
|
// perform transformation of dimension 8x8
|
|
|
|
// note the input and output index mapping
|
|
|
|
int idx = (ib & 0x02) ? (ib + 2) : ib;
|
2013-04-24 01:22:47 +02:00
|
|
|
const int src_stride = x->plane[0].src.stride;
|
|
|
|
uint8_t* const src =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, ib,
|
|
|
|
x->plane[0].src.buf, src_stride);
|
2013-04-23 17:26:10 +02:00
|
|
|
int16_t* const src_diff =
|
|
|
|
raster_block_offset_int16(xd, BLOCK_SIZE_MB16X16, 0, ib,
|
|
|
|
x->plane[0].src_diff);
|
2013-04-23 23:38:52 +02:00
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff, idx, 16);
|
2013-04-25 20:15:38 +02:00
|
|
|
uint8_t* const dst =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, ib,
|
|
|
|
xd->plane[0].dst.buf, xd->plane[0].dst.stride);
|
2012-08-01 19:18:25 +02:00
|
|
|
|
2013-04-02 23:50:40 +02:00
|
|
|
assert(ib < 16);
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(ta, a, sizeof(ta));
|
|
|
|
vpx_memcpy(tl, l, sizeof(tl));
|
2012-07-14 00:21:29 +02:00
|
|
|
for (mode = DC_PRED; mode <= TM_PRED; mode++) {
|
2013-01-14 23:37:53 +01:00
|
|
|
int64_t this_rd;
|
|
|
|
int rate_t = 0;
|
2011-08-05 01:30:27 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
// FIXME rate for compound mode and second intrapred mode
|
|
|
|
rate = mode_costs[mode];
|
2013-04-26 16:19:43 +02:00
|
|
|
xd->mode_info_context->bmi[ib].as_mode.first = mode;
|
2011-08-05 01:30:27 +02:00
|
|
|
|
2013-04-25 20:15:38 +02:00
|
|
|
vp9_intra8x8_predict(xd, ib, mode, dst, xd->plane[0].dst.stride);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-23 17:26:10 +02:00
|
|
|
vp9_subtract_block(8, 8, src_diff, 16,
|
2013-04-24 01:22:47 +02:00
|
|
|
src, src_stride,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst, xd->plane[0].dst.stride);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(ta_temp, ta, sizeof(ta));
|
|
|
|
vpx_memcpy(tl_temp, tl, sizeof(tl));
|
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
if (xd->mode_info_context->mbmi.txfm_size == TX_8X8) {
|
2013-03-06 00:18:06 +01:00
|
|
|
TX_TYPE tx_type = get_tx_type_8x8(xd, ib);
|
2013-01-14 23:37:53 +01:00
|
|
|
if (tx_type != DCT_DCT)
|
2013-04-23 23:38:52 +02:00
|
|
|
vp9_short_fht8x8(src_diff, coeff, 16, tx_type);
|
2013-01-14 23:37:53 +01:00
|
|
|
else
|
2013-04-23 23:38:52 +02:00
|
|
|
x->fwd_txm8x8(src_diff, coeff, 32);
|
2013-04-04 21:03:27 +02:00
|
|
|
x->quantize_b_8x8(x, idx, tx_type, 16);
|
2012-09-21 23:20:15 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
// compute quantization mse of 8x8 block
|
2013-04-23 23:38:52 +02:00
|
|
|
distortion = vp9_block_error_c(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, idx, 16), 64);
|
2013-02-26 02:01:30 +01:00
|
|
|
|
2013-02-20 19:16:24 +01:00
|
|
|
rate_t = cost_coeffs(cm, x, idx, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
ta_temp, tl_temp, TX_8X8, 16);
|
2012-08-01 19:18:25 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
rate += rate_t;
|
|
|
|
} else {
|
|
|
|
static const int iblock[4] = {0, 1, 4, 5};
|
|
|
|
TX_TYPE tx_type;
|
|
|
|
int i;
|
2013-04-29 19:37:25 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
distortion = 0;
|
|
|
|
rate_t = 0;
|
|
|
|
for (i = 0; i < 4; ++i) {
|
2013-04-23 17:26:10 +02:00
|
|
|
int16_t* const src_diff =
|
|
|
|
raster_block_offset_int16(xd, BLOCK_SIZE_MB16X16,
|
|
|
|
0, ib + iblock[i],
|
|
|
|
x->plane[0].src_diff);
|
2013-04-23 23:38:52 +02:00
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff,
|
|
|
|
ib + iblock[i], 16);
|
2013-02-06 23:13:05 +01:00
|
|
|
int do_two = 0;
|
2013-03-06 00:18:06 +01:00
|
|
|
tx_type = get_tx_type_4x4(xd, ib + iblock[i]);
|
2013-01-14 23:37:53 +01:00
|
|
|
if (tx_type != DCT_DCT) {
|
2013-04-23 23:38:52 +02:00
|
|
|
vp9_short_fht4x4(src_diff, coeff, 16, tx_type);
|
2013-04-25 23:44:44 +02:00
|
|
|
x->quantize_b_4x4(x, ib + iblock[i], tx_type, 16);
|
2013-03-06 00:18:06 +01:00
|
|
|
} else if (!(i & 1) &&
|
|
|
|
get_tx_type_4x4(xd, ib + iblock[i] + 1) == DCT_DCT) {
|
2013-04-23 23:38:52 +02:00
|
|
|
x->fwd_txm8x4(src_diff, coeff, 32);
|
2013-04-04 21:03:27 +02:00
|
|
|
x->quantize_b_4x4_pair(x, ib + iblock[i], ib + iblock[i] + 1, 16);
|
2013-02-06 23:13:05 +01:00
|
|
|
do_two = 1;
|
2013-01-14 23:37:53 +01:00
|
|
|
} else {
|
2013-04-23 23:38:52 +02:00
|
|
|
x->fwd_txm4x4(src_diff, coeff, 32);
|
2013-04-25 23:44:44 +02:00
|
|
|
x->quantize_b_4x4(x, ib + iblock[i], tx_type, 16);
|
2012-11-16 00:14:38 +01:00
|
|
|
}
|
2013-04-23 23:38:52 +02:00
|
|
|
distortion += vp9_block_error_c(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, ib + iblock[i], 16),
|
|
|
|
16 << do_two);
|
2013-02-20 19:16:24 +01:00
|
|
|
rate_t += cost_coeffs(cm, x, ib + iblock[i], PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
&ta_temp[i & 1], &tl_temp[i >> 1],
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_4X4, 16);
|
2013-02-06 23:13:05 +01:00
|
|
|
if (do_two) {
|
2013-02-26 02:01:30 +01:00
|
|
|
i++;
|
2013-02-20 19:16:24 +01:00
|
|
|
rate_t += cost_coeffs(cm, x, ib + iblock[i], PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
&ta_temp[i & 1], &tl_temp[i >> 1],
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_4X4, 16);
|
2013-02-06 23:13:05 +01:00
|
|
|
}
|
2012-10-11 02:18:22 +02:00
|
|
|
}
|
2013-01-14 23:37:53 +01:00
|
|
|
rate += rate_t;
|
|
|
|
}
|
2012-08-01 19:18:25 +02:00
|
|
|
|
2013-01-14 23:37:53 +01:00
|
|
|
distortion >>= 2;
|
|
|
|
this_rd = RDCOST(x->rdmult, x->rddiv, rate, distortion);
|
|
|
|
if (this_rd < best_rd) {
|
|
|
|
*bestrate = rate;
|
|
|
|
*bestratey = rate_t;
|
|
|
|
*bestdistortion = distortion;
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(a, ta_temp, sizeof(ta_temp));
|
|
|
|
vpx_memcpy(l, tl_temp, sizeof(tl_temp));
|
2013-01-14 23:37:53 +01:00
|
|
|
best_rd = this_rd;
|
|
|
|
*best_mode = mode;
|
2011-08-05 01:30:27 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2013-04-26 16:19:43 +02:00
|
|
|
xd->mode_info_context->bmi[ib].as_mode.first = (*best_mode);
|
2012-11-25 04:33:58 +01:00
|
|
|
vp9_encode_intra8x8(x, ib);
|
2012-08-01 19:18:25 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return best_rd;
|
2011-08-05 01:30:27 +02:00
|
|
|
}
|
|
|
|
|
2012-10-31 01:53:32 +01:00
|
|
|
static int64_t rd_pick_intra8x8mby_modes(VP9_COMP *cpi, MACROBLOCK *mb,
|
2012-10-30 02:04:33 +01:00
|
|
|
int *Rate, int *rate_y,
|
|
|
|
int *Distortion, int64_t best_rd) {
|
2012-07-14 00:21:29 +02:00
|
|
|
MACROBLOCKD *const xd = &mb->e_mbd;
|
|
|
|
int i, ib;
|
|
|
|
int cost = mb->mbmode_cost [xd->frame_type] [I8X8_PRED];
|
|
|
|
int distortion = 0;
|
|
|
|
int tot_rate_y = 0;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t total_rd = 0;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT t_above[4], t_left[4];
|
2012-07-14 00:21:29 +02:00
|
|
|
int *i8x8mode_costs;
|
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, xd->plane[0].above_context, sizeof(t_above));
|
|
|
|
vpx_memcpy(t_left, xd->plane[0].left_context, sizeof(t_left));
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-10-15 22:49:45 +02:00
|
|
|
xd->mode_info_context->mbmi.mode = I8X8_PRED;
|
2012-07-14 00:21:29 +02:00
|
|
|
i8x8mode_costs = mb->i8x8_mode_costs;
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++) {
|
2013-04-29 19:37:25 +02:00
|
|
|
const int x_idx = i & 1, y_idx = i >> 1;
|
2012-07-14 00:21:29 +02:00
|
|
|
MODE_INFO *const mic = xd->mode_info_context;
|
|
|
|
B_PREDICTION_MODE UNINITIALIZED_IS_SAFE(best_mode);
|
|
|
|
int UNINITIALIZED_IS_SAFE(r), UNINITIALIZED_IS_SAFE(ry), UNINITIALIZED_IS_SAFE(d);
|
2011-08-05 01:30:27 +02:00
|
|
|
|
2012-10-31 01:12:12 +01:00
|
|
|
ib = vp9_i8x8_block[i];
|
2013-04-29 19:37:25 +02:00
|
|
|
total_rd += rd_pick_intra8x8block(cpi, mb, ib, &best_mode, i8x8mode_costs,
|
|
|
|
t_above + x_idx * 2, t_left + y_idx * 2,
|
|
|
|
&r, &ry, &d);
|
2012-07-14 00:21:29 +02:00
|
|
|
cost += r;
|
|
|
|
distortion += d;
|
|
|
|
tot_rate_y += ry;
|
|
|
|
mic->bmi[ib].as_mode.first = best_mode;
|
|
|
|
}
|
2012-12-14 18:49:46 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
*Rate = cost;
|
2012-11-05 21:50:16 +01:00
|
|
|
*rate_y = tot_rate_y;
|
2012-07-14 00:21:29 +02:00
|
|
|
*Distortion = distortion;
|
|
|
|
return RDCOST(mb->rdmult, mb->rddiv, cost, distortion);
|
2011-08-05 01:30:27 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-03-07 19:45:25 +01:00
|
|
|
static int64_t rd_pick_intra8x8mby_modes_and_txsz(VP9_COMP *cpi, MACROBLOCK *x,
|
|
|
|
int *rate, int *rate_y,
|
|
|
|
int *distortion,
|
|
|
|
int *mode8x8,
|
|
|
|
int64_t best_yrd,
|
|
|
|
int64_t *txfm_cache) {
|
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
MB_MODE_INFO *mbmi = &xd->mode_info_context->mbmi;
|
|
|
|
int cost0 = vp9_cost_bit(cm->prob_tx[0], 0);
|
|
|
|
int cost1 = vp9_cost_bit(cm->prob_tx[0], 1);
|
|
|
|
int64_t tmp_rd_4x4s, tmp_rd_8x8s;
|
|
|
|
int64_t tmp_rd_4x4, tmp_rd_8x8, tmp_rd;
|
|
|
|
int r4x4, tok4x4, d4x4, r8x8, tok8x8, d8x8;
|
|
|
|
|
|
|
|
mbmi->txfm_size = TX_4X4;
|
|
|
|
tmp_rd_4x4 = rd_pick_intra8x8mby_modes(cpi, x, &r4x4, &tok4x4,
|
|
|
|
&d4x4, best_yrd);
|
|
|
|
mode8x8[0] = xd->mode_info_context->bmi[0].as_mode.first;
|
|
|
|
mode8x8[1] = xd->mode_info_context->bmi[2].as_mode.first;
|
|
|
|
mode8x8[2] = xd->mode_info_context->bmi[8].as_mode.first;
|
|
|
|
mode8x8[3] = xd->mode_info_context->bmi[10].as_mode.first;
|
|
|
|
mbmi->txfm_size = TX_8X8;
|
|
|
|
tmp_rd_8x8 = rd_pick_intra8x8mby_modes(cpi, x, &r8x8, &tok8x8,
|
|
|
|
&d8x8, best_yrd);
|
|
|
|
txfm_cache[ONLY_4X4] = tmp_rd_4x4;
|
|
|
|
txfm_cache[ALLOW_8X8] = tmp_rd_8x8;
|
|
|
|
txfm_cache[ALLOW_16X16] = tmp_rd_8x8;
|
|
|
|
tmp_rd_4x4s = tmp_rd_4x4 + RDCOST(x->rdmult, x->rddiv, cost0, 0);
|
|
|
|
tmp_rd_8x8s = tmp_rd_8x8 + RDCOST(x->rdmult, x->rddiv, cost1, 0);
|
|
|
|
txfm_cache[TX_MODE_SELECT] = tmp_rd_4x4s < tmp_rd_8x8s ?
|
|
|
|
tmp_rd_4x4s : tmp_rd_8x8s;
|
|
|
|
if (cm->txfm_mode == TX_MODE_SELECT) {
|
|
|
|
if (tmp_rd_4x4s < tmp_rd_8x8s) {
|
|
|
|
*rate = r4x4 + cost0;
|
|
|
|
*rate_y = tok4x4 + cost0;
|
|
|
|
*distortion = d4x4;
|
|
|
|
mbmi->txfm_size = TX_4X4;
|
|
|
|
tmp_rd = tmp_rd_4x4s;
|
|
|
|
} else {
|
|
|
|
*rate = r8x8 + cost1;
|
|
|
|
*rate_y = tok8x8 + cost1;
|
|
|
|
*distortion = d8x8;
|
|
|
|
mbmi->txfm_size = TX_8X8;
|
|
|
|
tmp_rd = tmp_rd_8x8s;
|
|
|
|
|
|
|
|
mode8x8[0] = xd->mode_info_context->bmi[0].as_mode.first;
|
|
|
|
mode8x8[1] = xd->mode_info_context->bmi[2].as_mode.first;
|
|
|
|
mode8x8[2] = xd->mode_info_context->bmi[8].as_mode.first;
|
|
|
|
mode8x8[3] = xd->mode_info_context->bmi[10].as_mode.first;
|
|
|
|
}
|
|
|
|
} else if (cm->txfm_mode == ONLY_4X4) {
|
|
|
|
*rate = r4x4;
|
|
|
|
*rate_y = tok4x4;
|
|
|
|
*distortion = d4x4;
|
|
|
|
mbmi->txfm_size = TX_4X4;
|
|
|
|
tmp_rd = tmp_rd_4x4;
|
|
|
|
} else {
|
|
|
|
*rate = r8x8;
|
|
|
|
*rate_y = tok8x8;
|
|
|
|
*distortion = d8x8;
|
|
|
|
mbmi->txfm_size = TX_8X8;
|
|
|
|
tmp_rd = tmp_rd_8x8;
|
|
|
|
|
|
|
|
mode8x8[0] = xd->mode_info_context->bmi[0].as_mode.first;
|
|
|
|
mode8x8[1] = xd->mode_info_context->bmi[2].as_mode.first;
|
|
|
|
mode8x8[2] = xd->mode_info_context->bmi[8].as_mode.first;
|
|
|
|
mode8x8[3] = xd->mode_info_context->bmi[10].as_mode.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
return tmp_rd;
|
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif // !CONFIG_SB8X8
|
2013-03-07 19:45:25 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rd_cost_sbuv_4x4(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize) - 1, bw = 1 << bwl;
|
|
|
|
const int bh = 1 << (b_height_log2(bsize) - 1);
|
2013-04-11 00:55:59 +02:00
|
|
|
int yoff = 4 * bw * bh;
|
|
|
|
int p, b, cost = 0;
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2012-01-21 00:30:31 +01:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
for (p = 1; p < MAX_MB_PLANE; p++) {
|
|
|
|
ENTROPY_CONTEXT t_above[8], t_left[8];
|
2013-04-11 00:55:59 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, xd->plane[p].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 2 * bw >> xd->plane[p].subsampling_x);
|
|
|
|
vpx_memcpy(t_left, xd->plane[p].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 2 * bh >> xd->plane[p].subsampling_y);
|
2013-04-11 00:55:59 +02:00
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b & (bw - 1), y_idx = b >> bwl;
|
|
|
|
cost += cost_coeffs(cm, x, yoff + b, PLANE_TYPE_UV,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx, t_left + y_idx,
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_4X4, bw * bh * 4);
|
|
|
|
}
|
|
|
|
yoff = (yoff * 5) >> 2; // u -> v
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
2012-01-21 00:30:31 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return cost;
|
2012-01-21 00:30:31 +01:00
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static void super_block_uvrd_4x4(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
int *rate, int *distortion, int *skip,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2012-11-08 20:03:00 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
vp9_transform_sbuv_4x4(x, bsize);
|
|
|
|
vp9_quantize_sbuv_4x4(x, bsize);
|
2012-11-08 20:03:00 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
*rate = rd_cost_sbuv_4x4(cm, x, bsize);
|
2013-04-23 18:51:09 +02:00
|
|
|
*distortion = block_error_sbuv(x, bsize, 2);
|
2013-04-11 20:14:31 +02:00
|
|
|
*skip = vp9_sbuv_is_skippable(xd, bsize);
|
2012-11-08 20:03:00 +01:00
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rd_cost_sbuv_8x8(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize) - 2, bw = 1 << bwl;
|
|
|
|
const int bh = 1 << (b_height_log2(bsize) - 2);
|
2013-04-11 00:55:59 +02:00
|
|
|
int yoff = 16 * bw * bh;
|
|
|
|
int p, b, cost = 0;
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
for (p = 1; p < MAX_MB_PLANE; p++) {
|
|
|
|
ENTROPY_CONTEXT t_above[8], t_left[8];
|
2013-04-11 00:55:59 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, xd->plane[p].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 4 * bw >> xd->plane[p].subsampling_x);
|
|
|
|
vpx_memcpy(t_left, xd->plane[p].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 4 * bh >> xd->plane[p].subsampling_y);
|
2013-04-11 00:55:59 +02:00
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b & (bw - 1), y_idx = b >> bwl;
|
|
|
|
cost += cost_coeffs(cm, x, yoff + b * 4, PLANE_TYPE_UV,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx * 2, t_left + y_idx * 2,
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_8X8, bw * bh * 16);
|
|
|
|
}
|
|
|
|
yoff = (yoff * 5) >> 2; // u -> v
|
2013-01-06 03:20:25 +01:00
|
|
|
}
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
|
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static void super_block_uvrd_8x8(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
int *rate, int *distortion, int *skip,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
vp9_transform_sbuv_8x8(x, bsize);
|
|
|
|
vp9_quantize_sbuv_8x8(x, bsize);
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
*rate = rd_cost_sbuv_8x8(cm, x, bsize);
|
2013-04-23 18:51:09 +02:00
|
|
|
*distortion = block_error_sbuv(x, bsize, 2);
|
2013-04-11 20:14:31 +02:00
|
|
|
*skip = vp9_sbuv_is_skippable(xd, bsize);
|
2013-01-06 03:20:25 +01:00
|
|
|
}
|
2012-01-21 00:30:31 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rd_cost_sbuv_16x16(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize) - 3, bw = 1 << bwl;
|
|
|
|
const int bh = 1 << (b_height_log2(bsize) - 3);
|
2013-04-11 00:55:59 +02:00
|
|
|
int yoff = 64 * bw * bh;
|
|
|
|
int p, b, cost = 0;
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
for (p = 1; p < MAX_MB_PLANE; p++) {
|
|
|
|
ENTROPY_CONTEXT t_above[8], t_left[8];
|
2013-04-11 00:55:59 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, xd->plane[p].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 8 * bw >> xd->plane[p].subsampling_x);
|
|
|
|
vpx_memcpy(t_left, xd->plane[p].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 8 * bh >> xd->plane[p].subsampling_y);
|
2013-04-11 00:55:59 +02:00
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b & (bw - 1), y_idx = b >> bwl;
|
|
|
|
cost += cost_coeffs(cm, x, yoff + b * 16, PLANE_TYPE_UV,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx * 4, t_left + y_idx * 4,
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_16X16, bw * bh * 64);
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2013-04-11 00:55:59 +02:00
|
|
|
yoff = (yoff * 5) >> 2; // u -> v
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
return cost;
|
2012-02-22 03:10:18 +01:00
|
|
|
}
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static void super_block_uvrd_16x16(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
int *rate, int *distortion, int *skip,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2012-08-20 23:43:34 +02:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
vp9_transform_sbuv_16x16(x, bsize);
|
|
|
|
vp9_quantize_sbuv_16x16(x, bsize);
|
2013-01-06 03:20:25 +01:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
*rate = rd_cost_sbuv_16x16(cm, x, bsize);
|
2013-04-23 18:51:09 +02:00
|
|
|
*distortion = block_error_sbuv(x, bsize, 2);
|
2013-04-11 20:14:31 +02:00
|
|
|
*skip = vp9_sbuv_is_skippable(xd, bsize);
|
2013-01-06 03:20:25 +01:00
|
|
|
}
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
static int rd_cost_sbuv_32x32(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
2013-04-24 18:18:16 +02:00
|
|
|
const int bwl = b_width_log2(bsize) - 4, bw = 1 << bwl;
|
|
|
|
const int bh = 1 << (b_height_log2(bsize) - 4);
|
2013-04-11 00:55:59 +02:00
|
|
|
int yoff = 256 * bh * bw;
|
|
|
|
int p, b, cost = 0;
|
2013-03-04 23:12:17 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
for (p = 1; p < MAX_MB_PLANE; p++) {
|
|
|
|
ENTROPY_CONTEXT t_above[8], t_left[8];
|
2013-04-11 00:55:59 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, xd->plane[p].above_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 16 * bw >> xd->plane[p].subsampling_x);
|
|
|
|
vpx_memcpy(t_left, xd->plane[p].left_context,
|
|
|
|
sizeof(ENTROPY_CONTEXT) * 16 * bh >> xd->plane[p].subsampling_y);
|
2013-04-11 00:55:59 +02:00
|
|
|
for (b = 0; b < bw * bh; b++) {
|
|
|
|
const int x_idx = b * (bw - 1), y_idx = b >> bwl;
|
|
|
|
cost += cost_coeffs(cm, x, yoff + b * 64, PLANE_TYPE_UV,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above + x_idx * 8, t_left + y_idx * 8,
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_32X32, 256 * bh * bw);
|
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
yoff = (yoff * 5) >> 2; // u -> v
|
|
|
|
}
|
|
|
|
|
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
#undef UVCTX
|
|
|
|
|
|
|
|
static void super_block_uvrd_32x32(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
int *rate, int *distortion, int *skip,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
|
|
|
|
vp9_transform_sbuv_32x32(x, bsize);
|
|
|
|
vp9_quantize_sbuv_32x32(x, bsize);
|
|
|
|
|
|
|
|
*rate = rd_cost_sbuv_32x32(cm, x, bsize);
|
|
|
|
*distortion = block_error_sbuv(x, bsize, 0);
|
|
|
|
*skip = vp9_sbuv_is_skippable(xd, bsize);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void super_block_uvrd(VP9_COMMON *const cm, MACROBLOCK *x,
|
|
|
|
int *rate, int *distortion, int *skippable,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
MB_MODE_INFO *const mbmi = &xd->mode_info_context->mbmi;
|
|
|
|
|
|
|
|
vp9_subtract_sbuv(x, bsize);
|
|
|
|
|
|
|
|
if (mbmi->txfm_size >= TX_32X32 && bsize >= BLOCK_SIZE_SB64X64) {
|
|
|
|
super_block_uvrd_32x32(cm, x, rate, distortion, skippable, bsize);
|
|
|
|
} else if (mbmi->txfm_size >= TX_16X16 && bsize >= BLOCK_SIZE_SB32X32) {
|
|
|
|
super_block_uvrd_16x16(cm, x, rate, distortion, skippable, bsize);
|
|
|
|
} else if (mbmi->txfm_size >= TX_8X8 && bsize >= BLOCK_SIZE_MB16X16) {
|
|
|
|
super_block_uvrd_8x8(cm, x, rate, distortion, skippable, bsize);
|
|
|
|
} else {
|
|
|
|
super_block_uvrd_4x4(cm, x, rate, distortion, skippable, bsize);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int64_t rd_pick_intra_sbuv_mode(VP9_COMP *cpi, MACROBLOCK *x,
|
|
|
|
int *rate, int *rate_tokenonly,
|
|
|
|
int *distortion, int *skippable,
|
|
|
|
BLOCK_SIZE_TYPE bsize) {
|
|
|
|
MB_PREDICTION_MODE mode;
|
|
|
|
MB_PREDICTION_MODE UNINITIALIZED_IS_SAFE(mode_selected);
|
|
|
|
int64_t best_rd = INT64_MAX, this_rd;
|
|
|
|
int this_rate_tokenonly, this_rate;
|
|
|
|
int this_distortion, s;
|
|
|
|
|
|
|
|
for (mode = DC_PRED; mode <= TM_PRED; mode++) {
|
|
|
|
x->e_mbd.mode_info_context->mbmi.uv_mode = mode;
|
|
|
|
vp9_build_intra_predictors_sbuv_s(&x->e_mbd, bsize);
|
|
|
|
|
|
|
|
super_block_uvrd(&cpi->common, x, &this_rate_tokenonly,
|
|
|
|
&this_distortion, &s, bsize);
|
|
|
|
this_rate = this_rate_tokenonly +
|
|
|
|
x->intra_uv_mode_cost[x->e_mbd.frame_type][mode];
|
|
|
|
this_rd = RDCOST(x->rdmult, x->rddiv, this_rate, this_distortion);
|
|
|
|
|
|
|
|
if (this_rd < best_rd) {
|
|
|
|
mode_selected = mode;
|
|
|
|
best_rd = this_rd;
|
|
|
|
*rate = this_rate;
|
|
|
|
*rate_tokenonly = this_rate_tokenonly;
|
|
|
|
*distortion = this_distortion;
|
|
|
|
*skippable = s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
x->e_mbd.mode_info_context->mbmi.uv_mode = mode_selected;
|
|
|
|
|
|
|
|
return best_rd;
|
|
|
|
}
|
|
|
|
|
|
|
|
int vp9_cost_mv_ref(VP9_COMP *cpi,
|
|
|
|
MB_PREDICTION_MODE m,
|
|
|
|
const int mode_context) {
|
|
|
|
MACROBLOCKD *xd = &cpi->mb.e_mbd;
|
|
|
|
int segment_id = xd->mode_info_context->mbmi.segment_id;
|
|
|
|
|
|
|
|
// Dont account for mode here if segment skip is enabled.
|
|
|
|
if (!vp9_segfeature_active(xd, segment_id, SEG_LVL_SKIP)) {
|
|
|
|
VP9_COMMON *pc = &cpi->common;
|
|
|
|
|
|
|
|
vp9_prob p [VP9_MVREFS - 1];
|
|
|
|
assert(NEARESTMV <= m && m <= SPLITMV);
|
|
|
|
vp9_mv_ref_probs(pc, p, mode_context);
|
|
|
|
return cost_token(vp9_mv_ref_tree, p,
|
|
|
|
vp9_mv_ref_encoding_array - NEARESTMV + m);
|
|
|
|
} else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void vp9_set_mbmode_and_mvs(MACROBLOCK *x, MB_PREDICTION_MODE mb, int_mv *mv) {
|
|
|
|
x->e_mbd.mode_info_context->mbmi.mode = mb;
|
|
|
|
x->e_mbd.mode_info_context->mbmi.mv[0].as_int = mv->as_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
static int labels2mode(MACROBLOCK *x,
|
|
|
|
int const *labelings, int which_label,
|
|
|
|
B_PREDICTION_MODE this_mode,
|
|
|
|
int_mv *this_mv, int_mv *this_second_mv,
|
|
|
|
int_mv seg_mvs[MAX_REF_FRAMES - 1],
|
|
|
|
int_mv *best_ref_mv,
|
|
|
|
int_mv *second_best_ref_mv,
|
|
|
|
int *mvjcost, int *mvcost[2], VP9_COMP *cpi) {
|
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
MODE_INFO *const mic = xd->mode_info_context;
|
|
|
|
MB_MODE_INFO * mbmi = &mic->mbmi;
|
|
|
|
const int mis = xd->mode_info_stride;
|
|
|
|
int i, cost = 0, thismvcost = 0;
|
|
|
|
|
|
|
|
/* We have to be careful retrieving previously-encoded motion vectors.
|
|
|
|
Ones from this macroblock have to be pulled from the BLOCKD array
|
|
|
|
as they have not yet made it to the bmi array in our MB_MODE_INFO. */
|
|
|
|
for (i = 0; i < 4; ++i) {
|
|
|
|
const int row = i >> 1, col = i & 1;
|
|
|
|
B_PREDICTION_MODE m;
|
|
|
|
|
|
|
|
if (labelings[i] != which_label)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (col && labelings[i] == labelings[i - 1])
|
|
|
|
m = LEFT4X4;
|
|
|
|
else if (row && labelings[i] == labelings[i - 2])
|
|
|
|
m = ABOVE4X4;
|
|
|
|
else {
|
|
|
|
// the only time we should do costing for new motion vector or mode
|
|
|
|
// is when we are on a new label (jbb May 08, 2007)
|
|
|
|
switch (m = this_mode) {
|
|
|
|
case NEW4X4 :
|
|
|
|
if (mbmi->second_ref_frame > 0) {
|
|
|
|
this_mv->as_int = seg_mvs[mbmi->ref_frame - 1].as_int;
|
|
|
|
this_second_mv->as_int =
|
|
|
|
seg_mvs[mbmi->second_ref_frame - 1].as_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
thismvcost = vp9_mv_bit_cost(this_mv, best_ref_mv, mvjcost, mvcost,
|
|
|
|
102, xd->allow_high_precision_mv);
|
|
|
|
if (mbmi->second_ref_frame > 0) {
|
|
|
|
thismvcost += vp9_mv_bit_cost(this_second_mv, second_best_ref_mv,
|
|
|
|
mvjcost, mvcost, 102,
|
|
|
|
xd->allow_high_precision_mv);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case LEFT4X4:
|
|
|
|
this_mv->as_int = col ? mic->bmi[i - 1].as_mv[0].as_int :
|
|
|
|
left_block_mv(xd, mic, i);
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
this_second_mv->as_int = col ? mic->bmi[i - 1].as_mv[1].as_int :
|
|
|
|
left_block_second_mv(xd, mic, i);
|
|
|
|
break;
|
|
|
|
case ABOVE4X4:
|
|
|
|
this_mv->as_int = row ? mic->bmi[i - 2].as_mv[0].as_int :
|
|
|
|
above_block_mv(mic, i, mis);
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
this_second_mv->as_int = row ? mic->bmi[i - 2].as_mv[1].as_int :
|
|
|
|
above_block_second_mv(mic, i, mis);
|
|
|
|
break;
|
|
|
|
case ZERO4X4:
|
|
|
|
this_mv->as_int = 0;
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
this_second_mv->as_int = 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m == ABOVE4X4) { // replace above with left if same
|
|
|
|
int_mv left_mv, left_second_mv;
|
|
|
|
|
|
|
|
left_second_mv.as_int = 0;
|
|
|
|
left_mv.as_int = col ? mic->bmi[i - 1].as_mv[0].as_int :
|
|
|
|
left_block_mv(xd, mic, i);
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
left_second_mv.as_int = col ? mic->bmi[i - 1].as_mv[1].as_int :
|
|
|
|
left_block_second_mv(xd, mic, i);
|
|
|
|
|
|
|
|
if (left_mv.as_int == this_mv->as_int &&
|
|
|
|
(mbmi->second_ref_frame <= 0 ||
|
|
|
|
left_second_mv.as_int == this_second_mv->as_int))
|
|
|
|
m = LEFT4X4;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if CONFIG_NEWBINTRAMODES
|
|
|
|
cost = x->inter_bmode_costs[m == B_CONTEXT_PRED ?
|
|
|
|
m - CONTEXT_PRED_REPLACEMENTS : m];
|
|
|
|
#else
|
|
|
|
cost = x->inter_bmode_costs[m];
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
mic->bmi[i].as_mv[0].as_int = this_mv->as_int;
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
mic->bmi[i].as_mv[1].as_int = this_second_mv->as_int;
|
|
|
|
|
|
|
|
x->partition_info->bmi[i].mode = m;
|
|
|
|
x->partition_info->bmi[i].mv.as_int = this_mv->as_int;
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
x->partition_info->bmi[i].second_mv.as_int = this_second_mv->as_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
cost += thismvcost;
|
|
|
|
return cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int64_t encode_inter_mb_segment(VP9_COMMON *const cm,
|
|
|
|
MACROBLOCK *x,
|
|
|
|
int const *labels,
|
|
|
|
int which_label,
|
|
|
|
int *labelyrate,
|
|
|
|
int *distortion,
|
|
|
|
ENTROPY_CONTEXT *ta,
|
|
|
|
ENTROPY_CONTEXT *tl) {
|
|
|
|
int i;
|
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
|
|
|
|
|
|
|
*labelyrate = 0;
|
|
|
|
*distortion = 0;
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
if (labels[i] == which_label) {
|
|
|
|
const int src_stride = x->plane[0].src.stride;
|
|
|
|
uint8_t* const src =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_SB8X8, 0, i,
|
|
|
|
x->plane[0].src.buf, src_stride);
|
|
|
|
int16_t* const src_diff =
|
|
|
|
raster_block_offset_int16(xd, BLOCK_SIZE_SB8X8, 0, i,
|
|
|
|
x->plane[0].src_diff);
|
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff, 16, i);
|
|
|
|
uint8_t* const pre =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_SB8X8, 0, i,
|
|
|
|
xd->plane[0].pre[0].buf,
|
|
|
|
xd->plane[0].pre[0].stride);
|
|
|
|
uint8_t* const dst =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_SB8X8, 0, i,
|
|
|
|
xd->plane[0].dst.buf,
|
|
|
|
xd->plane[0].dst.stride);
|
|
|
|
int thisdistortion;
|
|
|
|
|
|
|
|
vp9_build_inter_predictor(pre,
|
|
|
|
xd->plane[0].pre[0].stride,
|
|
|
|
dst,
|
|
|
|
xd->plane[0].dst.stride,
|
|
|
|
&xd->mode_info_context->bmi[i].as_mv[0],
|
|
|
|
&xd->scale_factor[0],
|
|
|
|
4, 4, 0 /* no avg */, &xd->subpix);
|
|
|
|
|
|
|
|
// TODO(debargha): Make this work properly with the
|
|
|
|
// implicit-compoundinter-weight experiment when implicit
|
|
|
|
// weighting for splitmv modes is turned on.
|
|
|
|
if (xd->mode_info_context->mbmi.second_ref_frame > 0) {
|
|
|
|
uint8_t* const second_pre =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_SB8X8, 0, i,
|
|
|
|
xd->plane[0].pre[1].buf,
|
|
|
|
xd->plane[0].pre[1].stride);
|
|
|
|
vp9_build_inter_predictor(second_pre, xd->plane[0].pre[1].stride,
|
|
|
|
dst, xd->plane[0].dst.stride,
|
|
|
|
&xd->mode_info_context->bmi[i].as_mv[1],
|
|
|
|
&xd->scale_factor[1], 4, 4, 1,
|
|
|
|
&xd->subpix);
|
|
|
|
}
|
|
|
|
|
|
|
|
vp9_subtract_block(4, 4, src_diff, 8,
|
|
|
|
src, src_stride,
|
|
|
|
dst, xd->plane[0].dst.stride);
|
|
|
|
x->fwd_txm4x4(src_diff, coeff, 16);
|
|
|
|
x->quantize_b_4x4(x, i, DCT_DCT, 16);
|
|
|
|
thisdistortion = vp9_block_error(coeff,
|
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff,
|
|
|
|
i, 16), 16);
|
|
|
|
*distortion += thisdistortion;
|
|
|
|
*labelyrate += cost_coeffs(cm, x, i, PLANE_TYPE_Y_WITH_DC,
|
|
|
|
ta + (i & 1),
|
|
|
|
tl + (i >> 1), TX_4X4, 16);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*distortion >>= 2;
|
|
|
|
return RDCOST(x->rdmult, x->rddiv, *labelyrate, *distortion);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
int_mv *ref_mv, *second_ref_mv;
|
|
|
|
int_mv mvp;
|
|
|
|
|
|
|
|
int64_t segment_rd;
|
|
|
|
int r;
|
|
|
|
int d;
|
|
|
|
int segment_yrate;
|
|
|
|
B_PREDICTION_MODE modes[4];
|
|
|
|
int_mv mvs[4], second_mvs[4];
|
|
|
|
int eobs[4];
|
|
|
|
|
|
|
|
int mvthresh;
|
|
|
|
int *mdcounts;
|
|
|
|
} BEST_SEG_INFO;
|
|
|
|
#endif // CONFIG_SB8X8
|
|
|
|
|
|
|
|
static INLINE int mv_check_bounds(MACROBLOCK *x, int_mv *mv) {
|
|
|
|
int r = 0;
|
|
|
|
r |= (mv->as_mv.row >> 3) < x->mv_row_min;
|
|
|
|
r |= (mv->as_mv.row >> 3) > x->mv_row_max;
|
|
|
|
r |= (mv->as_mv.col >> 3) < x->mv_col_min;
|
|
|
|
r |= (mv->as_mv.col >> 3) > x->mv_col_max;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
static void rd_check_segment_txsize(VP9_COMP *cpi, MACROBLOCK *x,
|
|
|
|
BEST_SEG_INFO *bsi,
|
|
|
|
int_mv seg_mvs[4][MAX_REF_FRAMES - 1]) {
|
|
|
|
int i, j;
|
|
|
|
static const int labels[4] = { 0, 1, 2, 3 };
|
|
|
|
int br = 0, bd = 0;
|
|
|
|
B_PREDICTION_MODE this_mode;
|
|
|
|
MB_MODE_INFO * mbmi = &x->e_mbd.mode_info_context->mbmi;
|
|
|
|
const int label_count = 4;
|
|
|
|
int64_t this_segment_rd = 0, other_segment_rd;
|
|
|
|
int label_mv_thresh;
|
|
|
|
int rate = 0;
|
|
|
|
int sbr = 0, sbd = 0;
|
|
|
|
int segmentyrate = 0;
|
|
|
|
int best_eobs[4] = { 0 };
|
|
|
|
|
|
|
|
vp9_variance_fn_ptr_t *v_fn_ptr;
|
|
|
|
|
|
|
|
ENTROPY_CONTEXT t_above[2], t_left[2];
|
|
|
|
ENTROPY_CONTEXT t_above_b[2], t_left_b[2];
|
|
|
|
|
|
|
|
vpx_memcpy(t_above, x->e_mbd.plane[0].above_context, sizeof(t_above));
|
|
|
|
vpx_memcpy(t_left, x->e_mbd.plane[0].left_context, sizeof(t_left));
|
|
|
|
|
|
|
|
v_fn_ptr = &cpi->fn_ptr[BLOCK_4X4];
|
|
|
|
|
|
|
|
// 64 makes this threshold really big effectively
|
|
|
|
// making it so that we very rarely check mvs on
|
|
|
|
// segments. setting this to 1 would make mv thresh
|
|
|
|
// roughly equal to what it is for macroblocks
|
|
|
|
label_mv_thresh = 1 * bsi->mvthresh / label_count;
|
|
|
|
|
|
|
|
// Segmentation method overheads
|
|
|
|
rate += vp9_cost_mv_ref(cpi, SPLITMV,
|
|
|
|
mbmi->mb_mode_context[mbmi->ref_frame]);
|
|
|
|
this_segment_rd += RDCOST(x->rdmult, x->rddiv, rate, 0);
|
|
|
|
br += rate;
|
|
|
|
other_segment_rd = this_segment_rd;
|
|
|
|
|
|
|
|
for (i = 0; i < label_count && this_segment_rd < bsi->segment_rd; i++) {
|
|
|
|
int_mv mode_mv[B_MODE_COUNT], second_mode_mv[B_MODE_COUNT];
|
|
|
|
int64_t best_label_rd = INT64_MAX, best_other_rd = INT64_MAX;
|
|
|
|
B_PREDICTION_MODE mode_selected = ZERO4X4;
|
|
|
|
int bestlabelyrate = 0;
|
|
|
|
|
|
|
|
// search for the best motion vector on this segment
|
|
|
|
for (this_mode = LEFT4X4; this_mode <= NEW4X4; this_mode ++) {
|
|
|
|
int64_t this_rd;
|
|
|
|
int distortion;
|
|
|
|
int labelyrate;
|
|
|
|
ENTROPY_CONTEXT t_above_s[2], t_left_s[2];
|
|
|
|
|
|
|
|
vpx_memcpy(t_above_s, t_above, sizeof(t_above_s));
|
|
|
|
vpx_memcpy(t_left_s, t_left, sizeof(t_left_s));
|
|
|
|
|
|
|
|
// motion search for newmv (single predictor case only)
|
|
|
|
if (mbmi->second_ref_frame <= 0 && this_mode == NEW4X4) {
|
|
|
|
int sseshift, n;
|
|
|
|
int step_param = 0;
|
|
|
|
int further_steps;
|
|
|
|
int thissme, bestsme = INT_MAX;
|
|
|
|
const struct buf_2d orig_src = x->plane[0].src;
|
|
|
|
const struct buf_2d orig_pre = x->e_mbd.plane[0].pre[0];
|
|
|
|
|
|
|
|
/* Is the best so far sufficiently good that we cant justify doing
|
|
|
|
* and new motion search. */
|
|
|
|
if (best_label_rd < label_mv_thresh)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (cpi->compressor_speed) {
|
|
|
|
// use previous block's result as next block's MV predictor.
|
|
|
|
if (i > 0) {
|
|
|
|
bsi->mvp.as_int =
|
|
|
|
x->e_mbd.mode_info_context->bmi[i - 1].as_mv[0].as_int;
|
|
|
|
if (i == 2)
|
|
|
|
bsi->mvp.as_int =
|
|
|
|
x->e_mbd.mode_info_context->bmi[i - 2].as_mv[0].as_int;
|
|
|
|
step_param = 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
further_steps = (MAX_MVSEARCH_STEPS - 1) - step_param;
|
|
|
|
|
|
|
|
{
|
|
|
|
int sadpb = x->sadperbit4;
|
|
|
|
int_mv mvp_full;
|
|
|
|
|
|
|
|
mvp_full.as_mv.row = bsi->mvp.as_mv.row >> 3;
|
|
|
|
mvp_full.as_mv.col = bsi->mvp.as_mv.col >> 3;
|
|
|
|
|
|
|
|
// find first label
|
|
|
|
n = i;
|
|
|
|
|
|
|
|
// adjust src pointer for this segment
|
|
|
|
x->plane[0].src.buf =
|
|
|
|
raster_block_offset_uint8(&x->e_mbd, BLOCK_SIZE_SB8X8, 0, n,
|
|
|
|
x->plane[0].src.buf,
|
|
|
|
x->plane[0].src.stride);
|
|
|
|
assert(((intptr_t)x->e_mbd.plane[0].pre[0].buf & 0xf) == 0);
|
|
|
|
x->e_mbd.plane[0].pre[0].buf =
|
|
|
|
raster_block_offset_uint8(&x->e_mbd, BLOCK_SIZE_SB8X8, 0, n,
|
|
|
|
x->e_mbd.plane[0].pre[0].buf,
|
|
|
|
x->e_mbd.plane[0].pre[0].stride);
|
|
|
|
|
|
|
|
bestsme = vp9_full_pixel_diamond(cpi, x, &mvp_full, step_param,
|
|
|
|
sadpb, further_steps, 0, v_fn_ptr,
|
|
|
|
bsi->ref_mv, &mode_mv[NEW4X4]);
|
|
|
|
|
|
|
|
sseshift = 0;
|
|
|
|
|
|
|
|
// Should we do a full search (best quality only)
|
|
|
|
if ((cpi->compressor_speed == 0) && (bestsme >> sseshift) > 4000) {
|
|
|
|
/* Check if mvp_full is within the range. */
|
|
|
|
clamp_mv(&mvp_full, x->mv_col_min, x->mv_col_max,
|
|
|
|
x->mv_row_min, x->mv_row_max);
|
|
|
|
|
|
|
|
thissme = cpi->full_search_sad(x, &mvp_full,
|
|
|
|
sadpb, 16, v_fn_ptr,
|
|
|
|
x->nmvjointcost, x->mvcost,
|
|
|
|
bsi->ref_mv,
|
|
|
|
n);
|
|
|
|
|
|
|
|
if (thissme < bestsme) {
|
|
|
|
bestsme = thissme;
|
|
|
|
mode_mv[NEW4X4].as_int =
|
|
|
|
x->e_mbd.mode_info_context->bmi[n].as_mv[0].as_int;
|
|
|
|
} else {
|
|
|
|
/* The full search result is actually worse so re-instate the
|
|
|
|
* previous best vector */
|
|
|
|
x->e_mbd.mode_info_context->bmi[n].as_mv[0].as_int =
|
|
|
|
mode_mv[NEW4X4].as_int;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bestsme < INT_MAX) {
|
|
|
|
int distortion;
|
|
|
|
unsigned int sse;
|
|
|
|
cpi->find_fractional_mv_step(x, &mode_mv[NEW4X4],
|
|
|
|
bsi->ref_mv, x->errorperbit, v_fn_ptr,
|
|
|
|
x->nmvjointcost, x->mvcost,
|
|
|
|
&distortion, &sse);
|
|
|
|
|
|
|
|
// safe motion search result for use in compound prediction
|
|
|
|
seg_mvs[i][mbmi->ref_frame - 1].as_int = mode_mv[NEW4X4].as_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
// restore src pointers
|
|
|
|
x->plane[0].src = orig_src;
|
|
|
|
x->e_mbd.plane[0].pre[0] = orig_pre;
|
|
|
|
} else if (mbmi->second_ref_frame > 0 && this_mode == NEW4X4) {
|
|
|
|
/* NEW4X4 */
|
|
|
|
/* motion search not completed? Then skip newmv for this block with
|
|
|
|
* comppred */
|
|
|
|
if (seg_mvs[i][mbmi->second_ref_frame - 1].as_int == INVALID_MV ||
|
|
|
|
seg_mvs[i][mbmi->ref_frame - 1].as_int == INVALID_MV) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rate = labels2mode(x, labels, i, this_mode, &mode_mv[this_mode],
|
|
|
|
&second_mode_mv[this_mode], seg_mvs[i],
|
|
|
|
bsi->ref_mv, bsi->second_ref_mv, x->nmvjointcost,
|
|
|
|
x->mvcost, cpi);
|
|
|
|
|
|
|
|
// Trap vectors that reach beyond the UMV borders
|
|
|
|
if (((mode_mv[this_mode].as_mv.row >> 3) < x->mv_row_min) ||
|
|
|
|
((mode_mv[this_mode].as_mv.row >> 3) > x->mv_row_max) ||
|
|
|
|
((mode_mv[this_mode].as_mv.col >> 3) < x->mv_col_min) ||
|
|
|
|
((mode_mv[this_mode].as_mv.col >> 3) > x->mv_col_max)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (mbmi->second_ref_frame > 0 &&
|
|
|
|
mv_check_bounds(x, &second_mode_mv[this_mode]))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
this_rd = encode_inter_mb_segment(&cpi->common,
|
|
|
|
x, labels, i, &labelyrate,
|
|
|
|
&distortion, t_above_s, t_left_s);
|
|
|
|
this_rd += RDCOST(x->rdmult, x->rddiv, rate, 0);
|
|
|
|
rate += labelyrate;
|
|
|
|
|
|
|
|
if (this_rd < best_label_rd) {
|
|
|
|
sbr = rate;
|
|
|
|
sbd = distortion;
|
|
|
|
bestlabelyrate = labelyrate;
|
|
|
|
mode_selected = this_mode;
|
|
|
|
best_label_rd = this_rd;
|
|
|
|
for (j = 0; j < 4; j++)
|
|
|
|
if (labels[j] == i)
|
|
|
|
best_eobs[j] = x->e_mbd.plane[0].eobs[j];
|
|
|
|
|
|
|
|
vpx_memcpy(t_above_b, t_above_s, sizeof(t_above_s));
|
|
|
|
vpx_memcpy(t_left_b, t_left_s, sizeof(t_left_s));
|
|
|
|
}
|
|
|
|
} /*for each 4x4 mode*/
|
|
|
|
|
|
|
|
vpx_memcpy(t_above, t_above_b, sizeof(t_above));
|
|
|
|
vpx_memcpy(t_left, t_left_b, sizeof(t_left));
|
|
|
|
|
|
|
|
labels2mode(x, labels, i, mode_selected, &mode_mv[mode_selected],
|
|
|
|
&second_mode_mv[mode_selected], seg_mvs[i],
|
|
|
|
bsi->ref_mv, bsi->second_ref_mv, x->nmvjointcost,
|
|
|
|
x->mvcost, cpi);
|
|
|
|
|
|
|
|
br += sbr;
|
|
|
|
bd += sbd;
|
|
|
|
segmentyrate += bestlabelyrate;
|
|
|
|
this_segment_rd += best_label_rd;
|
|
|
|
other_segment_rd += best_other_rd;
|
|
|
|
} /* for each label */
|
|
|
|
|
|
|
|
if (this_segment_rd < bsi->segment_rd) {
|
|
|
|
bsi->r = br;
|
|
|
|
bsi->d = bd;
|
|
|
|
bsi->segment_yrate = segmentyrate;
|
|
|
|
bsi->segment_rd = this_segment_rd;
|
|
|
|
|
|
|
|
// store everything needed to come back to this!!
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
bsi->mvs[i].as_mv = x->partition_info->bmi[i].mv.as_mv;
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
bsi->second_mvs[i].as_mv = x->partition_info->bmi[i].second_mv.as_mv;
|
|
|
|
bsi->modes[i] = x->partition_info->bmi[i].mode;
|
|
|
|
bsi->eobs[i] = best_eobs[i];
|
|
|
|
}
|
2013-03-04 23:12:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
static void rd_check_segment(VP9_COMP *cpi, MACROBLOCK *x,
|
|
|
|
BEST_SEG_INFO *bsi,
|
|
|
|
int_mv seg_mvs[4][MAX_REF_FRAMES - 1]) {
|
|
|
|
rd_check_segment_txsize(cpi, x, bsi, seg_mvs);
|
2013-03-04 23:12:17 +01:00
|
|
|
}
|
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
static int rd_pick_best_mbsegmentation(VP9_COMP *cpi, MACROBLOCK *x,
|
|
|
|
int_mv *best_ref_mv,
|
|
|
|
int_mv *second_best_ref_mv,
|
|
|
|
int64_t best_rd,
|
|
|
|
int *mdcounts,
|
|
|
|
int *returntotrate,
|
|
|
|
int *returnyrate,
|
|
|
|
int *returndistortion,
|
|
|
|
int *skippable, int mvthresh,
|
|
|
|
int_mv seg_mvs[4][MAX_REF_FRAMES - 1]) {
|
|
|
|
int i;
|
|
|
|
BEST_SEG_INFO bsi;
|
|
|
|
MB_MODE_INFO * mbmi = &x->e_mbd.mode_info_context->mbmi;
|
2013-01-06 03:20:25 +01:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
vpx_memset(&bsi, 0, sizeof(bsi));
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
bsi.segment_rd = best_rd;
|
|
|
|
bsi.ref_mv = best_ref_mv;
|
|
|
|
bsi.second_ref_mv = second_best_ref_mv;
|
|
|
|
bsi.mvp.as_int = best_ref_mv->as_int;
|
|
|
|
bsi.mvthresh = mvthresh;
|
|
|
|
bsi.mdcounts = mdcounts;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
bsi.modes[i] = ZERO4X4;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, seg_mvs);
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
/* set it to the best */
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
x->e_mbd.mode_info_context->bmi[i].as_mv[0].as_int = bsi.mvs[i].as_int;
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
x->e_mbd.mode_info_context->bmi[i].as_mv[1].as_int =
|
|
|
|
bsi.second_mvs[i].as_int;
|
|
|
|
x->e_mbd.plane[0].eobs[i] = bsi.eobs[i];
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
/* save partitions */
|
|
|
|
x->partition_info->count = 4;
|
2013-01-06 03:20:25 +01:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
for (i = 0; i < x->partition_info->count; i++) {
|
|
|
|
x->partition_info->bmi[i].mode = bsi.modes[i];
|
|
|
|
x->partition_info->bmi[i].mv.as_mv = bsi.mvs[i].as_mv;
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
x->partition_info->bmi[i].second_mv.as_mv = bsi.second_mvs[i].as_mv;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* used to set mbmi->mv.as_int
|
|
|
|
*/
|
|
|
|
x->partition_info->bmi[3].mv.as_int = bsi.mvs[3].as_int;
|
|
|
|
if (mbmi->second_ref_frame > 0)
|
|
|
|
x->partition_info->bmi[3].second_mv.as_int = bsi.second_mvs[3].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
*returntotrate = bsi.r;
|
|
|
|
*returndistortion = bsi.d;
|
|
|
|
*returnyrate = bsi.segment_yrate;
|
|
|
|
*skippable = vp9_sby_is_skippable(&x->e_mbd, BLOCK_SIZE_SB8X8);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
return (int)(bsi.segment_rd);
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#else // !CONFIG_SB8X8
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-26 22:42:07 +02:00
|
|
|
static int labels2mode(
|
|
|
|
MACROBLOCK *x,
|
|
|
|
int const *labelings, int which_label,
|
|
|
|
B_PREDICTION_MODE this_mode,
|
|
|
|
int_mv *this_mv, int_mv *this_second_mv,
|
|
|
|
int_mv seg_mvs[MAX_REF_FRAMES - 1],
|
|
|
|
int_mv *best_ref_mv,
|
|
|
|
int_mv *second_best_ref_mv,
|
2013-04-26 20:57:17 +02:00
|
|
|
int *mvjcost, int *mvcost[2], VP9_COMP *cpi) {
|
2012-10-17 23:51:27 +02:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2012-07-14 00:21:29 +02:00
|
|
|
MODE_INFO *const mic = xd->mode_info_context;
|
2012-08-14 12:32:29 +02:00
|
|
|
MB_MODE_INFO * mbmi = &mic->mbmi;
|
2012-07-14 00:21:29 +02:00
|
|
|
const int mis = xd->mode_info_stride;
|
|
|
|
|
2012-08-07 01:21:23 +02:00
|
|
|
int i, cost = 0, thismvcost = 0;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
/* We have to be careful retrieving previously-encoded motion vectors.
|
|
|
|
Ones from this macroblock have to be pulled from the BLOCKD array
|
|
|
|
as they have not yet made it to the bmi array in our MB_MODE_INFO. */
|
2012-08-07 01:21:23 +02:00
|
|
|
for (i = 0; i < 16; ++i) {
|
2012-07-14 00:21:29 +02:00
|
|
|
const int row = i >> 2, col = i & 3;
|
|
|
|
|
|
|
|
B_PREDICTION_MODE m;
|
|
|
|
|
|
|
|
if (labelings[i] != which_label)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (col && labelings[i] == labelings[i - 1])
|
|
|
|
m = LEFT4X4;
|
|
|
|
else if (row && labelings[i] == labelings[i - 4])
|
|
|
|
m = ABOVE4X4;
|
|
|
|
else {
|
|
|
|
// the only time we should do costing for new motion vector or mode
|
|
|
|
// is when we are on a new label (jbb May 08, 2007)
|
|
|
|
switch (m = this_mode) {
|
|
|
|
case NEW4X4 :
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0) {
|
2012-08-10 15:12:43 +02:00
|
|
|
this_mv->as_int = seg_mvs[mbmi->ref_frame - 1].as_int;
|
|
|
|
this_second_mv->as_int =
|
|
|
|
seg_mvs[mbmi->second_ref_frame - 1].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2012-04-18 22:51:58 +02:00
|
|
|
|
2012-11-09 00:44:39 +01:00
|
|
|
thismvcost = vp9_mv_bit_cost(this_mv, best_ref_mv, mvjcost, mvcost,
|
2012-07-14 00:21:29 +02:00
|
|
|
102, xd->allow_high_precision_mv);
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0) {
|
2012-10-30 20:58:42 +01:00
|
|
|
thismvcost += vp9_mv_bit_cost(this_second_mv, second_best_ref_mv,
|
2012-11-09 00:44:39 +01:00
|
|
|
mvjcost, mvcost, 102,
|
2012-08-03 21:17:18 +02:00
|
|
|
xd->allow_high_precision_mv);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case LEFT4X4:
|
2013-04-26 16:19:43 +02:00
|
|
|
this_mv->as_int = col ? mic->bmi[i - 1].as_mv[0].as_int :
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 18:35:28 +01:00
|
|
|
left_block_mv(xd, mic, i);
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2013-04-26 16:19:43 +02:00
|
|
|
this_second_mv->as_int = col ? mic->bmi[i - 1].as_mv[1].as_int :
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 18:35:28 +01:00
|
|
|
left_block_second_mv(xd, mic, i);
|
2012-07-14 00:21:29 +02:00
|
|
|
break;
|
|
|
|
case ABOVE4X4:
|
2013-04-26 16:19:43 +02:00
|
|
|
this_mv->as_int = row ? mic->bmi[i - 4].as_mv[0].as_int :
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 18:35:28 +01:00
|
|
|
above_block_mv(mic, i, mis);
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2013-04-26 16:19:43 +02:00
|
|
|
this_second_mv->as_int = row ? mic->bmi[i - 4].as_mv[1].as_int :
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 18:35:28 +01:00
|
|
|
above_block_second_mv(mic, i, mis);
|
2012-07-14 00:21:29 +02:00
|
|
|
break;
|
|
|
|
case ZERO4X4:
|
|
|
|
this_mv->as_int = 0;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2012-07-14 00:21:29 +02:00
|
|
|
this_second_mv->as_int = 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (m == ABOVE4X4) { // replace above with left if same
|
|
|
|
int_mv left_mv, left_second_mv;
|
2011-05-24 19:24:52 +02:00
|
|
|
|
2012-08-14 01:20:20 +02:00
|
|
|
left_second_mv.as_int = 0;
|
2013-04-26 16:19:43 +02:00
|
|
|
left_mv.as_int = col ? mic->bmi[i - 1].as_mv[0].as_int :
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 18:35:28 +01:00
|
|
|
left_block_mv(xd, mic, i);
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2013-04-26 16:19:43 +02:00
|
|
|
left_second_mv.as_int = col ? mic->bmi[i - 1].as_mv[1].as_int :
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 18:35:28 +01:00
|
|
|
left_block_second_mv(xd, mic, i);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (left_mv.as_int == this_mv->as_int &&
|
2012-11-07 15:50:25 +01:00
|
|
|
(mbmi->second_ref_frame <= 0 ||
|
2012-07-14 00:21:29 +02:00
|
|
|
left_second_mv.as_int == this_second_mv->as_int))
|
|
|
|
m = LEFT4X4;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-09 22:19:15 +02:00
|
|
|
#if CONFIG_NEWBINTRAMODES
|
|
|
|
cost = x->inter_bmode_costs[
|
|
|
|
m == B_CONTEXT_PRED ? m - CONTEXT_PRED_REPLACEMENTS : m];
|
|
|
|
#else
|
|
|
|
cost = x->inter_bmode_costs[m];
|
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-04-26 16:19:43 +02:00
|
|
|
mic->bmi[i].as_mv[0].as_int = this_mv->as_int;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2013-04-26 16:19:43 +02:00
|
|
|
mic->bmi[i].as_mv[1].as_int = this_second_mv->as_int;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
x->partition_info->bmi[i].mode = m;
|
|
|
|
x->partition_info->bmi[i].mv.as_int = this_mv->as_int;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2012-07-14 00:21:29 +02:00
|
|
|
x->partition_info->bmi[i].second_mv.as_int = this_second_mv->as_int;
|
2012-08-07 01:21:23 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
cost += thismvcost;
|
|
|
|
return cost;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-02-20 19:16:24 +01:00
|
|
|
static int64_t encode_inter_mb_segment(VP9_COMMON *const cm,
|
|
|
|
MACROBLOCK *x,
|
2012-10-17 20:40:00 +02:00
|
|
|
int const *labels,
|
|
|
|
int which_label,
|
|
|
|
int *labelyrate,
|
|
|
|
int *distortion,
|
|
|
|
ENTROPY_CONTEXT *ta,
|
2012-11-25 04:33:58 +01:00
|
|
|
ENTROPY_CONTEXT *tl) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int i;
|
2012-08-14 12:32:29 +02:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-10-17 20:40:00 +02:00
|
|
|
*labelyrate = 0;
|
|
|
|
*distortion = 0;
|
2012-07-14 00:21:29 +02:00
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
if (labels[i] == which_label) {
|
2013-04-24 01:22:47 +02:00
|
|
|
const int src_stride = x->plane[0].src.stride;
|
|
|
|
uint8_t* const src =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, i,
|
|
|
|
x->plane[0].src.buf, src_stride);
|
2013-04-23 17:26:10 +02:00
|
|
|
int16_t* const src_diff =
|
|
|
|
raster_block_offset_int16(xd, BLOCK_SIZE_MB16X16, 0, i,
|
|
|
|
x->plane[0].src_diff);
|
2013-04-23 23:38:52 +02:00
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff, 16, i);
|
2013-04-25 20:15:38 +02:00
|
|
|
uint8_t* const pre =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, i,
|
|
|
|
xd->plane[0].pre[0].buf,
|
|
|
|
xd->plane[0].pre[0].stride);
|
2013-04-25 20:15:38 +02:00
|
|
|
uint8_t* const dst =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, i,
|
|
|
|
xd->plane[0].dst.buf,
|
|
|
|
xd->plane[0].dst.stride);
|
2012-07-14 00:21:29 +02:00
|
|
|
int thisdistortion;
|
|
|
|
|
2013-04-25 20:15:38 +02:00
|
|
|
vp9_build_inter_predictor(pre,
|
|
|
|
xd->plane[0].pre[0].stride,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst,
|
|
|
|
xd->plane[0].dst.stride,
|
2013-04-26 16:19:43 +02:00
|
|
|
&xd->mode_info_context->bmi[i].as_mv[0],
|
2013-02-09 02:49:44 +01:00
|
|
|
&xd->scale_factor[0],
|
|
|
|
4, 4, 0 /* no avg */, &xd->subpix);
|
|
|
|
|
Implicit weighted prediction experiment
Adds an experiment to use a weighted prediction of two INTER
predictors, where the weight is one of (1/4, 3/4), (3/8, 5/8),
(1/2, 1/2), (5/8, 3/8) or (3/4, 1/4), and is chosen implicitly
based on consistency of the predictors to the already
reconstructed pixels to the top and left of the current macroblock
or superblock.
Currently the weighting is not applied to SPLITMV modes, which
default to the usual (1/2, 1/2) weighting. However the code is in
place controlled by a macro. The same weighting is used for Y and
UV components, where the weight is derived from analyzing the Y
component only.
Results (over compound inter-intra experiment)
derf: +0.18%
yt: +0.34%
hd: +0.49%
stdhd: +0.23%
The experiment suggests bigger benefit for explicitly signaled weights.
Change-Id: I5438539ff4485c5752874cd1eb078ff14bf5235a
2013-03-12 22:21:08 +01:00
|
|
|
// TODO(debargha): Make this work properly with the
|
|
|
|
// implicit-compoundinter-weight experiment when implicit
|
|
|
|
// weighting for splitmv modes is turned on.
|
2013-02-09 02:49:44 +01:00
|
|
|
if (xd->mode_info_context->mbmi.second_ref_frame > 0) {
|
2013-04-25 20:15:38 +02:00
|
|
|
uint8_t* const second_pre =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, i,
|
|
|
|
xd->plane[0].pre[1].buf,
|
|
|
|
xd->plane[0].pre[1].stride);
|
Implicit weighted prediction experiment
Adds an experiment to use a weighted prediction of two INTER
predictors, where the weight is one of (1/4, 3/4), (3/8, 5/8),
(1/2, 1/2), (5/8, 3/8) or (3/4, 1/4), and is chosen implicitly
based on consistency of the predictors to the already
reconstructed pixels to the top and left of the current macroblock
or superblock.
Currently the weighting is not applied to SPLITMV modes, which
default to the usual (1/2, 1/2) weighting. However the code is in
place controlled by a macro. The same weighting is used for Y and
UV components, where the weight is derived from analyzing the Y
component only.
Results (over compound inter-intra experiment)
derf: +0.18%
yt: +0.34%
hd: +0.49%
stdhd: +0.23%
The experiment suggests bigger benefit for explicitly signaled weights.
Change-Id: I5438539ff4485c5752874cd1eb078ff14bf5235a
2013-03-12 22:21:08 +01:00
|
|
|
vp9_build_inter_predictor(
|
2013-04-25 20:15:38 +02:00
|
|
|
second_pre, xd->plane[0].pre[1].stride,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst, xd->plane[0].dst.stride,
|
2013-04-26 16:19:43 +02:00
|
|
|
&xd->mode_info_context->bmi[i].as_mv[1],
|
|
|
|
&xd->scale_factor[1], 4, 4, 1,
|
Implicit weighted prediction experiment
Adds an experiment to use a weighted prediction of two INTER
predictors, where the weight is one of (1/4, 3/4), (3/8, 5/8),
(1/2, 1/2), (5/8, 3/8) or (3/4, 1/4), and is chosen implicitly
based on consistency of the predictors to the already
reconstructed pixels to the top and left of the current macroblock
or superblock.
Currently the weighting is not applied to SPLITMV modes, which
default to the usual (1/2, 1/2) weighting. However the code is in
place controlled by a macro. The same weighting is used for Y and
UV components, where the weight is derived from analyzing the Y
component only.
Results (over compound inter-intra experiment)
derf: +0.18%
yt: +0.34%
hd: +0.49%
stdhd: +0.23%
The experiment suggests bigger benefit for explicitly signaled weights.
Change-Id: I5438539ff4485c5752874cd1eb078ff14bf5235a
2013-03-12 22:21:08 +01:00
|
|
|
&xd->subpix);
|
2013-02-09 02:49:44 +01:00
|
|
|
}
|
|
|
|
|
2013-04-23 17:26:10 +02:00
|
|
|
vp9_subtract_block(4, 4, src_diff, 16,
|
2013-04-24 01:22:47 +02:00
|
|
|
src, src_stride,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst, xd->plane[0].dst.stride);
|
2013-04-23 23:38:52 +02:00
|
|
|
x->fwd_txm4x4(src_diff, coeff, 32);
|
2013-04-25 23:44:44 +02:00
|
|
|
x->quantize_b_4x4(x, i, DCT_DCT, 16);
|
2013-04-23 23:38:52 +02:00
|
|
|
thisdistortion = vp9_block_error(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, i, 16), 16);
|
2012-10-17 20:40:00 +02:00
|
|
|
*distortion += thisdistortion;
|
2013-02-20 19:16:24 +01:00
|
|
|
*labelyrate += cost_coeffs(cm, x, i, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
ta + (i & 3),
|
|
|
|
tl + (i >> 2), TX_4X4, 16);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
}
|
2012-10-17 20:40:00 +02:00
|
|
|
*distortion >>= 2;
|
|
|
|
return RDCOST(x->rdmult, x->rddiv, *labelyrate, *distortion);
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-02-20 19:16:24 +01:00
|
|
|
static int64_t encode_inter_mb_segment_8x8(VP9_COMMON *const cm,
|
|
|
|
MACROBLOCK *x,
|
2012-10-17 20:40:00 +02:00
|
|
|
int const *labels,
|
|
|
|
int which_label,
|
|
|
|
int *labelyrate,
|
|
|
|
int *distortion,
|
2012-10-22 20:49:00 +02:00
|
|
|
int64_t *otherrd,
|
2012-10-17 20:40:00 +02:00
|
|
|
ENTROPY_CONTEXT *ta,
|
2012-11-25 04:33:58 +01:00
|
|
|
ENTROPY_CONTEXT *tl) {
|
2012-10-17 20:40:00 +02:00
|
|
|
int i, j;
|
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
|
|
|
const int iblock[4] = { 0, 1, 4, 5 };
|
2012-10-22 20:49:00 +02:00
|
|
|
int othercost = 0, otherdist = 0;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT tac[4], tlc[4];
|
2012-10-22 20:49:00 +02:00
|
|
|
|
|
|
|
if (otherrd) {
|
2013-04-29 19:37:25 +02:00
|
|
|
memcpy(&tac, ta, sizeof(tac));
|
|
|
|
memcpy(&tlc, tl, sizeof(tlc));
|
2012-10-22 20:49:00 +02:00
|
|
|
}
|
2012-10-17 20:40:00 +02:00
|
|
|
|
|
|
|
*distortion = 0;
|
|
|
|
*labelyrate = 0;
|
|
|
|
for (i = 0; i < 4; i++) {
|
2012-10-31 01:12:12 +01:00
|
|
|
int ib = vp9_i8x8_block[i];
|
2012-10-17 20:40:00 +02:00
|
|
|
|
|
|
|
if (labels[ib] == which_label) {
|
2013-02-09 02:49:44 +01:00
|
|
|
const int use_second_ref =
|
|
|
|
xd->mode_info_context->mbmi.second_ref_frame > 0;
|
|
|
|
int which_mv;
|
2013-04-02 23:50:40 +02:00
|
|
|
const int idx = (ib & 8) + ((ib & 2) << 1);
|
2013-04-24 01:22:47 +02:00
|
|
|
const int src_stride = x->plane[0].src.stride;
|
|
|
|
uint8_t* const src =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, ib,
|
|
|
|
x->plane[0].src.buf, src_stride);
|
2013-04-23 17:26:10 +02:00
|
|
|
int16_t* const src_diff =
|
|
|
|
raster_block_offset_int16(xd, BLOCK_SIZE_MB16X16, 0, ib,
|
|
|
|
x->plane[0].src_diff);
|
2013-04-23 23:38:52 +02:00
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff, idx, 16);
|
2012-10-17 20:40:00 +02:00
|
|
|
int thisdistortion;
|
2013-04-25 20:15:38 +02:00
|
|
|
uint8_t* const dst =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, ib,
|
|
|
|
xd->plane[0].dst.buf,
|
|
|
|
xd->plane[0].dst.stride);
|
2012-10-17 20:40:00 +02:00
|
|
|
|
2013-04-02 23:50:40 +02:00
|
|
|
assert(idx < 16);
|
2013-02-09 02:49:44 +01:00
|
|
|
for (which_mv = 0; which_mv < 1 + use_second_ref; ++which_mv) {
|
2013-04-25 20:15:38 +02:00
|
|
|
uint8_t* const pre =
|
|
|
|
raster_block_offset_uint8(xd, BLOCK_SIZE_MB16X16, 0, ib,
|
|
|
|
xd->plane[0].pre[which_mv].buf,
|
|
|
|
xd->plane[0].pre[which_mv].stride);
|
2013-02-09 02:49:44 +01:00
|
|
|
|
Implicit weighted prediction experiment
Adds an experiment to use a weighted prediction of two INTER
predictors, where the weight is one of (1/4, 3/4), (3/8, 5/8),
(1/2, 1/2), (5/8, 3/8) or (3/4, 1/4), and is chosen implicitly
based on consistency of the predictors to the already
reconstructed pixels to the top and left of the current macroblock
or superblock.
Currently the weighting is not applied to SPLITMV modes, which
default to the usual (1/2, 1/2) weighting. However the code is in
place controlled by a macro. The same weighting is used for Y and
UV components, where the weight is derived from analyzing the Y
component only.
Results (over compound inter-intra experiment)
derf: +0.18%
yt: +0.34%
hd: +0.49%
stdhd: +0.23%
The experiment suggests bigger benefit for explicitly signaled weights.
Change-Id: I5438539ff4485c5752874cd1eb078ff14bf5235a
2013-03-12 22:21:08 +01:00
|
|
|
// TODO(debargha): Make this work properly with the
|
|
|
|
// implicit-compoundinter-weight experiment when implicit
|
|
|
|
// weighting for splitmv modes is turned on.
|
|
|
|
vp9_build_inter_predictor(
|
2013-04-25 20:15:38 +02:00
|
|
|
pre, xd->plane[0].pre[which_mv].stride,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst, xd->plane[0].dst.stride,
|
2013-04-26 16:19:43 +02:00
|
|
|
&xd->mode_info_context->bmi[ib].as_mv[which_mv],
|
|
|
|
&xd->scale_factor[which_mv], 8, 8,
|
2013-04-23 01:29:34 +02:00
|
|
|
which_mv, &xd->subpix);
|
2013-02-09 02:49:44 +01:00
|
|
|
}
|
|
|
|
|
2013-04-23 17:26:10 +02:00
|
|
|
vp9_subtract_block(8, 8, src_diff, 16,
|
2013-04-24 01:22:47 +02:00
|
|
|
src, src_stride,
|
2013-04-25 20:15:38 +02:00
|
|
|
dst, xd->plane[0].dst.stride);
|
2012-10-17 20:40:00 +02:00
|
|
|
|
2012-10-22 20:49:00 +02:00
|
|
|
if (xd->mode_info_context->mbmi.txfm_size == TX_4X4) {
|
|
|
|
if (otherrd) {
|
2013-04-23 23:38:52 +02:00
|
|
|
x->fwd_txm8x8(src_diff, coeff, 32);
|
2013-04-04 21:03:27 +02:00
|
|
|
x->quantize_b_8x8(x, idx, DCT_DCT, 16);
|
2013-04-23 23:38:52 +02:00
|
|
|
thisdistortion = vp9_block_error_c(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, idx, 16), 64);
|
2012-10-22 20:49:00 +02:00
|
|
|
otherdist += thisdistortion;
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_8X8;
|
|
|
|
othercost += cost_coeffs(cm, x, idx, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
tac + (i & 1) * 2,
|
|
|
|
tlc + (i & 2),
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_8X8, 16);
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_4X4;
|
2012-10-22 20:49:00 +02:00
|
|
|
}
|
|
|
|
for (j = 0; j < 4; j += 2) {
|
2013-04-23 17:26:10 +02:00
|
|
|
int16_t* const src_diff =
|
|
|
|
raster_block_offset_int16(xd, BLOCK_SIZE_MB16X16,
|
|
|
|
0, ib + iblock[j],
|
|
|
|
x->plane[0].src_diff);
|
2013-04-23 23:38:52 +02:00
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff,
|
|
|
|
ib + iblock[j], 16);
|
|
|
|
x->fwd_txm8x4(src_diff, coeff, 32);
|
2013-04-04 21:03:27 +02:00
|
|
|
x->quantize_b_4x4_pair(x, ib + iblock[j], ib + iblock[j] + 1, 16);
|
2013-04-23 23:38:52 +02:00
|
|
|
thisdistortion = vp9_block_error_c(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, ib + iblock[j], 16), 32);
|
2012-10-22 20:49:00 +02:00
|
|
|
*distortion += thisdistortion;
|
2013-02-20 19:16:24 +01:00
|
|
|
*labelyrate +=
|
|
|
|
cost_coeffs(cm, x, ib + iblock[j], PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
ta + (i & 1) * 2,
|
|
|
|
tl + (i & 2) + ((j & 2) >> 1),
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_4X4, 16);
|
2013-02-20 19:16:24 +01:00
|
|
|
*labelyrate +=
|
|
|
|
cost_coeffs(cm, x, ib + iblock[j] + 1,
|
|
|
|
PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
ta + (i & 1) * 2 + 1,
|
|
|
|
tl + (i & 2) + ((j & 2) >> 1),
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_4X4, 16);
|
2012-10-22 20:49:00 +02:00
|
|
|
}
|
|
|
|
} else /* 8x8 */ {
|
|
|
|
if (otherrd) {
|
|
|
|
for (j = 0; j < 4; j += 2) {
|
2013-04-23 17:26:10 +02:00
|
|
|
int16_t* const src_diff =
|
|
|
|
raster_block_offset_int16(xd, BLOCK_SIZE_MB16X16,
|
|
|
|
0, ib + iblock[j],
|
|
|
|
x->plane[0].src_diff);
|
2013-04-23 23:38:52 +02:00
|
|
|
int16_t* const coeff = BLOCK_OFFSET(x->plane[0].coeff,
|
|
|
|
ib + iblock[j], 16);
|
|
|
|
x->fwd_txm8x4(src_diff, coeff, 32);
|
2013-04-04 21:03:27 +02:00
|
|
|
x->quantize_b_4x4_pair(x, ib + iblock[j], ib + iblock[j] + 1, 16);
|
2013-04-23 23:38:52 +02:00
|
|
|
thisdistortion = vp9_block_error_c(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, ib + iblock[j], 16), 32);
|
2012-10-22 20:49:00 +02:00
|
|
|
otherdist += thisdistortion;
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_4X4;
|
|
|
|
othercost +=
|
|
|
|
cost_coeffs(cm, x, ib + iblock[j], PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
tac + (i & 1) * 2,
|
|
|
|
tlc + (i & 2) + ((j & 2) >> 1),
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_4X4, 16);
|
2013-02-20 19:16:24 +01:00
|
|
|
othercost +=
|
|
|
|
cost_coeffs(cm, x, ib + iblock[j] + 1,
|
|
|
|
PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
tac + (i & 1) * 2 + 1,
|
|
|
|
tlc + (i & 2) + ((j & 2) >> 1),
|
2013-04-04 21:03:27 +02:00
|
|
|
TX_4X4, 16);
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_8X8;
|
2012-10-22 20:49:00 +02:00
|
|
|
}
|
|
|
|
}
|
2013-04-23 23:38:52 +02:00
|
|
|
x->fwd_txm8x8(src_diff, coeff, 32);
|
2013-04-04 21:03:27 +02:00
|
|
|
x->quantize_b_8x8(x, idx, DCT_DCT, 16);
|
2013-04-23 23:38:52 +02:00
|
|
|
thisdistortion = vp9_block_error_c(coeff,
|
2013-04-02 23:50:40 +02:00
|
|
|
BLOCK_OFFSET(xd->plane[0].dqcoeff, idx, 16), 64);
|
2012-10-17 20:40:00 +02:00
|
|
|
*distortion += thisdistortion;
|
2013-02-20 19:16:24 +01:00
|
|
|
*labelyrate += cost_coeffs(cm, x, idx, PLANE_TYPE_Y_WITH_DC,
|
2013-04-29 19:37:25 +02:00
|
|
|
ta + (i & 1) * 2,
|
|
|
|
tl + (i & 2),
|
|
|
|
TX_8X8, 16);
|
2012-10-17 20:40:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*distortion >>= 2;
|
2012-10-22 20:49:00 +02:00
|
|
|
if (otherrd) {
|
2012-10-25 18:17:47 +02:00
|
|
|
otherdist >>= 2;
|
2012-10-22 20:49:00 +02:00
|
|
|
*otherrd = RDCOST(x->rdmult, x->rddiv, othercost, otherdist);
|
|
|
|
}
|
2012-10-17 20:40:00 +02:00
|
|
|
return RDCOST(x->rdmult, x->rddiv, *labelyrate, *distortion);
|
|
|
|
}
|
2011-06-23 00:07:04 +02:00
|
|
|
|
2010-12-06 22:42:52 +01:00
|
|
|
static const unsigned int segmentation_to_sseshift[4] = {3, 3, 2, 0};
|
|
|
|
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
typedef struct {
|
2012-04-18 22:51:58 +02:00
|
|
|
int_mv *ref_mv, *second_ref_mv;
|
2011-05-12 16:50:16 +02:00
|
|
|
int_mv mvp;
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2012-08-02 19:07:33 +02:00
|
|
|
int64_t segment_rd;
|
2012-10-22 20:25:48 +02:00
|
|
|
SPLITMV_PARTITIONING_TYPE segment_num;
|
2012-10-22 20:49:00 +02:00
|
|
|
TX_SIZE txfm_size;
|
2010-12-06 22:42:52 +01:00
|
|
|
int r;
|
|
|
|
int d;
|
|
|
|
int segment_yrate;
|
|
|
|
B_PREDICTION_MODE modes[16];
|
2012-04-18 22:51:58 +02:00
|
|
|
int_mv mvs[16], second_mvs[16];
|
2012-10-23 18:23:23 +02:00
|
|
|
int eobs[16];
|
2010-12-06 22:42:52 +01:00
|
|
|
|
|
|
|
int mvthresh;
|
|
|
|
int *mdcounts;
|
|
|
|
|
2011-05-12 16:50:16 +02:00
|
|
|
int_mv sv_mvp[4]; // save 4 mvp from 8x8
|
2010-12-23 17:23:03 +01:00
|
|
|
int sv_istep[2]; // save 2 initial step_param for 16x8/8x16
|
|
|
|
|
2010-12-06 22:42:52 +01:00
|
|
|
} BEST_SEG_INFO;
|
|
|
|
|
2012-10-31 01:53:32 +01:00
|
|
|
static void rd_check_segment_txsize(VP9_COMP *cpi, MACROBLOCK *x,
|
2012-10-22 20:49:00 +02:00
|
|
|
BEST_SEG_INFO *bsi,
|
|
|
|
SPLITMV_PARTITIONING_TYPE segmentation,
|
|
|
|
TX_SIZE tx_size, int64_t *otherrds,
|
|
|
|
int64_t *rds, int *completed,
|
|
|
|
/* 16 = n_blocks */
|
|
|
|
int_mv seg_mvs[16 /* n_blocks */]
|
|
|
|
[MAX_REF_FRAMES - 1]) {
|
2012-10-17 23:32:17 +02:00
|
|
|
int i, j;
|
2012-07-14 00:21:29 +02:00
|
|
|
int const *labels;
|
2012-08-07 01:21:23 +02:00
|
|
|
int br = 0, bd = 0;
|
2012-07-14 00:21:29 +02:00
|
|
|
B_PREDICTION_MODE this_mode;
|
2012-08-10 15:12:43 +02:00
|
|
|
MB_MODE_INFO * mbmi = &x->e_mbd.mode_info_context->mbmi;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
int label_count;
|
2012-10-22 20:49:00 +02:00
|
|
|
int64_t this_segment_rd = 0, other_segment_rd;
|
2012-07-14 00:21:29 +02:00
|
|
|
int label_mv_thresh;
|
|
|
|
int rate = 0;
|
2012-08-07 01:21:23 +02:00
|
|
|
int sbr = 0, sbd = 0;
|
2012-07-14 00:21:29 +02:00
|
|
|
int segmentyrate = 0;
|
2012-10-23 18:23:23 +02:00
|
|
|
int best_eobs[16] = { 0 };
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_variance_fn_ptr_t *v_fn_ptr;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT t_above[4], t_left[4];
|
|
|
|
ENTROPY_CONTEXT t_above_b[4], t_left_b[4];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, x->e_mbd.plane[0].above_context, sizeof(t_above));
|
|
|
|
vpx_memcpy(t_left, x->e_mbd.plane[0].left_context, sizeof(t_left));
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
v_fn_ptr = &cpi->fn_ptr[segmentation];
|
2012-10-31 01:12:12 +01:00
|
|
|
labels = vp9_mbsplits[segmentation];
|
|
|
|
label_count = vp9_mbsplit_count[segmentation];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// 64 makes this threshold really big effectively
|
|
|
|
// making it so that we very rarely check mvs on
|
|
|
|
// segments. setting this to 1 would make mv thresh
|
|
|
|
// roughly equal to what it is for macroblocks
|
|
|
|
label_mv_thresh = 1 * bsi->mvthresh / label_count;
|
|
|
|
|
|
|
|
// Segmentation method overheads
|
2012-10-31 22:40:53 +01:00
|
|
|
rate = cost_token(vp9_mbsplit_tree, vp9_mbsplit_probs,
|
|
|
|
vp9_mbsplit_encodings + segmentation);
|
2012-11-12 16:09:25 +01:00
|
|
|
rate += vp9_cost_mv_ref(cpi, SPLITMV,
|
|
|
|
mbmi->mb_mode_context[mbmi->ref_frame]);
|
2012-07-14 00:21:29 +02:00
|
|
|
this_segment_rd += RDCOST(x->rdmult, x->rddiv, rate, 0);
|
|
|
|
br += rate;
|
2012-10-22 20:49:00 +02:00
|
|
|
other_segment_rd = this_segment_rd;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-10-22 20:49:00 +02:00
|
|
|
mbmi->txfm_size = tx_size;
|
|
|
|
for (i = 0; i < label_count && this_segment_rd < bsi->segment_rd; i++) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int_mv mode_mv[B_MODE_COUNT], second_mode_mv[B_MODE_COUNT];
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_label_rd = INT64_MAX, best_other_rd = INT64_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
B_PREDICTION_MODE mode_selected = ZERO4X4;
|
|
|
|
int bestlabelyrate = 0;
|
|
|
|
|
|
|
|
// search for the best motion vector on this segment
|
|
|
|
for (this_mode = LEFT4X4; this_mode <= NEW4X4; this_mode ++) {
|
2012-10-22 20:49:00 +02:00
|
|
|
int64_t this_rd, other_rd;
|
2012-07-14 00:21:29 +02:00
|
|
|
int distortion;
|
|
|
|
int labelyrate;
|
2013-04-29 19:37:25 +02:00
|
|
|
ENTROPY_CONTEXT t_above_s[4], t_left_s[4];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above_s, t_above, sizeof(t_above_s));
|
|
|
|
vpx_memcpy(t_left_s, t_left, sizeof(t_left_s));
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// motion search for newmv (single predictor case only)
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame <= 0 && this_mode == NEW4X4) {
|
2012-08-08 01:44:26 +02:00
|
|
|
int sseshift, n;
|
2012-07-14 00:21:29 +02:00
|
|
|
int step_param = 0;
|
|
|
|
int further_steps;
|
2012-08-07 01:21:23 +02:00
|
|
|
int thissme, bestsme = INT_MAX;
|
2013-04-24 01:22:47 +02:00
|
|
|
const struct buf_2d orig_src = x->plane[0].src;
|
2013-04-25 20:15:38 +02:00
|
|
|
const struct buf_2d orig_pre = x->e_mbd.plane[0].pre[0];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
/* Is the best so far sufficiently good that we cant justify doing
|
|
|
|
* and new motion search. */
|
2012-07-14 00:21:29 +02:00
|
|
|
if (best_label_rd < label_mv_thresh)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (cpi->compressor_speed) {
|
2012-10-22 20:49:00 +02:00
|
|
|
if (segmentation == PARTITIONING_8X16 ||
|
|
|
|
segmentation == PARTITIONING_16X8) {
|
2012-07-14 00:21:29 +02:00
|
|
|
bsi->mvp.as_int = bsi->sv_mvp[i].as_int;
|
2012-10-22 20:49:00 +02:00
|
|
|
if (i == 1 && segmentation == PARTITIONING_16X8)
|
2012-07-14 00:21:29 +02:00
|
|
|
bsi->mvp.as_int = bsi->sv_mvp[2].as_int;
|
|
|
|
|
|
|
|
step_param = bsi->sv_istep[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
// use previous block's result as next block's MV predictor.
|
2012-10-22 20:49:00 +02:00
|
|
|
if (segmentation == PARTITIONING_4X4 && i > 0) {
|
2013-04-26 16:19:43 +02:00
|
|
|
bsi->mvp.as_int =
|
|
|
|
x->e_mbd.mode_info_context->bmi[i - 1].as_mv[0].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
if (i == 4 || i == 8 || i == 12)
|
2013-04-26 16:19:43 +02:00
|
|
|
bsi->mvp.as_int =
|
|
|
|
x->e_mbd.mode_info_context->bmi[i - 4].as_mv[0].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
step_param = 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
further_steps = (MAX_MVSEARCH_STEPS - 1) - step_param;
|
|
|
|
|
|
|
|
{
|
|
|
|
int sadpb = x->sadperbit4;
|
|
|
|
int_mv mvp_full;
|
|
|
|
|
|
|
|
mvp_full.as_mv.row = bsi->mvp.as_mv.row >> 3;
|
|
|
|
mvp_full.as_mv.col = bsi->mvp.as_mv.col >> 3;
|
|
|
|
|
|
|
|
// find first label
|
2012-10-31 01:12:12 +01:00
|
|
|
n = vp9_mbsplit_offset[segmentation][i];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
// adjust src pointer for this segment
|
|
|
|
x->plane[0].src.buf =
|
|
|
|
raster_block_offset_uint8(&x->e_mbd, BLOCK_SIZE_MB16X16, 0, n,
|
|
|
|
x->plane[0].src.buf,
|
|
|
|
x->plane[0].src.stride);
|
2013-04-25 20:15:38 +02:00
|
|
|
assert(((intptr_t)x->e_mbd.plane[0].pre[0].buf & 0xf) == 0);
|
|
|
|
x->e_mbd.plane[0].pre[0].buf =
|
|
|
|
raster_block_offset_uint8(&x->e_mbd, BLOCK_SIZE_MB16X16, 0, n,
|
|
|
|
x->e_mbd.plane[0].pre[0].buf,
|
|
|
|
x->e_mbd.plane[0].pre[0].stride);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
bestsme = vp9_full_pixel_diamond(cpi, x, &mvp_full, step_param,
|
2012-08-08 20:52:22 +02:00
|
|
|
sadpb, further_steps, 0, v_fn_ptr,
|
|
|
|
bsi->ref_mv, &mode_mv[NEW4X4]);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
sseshift = segmentation_to_sseshift[segmentation];
|
2010-10-26 21:34:16 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// Should we do a full search (best quality only)
|
|
|
|
if ((cpi->compressor_speed == 0) && (bestsme >> sseshift) > 4000) {
|
|
|
|
/* Check if mvp_full is within the range. */
|
2012-10-31 22:40:53 +01:00
|
|
|
clamp_mv(&mvp_full, x->mv_col_min, x->mv_col_max,
|
|
|
|
x->mv_row_min, x->mv_row_max);
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
thissme = cpi->full_search_sad(x, &mvp_full,
|
2012-07-14 00:21:29 +02:00
|
|
|
sadpb, 16, v_fn_ptr,
|
2012-11-09 00:44:39 +01:00
|
|
|
x->nmvjointcost, x->mvcost,
|
2013-04-26 16:19:43 +02:00
|
|
|
bsi->ref_mv,
|
|
|
|
n);
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (thissme < bestsme) {
|
|
|
|
bestsme = thissme;
|
2013-04-26 16:19:43 +02:00
|
|
|
mode_mv[NEW4X4].as_int =
|
|
|
|
x->e_mbd.mode_info_context->bmi[n].as_mv[0].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
2012-10-22 20:25:48 +02:00
|
|
|
/* The full search result is actually worse so re-instate the
|
|
|
|
* previous best vector */
|
2013-04-26 16:19:43 +02:00
|
|
|
x->e_mbd.mode_info_context->bmi[n].as_mv[0].as_int =
|
|
|
|
mode_mv[NEW4X4].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (bestsme < INT_MAX) {
|
|
|
|
int distortion;
|
|
|
|
unsigned int sse;
|
2013-04-24 01:22:47 +02:00
|
|
|
cpi->find_fractional_mv_step(x, &mode_mv[NEW4X4],
|
2012-10-22 20:25:48 +02:00
|
|
|
bsi->ref_mv, x->errorperbit, v_fn_ptr,
|
2012-11-09 00:44:39 +01:00
|
|
|
x->nmvjointcost, x->mvcost,
|
|
|
|
&distortion, &sse);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// safe motion search result for use in compound prediction
|
2012-08-10 15:12:43 +02:00
|
|
|
seg_mvs[i][mbmi->ref_frame - 1].as_int = mode_mv[NEW4X4].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2013-04-24 01:22:47 +02:00
|
|
|
|
2013-04-25 20:15:38 +02:00
|
|
|
// restore src pointers
|
2013-04-24 01:22:47 +02:00
|
|
|
x->plane[0].src = orig_src;
|
2013-04-25 20:15:38 +02:00
|
|
|
x->e_mbd.plane[0].pre[0] = orig_pre;
|
2012-11-07 15:50:25 +01:00
|
|
|
} else if (mbmi->second_ref_frame > 0 && this_mode == NEW4X4) {
|
|
|
|
/* NEW4X4 */
|
2012-10-22 20:25:48 +02:00
|
|
|
/* motion search not completed? Then skip newmv for this block with
|
|
|
|
* comppred */
|
2012-08-10 15:12:43 +02:00
|
|
|
if (seg_mvs[i][mbmi->second_ref_frame - 1].as_int == INVALID_MV ||
|
|
|
|
seg_mvs[i][mbmi->ref_frame - 1].as_int == INVALID_MV) {
|
2012-07-14 00:21:29 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
rate = labels2mode(x, labels, i, this_mode, &mode_mv[this_mode],
|
2012-08-10 15:12:43 +02:00
|
|
|
&second_mode_mv[this_mode], seg_mvs[i],
|
2012-11-09 00:44:39 +01:00
|
|
|
bsi->ref_mv, bsi->second_ref_mv, x->nmvjointcost,
|
2013-04-26 20:57:17 +02:00
|
|
|
x->mvcost, cpi);
|
2010-06-11 20:33:49 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// Trap vectors that reach beyond the UMV borders
|
2012-08-10 15:12:43 +02:00
|
|
|
if (((mode_mv[this_mode].as_mv.row >> 3) < x->mv_row_min) ||
|
|
|
|
((mode_mv[this_mode].as_mv.row >> 3) > x->mv_row_max) ||
|
|
|
|
((mode_mv[this_mode].as_mv.col >> 3) < x->mv_col_min) ||
|
|
|
|
((mode_mv[this_mode].as_mv.col >> 3) > x->mv_col_max)) {
|
2012-07-14 00:21:29 +02:00
|
|
|
continue;
|
|
|
|
}
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0 &&
|
2012-08-08 20:13:53 +02:00
|
|
|
mv_check_bounds(x, &second_mode_mv[this_mode]))
|
|
|
|
continue;
|
2010-06-11 20:33:49 +02:00
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
if (segmentation == PARTITIONING_4X4) {
|
2013-02-20 19:16:24 +01:00
|
|
|
this_rd = encode_inter_mb_segment(&cpi->common,
|
|
|
|
x, labels, i, &labelyrate,
|
2013-04-29 19:37:25 +02:00
|
|
|
&distortion, t_above_s, t_left_s);
|
2012-10-22 20:49:00 +02:00
|
|
|
other_rd = this_rd;
|
2012-10-17 20:40:00 +02:00
|
|
|
} else {
|
2013-02-20 19:16:24 +01:00
|
|
|
this_rd = encode_inter_mb_segment_8x8(&cpi->common,
|
|
|
|
x, labels, i, &labelyrate,
|
2012-10-22 20:49:00 +02:00
|
|
|
&distortion, &other_rd,
|
2013-04-29 19:37:25 +02:00
|
|
|
t_above_s, t_left_s);
|
2012-10-17 20:40:00 +02:00
|
|
|
}
|
|
|
|
this_rd += RDCOST(x->rdmult, x->rddiv, rate, 0);
|
2012-07-14 00:21:29 +02:00
|
|
|
rate += labelyrate;
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (this_rd < best_label_rd) {
|
|
|
|
sbr = rate;
|
|
|
|
sbd = distortion;
|
|
|
|
bestlabelyrate = labelyrate;
|
|
|
|
mode_selected = this_mode;
|
|
|
|
best_label_rd = this_rd;
|
2012-10-22 20:49:00 +02:00
|
|
|
if (x->e_mbd.mode_info_context->mbmi.txfm_size == TX_4X4) {
|
|
|
|
for (j = 0; j < 16; j++)
|
|
|
|
if (labels[j] == i)
|
2013-04-04 21:03:27 +02:00
|
|
|
best_eobs[j] = x->e_mbd.plane[0].eobs[j];
|
2012-10-22 20:49:00 +02:00
|
|
|
} else {
|
|
|
|
for (j = 0; j < 4; j++) {
|
2012-10-31 01:12:12 +01:00
|
|
|
int ib = vp9_i8x8_block[j], idx = j * 4;
|
2012-10-22 20:49:00 +02:00
|
|
|
|
|
|
|
if (labels[ib] == i)
|
2013-04-04 21:03:27 +02:00
|
|
|
best_eobs[idx] = x->e_mbd.plane[0].eobs[idx];
|
2012-10-22 20:49:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (other_rd < best_other_rd)
|
|
|
|
best_other_rd = other_rd;
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above_b, t_above_s, sizeof(t_above_s));
|
|
|
|
vpx_memcpy(t_left_b, t_left_s, sizeof(t_left_s));
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
} /*for each 4x4 mode*/
|
2012-04-18 22:51:58 +02:00
|
|
|
|
2013-04-29 19:37:25 +02:00
|
|
|
vpx_memcpy(t_above, t_above_b, sizeof(t_above));
|
|
|
|
vpx_memcpy(t_left, t_left_b, sizeof(t_left));
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
labels2mode(x, labels, i, mode_selected, &mode_mv[mode_selected],
|
2012-10-22 20:49:00 +02:00
|
|
|
&second_mode_mv[mode_selected], seg_mvs[i],
|
2013-04-26 20:57:17 +02:00
|
|
|
bsi->ref_mv, bsi->second_ref_mv, x->nmvjointcost,
|
|
|
|
x->mvcost, cpi);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
br += sbr;
|
|
|
|
bd += sbd;
|
|
|
|
segmentyrate += bestlabelyrate;
|
|
|
|
this_segment_rd += best_label_rd;
|
2012-10-22 20:49:00 +02:00
|
|
|
other_segment_rd += best_other_rd;
|
|
|
|
if (rds)
|
|
|
|
rds[i] = this_segment_rd;
|
|
|
|
if (otherrds)
|
2012-10-24 22:03:51 +02:00
|
|
|
otherrds[i] = other_segment_rd;
|
2012-07-14 00:21:29 +02:00
|
|
|
} /* for each label */
|
2012-02-16 18:29:54 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (this_segment_rd < bsi->segment_rd) {
|
|
|
|
bsi->r = br;
|
|
|
|
bsi->d = bd;
|
|
|
|
bsi->segment_yrate = segmentyrate;
|
|
|
|
bsi->segment_rd = this_segment_rd;
|
|
|
|
bsi->segment_num = segmentation;
|
2012-10-22 20:49:00 +02:00
|
|
|
bsi->txfm_size = mbmi->txfm_size;
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// store everything needed to come back to this!!
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
bsi->mvs[i].as_mv = x->partition_info->bmi[i].mv.as_mv;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2012-07-14 00:21:29 +02:00
|
|
|
bsi->second_mvs[i].as_mv = x->partition_info->bmi[i].second_mv.as_mv;
|
|
|
|
bsi->modes[i] = x->partition_info->bmi[i].mode;
|
2012-10-17 23:32:17 +02:00
|
|
|
bsi->eobs[i] = best_eobs[i];
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2012-10-22 20:49:00 +02:00
|
|
|
|
|
|
|
if (completed) {
|
|
|
|
*completed = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-31 01:53:32 +01:00
|
|
|
static void rd_check_segment(VP9_COMP *cpi, MACROBLOCK *x,
|
2012-10-22 20:49:00 +02:00
|
|
|
BEST_SEG_INFO *bsi,
|
|
|
|
unsigned int segmentation,
|
|
|
|
/* 16 = n_blocks */
|
|
|
|
int_mv seg_mvs[16][MAX_REF_FRAMES - 1],
|
|
|
|
int64_t txfm_cache[NB_TXFM_MODES]) {
|
2012-10-31 01:12:12 +01:00
|
|
|
int i, n, c = vp9_mbsplit_count[segmentation];
|
2012-10-22 20:49:00 +02:00
|
|
|
|
|
|
|
if (segmentation == PARTITIONING_4X4) {
|
|
|
|
int64_t rd[16];
|
|
|
|
|
|
|
|
rd_check_segment_txsize(cpi, x, bsi, segmentation, TX_4X4, NULL,
|
|
|
|
rd, &n, seg_mvs);
|
|
|
|
if (n == c) {
|
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
if (rd[c - 1] < txfm_cache[i])
|
|
|
|
txfm_cache[i] = rd[c - 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int64_t diff, base_rd;
|
2012-10-31 22:40:53 +01:00
|
|
|
int cost4x4 = vp9_cost_bit(cpi->common.prob_tx[0], 0);
|
|
|
|
int cost8x8 = vp9_cost_bit(cpi->common.prob_tx[0], 1);
|
2012-10-22 20:49:00 +02:00
|
|
|
|
|
|
|
if (cpi->common.txfm_mode == TX_MODE_SELECT) {
|
|
|
|
int64_t rd4x4[4], rd8x8[4];
|
|
|
|
int n4x4, n8x8, nmin;
|
|
|
|
BEST_SEG_INFO bsi4x4, bsi8x8;
|
|
|
|
|
|
|
|
/* factor in cost of cost4x4/8x8 in decision */
|
|
|
|
vpx_memcpy(&bsi4x4, bsi, sizeof(*bsi));
|
|
|
|
vpx_memcpy(&bsi8x8, bsi, sizeof(*bsi));
|
|
|
|
rd_check_segment_txsize(cpi, x, &bsi4x4, segmentation,
|
|
|
|
TX_4X4, NULL, rd4x4, &n4x4, seg_mvs);
|
|
|
|
rd_check_segment_txsize(cpi, x, &bsi8x8, segmentation,
|
|
|
|
TX_8X8, NULL, rd8x8, &n8x8, seg_mvs);
|
|
|
|
if (bsi4x4.segment_num == segmentation) {
|
|
|
|
bsi4x4.segment_rd += RDCOST(x->rdmult, x->rddiv, cost4x4, 0);
|
|
|
|
if (bsi4x4.segment_rd < bsi->segment_rd)
|
|
|
|
vpx_memcpy(bsi, &bsi4x4, sizeof(*bsi));
|
|
|
|
}
|
|
|
|
if (bsi8x8.segment_num == segmentation) {
|
|
|
|
bsi8x8.segment_rd += RDCOST(x->rdmult, x->rddiv, cost8x8, 0);
|
|
|
|
if (bsi8x8.segment_rd < bsi->segment_rd)
|
|
|
|
vpx_memcpy(bsi, &bsi8x8, sizeof(*bsi));
|
|
|
|
}
|
|
|
|
n = n4x4 > n8x8 ? n4x4 : n8x8;
|
|
|
|
if (n == c) {
|
|
|
|
nmin = n4x4 < n8x8 ? n4x4 : n8x8;
|
|
|
|
diff = rd8x8[nmin - 1] - rd4x4[nmin - 1];
|
|
|
|
if (n == n4x4) {
|
|
|
|
base_rd = rd4x4[c - 1];
|
|
|
|
} else {
|
|
|
|
base_rd = rd8x8[c - 1] - diff;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int64_t rd[4], otherrd[4];
|
|
|
|
|
|
|
|
if (cpi->common.txfm_mode == ONLY_4X4) {
|
|
|
|
rd_check_segment_txsize(cpi, x, bsi, segmentation, TX_4X4, otherrd,
|
|
|
|
rd, &n, seg_mvs);
|
|
|
|
if (n == c) {
|
|
|
|
base_rd = rd[c - 1];
|
|
|
|
diff = otherrd[c - 1] - rd[c - 1];
|
|
|
|
}
|
|
|
|
} else /* use 8x8 transform */ {
|
|
|
|
rd_check_segment_txsize(cpi, x, bsi, segmentation, TX_8X8, otherrd,
|
|
|
|
rd, &n, seg_mvs);
|
|
|
|
if (n == c) {
|
|
|
|
diff = rd[c - 1] - otherrd[c - 1];
|
|
|
|
base_rd = otherrd[c - 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (n == c) {
|
|
|
|
if (base_rd < txfm_cache[ONLY_4X4]) {
|
|
|
|
txfm_cache[ONLY_4X4] = base_rd;
|
|
|
|
}
|
2013-02-16 01:39:22 +01:00
|
|
|
if (base_rd + diff < txfm_cache[ALLOW_8X8]) {
|
2013-02-16 00:55:31 +01:00
|
|
|
txfm_cache[ALLOW_8X8] = txfm_cache[ALLOW_16X16] =
|
|
|
|
txfm_cache[ALLOW_32X32] = base_rd + diff;
|
2012-10-22 20:49:00 +02:00
|
|
|
}
|
|
|
|
if (diff < 0) {
|
|
|
|
base_rd += diff + RDCOST(x->rdmult, x->rddiv, cost8x8, 0);
|
|
|
|
} else {
|
|
|
|
base_rd += RDCOST(x->rdmult, x->rddiv, cost4x4, 0);
|
|
|
|
}
|
|
|
|
if (base_rd < txfm_cache[TX_MODE_SELECT]) {
|
|
|
|
txfm_cache[TX_MODE_SELECT] = base_rd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-12-06 22:42:52 +01:00
|
|
|
}
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2013-02-06 21:45:28 +01:00
|
|
|
static INLINE void cal_step_param(int sr, int *sp) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int step = 0;
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (sr > MAX_FIRST_STEP) sr = MAX_FIRST_STEP;
|
|
|
|
else if (sr < 1) sr = 1;
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
while (sr >>= 1)
|
|
|
|
step++;
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
*sp = MAX_MVSEARCH_STEPS - 1 - step;
|
2010-12-23 17:23:03 +01:00
|
|
|
}
|
|
|
|
|
2012-10-31 01:53:32 +01:00
|
|
|
static int rd_pick_best_mbsegmentation(VP9_COMP *cpi, MACROBLOCK *x,
|
2012-10-22 20:25:48 +02:00
|
|
|
int_mv *best_ref_mv,
|
|
|
|
int_mv *second_best_ref_mv,
|
|
|
|
int64_t best_rd,
|
|
|
|
int *mdcounts,
|
|
|
|
int *returntotrate,
|
|
|
|
int *returnyrate,
|
|
|
|
int *returndistortion,
|
|
|
|
int *skippable, int mvthresh,
|
|
|
|
int_mv seg_mvs[NB_PARTITIONINGS]
|
|
|
|
[16 /* n_blocks */]
|
2012-10-22 20:49:00 +02:00
|
|
|
[MAX_REF_FRAMES - 1],
|
|
|
|
int64_t txfm_cache[NB_TXFM_MODES]) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int i;
|
|
|
|
BEST_SEG_INFO bsi;
|
2012-08-10 15:12:43 +02:00
|
|
|
MB_MODE_INFO * mbmi = &x->e_mbd.mode_info_context->mbmi;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
vpx_memset(&bsi, 0, sizeof(bsi));
|
2012-10-22 20:49:00 +02:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++)
|
2013-01-14 20:49:30 +01:00
|
|
|
txfm_cache[i] = INT64_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
bsi.segment_rd = best_rd;
|
|
|
|
bsi.ref_mv = best_ref_mv;
|
|
|
|
bsi.second_ref_mv = second_best_ref_mv;
|
|
|
|
bsi.mvp.as_int = best_ref_mv->as_int;
|
|
|
|
bsi.mvthresh = mvthresh;
|
|
|
|
bsi.mdcounts = mdcounts;
|
2012-10-22 20:49:00 +02:00
|
|
|
bsi.txfm_size = TX_4X4;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-07 01:21:23 +02:00
|
|
|
for (i = 0; i < 16; i++)
|
2012-07-14 00:21:29 +02:00
|
|
|
bsi.modes[i] = ZERO4X4;
|
|
|
|
|
|
|
|
if (cpi->compressor_speed == 0) {
|
|
|
|
/* for now, we will keep the original segmentation order
|
|
|
|
when in best quality mode */
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_16X8,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_16X8], txfm_cache);
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_8X16,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_8X16], txfm_cache);
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_8X8,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_8X8], txfm_cache);
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_4X4,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_4X4], txfm_cache);
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
|
|
|
int sr;
|
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_8X8,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_8X8], txfm_cache);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
if (bsi.segment_rd < best_rd) {
|
|
|
|
int tmp_col_min = x->mv_col_min;
|
|
|
|
int tmp_col_max = x->mv_col_max;
|
|
|
|
int tmp_row_min = x->mv_row_min;
|
|
|
|
int tmp_row_max = x->mv_row_max;
|
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_clamp_mv_min_max(x, best_ref_mv);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
/* Get 8x8 result */
|
|
|
|
bsi.sv_mvp[0].as_int = bsi.mvs[0].as_int;
|
|
|
|
bsi.sv_mvp[1].as_int = bsi.mvs[2].as_int;
|
|
|
|
bsi.sv_mvp[2].as_int = bsi.mvs[8].as_int;
|
|
|
|
bsi.sv_mvp[3].as_int = bsi.mvs[10].as_int;
|
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
/* Use 8x8 result as 16x8/8x16's predictor MV. Adjust search range
|
|
|
|
* according to the closeness of 2 MV. */
|
2012-07-14 00:21:29 +02:00
|
|
|
/* block 8X16 */
|
2012-10-22 20:25:48 +02:00
|
|
|
sr = MAXF((abs(bsi.sv_mvp[0].as_mv.row - bsi.sv_mvp[2].as_mv.row)) >> 3,
|
|
|
|
(abs(bsi.sv_mvp[0].as_mv.col - bsi.sv_mvp[2].as_mv.col)) >> 3);
|
2012-10-31 22:40:53 +01:00
|
|
|
cal_step_param(sr, &bsi.sv_istep[0]);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
sr = MAXF((abs(bsi.sv_mvp[1].as_mv.row - bsi.sv_mvp[3].as_mv.row)) >> 3,
|
|
|
|
(abs(bsi.sv_mvp[1].as_mv.col - bsi.sv_mvp[3].as_mv.col)) >> 3);
|
2012-10-31 22:40:53 +01:00
|
|
|
cal_step_param(sr, &bsi.sv_istep[1]);
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_8X16,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_8X16], txfm_cache);
|
2012-02-16 18:29:54 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* block 16X8 */
|
2012-10-22 20:25:48 +02:00
|
|
|
sr = MAXF((abs(bsi.sv_mvp[0].as_mv.row - bsi.sv_mvp[1].as_mv.row)) >> 3,
|
|
|
|
(abs(bsi.sv_mvp[0].as_mv.col - bsi.sv_mvp[1].as_mv.col)) >> 3);
|
2012-10-31 22:40:53 +01:00
|
|
|
cal_step_param(sr, &bsi.sv_istep[0]);
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
sr = MAXF((abs(bsi.sv_mvp[2].as_mv.row - bsi.sv_mvp[3].as_mv.row)) >> 3,
|
|
|
|
(abs(bsi.sv_mvp[2].as_mv.col - bsi.sv_mvp[3].as_mv.col)) >> 3);
|
2012-10-31 22:40:53 +01:00
|
|
|
cal_step_param(sr, &bsi.sv_istep[1]);
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_16X8,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_16X8], txfm_cache);
|
2010-12-23 17:23:03 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* If 8x8 is better than 16x8/8x16, then do 4x4 search */
|
|
|
|
/* Not skip 4x4 if speed=0 (good quality) */
|
2012-10-22 20:49:00 +02:00
|
|
|
if (cpi->sf.no_skip_block4x4_search ||
|
|
|
|
bsi.segment_num == PARTITIONING_8X8) {
|
2012-10-22 20:25:48 +02:00
|
|
|
/* || (sv_segment_rd8x8-bsi.segment_rd) < sv_segment_rd8x8>>5) */
|
2012-07-14 00:21:29 +02:00
|
|
|
bsi.mvp.as_int = bsi.sv_mvp[0].as_int;
|
2012-10-22 20:25:48 +02:00
|
|
|
rd_check_segment(cpi, x, &bsi, PARTITIONING_4X4,
|
2012-10-22 20:49:00 +02:00
|
|
|
seg_mvs[PARTITIONING_4X4], txfm_cache);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-12-28 19:23:07 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* restore UMV window */
|
|
|
|
x->mv_col_min = tmp_col_min;
|
|
|
|
x->mv_col_max = tmp_col_max;
|
|
|
|
x->mv_row_min = tmp_row_min;
|
|
|
|
x->mv_row_max = tmp_row_max;
|
2010-12-16 23:01:27 +01:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-12-06 22:42:52 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* set it to the best */
|
|
|
|
for (i = 0; i < 16; i++) {
|
2013-04-26 16:19:43 +02:00
|
|
|
x->e_mbd.mode_info_context->bmi[i].as_mv[0].as_int = bsi.mvs[i].as_int;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2013-04-26 16:19:43 +02:00
|
|
|
x->e_mbd.mode_info_context->bmi[i].as_mv[1].as_int =
|
|
|
|
bsi.second_mvs[i].as_int;
|
2013-04-04 21:03:27 +02:00
|
|
|
x->e_mbd.plane[0].eobs[i] = bsi.eobs[i];
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* save partitions */
|
2012-10-22 20:49:00 +02:00
|
|
|
mbmi->txfm_size = bsi.txfm_size;
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->partitioning = bsi.segment_num;
|
2012-10-31 01:12:12 +01:00
|
|
|
x->partition_info->count = vp9_mbsplit_count[bsi.segment_num];
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
for (i = 0; i < x->partition_info->count; i++) {
|
|
|
|
int j;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-31 01:12:12 +01:00
|
|
|
j = vp9_mbsplit_offset[bsi.segment_num][i];
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
x->partition_info->bmi[i].mode = bsi.modes[j];
|
|
|
|
x->partition_info->bmi[i].mv.as_mv = bsi.mvs[j].as_mv;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2012-07-14 00:21:29 +02:00
|
|
|
x->partition_info->bmi[i].second_mv.as_mv = bsi.second_mvs[j].as_mv;
|
|
|
|
}
|
|
|
|
/*
|
2012-08-10 15:12:43 +02:00
|
|
|
* used to set mbmi->mv.as_int
|
2012-07-14 00:21:29 +02:00
|
|
|
*/
|
|
|
|
x->partition_info->bmi[15].mv.as_int = bsi.mvs[15].as_int;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2012-07-14 00:21:29 +02:00
|
|
|
x->partition_info->bmi[15].second_mv.as_int = bsi.second_mvs[15].as_int;
|
|
|
|
|
2013-04-11 20:14:31 +02:00
|
|
|
*returntotrate = bsi.r;
|
|
|
|
*returndistortion = bsi.d;
|
|
|
|
*returnyrate = bsi.segment_yrate;
|
|
|
|
*skippable = vp9_sby_is_skippable(&x->e_mbd, BLOCK_SIZE_MB16X16);
|
|
|
|
|
2012-11-05 23:22:59 +01:00
|
|
|
return (int)(bsi.segment_rd);
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif // !CONFIG_SB8X8
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
static void mv_pred(VP9_COMP *cpi, MACROBLOCK *x,
|
2012-12-19 00:31:19 +01:00
|
|
|
uint8_t *ref_y_buffer, int ref_y_stride,
|
2012-12-05 17:23:38 +01:00
|
|
|
int ref_frame, enum BlockSize block_size ) {
|
2012-11-26 12:29:08 +01:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
|
|
|
MB_MODE_INFO *mbmi = &xd->mode_info_context->mbmi;
|
|
|
|
int_mv this_mv;
|
|
|
|
int i;
|
2013-04-16 00:24:39 +02:00
|
|
|
int zero_seen = 0;
|
2012-11-29 18:02:47 +01:00
|
|
|
int best_index = 0;
|
2012-11-26 12:29:08 +01:00
|
|
|
int best_sad = INT_MAX;
|
|
|
|
int this_sad = INT_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
uint8_t *src_y_ptr = x->plane[0].src.buf;
|
2012-12-19 00:31:19 +01:00
|
|
|
uint8_t *ref_y_ptr;
|
2012-11-29 18:02:47 +01:00
|
|
|
int row_offset, col_offset;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
// Get the sad for each candidate reference mv
|
2013-04-19 12:10:16 +02:00
|
|
|
for (i = 0; i < MAX_MV_REF_CANDIDATES; i++) {
|
2012-11-26 12:29:08 +01:00
|
|
|
this_mv.as_int = mbmi->ref_mvs[ref_frame][i].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
// The list is at an end if we see 0 for a second time.
|
|
|
|
if (!this_mv.as_int && zero_seen)
|
|
|
|
break;
|
|
|
|
zero_seen = zero_seen || !this_mv.as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
row_offset = this_mv.as_mv.row >> 3;
|
|
|
|
col_offset = this_mv.as_mv.col >> 3;
|
|
|
|
ref_y_ptr = ref_y_buffer + (ref_y_stride * row_offset) + col_offset;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
// Find sad for current vector.
|
2013-04-24 01:22:47 +02:00
|
|
|
this_sad = cpi->fn_ptr[block_size].sdf(src_y_ptr, x->plane[0].src.stride,
|
2012-11-26 12:29:08 +01:00
|
|
|
ref_y_ptr, ref_y_stride,
|
|
|
|
0x7fffffff);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
// Note if it is the best so far.
|
|
|
|
if (this_sad < best_sad) {
|
|
|
|
best_sad = this_sad;
|
|
|
|
best_index = i;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-05 17:23:38 +01:00
|
|
|
// Note the index of the mv that worked best in the reference list.
|
|
|
|
x->mv_best_ref_index[ref_frame] = best_index;
|
2010-12-03 17:26:21 +01:00
|
|
|
}
|
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#if !CONFIG_SB8X8
|
2013-01-14 23:37:53 +01:00
|
|
|
static void set_i8x8_block_modes(MACROBLOCK *x, int modes[4]) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int i;
|
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
|
|
|
for (i = 0; i < 4; i++) {
|
2012-10-31 01:12:12 +01:00
|
|
|
int ib = vp9_i8x8_block[i];
|
2013-01-14 23:37:53 +01:00
|
|
|
xd->mode_info_context->bmi[ib + 0].as_mode.first = modes[i];
|
|
|
|
xd->mode_info_context->bmi[ib + 1].as_mode.first = modes[i];
|
|
|
|
xd->mode_info_context->bmi[ib + 4].as_mode.first = modes[i];
|
|
|
|
xd->mode_info_context->bmi[ib + 5].as_mode.first = modes[i];
|
|
|
|
// printf("%d,%d,%d,%d\n",
|
|
|
|
// modes[0], modes[1], modes[2], modes[3]);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2011-12-01 01:25:00 +01:00
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif
|
2011-12-01 01:25:00 +01:00
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
extern void vp9_calc_ref_probs(int *count, vp9_prob *probs);
|
|
|
|
static void estimate_curframe_refprobs(VP9_COMP *cpi, vp9_prob mod_refprobs[3], int pred_ref) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int norm_cnt[MAX_REF_FRAMES];
|
|
|
|
const int *const rfct = cpi->count_mb_ref_frame_usage;
|
|
|
|
int intra_count = rfct[INTRA_FRAME];
|
|
|
|
int last_count = rfct[LAST_FRAME];
|
|
|
|
int gf_count = rfct[GOLDEN_FRAME];
|
|
|
|
int arf_count = rfct[ALTREF_FRAME];
|
|
|
|
|
|
|
|
// Work out modified reference frame probabilities to use where prediction
|
|
|
|
// of the reference frame fails
|
|
|
|
if (pred_ref == INTRA_FRAME) {
|
|
|
|
norm_cnt[0] = 0;
|
|
|
|
norm_cnt[1] = last_count;
|
|
|
|
norm_cnt[2] = gf_count;
|
|
|
|
norm_cnt[3] = arf_count;
|
2012-10-29 14:44:18 +01:00
|
|
|
vp9_calc_ref_probs(norm_cnt, mod_refprobs);
|
2012-07-14 00:21:29 +02:00
|
|
|
mod_refprobs[0] = 0; // This branch implicit
|
|
|
|
} else if (pred_ref == LAST_FRAME) {
|
|
|
|
norm_cnt[0] = intra_count;
|
|
|
|
norm_cnt[1] = 0;
|
|
|
|
norm_cnt[2] = gf_count;
|
|
|
|
norm_cnt[3] = arf_count;
|
2012-10-29 14:44:18 +01:00
|
|
|
vp9_calc_ref_probs(norm_cnt, mod_refprobs);
|
2012-07-14 00:21:29 +02:00
|
|
|
mod_refprobs[1] = 0; // This branch implicit
|
|
|
|
} else if (pred_ref == GOLDEN_FRAME) {
|
|
|
|
norm_cnt[0] = intra_count;
|
|
|
|
norm_cnt[1] = last_count;
|
|
|
|
norm_cnt[2] = 0;
|
|
|
|
norm_cnt[3] = arf_count;
|
2012-10-29 14:44:18 +01:00
|
|
|
vp9_calc_ref_probs(norm_cnt, mod_refprobs);
|
2012-07-14 00:21:29 +02:00
|
|
|
mod_refprobs[2] = 0; // This branch implicit
|
|
|
|
} else {
|
|
|
|
norm_cnt[0] = intra_count;
|
|
|
|
norm_cnt[1] = last_count;
|
|
|
|
norm_cnt[2] = gf_count;
|
|
|
|
norm_cnt[3] = 0;
|
2012-10-29 14:44:18 +01:00
|
|
|
vp9_calc_ref_probs(norm_cnt, mod_refprobs);
|
2012-07-14 00:21:29 +02:00
|
|
|
mod_refprobs[2] = 0; // This branch implicit
|
|
|
|
}
|
2012-05-15 02:39:42 +02:00
|
|
|
}
|
|
|
|
|
2013-02-06 21:45:28 +01:00
|
|
|
static INLINE unsigned weighted_cost(vp9_prob *tab0, vp9_prob *tab1,
|
|
|
|
int idx, int val, int weight) {
|
2012-10-31 22:40:53 +01:00
|
|
|
unsigned cost0 = tab0[idx] ? vp9_cost_bit(tab0[idx], val) : 0;
|
|
|
|
unsigned cost1 = tab1[idx] ? vp9_cost_bit(tab1[idx], val) : 0;
|
2012-07-14 00:21:29 +02:00
|
|
|
// weight is 16-bit fixed point, so this basically calculates:
|
|
|
|
// 0.5 + weight * cost1 + (1.0 - weight) * cost0
|
|
|
|
return (0x8000 + weight * cost1 + (0x10000 - weight) * cost0) >> 16;
|
2012-05-15 02:39:42 +02:00
|
|
|
}
|
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
static void estimate_ref_frame_costs(VP9_COMP *cpi, int segment_id, unsigned int *ref_costs) {
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 00:21:29 +02:00
|
|
|
MACROBLOCKD *xd = &cpi->mb.e_mbd;
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_prob *mod_refprobs;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
unsigned int cost;
|
|
|
|
int pred_ref;
|
|
|
|
int pred_flag;
|
|
|
|
int pred_ctx;
|
|
|
|
int i;
|
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_prob pred_prob, new_pred_prob;
|
2012-07-14 00:21:29 +02:00
|
|
|
int seg_ref_active;
|
|
|
|
int seg_ref_count = 0;
|
2012-10-30 06:15:27 +01:00
|
|
|
seg_ref_active = vp9_segfeature_active(xd,
|
|
|
|
segment_id,
|
|
|
|
SEG_LVL_REF_FRAME);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
if (seg_ref_active) {
|
2012-10-30 06:15:27 +01:00
|
|
|
seg_ref_count = vp9_check_segref(xd, segment_id, INTRA_FRAME) +
|
|
|
|
vp9_check_segref(xd, segment_id, LAST_FRAME) +
|
|
|
|
vp9_check_segref(xd, segment_id, GOLDEN_FRAME) +
|
|
|
|
vp9_check_segref(xd, segment_id, ALTREF_FRAME);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the predicted reference for this mb
|
2012-10-29 14:44:18 +01:00
|
|
|
pred_ref = vp9_get_pred_ref(cm, xd);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// Get the context probability for the prediction flag (based on last frame)
|
2012-10-29 14:44:18 +01:00
|
|
|
pred_prob = vp9_get_pred_prob(cm, xd, PRED_REF);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// Predict probability for current frame based on stats so far
|
2012-10-29 14:44:18 +01:00
|
|
|
pred_ctx = vp9_get_pred_context(cm, xd, PRED_REF);
|
Consistently use get_prob(), clip_prob() and newly added clip_pixel().
Add a function clip_pixel() to clip a pixel value to the [0,255] range
of allowed values, and use this where-ever appropriate (e.g. prediction,
reconstruction). Likewise, consistently use the recently added function
clip_prob(), which calculates a binary probability in the [1,255] range.
If possible, try to use get_prob() or its sister get_binary_prob() to
calculate binary probabilities, for consistency.
Since in some places, this means that binary probability calculations
are changed (we use {255,256}*count0/(total) in a range of places,
and all of these are now changed to use 256*count0+(total>>1)/total),
this changes the encoding result, so this patch warrants some extensive
testing.
Change-Id: Ibeeff8d886496839b8e0c0ace9ccc552351f7628
2012-12-10 21:09:07 +01:00
|
|
|
new_pred_prob = get_binary_prob(cpi->ref_pred_count[pred_ctx][0],
|
|
|
|
cpi->ref_pred_count[pred_ctx][1]);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// Get the set of probabilities to use if prediction fails
|
|
|
|
mod_refprobs = cm->mod_refprobs[pred_ref];
|
|
|
|
|
|
|
|
// For each possible selected reference frame work out a cost.
|
|
|
|
for (i = 0; i < MAX_REF_FRAMES; i++) {
|
|
|
|
if (seg_ref_active && seg_ref_count == 1) {
|
|
|
|
cost = 0;
|
|
|
|
} else {
|
|
|
|
pred_flag = (i == pred_ref);
|
|
|
|
|
|
|
|
// Get the prediction for the current mb
|
|
|
|
cost = weighted_cost(&pred_prob, &new_pred_prob, 0,
|
|
|
|
pred_flag, cpi->seg0_progress);
|
|
|
|
if (cost > 1024) cost = 768; // i.e. account for 4 bits max.
|
|
|
|
|
|
|
|
// for incorrectly predicted cases
|
|
|
|
if (! pred_flag) {
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_prob curframe_mod_refprobs[3];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
if (cpi->seg0_progress) {
|
|
|
|
estimate_curframe_refprobs(cpi, curframe_mod_refprobs, pred_ref);
|
|
|
|
} else {
|
|
|
|
vpx_memset(curframe_mod_refprobs, 0, sizeof(curframe_mod_refprobs));
|
|
|
|
}
|
|
|
|
|
|
|
|
cost += weighted_cost(mod_refprobs, curframe_mod_refprobs, 0,
|
|
|
|
(i != INTRA_FRAME), cpi->seg0_progress);
|
|
|
|
if (i != INTRA_FRAME) {
|
|
|
|
cost += weighted_cost(mod_refprobs, curframe_mod_refprobs, 1,
|
|
|
|
(i != LAST_FRAME), cpi->seg0_progress);
|
|
|
|
if (i != LAST_FRAME) {
|
|
|
|
cost += weighted_cost(mod_refprobs, curframe_mod_refprobs, 2,
|
|
|
|
(i != GOLDEN_FRAME), cpi->seg0_progress);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ref_costs[i] = cost;
|
|
|
|
}
|
2012-01-28 13:20:14 +01:00
|
|
|
}
|
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
static void store_coding_context(MACROBLOCK *x, PICK_MODE_CONTEXT *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int mode_index,
|
|
|
|
PARTITION_INFO *partition,
|
|
|
|
int_mv *ref_mv,
|
2012-08-20 23:43:34 +02:00
|
|
|
int_mv *second_ref_mv,
|
2012-11-08 20:03:00 +01:00
|
|
|
int64_t comp_pred_diff[NB_PREDICTION_TYPES],
|
2012-10-09 18:18:21 +02:00
|
|
|
int64_t txfm_size_diff[NB_TXFM_MODES]) {
|
2012-11-08 20:03:00 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// Take a snapshot of the coding context so it can be
|
|
|
|
// restored if we decide to encode this way
|
2013-01-07 20:02:14 +01:00
|
|
|
ctx->skip = x->skip;
|
2012-08-20 23:43:34 +02:00
|
|
|
ctx->best_mode_index = mode_index;
|
|
|
|
vpx_memcpy(&ctx->mic, xd->mode_info_context,
|
2012-07-14 00:21:29 +02:00
|
|
|
sizeof(MODE_INFO));
|
2012-08-20 23:43:34 +02:00
|
|
|
if (partition)
|
|
|
|
vpx_memcpy(&ctx->partition_info, partition,
|
|
|
|
sizeof(PARTITION_INFO));
|
|
|
|
ctx->best_ref_mv.as_int = ref_mv->as_int;
|
|
|
|
ctx->second_best_ref_mv.as_int = second_ref_mv->as_int;
|
|
|
|
|
2013-01-18 18:44:23 +01:00
|
|
|
ctx->single_pred_diff = (int)comp_pred_diff[SINGLE_PREDICTION_ONLY];
|
|
|
|
ctx->comp_pred_diff = (int)comp_pred_diff[COMP_PREDICTION_ONLY];
|
|
|
|
ctx->hybrid_pred_diff = (int)comp_pred_diff[HYBRID_PREDICTION];
|
2012-10-09 18:18:21 +02:00
|
|
|
|
2012-11-08 20:03:00 +01:00
|
|
|
memcpy(ctx->txfm_rd_diff, txfm_size_diff, sizeof(ctx->txfm_rd_diff));
|
2012-04-13 19:26:49 +02:00
|
|
|
}
|
|
|
|
|
2012-10-31 01:53:32 +01:00
|
|
|
static void setup_buffer_inter(VP9_COMP *cpi, MACROBLOCK *x,
|
2012-11-26 12:29:08 +01:00
|
|
|
int idx, MV_REFERENCE_FRAME frame_type,
|
2013-04-13 19:01:46 +02:00
|
|
|
enum BlockSize block_size,
|
2013-04-26 20:57:17 +02:00
|
|
|
int mi_row, int mi_col,
|
2012-11-26 12:29:08 +01:00
|
|
|
int_mv frame_nearest_mv[MAX_REF_FRAMES],
|
|
|
|
int_mv frame_near_mv[MAX_REF_FRAMES],
|
2012-10-30 02:04:33 +01:00
|
|
|
int frame_mdcounts[4][4],
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
YV12_BUFFER_CONFIG yv12_mb[4],
|
|
|
|
struct scale_factors scale[MAX_REF_FRAMES]) {
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2013-02-22 20:22:03 +01:00
|
|
|
YV12_BUFFER_CONFIG *yv12 = &cm->yv12_fb[cpi->common.ref_frame_map[idx]];
|
2012-11-08 20:03:00 +01:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
|
|
|
MB_MODE_INFO *const mbmi = &xd->mode_info_context->mbmi;
|
2013-04-19 16:40:36 +02:00
|
|
|
int use_prev_in_find_mv_refs;
|
2012-09-07 13:46:41 +02:00
|
|
|
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
// set up scaling factors
|
|
|
|
scale[frame_type] = cpi->common.active_ref_scale[frame_type - 1];
|
|
|
|
scale[frame_type].x_offset_q4 =
|
2013-04-26 20:57:17 +02:00
|
|
|
(mi_col * MI_SIZE * scale[frame_type].x_num /
|
|
|
|
scale[frame_type].x_den) & 0xf;
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
scale[frame_type].y_offset_q4 =
|
2013-04-26 20:57:17 +02:00
|
|
|
(mi_row * MI_SIZE * scale[frame_type].y_num /
|
|
|
|
scale[frame_type].y_den) & 0xf;
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
|
|
|
|
// TODO(jkoleszar): Is the UV buffer ever used here? If so, need to make this
|
|
|
|
// use the UV scaling factors.
|
2013-04-26 20:57:17 +02:00
|
|
|
setup_pred_block(&yv12_mb[frame_type], yv12, mi_row, mi_col,
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
&scale[frame_type], &scale[frame_type]);
|
2012-08-24 16:44:01 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
// Gets an initial list of candidate vectors from neighbours and orders them
|
2013-03-21 00:41:30 +01:00
|
|
|
use_prev_in_find_mv_refs = cm->width == cm->last_width &&
|
|
|
|
cm->height == cm->last_height &&
|
2013-02-20 21:34:31 +01:00
|
|
|
!cpi->common.error_resilient_mode;
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 18:35:28 +01:00
|
|
|
vp9_find_mv_refs(&cpi->common, xd, xd->mode_info_context,
|
2013-02-20 21:34:31 +01:00
|
|
|
use_prev_in_find_mv_refs ? xd->prev_mode_info_context : NULL,
|
2012-10-29 14:44:18 +01:00
|
|
|
frame_type,
|
|
|
|
mbmi->ref_mvs[frame_type],
|
|
|
|
cpi->common.ref_frame_sign_bias);
|
2012-08-24 16:44:01 +02:00
|
|
|
|
2012-11-26 12:29:08 +01:00
|
|
|
// Candidate refinement carried out at encoder and decoder
|
2013-01-15 15:43:35 +01:00
|
|
|
vp9_find_best_ref_mvs(xd,
|
2012-09-07 13:46:41 +02:00
|
|
|
mbmi->ref_mvs[frame_type],
|
2012-08-06 19:51:20 +02:00
|
|
|
&frame_nearest_mv[frame_type],
|
|
|
|
&frame_near_mv[frame_type]);
|
2012-11-26 12:29:08 +01:00
|
|
|
|
|
|
|
// Further refinement that is encode side only to test the top few candidates
|
|
|
|
// in full and choose the best as the centre point for subsequent searches.
|
2013-03-14 01:09:05 +01:00
|
|
|
// The current implementation doesn't support scaling.
|
|
|
|
if (scale[frame_type].x_num == scale[frame_type].x_den &&
|
|
|
|
scale[frame_type].y_num == scale[frame_type].y_den)
|
|
|
|
mv_pred(cpi, x, yv12_mb[frame_type].y_buffer, yv12->y_stride,
|
|
|
|
frame_type, block_size);
|
2012-08-08 03:55:28 +02:00
|
|
|
}
|
|
|
|
|
2013-02-12 02:08:52 +01:00
|
|
|
static void model_rd_from_var_lapndz(int var, int n, int qstep,
|
|
|
|
int *rate, int *dist) {
|
|
|
|
// This function models the rate and distortion for a Laplacian
|
|
|
|
// source with given variance when quantized with a uniform quantizer
|
|
|
|
// with given stepsize. The closed form expressions are in:
|
|
|
|
// Hang and Chen, "Source Model for transform video coder and its
|
|
|
|
// application - Part I: Fundamental Theory", IEEE Trans. Circ.
|
|
|
|
// Sys. for Video Tech., April 1997.
|
|
|
|
// The function is implemented as piecewise approximation to the
|
|
|
|
// exact computation.
|
|
|
|
// TODO(debargha): Implement the functions by interpolating from a
|
|
|
|
// look-up table
|
|
|
|
vp9_clear_system_state();
|
|
|
|
{
|
|
|
|
double D, R;
|
|
|
|
double s2 = (double) var / n;
|
|
|
|
double s = sqrt(s2);
|
|
|
|
double x = qstep / s;
|
|
|
|
if (x > 1.0) {
|
|
|
|
double y = exp(-x / 2);
|
|
|
|
double y2 = y * y;
|
|
|
|
D = 2.069981728764738 * y2 - 2.764286806516079 * y + 1.003956960819275;
|
|
|
|
R = 0.924056758535089 * y2 + 2.738636469814024 * y - 0.005169662030017;
|
|
|
|
} else {
|
|
|
|
double x2 = x * x;
|
|
|
|
D = 0.075303187668830 * x2 + 0.004296954321112 * x - 0.000413209252807;
|
|
|
|
if (x > 0.125)
|
|
|
|
R = 1 / (-0.03459733614226 * x2 + 0.36561675733603 * x +
|
|
|
|
0.1626989668625);
|
|
|
|
else
|
|
|
|
R = -1.442252874826093 * log(x) + 1.944647760719664;
|
|
|
|
}
|
|
|
|
if (R < 0) {
|
|
|
|
*rate = 0;
|
|
|
|
*dist = var;
|
|
|
|
} else {
|
|
|
|
*rate = (n * R * 256 + 0.5);
|
|
|
|
*dist = (n * D * s2 + 0.5);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
vp9_clear_system_state();
|
|
|
|
}
|
|
|
|
|
2013-04-13 19:01:46 +02:00
|
|
|
static enum BlockSize y_to_uv_block_size(enum BlockSize bs) {
|
|
|
|
switch (bs) {
|
|
|
|
case BLOCK_64X64: return BLOCK_32X32;
|
2013-04-15 19:00:34 +02:00
|
|
|
case BLOCK_64X32: return BLOCK_32X16;
|
|
|
|
case BLOCK_32X64: return BLOCK_16X32;
|
2013-04-13 19:01:46 +02:00
|
|
|
case BLOCK_32X32: return BLOCK_16X16;
|
2013-04-15 19:00:34 +02:00
|
|
|
case BLOCK_32X16: return BLOCK_16X8;
|
|
|
|
case BLOCK_16X32: return BLOCK_8X16;
|
2013-04-13 19:01:46 +02:00
|
|
|
case BLOCK_16X16: return BLOCK_8X8;
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static enum BlockSize y_bsizet_to_block_size(BLOCK_SIZE_TYPE bs) {
|
|
|
|
switch (bs) {
|
|
|
|
case BLOCK_SIZE_SB64X64: return BLOCK_64X64;
|
2013-04-15 19:00:34 +02:00
|
|
|
case BLOCK_SIZE_SB64X32: return BLOCK_64X32;
|
|
|
|
case BLOCK_SIZE_SB32X64: return BLOCK_32X64;
|
2013-04-13 19:01:46 +02:00
|
|
|
case BLOCK_SIZE_SB32X32: return BLOCK_32X32;
|
2013-04-15 19:00:34 +02:00
|
|
|
case BLOCK_SIZE_SB32X16: return BLOCK_32X16;
|
|
|
|
case BLOCK_SIZE_SB16X32: return BLOCK_16X32;
|
2013-04-13 19:01:46 +02:00
|
|
|
case BLOCK_SIZE_MB16X16: return BLOCK_16X16;
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-31 01:53:32 +01:00
|
|
|
static int64_t handle_inter_mode(VP9_COMP *cpi, MACROBLOCK *x,
|
2013-04-11 00:55:59 +02:00
|
|
|
BLOCK_SIZE_TYPE bsize,
|
2012-10-30 01:58:18 +01:00
|
|
|
int mdcounts[4], int64_t txfm_cache[],
|
|
|
|
int *rate2, int *distortion, int *skippable,
|
|
|
|
int *compmode_cost,
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
int *compmode_interintra_cost,
|
|
|
|
#endif
|
2012-10-30 01:58:18 +01:00
|
|
|
int *rate_y, int *distortion_y,
|
|
|
|
int *rate_uv, int *distortion_uv,
|
|
|
|
int *mode_excluded, int *disable_skip,
|
2013-02-07 19:09:05 +01:00
|
|
|
int mode_index,
|
2013-02-12 02:08:52 +01:00
|
|
|
INTERPOLATIONFILTERTYPE *best_filter,
|
2012-12-05 17:23:38 +01:00
|
|
|
int_mv frame_mv[MB_MODE_COUNT]
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
[MAX_REF_FRAMES],
|
2013-02-20 21:34:31 +01:00
|
|
|
YV12_BUFFER_CONFIG *scaled_ref_frame,
|
2013-04-26 20:57:17 +02:00
|
|
|
int mi_row, int mi_col) {
|
|
|
|
const int bw = 1 << mi_width_log2(bsize), bh = 1 << mi_height_log2(bsize);
|
2013-04-13 19:01:46 +02:00
|
|
|
const enum BlockSize block_size = y_bsizet_to_block_size(bsize);
|
|
|
|
const enum BlockSize uv_block_size = y_to_uv_block_size(block_size);
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-10-30 01:58:18 +01:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
|
|
|
MB_MODE_INFO *mbmi = &xd->mode_info_context->mbmi;
|
2012-11-07 15:50:25 +01:00
|
|
|
const int is_comp_pred = (mbmi->second_ref_frame > 0);
|
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
const int is_comp_interintra_pred = (mbmi->second_ref_frame == INTRA_FRAME);
|
|
|
|
#endif
|
2012-10-30 01:58:18 +01:00
|
|
|
const int num_refs = is_comp_pred ? 2 : 1;
|
|
|
|
const int this_mode = mbmi->mode;
|
|
|
|
int i;
|
2012-11-07 15:50:25 +01:00
|
|
|
int refs[2] = { mbmi->ref_frame,
|
|
|
|
(mbmi->second_ref_frame < 0 ? 0 : mbmi->second_ref_frame) };
|
2012-10-30 01:58:18 +01:00
|
|
|
int_mv cur_mv[2];
|
2012-12-05 17:23:38 +01:00
|
|
|
int_mv ref_mv[2];
|
2012-10-30 01:58:18 +01:00
|
|
|
int64_t this_rd = 0;
|
2013-02-12 02:08:52 +01:00
|
|
|
unsigned char tmp_ybuf[64 * 64];
|
|
|
|
unsigned char tmp_ubuf[32 * 32];
|
|
|
|
unsigned char tmp_vbuf[32 * 32];
|
|
|
|
int pred_exists = 0;
|
|
|
|
int interpolating_intpel_seen = 0;
|
|
|
|
int intpel_mv;
|
|
|
|
int64_t rd, best_rd = INT64_MAX;
|
2012-10-30 01:58:18 +01:00
|
|
|
|
|
|
|
switch (this_mode) {
|
|
|
|
case NEWMV:
|
2012-12-05 17:23:38 +01:00
|
|
|
ref_mv[0] = mbmi->ref_mvs[refs[0]][0];
|
|
|
|
ref_mv[1] = mbmi->ref_mvs[refs[1]][0];
|
2012-12-10 13:38:48 +01:00
|
|
|
|
2012-10-30 01:58:18 +01:00
|
|
|
if (is_comp_pred) {
|
|
|
|
if (frame_mv[NEWMV][refs[0]].as_int == INVALID_MV ||
|
|
|
|
frame_mv[NEWMV][refs[1]].as_int == INVALID_MV)
|
2013-01-14 20:49:30 +01:00
|
|
|
return INT64_MAX;
|
2012-10-30 20:58:42 +01:00
|
|
|
*rate2 += vp9_mv_bit_cost(&frame_mv[NEWMV][refs[0]],
|
2012-12-05 17:23:38 +01:00
|
|
|
&ref_mv[0],
|
2012-11-09 00:44:39 +01:00
|
|
|
x->nmvjointcost, x->mvcost, 96,
|
2012-10-30 01:58:18 +01:00
|
|
|
x->e_mbd.allow_high_precision_mv);
|
2012-10-30 20:58:42 +01:00
|
|
|
*rate2 += vp9_mv_bit_cost(&frame_mv[NEWMV][refs[1]],
|
2012-12-05 17:23:38 +01:00
|
|
|
&ref_mv[1],
|
2012-11-09 00:44:39 +01:00
|
|
|
x->nmvjointcost, x->mvcost, 96,
|
2012-10-30 01:58:18 +01:00
|
|
|
x->e_mbd.allow_high_precision_mv);
|
|
|
|
} else {
|
2013-04-20 04:16:14 +02:00
|
|
|
struct buf_2d backup_yv12[MAX_MB_PLANE] = {{0}};
|
2012-10-30 01:58:18 +01:00
|
|
|
int bestsme = INT_MAX;
|
|
|
|
int further_steps, step_param = cpi->sf.first_step;
|
|
|
|
int sadpb = x->sadperbit16;
|
|
|
|
int_mv mvp_full, tmp_mv;
|
|
|
|
int sr = 0;
|
|
|
|
|
|
|
|
int tmp_col_min = x->mv_col_min;
|
|
|
|
int tmp_col_max = x->mv_col_max;
|
|
|
|
int tmp_row_min = x->mv_row_min;
|
|
|
|
int tmp_row_max = x->mv_row_max;
|
|
|
|
|
2013-02-20 21:34:31 +01:00
|
|
|
if (scaled_ref_frame) {
|
2013-04-20 04:16:14 +02:00
|
|
|
int i;
|
|
|
|
|
2013-02-20 21:34:31 +01:00
|
|
|
// Swap out the reference frame for a version that's been scaled to
|
|
|
|
// match the resolution of the current frame, allowing the existing
|
|
|
|
// motion search code to be used without additional modifications.
|
2013-04-20 04:16:14 +02:00
|
|
|
for (i = 0; i < MAX_MB_PLANE; i++)
|
|
|
|
backup_yv12[i] = xd->plane[i].pre[0];
|
|
|
|
|
2013-04-26 20:57:17 +02:00
|
|
|
setup_pre_planes(xd, scaled_ref_frame, NULL, mi_row, mi_col,
|
2013-04-20 04:16:14 +02:00
|
|
|
NULL, NULL);
|
2013-02-20 21:34:31 +01:00
|
|
|
}
|
|
|
|
|
2012-12-05 17:23:38 +01:00
|
|
|
vp9_clamp_mv_min_max(x, &ref_mv[0]);
|
2012-10-30 01:58:18 +01:00
|
|
|
|
2013-03-21 00:41:30 +01:00
|
|
|
sr = vp9_init_search_range(cpi->common.width, cpi->common.height);
|
2013-03-06 21:02:15 +01:00
|
|
|
|
2012-12-10 13:38:48 +01:00
|
|
|
// mvp_full.as_int = ref_mv[0].as_int;
|
2012-12-05 17:23:38 +01:00
|
|
|
mvp_full.as_int =
|
2012-12-10 13:38:48 +01:00
|
|
|
mbmi->ref_mvs[refs[0]][x->mv_best_ref_index[refs[0]]].as_int;
|
|
|
|
|
2012-12-05 17:23:38 +01:00
|
|
|
mvp_full.as_mv.col >>= 3;
|
|
|
|
mvp_full.as_mv.row >>= 3;
|
2012-10-30 01:58:18 +01:00
|
|
|
|
|
|
|
// adjust search range according to sr from mv prediction
|
|
|
|
step_param = MAX(step_param, sr);
|
|
|
|
|
|
|
|
// Further step/diamond searches as necessary
|
|
|
|
further_steps = (cpi->sf.max_step_search_steps - 1) - step_param;
|
|
|
|
|
2013-04-24 01:22:47 +02:00
|
|
|
bestsme = vp9_full_pixel_diamond(cpi, x, &mvp_full, step_param,
|
2012-10-30 01:58:18 +01:00
|
|
|
sadpb, further_steps, 1,
|
|
|
|
&cpi->fn_ptr[block_size],
|
2012-12-05 17:23:38 +01:00
|
|
|
&ref_mv[0], &tmp_mv);
|
2012-10-30 01:58:18 +01:00
|
|
|
|
|
|
|
x->mv_col_min = tmp_col_min;
|
|
|
|
x->mv_col_max = tmp_col_max;
|
|
|
|
x->mv_row_min = tmp_row_min;
|
|
|
|
x->mv_row_max = tmp_row_max;
|
|
|
|
|
|
|
|
if (bestsme < INT_MAX) {
|
|
|
|
int dis; /* TODO: use dis in distortion calculation later. */
|
|
|
|
unsigned int sse;
|
2013-04-24 01:22:47 +02:00
|
|
|
cpi->find_fractional_mv_step(x, &tmp_mv,
|
2012-12-05 17:23:38 +01:00
|
|
|
&ref_mv[0],
|
2012-10-30 01:58:18 +01:00
|
|
|
x->errorperbit,
|
|
|
|
&cpi->fn_ptr[block_size],
|
2012-11-09 00:44:39 +01:00
|
|
|
x->nmvjointcost, x->mvcost,
|
|
|
|
&dis, &sse);
|
2012-10-30 01:58:18 +01:00
|
|
|
}
|
2013-04-26 16:19:43 +02:00
|
|
|
frame_mv[NEWMV][refs[0]].as_int =
|
|
|
|
xd->mode_info_context->bmi[0].as_mv[0].as_int = tmp_mv.as_int;
|
2012-10-30 01:58:18 +01:00
|
|
|
|
|
|
|
// Add the new motion vector cost to our rolling cost variable
|
2012-12-05 17:23:38 +01:00
|
|
|
*rate2 += vp9_mv_bit_cost(&tmp_mv, &ref_mv[0],
|
2012-11-09 00:44:39 +01:00
|
|
|
x->nmvjointcost, x->mvcost,
|
|
|
|
96, xd->allow_high_precision_mv);
|
2013-02-20 21:34:31 +01:00
|
|
|
|
|
|
|
// restore the predictor, if required
|
|
|
|
if (scaled_ref_frame) {
|
2013-04-20 04:16:14 +02:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_MB_PLANE; i++)
|
|
|
|
xd->plane[i].pre[0] = backup_yv12[i];
|
2013-02-20 21:34:31 +01:00
|
|
|
}
|
2012-10-30 01:58:18 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case NEARMV:
|
2013-02-05 11:28:26 +01:00
|
|
|
case NEARESTMV:
|
2012-10-30 01:58:18 +01:00
|
|
|
case ZEROMV:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
for (i = 0; i < num_refs; ++i) {
|
|
|
|
cur_mv[i] = frame_mv[this_mode][refs[i]];
|
|
|
|
// Clip "next_nearest" so that it does not extend to far out of image
|
2013-04-06 19:00:53 +02:00
|
|
|
if (this_mode == NEWMV)
|
|
|
|
assert(!clamp_mv2(&cur_mv[i], xd));
|
|
|
|
else
|
|
|
|
clamp_mv2(&cur_mv[i], xd);
|
|
|
|
|
2012-10-30 01:58:18 +01:00
|
|
|
if (mv_check_bounds(x, &cur_mv[i]))
|
2013-01-14 20:49:30 +01:00
|
|
|
return INT64_MAX;
|
2012-10-30 01:58:18 +01:00
|
|
|
mbmi->mv[i].as_int = cur_mv[i].as_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* We don't include the cost of the second reference here, because there
|
|
|
|
* are only three options: Last/Golden, ARF/Last or Golden/ARF, or in other
|
|
|
|
* words if you present them in that order, the second one is always known
|
|
|
|
* if the first is known */
|
2012-10-31 22:40:53 +01:00
|
|
|
*compmode_cost = vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_COMP),
|
2012-10-30 01:58:18 +01:00
|
|
|
is_comp_pred);
|
2012-11-12 16:09:25 +01:00
|
|
|
*rate2 += vp9_cost_mv_ref(cpi, this_mode,
|
|
|
|
mbmi->mb_mode_context[mbmi->ref_frame]);
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if (!is_comp_pred) {
|
|
|
|
*compmode_interintra_cost = vp9_cost_bit(cm->fc.interintra_prob,
|
|
|
|
is_comp_interintra_pred);
|
|
|
|
if (is_comp_interintra_pred) {
|
|
|
|
*compmode_interintra_cost +=
|
|
|
|
x->mbmode_cost[xd->frame_type][mbmi->interintra_mode];
|
|
|
|
#if SEPARATE_INTERINTRA_UV
|
|
|
|
*compmode_interintra_cost +=
|
|
|
|
x->intra_uv_mode_cost[xd->frame_type][mbmi->interintra_uv_mode];
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2012-10-30 01:58:18 +01:00
|
|
|
|
2013-02-12 02:08:52 +01:00
|
|
|
pred_exists = 0;
|
|
|
|
interpolating_intpel_seen = 0;
|
|
|
|
// Are all MVs integer pel for Y and UV
|
|
|
|
intpel_mv = (mbmi->mv[0].as_mv.row & 15) == 0 &&
|
|
|
|
(mbmi->mv[0].as_mv.col & 15) == 0;
|
|
|
|
if (is_comp_pred)
|
|
|
|
intpel_mv &= (mbmi->mv[1].as_mv.row & 15) == 0 &&
|
|
|
|
(mbmi->mv[1].as_mv.col & 15) == 0;
|
|
|
|
// Search for best switchable filter by checking the variance of
|
|
|
|
// pred error irrespective of whether the filter will be used
|
2013-04-25 23:17:08 +02:00
|
|
|
if (1) {
|
2013-02-12 02:08:52 +01:00
|
|
|
int switchable_filter_index, newbest;
|
|
|
|
int tmp_rate_y_i = 0, tmp_rate_u_i = 0, tmp_rate_v_i = 0;
|
|
|
|
int tmp_dist_y_i = 0, tmp_dist_u_i = 0, tmp_dist_v_i = 0;
|
|
|
|
for (switchable_filter_index = 0;
|
|
|
|
switchable_filter_index < VP9_SWITCHABLE_FILTERS;
|
|
|
|
++switchable_filter_index) {
|
|
|
|
int rs = 0;
|
|
|
|
mbmi->interp_filter = vp9_switchable_interp[switchable_filter_index];
|
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
|
|
|
|
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE) {
|
|
|
|
const int c = vp9_get_pred_context(cm, xd, PRED_SWITCHABLE_INTERP);
|
|
|
|
const int m = vp9_switchable_interp_map[mbmi->interp_filter];
|
|
|
|
rs = SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs[c][m];
|
|
|
|
}
|
|
|
|
if (interpolating_intpel_seen && intpel_mv &&
|
|
|
|
vp9_is_interpolating_filter[mbmi->interp_filter]) {
|
|
|
|
rd = RDCOST(x->rdmult, x->rddiv,
|
|
|
|
rs + tmp_rate_y_i + tmp_rate_u_i + tmp_rate_v_i,
|
|
|
|
tmp_dist_y_i + tmp_dist_u_i + tmp_dist_v_i);
|
|
|
|
} else {
|
|
|
|
unsigned int sse, var;
|
|
|
|
int tmp_rate_y, tmp_rate_u, tmp_rate_v;
|
|
|
|
int tmp_dist_y, tmp_dist_u, tmp_dist_v;
|
2013-04-26 20:57:17 +02:00
|
|
|
vp9_build_inter_predictors_sb(xd, mi_row, mi_col, bsize);
|
2013-04-24 01:22:47 +02:00
|
|
|
var = cpi->fn_ptr[block_size].vf(x->plane[0].src.buf,
|
|
|
|
x->plane[0].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[0].dst.buf,
|
|
|
|
xd->plane[0].dst.stride,
|
2013-04-13 19:01:46 +02:00
|
|
|
&sse);
|
2013-02-12 02:08:52 +01:00
|
|
|
// Note our transform coeffs are 8 times an orthogonal transform.
|
|
|
|
// Hence quantizer step is also 8 times. To get effective quantizer
|
|
|
|
// we need to divide by 8 before sending to modeling function.
|
2013-04-26 20:57:17 +02:00
|
|
|
model_rd_from_var_lapndz(var, MI_SIZE * bw * MI_SIZE * bh,
|
2013-04-24 23:48:17 +02:00
|
|
|
xd->plane[0].dequant[1] >> 3,
|
2013-02-12 02:08:52 +01:00
|
|
|
&tmp_rate_y, &tmp_dist_y);
|
2013-04-24 01:22:47 +02:00
|
|
|
var = cpi->fn_ptr[uv_block_size].vf(x->plane[1].src.buf,
|
|
|
|
x->plane[1].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[1].dst.buf,
|
|
|
|
xd->plane[1].dst.stride,
|
2013-04-13 19:01:46 +02:00
|
|
|
&sse);
|
2013-04-26 20:57:17 +02:00
|
|
|
model_rd_from_var_lapndz(var, MI_UV_SIZE * bw * MI_UV_SIZE * bh,
|
2013-04-24 23:48:17 +02:00
|
|
|
xd->plane[1].dequant[1] >> 3,
|
2013-02-12 02:08:52 +01:00
|
|
|
&tmp_rate_u, &tmp_dist_u);
|
2013-04-24 01:22:47 +02:00
|
|
|
var = cpi->fn_ptr[uv_block_size].vf(x->plane[2].src.buf,
|
|
|
|
x->plane[1].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[2].dst.buf,
|
|
|
|
xd->plane[1].dst.stride,
|
2013-04-13 19:01:46 +02:00
|
|
|
&sse);
|
2013-04-26 20:57:17 +02:00
|
|
|
model_rd_from_var_lapndz(var, MI_UV_SIZE * bw * MI_UV_SIZE * bh,
|
2013-04-24 23:48:17 +02:00
|
|
|
xd->plane[2].dequant[1] >> 3,
|
2013-02-12 02:08:52 +01:00
|
|
|
&tmp_rate_v, &tmp_dist_v);
|
|
|
|
rd = RDCOST(x->rdmult, x->rddiv,
|
|
|
|
rs + tmp_rate_y + tmp_rate_u + tmp_rate_v,
|
|
|
|
tmp_dist_y + tmp_dist_u + tmp_dist_v);
|
|
|
|
if (!interpolating_intpel_seen && intpel_mv &&
|
|
|
|
vp9_is_interpolating_filter[mbmi->interp_filter]) {
|
|
|
|
tmp_rate_y_i = tmp_rate_y;
|
|
|
|
tmp_rate_u_i = tmp_rate_u;
|
|
|
|
tmp_rate_v_i = tmp_rate_v;
|
|
|
|
tmp_dist_y_i = tmp_dist_y;
|
|
|
|
tmp_dist_u_i = tmp_dist_u;
|
|
|
|
tmp_dist_v_i = tmp_dist_v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
newbest = (switchable_filter_index == 0 || rd < best_rd);
|
|
|
|
if (newbest) {
|
|
|
|
best_rd = rd;
|
|
|
|
*best_filter = mbmi->interp_filter;
|
|
|
|
}
|
|
|
|
if ((cm->mcomp_filter_type == SWITCHABLE && newbest) ||
|
|
|
|
(cm->mcomp_filter_type != SWITCHABLE &&
|
|
|
|
cm->mcomp_filter_type == mbmi->interp_filter)) {
|
2013-04-15 18:31:27 +02:00
|
|
|
int i;
|
2013-04-26 20:57:17 +02:00
|
|
|
for (i = 0; i < MI_SIZE * bh; ++i)
|
|
|
|
vpx_memcpy(tmp_ybuf + i * MI_SIZE * bw,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[0].dst.buf + i * xd->plane[0].dst.stride,
|
2013-04-26 20:57:17 +02:00
|
|
|
sizeof(unsigned char) * MI_SIZE * bw);
|
|
|
|
for (i = 0; i < MI_UV_SIZE * bh; ++i)
|
|
|
|
vpx_memcpy(tmp_ubuf + i * MI_UV_SIZE * bw,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[1].dst.buf + i * xd->plane[1].dst.stride,
|
2013-04-26 20:57:17 +02:00
|
|
|
sizeof(unsigned char) * MI_UV_SIZE * bw);
|
|
|
|
for (i = 0; i < MI_UV_SIZE * bh; ++i)
|
|
|
|
vpx_memcpy(tmp_vbuf + i * MI_UV_SIZE * bw,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[2].dst.buf + i * xd->plane[1].dst.stride,
|
2013-04-26 20:57:17 +02:00
|
|
|
sizeof(unsigned char) * MI_UV_SIZE * bw);
|
2013-02-12 02:08:52 +01:00
|
|
|
pred_exists = 1;
|
|
|
|
}
|
|
|
|
interpolating_intpel_seen |=
|
|
|
|
intpel_mv && vp9_is_interpolating_filter[mbmi->interp_filter];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the appripriate filter
|
|
|
|
if (cm->mcomp_filter_type != SWITCHABLE)
|
|
|
|
mbmi->interp_filter = cm->mcomp_filter_type;
|
|
|
|
else
|
|
|
|
mbmi->interp_filter = *best_filter;
|
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
|
|
|
|
|
|
|
if (pred_exists) {
|
2013-04-13 19:01:46 +02:00
|
|
|
// FIXME(rbultje): mb code still predicts into xd->predictor
|
2013-04-26 20:57:17 +02:00
|
|
|
for (i = 0; i < bh * MI_SIZE; ++i)
|
2013-04-20 00:52:17 +02:00
|
|
|
vpx_memcpy(xd->plane[0].dst.buf + i * xd->plane[0].dst.stride,
|
2013-04-26 20:57:17 +02:00
|
|
|
tmp_ybuf + i * bw * MI_SIZE,
|
|
|
|
sizeof(unsigned char) * bw * MI_SIZE);
|
|
|
|
for (i = 0; i < bh * MI_UV_SIZE; ++i)
|
2013-04-20 00:52:17 +02:00
|
|
|
vpx_memcpy(xd->plane[1].dst.buf + i * xd->plane[1].dst.stride,
|
2013-04-26 20:57:17 +02:00
|
|
|
tmp_ubuf + i * bw * MI_UV_SIZE,
|
|
|
|
sizeof(unsigned char) * bw * MI_UV_SIZE);
|
|
|
|
for (i = 0; i < bh * MI_UV_SIZE; ++i)
|
2013-04-20 00:52:17 +02:00
|
|
|
vpx_memcpy(xd->plane[2].dst.buf + i * xd->plane[1].dst.stride,
|
2013-04-26 20:57:17 +02:00
|
|
|
tmp_vbuf + i * bw * MI_UV_SIZE,
|
|
|
|
sizeof(unsigned char) * bw * MI_UV_SIZE);
|
2013-02-12 02:08:52 +01:00
|
|
|
} else {
|
|
|
|
// Handles the special case when a filter that is not in the
|
|
|
|
// switchable list (ex. bilinear, 6-tap) is indicated at the frame level
|
2013-04-26 20:57:17 +02:00
|
|
|
vp9_build_inter_predictors_sb(xd, mi_row, mi_col, bsize);
|
2013-02-12 02:08:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE) {
|
|
|
|
const int c = vp9_get_pred_context(cm, xd, PRED_SWITCHABLE_INTERP);
|
|
|
|
const int m = vp9_switchable_interp_map[mbmi->interp_filter];
|
|
|
|
*rate2 += SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs[c][m];
|
2012-10-30 01:58:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (cpi->active_map_enabled && x->active_ptr[0] == 0)
|
|
|
|
x->skip = 1;
|
|
|
|
else if (x->encode_breakout) {
|
2013-02-12 02:08:52 +01:00
|
|
|
unsigned int var, sse;
|
2013-04-24 23:48:17 +02:00
|
|
|
int threshold = (xd->plane[0].dequant[1]
|
|
|
|
* xd->plane[0].dequant[1] >> 4);
|
2012-10-30 01:58:18 +01:00
|
|
|
|
|
|
|
if (threshold < x->encode_breakout)
|
|
|
|
threshold = x->encode_breakout;
|
|
|
|
|
2013-04-13 19:01:46 +02:00
|
|
|
if (bsize != BLOCK_SIZE_MB16X16) {
|
2013-04-24 01:22:47 +02:00
|
|
|
var = cpi->fn_ptr[block_size].vf(x->plane[0].src.buf,
|
|
|
|
x->plane[0].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[0].dst.buf,
|
|
|
|
xd->plane[0].dst.stride,
|
2013-04-13 19:01:46 +02:00
|
|
|
&sse);
|
2013-01-08 19:29:22 +01:00
|
|
|
} else {
|
2013-04-24 01:22:47 +02:00
|
|
|
var = vp9_variance16x16(x->plane[0].src.buf, x->plane[0].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[0].dst.buf, xd->plane[0].dst.stride,
|
|
|
|
&sse);
|
2012-10-30 01:58:18 +01:00
|
|
|
}
|
|
|
|
|
2012-11-05 23:22:59 +01:00
|
|
|
if ((int)sse < threshold) {
|
2013-04-24 23:48:17 +02:00
|
|
|
unsigned int q2dc = xd->plane[0].dequant[0];
|
2012-10-30 01:58:18 +01:00
|
|
|
/* If there is no codeable 2nd order dc
|
2013-02-12 02:08:52 +01:00
|
|
|
or a very small uniform pixel change change */
|
2012-10-30 01:58:18 +01:00
|
|
|
if ((sse - var < q2dc * q2dc >> 4) ||
|
|
|
|
(sse / 2 > var && sse - var < 64)) {
|
|
|
|
// Check u and v to make sure skip is ok
|
|
|
|
int sse2;
|
|
|
|
|
2013-04-13 19:01:46 +02:00
|
|
|
if (bsize != BLOCK_SIZE_MB16X16) {
|
2013-01-06 03:20:25 +01:00
|
|
|
unsigned int sse2u, sse2v;
|
2013-04-13 19:01:46 +02:00
|
|
|
// FIXME(rbultje): mb predictors predict into xd->predictor
|
2013-04-24 01:22:47 +02:00
|
|
|
var = cpi->fn_ptr[uv_block_size].vf(x->plane[1].src.buf,
|
|
|
|
x->plane[1].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[1].dst.buf,
|
|
|
|
xd->plane[1].dst.stride, &sse2u);
|
2013-04-24 01:22:47 +02:00
|
|
|
var = cpi->fn_ptr[uv_block_size].vf(x->plane[2].src.buf,
|
|
|
|
x->plane[1].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[2].dst.buf,
|
|
|
|
xd->plane[1].dst.stride, &sse2v);
|
2013-01-06 03:20:25 +01:00
|
|
|
sse2 = sse2u + sse2v;
|
2013-04-13 19:01:46 +02:00
|
|
|
} else {
|
2012-10-30 01:58:18 +01:00
|
|
|
unsigned int sse2u, sse2v;
|
2013-04-24 01:22:47 +02:00
|
|
|
var = vp9_variance8x8(x->plane[1].src.buf, x->plane[1].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[1].dst.buf, xd->plane[1].dst.stride,
|
|
|
|
&sse2u);
|
2013-04-24 01:22:47 +02:00
|
|
|
var = vp9_variance8x8(x->plane[2].src.buf, x->plane[1].src.stride,
|
2013-04-20 00:52:17 +02:00
|
|
|
xd->plane[2].dst.buf, xd->plane[1].dst.stride,
|
|
|
|
&sse2v);
|
2012-10-30 01:58:18 +01:00
|
|
|
sse2 = sse2u + sse2v;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sse2 * 2 < threshold) {
|
|
|
|
x->skip = 1;
|
|
|
|
*distortion = sse + sse2;
|
|
|
|
*rate2 = 500;
|
|
|
|
|
|
|
|
/* for best_yrd calculation */
|
|
|
|
*rate_uv = 0;
|
|
|
|
*distortion_uv = sse2;
|
|
|
|
|
|
|
|
*disable_skip = 1;
|
|
|
|
this_rd = RDCOST(x->rdmult, x->rddiv, *rate2, *distortion);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!x->skip) {
|
2013-04-11 00:55:59 +02:00
|
|
|
int skippable_y, skippable_uv;
|
|
|
|
|
|
|
|
// Y cost and distortion
|
|
|
|
super_block_yrd(cpi, x, rate_y, distortion_y, &skippable_y,
|
|
|
|
bsize, txfm_cache);
|
|
|
|
*rate2 += *rate_y;
|
|
|
|
*distortion += *distortion_y;
|
|
|
|
|
|
|
|
super_block_uvrd(cm, x, rate_uv, distortion_uv,
|
|
|
|
&skippable_uv, bsize);
|
|
|
|
|
|
|
|
*rate2 += *rate_uv;
|
|
|
|
*distortion += *distortion_uv;
|
|
|
|
*skippable = skippable_y && skippable_uv;
|
2012-10-30 01:58:18 +01:00
|
|
|
}
|
2013-02-12 02:08:52 +01:00
|
|
|
|
|
|
|
if (!(*mode_excluded)) {
|
|
|
|
if (is_comp_pred) {
|
|
|
|
*mode_excluded = (cpi->common.comp_pred_mode == SINGLE_PREDICTION_ONLY);
|
|
|
|
} else {
|
|
|
|
*mode_excluded = (cpi->common.comp_pred_mode == COMP_PREDICTION_ONLY);
|
|
|
|
}
|
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if (is_comp_interintra_pred && !cm->use_interintra) *mode_excluded = 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2012-10-30 01:58:18 +01:00
|
|
|
return this_rd; // if 0, this will be re-calculated by caller
|
|
|
|
}
|
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#if !CONFIG_SB8X8
|
2012-11-02 19:22:57 +01:00
|
|
|
static void rd_pick_inter_mode(VP9_COMP *cpi, MACROBLOCK *x,
|
2013-04-26 20:57:17 +02:00
|
|
|
int mi_row, int mi_col,
|
2012-11-02 19:22:57 +01:00
|
|
|
int *returnrate, int *returndistortion,
|
|
|
|
int64_t *returnintra) {
|
2013-02-07 00:54:52 +01:00
|
|
|
static const int flag_list[4] = { 0, VP9_LAST_FLAG, VP9_GOLD_FLAG,
|
2013-02-12 02:08:52 +01:00
|
|
|
VP9_ALT_FLAG };
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 00:21:29 +02:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
|
|
|
union b_mode_info best_bmodes[16];
|
|
|
|
MB_MODE_INFO best_mbmode;
|
|
|
|
PARTITION_INFO best_partition;
|
|
|
|
int_mv best_ref_mv, second_best_ref_mv;
|
|
|
|
MB_PREDICTION_MODE this_mode;
|
2012-12-20 23:56:19 +01:00
|
|
|
MB_PREDICTION_MODE best_mode = DC_PRED;
|
2012-10-29 11:36:11 +01:00
|
|
|
MB_MODE_INFO * mbmi = &xd->mode_info_context->mbmi;
|
2012-08-08 01:44:26 +02:00
|
|
|
int i, best_mode_index = 0;
|
2013-01-14 23:37:53 +01:00
|
|
|
int mode8x8[4];
|
2012-08-10 15:12:43 +02:00
|
|
|
unsigned char segment_id = mbmi->segment_id;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
int mode_index;
|
|
|
|
int mdcounts[4];
|
2012-08-07 01:21:23 +02:00
|
|
|
int rate, distortion;
|
|
|
|
int rate2, distortion2;
|
2012-10-09 18:18:21 +02:00
|
|
|
int64_t best_txfm_rd[NB_TXFM_MODES];
|
|
|
|
int64_t best_txfm_diff[NB_TXFM_MODES];
|
2012-08-10 03:25:29 +02:00
|
|
|
int64_t best_pred_diff[NB_PREDICTION_TYPES];
|
|
|
|
int64_t best_pred_rd[NB_PREDICTION_TYPES];
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_rd = INT64_MAX, best_intra_rd = INT64_MAX;
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
int is_best_interintra = 0;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_intra16_rd = INT64_MAX;
|
2013-02-20 19:16:24 +01:00
|
|
|
int best_intra16_mode = DC_PRED;
|
|
|
|
#if SEPARATE_INTERINTRA_UV
|
|
|
|
int best_intra16_uv_mode = DC_PRED;
|
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
#endif
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_overall_rd = INT64_MAX;
|
2012-12-20 23:56:19 +01:00
|
|
|
INTERPOLATIONFILTERTYPE best_filter = SWITCHABLE;
|
2013-02-12 02:08:52 +01:00
|
|
|
INTERPOLATIONFILTERTYPE tmp_best_filter = SWITCHABLE;
|
2013-04-11 00:55:59 +02:00
|
|
|
int uv_intra_rate[2], uv_intra_distortion[2], uv_intra_rate_tokenonly[2];
|
|
|
|
int uv_intra_skippable[2];
|
|
|
|
MB_PREDICTION_MODE uv_intra_mode[2];
|
2012-07-14 00:21:29 +02:00
|
|
|
int rate_y, UNINITIALIZED_IS_SAFE(rate_uv);
|
2012-08-22 00:08:54 +02:00
|
|
|
int distortion_uv = INT_MAX;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_yrd = INT64_MAX;
|
2011-01-11 21:00:00 +01:00
|
|
|
|
2012-08-10 01:07:41 +02:00
|
|
|
int_mv frame_mv[MB_MODE_COUNT][MAX_REF_FRAMES];
|
2012-07-14 00:21:29 +02:00
|
|
|
int frame_mdcounts[4][4];
|
2013-02-07 19:09:05 +01:00
|
|
|
YV12_BUFFER_CONFIG yv12_mb[4];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
unsigned int ref_costs[MAX_REF_FRAMES];
|
2012-10-22 20:25:48 +02:00
|
|
|
int_mv seg_mvs[NB_PARTITIONINGS][16 /* n_blocks */][MAX_REF_FRAMES - 1];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-12-14 18:49:46 +01:00
|
|
|
int intra_cost_penalty = 20 * vp9_dc_quant(cpi->common.base_qindex,
|
2013-04-17 00:05:52 +02:00
|
|
|
cpi->common.y_dc_delta_q);
|
2013-04-23 00:42:41 +02:00
|
|
|
int64_t mode_distortions[MB_MODE_COUNT] = {-1};
|
|
|
|
int64_t frame_distortions[MAX_REF_FRAMES] = {-1};
|
|
|
|
int ref_frame;
|
2012-12-14 18:49:46 +01:00
|
|
|
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
struct scale_factors scale_factor[4];
|
|
|
|
|
2012-08-14 01:20:20 +02:00
|
|
|
vpx_memset(mode8x8, 0, sizeof(mode8x8));
|
2012-08-10 01:07:41 +02:00
|
|
|
vpx_memset(&frame_mv, 0, sizeof(frame_mv));
|
2012-07-14 00:21:29 +02:00
|
|
|
vpx_memset(&best_mbmode, 0, sizeof(best_mbmode));
|
|
|
|
vpx_memset(&best_bmodes, 0, sizeof(best_bmodes));
|
2013-01-06 03:20:25 +01:00
|
|
|
vpx_memset(&x->mb_context[xd->sb_index][xd->mb_index], 0,
|
|
|
|
sizeof(PICK_MODE_CONTEXT));
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-23 00:42:41 +02:00
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].frames_with_high_error = 0;
|
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].modes_with_high_error = 0;
|
|
|
|
|
2012-08-10 01:07:41 +02:00
|
|
|
for (i = 0; i < MAX_REF_FRAMES; i++)
|
|
|
|
frame_mv[NEWMV][i].as_int = INVALID_MV;
|
2012-08-10 03:25:29 +02:00
|
|
|
for (i = 0; i < NB_PREDICTION_TYPES; ++i)
|
2013-01-14 20:49:30 +01:00
|
|
|
best_pred_rd[i] = INT64_MAX;
|
2012-10-09 18:18:21 +02:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++)
|
2013-01-14 20:49:30 +01:00
|
|
|
best_txfm_rd[i] = INT64_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-10-22 20:25:48 +02:00
|
|
|
for (i = 0; i < NB_PARTITIONINGS; i++) {
|
2012-08-07 01:21:23 +02:00
|
|
|
int j, k;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-07 01:21:23 +02:00
|
|
|
for (j = 0; j < 16; j++)
|
|
|
|
for (k = 0; k < MAX_REF_FRAMES - 1; k++)
|
2012-07-14 00:21:29 +02:00
|
|
|
seg_mvs[i][j][k].as_int = INVALID_MV;
|
|
|
|
}
|
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
if (cpi->ref_frame_flags & VP9_LAST_FLAG) {
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
setup_buffer_inter(cpi, x, cpi->lst_fb_idx,
|
2013-04-26 20:57:17 +02:00
|
|
|
LAST_FRAME, BLOCK_16X16, mi_row, mi_col,
|
2012-12-05 17:23:38 +01:00
|
|
|
frame_mv[NEARESTMV], frame_mv[NEARMV],
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
frame_mdcounts, yv12_mb, scale_factor);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
if (cpi->ref_frame_flags & VP9_GOLD_FLAG) {
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
setup_buffer_inter(cpi, x, cpi->gld_fb_idx,
|
2013-04-26 20:57:17 +02:00
|
|
|
GOLDEN_FRAME, BLOCK_16X16, mi_row, mi_col,
|
2012-12-05 17:23:38 +01:00
|
|
|
frame_mv[NEARESTMV], frame_mv[NEARMV],
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
frame_mdcounts, yv12_mb, scale_factor);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2012-10-31 22:40:53 +01:00
|
|
|
if (cpi->ref_frame_flags & VP9_ALT_FLAG) {
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
setup_buffer_inter(cpi, x, cpi->alt_fb_idx,
|
2013-04-26 20:57:17 +02:00
|
|
|
ALTREF_FRAME, BLOCK_16X16, mi_row, mi_col,
|
2012-12-05 17:23:38 +01:00
|
|
|
frame_mv[NEARESTMV], frame_mv[NEARMV],
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
frame_mdcounts, yv12_mb, scale_factor);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2013-01-14 20:49:30 +01:00
|
|
|
*returnintra = INT64_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->ref_frame = INTRA_FRAME;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
/* Initialize zbin mode boost for uv costing */
|
|
|
|
cpi->zbin_mode_boost = 0;
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_update_zbin_extra(cpi, x);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.mode = DC_PRED;
|
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
for (i = 0; i <= TX_8X8; i++) {
|
|
|
|
mbmi->txfm_size = i;
|
|
|
|
rd_pick_intra_sbuv_mode(cpi, x, &uv_intra_rate[i],
|
|
|
|
&uv_intra_rate_tokenonly[i],
|
|
|
|
&uv_intra_distortion[i],
|
|
|
|
&uv_intra_skippable[i],
|
|
|
|
BLOCK_SIZE_MB16X16);
|
|
|
|
uv_intra_mode[i] = mbmi->uv_mode;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get estimates of reference frame costs for each reference frame
|
|
|
|
// that depend on the current prediction etc.
|
2012-10-31 22:40:53 +01:00
|
|
|
estimate_ref_frame_costs(cpi, segment_id, ref_costs);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-02-12 02:08:52 +01:00
|
|
|
for (mode_index = 0; mode_index < MAX_MODES; ++mode_index) {
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t this_rd = INT64_MAX;
|
2012-10-09 18:18:21 +02:00
|
|
|
int disable_skip = 0, skippable = 0;
|
2012-07-14 00:21:29 +02:00
|
|
|
int other_cost = 0;
|
|
|
|
int compmode_cost = 0;
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
int compmode_interintra_cost = 0;
|
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
int mode_excluded = 0;
|
2012-10-11 02:18:22 +02:00
|
|
|
int64_t txfm_cache[NB_TXFM_MODES] = { 0 };
|
2013-02-20 21:34:31 +01:00
|
|
|
YV12_BUFFER_CONFIG *scaled_ref_frame;
|
2012-02-22 03:10:18 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// These variables hold are rolling total cost and distortion for this mode
|
|
|
|
rate2 = 0;
|
|
|
|
distortion2 = 0;
|
|
|
|
rate_y = 0;
|
|
|
|
rate_uv = 0;
|
|
|
|
|
2013-01-14 21:43:12 +01:00
|
|
|
x->skip = 0;
|
|
|
|
|
2012-10-31 01:12:12 +01:00
|
|
|
this_mode = vp9_mode_order[mode_index].mode;
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->mode = this_mode;
|
|
|
|
mbmi->uv_mode = DC_PRED;
|
2012-10-31 01:12:12 +01:00
|
|
|
mbmi->ref_frame = vp9_mode_order[mode_index].ref_frame;
|
|
|
|
mbmi->second_ref_frame = vp9_mode_order[mode_index].second_ref_frame;
|
2012-12-20 23:56:19 +01:00
|
|
|
|
2013-02-12 02:08:52 +01:00
|
|
|
mbmi->interp_filter = cm->mcomp_filter_type;
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
|
|
|
|
set_scale_factors(xd, mbmi->ref_frame, mbmi->second_ref_frame,
|
|
|
|
scale_factor);
|
|
|
|
|
2013-02-12 02:08:52 +01:00
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
2012-07-18 22:43:01 +02:00
|
|
|
|
|
|
|
// Test best rd so far against threshold for trying this mode.
|
2012-08-07 01:21:23 +02:00
|
|
|
if (best_rd <= cpi->rd_threshes[mode_index])
|
2012-07-18 22:43:01 +02:00
|
|
|
continue;
|
2012-04-07 01:38:34 +02:00
|
|
|
|
2013-02-07 00:54:52 +01:00
|
|
|
// Ensure that the references used by this mode are available.
|
|
|
|
if (mbmi->ref_frame &&
|
|
|
|
!(cpi->ref_frame_flags & flag_list[mbmi->ref_frame]))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (mbmi->second_ref_frame > 0 &&
|
|
|
|
!(cpi->ref_frame_flags & flag_list[mbmi->second_ref_frame]))
|
|
|
|
continue;
|
|
|
|
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
// only scale on zeromv.
|
|
|
|
if (mbmi->ref_frame > 0 &&
|
|
|
|
(yv12_mb[mbmi->ref_frame].y_width != cm->mb_cols * 16 ||
|
|
|
|
yv12_mb[mbmi->ref_frame].y_height != cm->mb_rows * 16) &&
|
|
|
|
this_mode != ZEROMV)
|
|
|
|
continue;
|
2013-04-19 17:04:57 +02:00
|
|
|
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
if (mbmi->second_ref_frame > 0 &&
|
|
|
|
(yv12_mb[mbmi->second_ref_frame].y_width != cm->mb_cols * 16 ||
|
|
|
|
yv12_mb[mbmi->second_ref_frame].y_height != cm->mb_rows * 16) &&
|
|
|
|
this_mode != ZEROMV)
|
|
|
|
continue;
|
|
|
|
|
2012-06-25 21:26:09 +02:00
|
|
|
// current coding mode under rate-distortion optimization test loop
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
mbmi->interintra_mode = (MB_PREDICTION_MODE)(DC_PRED - 1);
|
|
|
|
mbmi->interintra_uv_mode = (MB_PREDICTION_MODE)(DC_PRED - 1);
|
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// If the segment reference frame feature is enabled....
|
|
|
|
// then do nothing if the current ref frame is not allowed..
|
2012-10-30 06:15:27 +01:00
|
|
|
if (vp9_segfeature_active(xd, segment_id, SEG_LVL_REF_FRAME) &&
|
|
|
|
!vp9_check_segref(xd, segment_id, mbmi->ref_frame)) {
|
2012-07-14 00:21:29 +02:00
|
|
|
continue;
|
2013-01-28 16:22:53 +01:00
|
|
|
// If the segment skip feature is enabled....
|
2012-07-14 00:21:29 +02:00
|
|
|
// then do nothing if the current mode is not allowed..
|
2013-01-28 16:22:53 +01:00
|
|
|
} else if (vp9_segfeature_active(xd, segment_id, SEG_LVL_SKIP) &&
|
|
|
|
(this_mode != ZEROMV)) {
|
2012-07-14 00:21:29 +02:00
|
|
|
continue;
|
2013-01-28 16:22:53 +01:00
|
|
|
// Disable this drop out case if the ref frame segment
|
|
|
|
// level feature is enabled for this segment. This is to
|
2012-07-14 00:21:29 +02:00
|
|
|
// prevent the possibility that the we end up unable to pick any mode.
|
2013-01-28 16:22:53 +01:00
|
|
|
} else if (!vp9_segfeature_active(xd, segment_id, SEG_LVL_REF_FRAME)) {
|
|
|
|
// Only consider ZEROMV/ALTREF_FRAME for alt ref frame overlay,
|
2012-07-14 00:21:29 +02:00
|
|
|
// unless ARNR filtering is enabled in which case we want
|
|
|
|
// an unfiltered alternative
|
|
|
|
if (cpi->is_src_frame_alt_ref && (cpi->oxcf.arnr_max_frames == 0)) {
|
|
|
|
if (this_mode != ZEROMV ||
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->ref_frame != ALTREF_FRAME) {
|
2012-07-14 00:21:29 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* everything but intra */
|
2013-02-20 21:34:31 +01:00
|
|
|
scaled_ref_frame = NULL;
|
2012-08-10 15:12:43 +02:00
|
|
|
if (mbmi->ref_frame) {
|
|
|
|
int ref = mbmi->ref_frame;
|
2013-02-20 21:34:31 +01:00
|
|
|
int fb;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-12-05 17:23:38 +01:00
|
|
|
best_ref_mv = mbmi->ref_mvs[ref][0];
|
2012-07-14 00:21:29 +02:00
|
|
|
vpx_memcpy(mdcounts, frame_mdcounts[ref], sizeof(mdcounts));
|
2013-02-20 21:34:31 +01:00
|
|
|
|
|
|
|
if (mbmi->ref_frame == LAST_FRAME) {
|
|
|
|
fb = cpi->lst_fb_idx;
|
|
|
|
} else if (mbmi->ref_frame == GOLDEN_FRAME) {
|
|
|
|
fb = cpi->gld_fb_idx;
|
|
|
|
} else {
|
|
|
|
fb = cpi->alt_fb_idx;
|
|
|
|
}
|
|
|
|
|
2013-02-22 20:22:03 +01:00
|
|
|
if (cpi->scaled_ref_idx[fb] != cm->ref_frame_map[fb])
|
2013-02-20 21:34:31 +01:00
|
|
|
scaled_ref_frame = &cm->yv12_fb[cpi->scaled_ref_idx[fb]];
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0) {
|
2012-08-10 15:12:43 +02:00
|
|
|
int ref = mbmi->second_ref_frame;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-12-05 17:23:38 +01:00
|
|
|
second_best_ref_mv = mbmi->ref_mvs[ref][0];
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2013-04-20 04:16:14 +02:00
|
|
|
// TODO(jkoleszar) scaling/translation handled during creation of yv12_mb
|
|
|
|
// currently.
|
|
|
|
setup_pre_planes(xd, &yv12_mb[mbmi->ref_frame],
|
|
|
|
mbmi->second_ref_frame > 0 ? &yv12_mb[mbmi->second_ref_frame] : NULL,
|
|
|
|
0, 0, NULL, NULL);
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// Experimental code. Special case for gf and arf zeromv modes.
|
|
|
|
// Increase zbin size to suppress noise
|
|
|
|
if (cpi->zbin_mode_boost_enabled) {
|
2012-10-31 01:12:12 +01:00
|
|
|
if (vp9_mode_order[mode_index].ref_frame == INTRA_FRAME)
|
2012-07-14 00:21:29 +02:00
|
|
|
cpi->zbin_mode_boost = 0;
|
|
|
|
else {
|
2012-10-31 01:12:12 +01:00
|
|
|
if (vp9_mode_order[mode_index].mode == ZEROMV) {
|
|
|
|
if (vp9_mode_order[mode_index].ref_frame != LAST_FRAME)
|
2012-07-14 00:21:29 +02:00
|
|
|
cpi->zbin_mode_boost = GF_ZEROMV_ZBIN_BOOST;
|
|
|
|
else
|
|
|
|
cpi->zbin_mode_boost = LF_ZEROMV_ZBIN_BOOST;
|
2012-10-31 01:12:12 +01:00
|
|
|
} else if (vp9_mode_order[mode_index].mode == SPLITMV)
|
2012-07-14 00:21:29 +02:00
|
|
|
cpi->zbin_mode_boost = 0;
|
|
|
|
else
|
|
|
|
cpi->zbin_mode_boost = MV_ZBIN_BOOST;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_update_zbin_extra(cpi, x);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-08-08 20:13:53 +02:00
|
|
|
// Intra
|
2012-08-10 15:12:43 +02:00
|
|
|
if (!mbmi->ref_frame) {
|
2012-07-14 00:21:29 +02:00
|
|
|
switch (this_mode) {
|
2012-08-08 20:13:53 +02:00
|
|
|
default:
|
2012-08-07 23:33:15 +02:00
|
|
|
case V_PRED:
|
|
|
|
case H_PRED:
|
|
|
|
case D45_PRED:
|
|
|
|
case D135_PRED:
|
|
|
|
case D117_PRED:
|
|
|
|
case D153_PRED:
|
|
|
|
case D27_PRED:
|
|
|
|
case D63_PRED:
|
2012-12-14 18:49:46 +01:00
|
|
|
rate2 += intra_cost_penalty;
|
|
|
|
case DC_PRED:
|
|
|
|
case TM_PRED:
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->ref_frame = INTRA_FRAME;
|
2012-08-07 23:33:15 +02:00
|
|
|
// FIXME compound intra prediction
|
2013-04-15 18:31:27 +02:00
|
|
|
vp9_build_intra_predictors_sby_s(&x->e_mbd, BLOCK_SIZE_MB16X16);
|
|
|
|
// vp9_build_intra_predictors_mby(&x->e_mbd);
|
2013-04-11 00:55:59 +02:00
|
|
|
super_block_yrd(cpi, x, &rate_y, &distortion, &skippable,
|
|
|
|
BLOCK_SIZE_MB16X16, txfm_cache);
|
2012-08-07 23:33:15 +02:00
|
|
|
rate2 += rate_y;
|
|
|
|
distortion2 += distortion;
|
2012-10-29 11:36:11 +01:00
|
|
|
rate2 += x->mbmode_cost[xd->frame_type][mbmi->mode];
|
2013-04-11 00:55:59 +02:00
|
|
|
|
|
|
|
rate2 += uv_intra_rate[mbmi->txfm_size != TX_4X4];
|
|
|
|
rate_uv = uv_intra_rate_tokenonly[mbmi->txfm_size != TX_4X4];
|
|
|
|
distortion2 += uv_intra_distortion[mbmi->txfm_size != TX_4X4];
|
|
|
|
distortion_uv = uv_intra_distortion[mbmi->txfm_size != TX_4X4];
|
|
|
|
skippable = skippable &&
|
|
|
|
uv_intra_skippable[mbmi->txfm_size != TX_4X4];
|
2012-08-07 23:33:15 +02:00
|
|
|
break;
|
2013-04-11 21:16:35 +02:00
|
|
|
case I4X4_PRED: {
|
2012-08-02 19:07:33 +02:00
|
|
|
int64_t tmp_rd;
|
2011-02-08 22:50:43 +01:00
|
|
|
|
2012-08-08 20:13:53 +02:00
|
|
|
// Note the rate value returned here includes the cost of coding
|
2013-04-11 21:16:35 +02:00
|
|
|
// the I4X4_PRED mode : x->mbmode_cost[xd->frame_type][I4X4_PRED];
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = TX_4X4;
|
2013-01-14 23:37:53 +01:00
|
|
|
tmp_rd = rd_pick_intra4x4mby_modes(cpi, x, &rate, &rate_y,
|
2013-01-25 01:28:53 +01:00
|
|
|
&distortion, best_yrd);
|
2012-07-14 00:21:29 +02:00
|
|
|
rate2 += rate;
|
2012-12-14 18:49:46 +01:00
|
|
|
rate2 += intra_cost_penalty;
|
2012-07-14 00:21:29 +02:00
|
|
|
distortion2 += distortion;
|
|
|
|
|
|
|
|
if (tmp_rd < best_yrd) {
|
2013-04-11 00:55:59 +02:00
|
|
|
rate2 += uv_intra_rate[TX_4X4];
|
|
|
|
rate_uv = uv_intra_rate_tokenonly[TX_4X4];
|
|
|
|
distortion2 += uv_intra_distortion[TX_4X4];
|
|
|
|
distortion_uv = uv_intra_distortion[TX_4X4];
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
2013-01-14 20:49:30 +01:00
|
|
|
this_rd = INT64_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
disable_skip = 1;
|
|
|
|
}
|
2011-02-08 22:50:43 +01:00
|
|
|
}
|
|
|
|
break;
|
2012-07-14 00:21:29 +02:00
|
|
|
case I8X8_PRED: {
|
2013-03-07 19:45:25 +01:00
|
|
|
int64_t tmp_rd;
|
2012-10-11 02:18:22 +02:00
|
|
|
|
2013-03-07 19:45:25 +01:00
|
|
|
tmp_rd = rd_pick_intra8x8mby_modes_and_txsz(cpi, x, &rate, &rate_y,
|
|
|
|
&distortion, mode8x8,
|
|
|
|
best_yrd, txfm_cache);
|
2012-10-11 02:18:22 +02:00
|
|
|
rate2 += rate;
|
2012-12-14 18:49:46 +01:00
|
|
|
rate2 += intra_cost_penalty;
|
2012-10-11 02:18:22 +02:00
|
|
|
distortion2 += distortion;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
/* TODO: uv rate maybe over-estimated here since there is UV intra
|
|
|
|
mode coded in I8X8_PRED prediction */
|
|
|
|
if (tmp_rd < best_yrd) {
|
2013-04-11 00:55:59 +02:00
|
|
|
rate2 += uv_intra_rate[TX_4X4];
|
|
|
|
rate_uv = uv_intra_rate_tokenonly[TX_4X4];
|
|
|
|
distortion2 += uv_intra_distortion[TX_4X4];
|
|
|
|
distortion_uv = uv_intra_distortion[TX_4X4];
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
2013-01-14 20:49:30 +01:00
|
|
|
this_rd = INT64_MAX;
|
2012-07-14 00:21:29 +02:00
|
|
|
disable_skip = 1;
|
|
|
|
}
|
2011-12-01 01:25:00 +01:00
|
|
|
}
|
|
|
|
break;
|
2012-08-08 20:13:53 +02:00
|
|
|
}
|
|
|
|
}
|
2012-08-08 20:49:56 +02:00
|
|
|
// Split MV. The code is very different from the other inter modes so
|
|
|
|
// special case it.
|
|
|
|
else if (this_mode == SPLITMV) {
|
2012-11-07 15:50:25 +01:00
|
|
|
const int is_comp_pred = mbmi->second_ref_frame > 0;
|
2013-02-12 02:08:52 +01:00
|
|
|
int64_t this_rd_thresh;
|
|
|
|
int64_t tmp_rd, tmp_best_rd = INT64_MAX, tmp_best_rdu = INT64_MAX;
|
|
|
|
int tmp_best_rate = INT_MAX, tmp_best_ratey = INT_MAX;
|
|
|
|
int tmp_best_distortion = INT_MAX, tmp_best_skippable = 0;
|
|
|
|
int switchable_filter_index;
|
2012-08-08 20:49:56 +02:00
|
|
|
int_mv *second_ref = is_comp_pred ? &second_best_ref_mv : NULL;
|
2013-02-12 02:08:52 +01:00
|
|
|
union b_mode_info tmp_best_bmodes[16];
|
|
|
|
MB_MODE_INFO tmp_best_mbmode;
|
|
|
|
PARTITION_INFO tmp_best_partition;
|
|
|
|
int pred_exists = 0;
|
2012-08-08 20:49:56 +02:00
|
|
|
|
|
|
|
this_rd_thresh =
|
2013-02-12 02:08:52 +01:00
|
|
|
(mbmi->ref_frame == LAST_FRAME) ?
|
2012-08-08 20:49:56 +02:00
|
|
|
cpi->rd_threshes[THR_NEWMV] : cpi->rd_threshes[THR_NEWA];
|
|
|
|
this_rd_thresh =
|
2013-02-12 02:08:52 +01:00
|
|
|
(mbmi->ref_frame == GOLDEN_FRAME) ?
|
2012-08-08 20:49:56 +02:00
|
|
|
cpi->rd_threshes[THR_NEWG] : this_rd_thresh;
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_4X4;
|
2012-08-08 20:49:56 +02:00
|
|
|
|
2013-02-12 02:08:52 +01:00
|
|
|
for (switchable_filter_index = 0;
|
|
|
|
switchable_filter_index < VP9_SWITCHABLE_FILTERS;
|
|
|
|
++switchable_filter_index) {
|
|
|
|
int newbest;
|
|
|
|
mbmi->interp_filter =
|
|
|
|
vp9_switchable_interp[switchable_filter_index];
|
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
|
|
|
|
|
|
|
tmp_rd = rd_pick_best_mbsegmentation(cpi, x, &best_ref_mv,
|
|
|
|
second_ref, best_yrd, mdcounts,
|
|
|
|
&rate, &rate_y, &distortion,
|
|
|
|
&skippable,
|
|
|
|
(int)this_rd_thresh, seg_mvs,
|
|
|
|
txfm_cache);
|
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE) {
|
|
|
|
int rs = SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs
|
|
|
|
[vp9_get_pred_context(&cpi->common, xd,
|
|
|
|
PRED_SWITCHABLE_INTERP)]
|
|
|
|
[vp9_switchable_interp_map[mbmi->interp_filter]];
|
|
|
|
tmp_rd += RDCOST(x->rdmult, x->rddiv, rs, 0);
|
|
|
|
}
|
|
|
|
newbest = (tmp_rd < tmp_best_rd);
|
|
|
|
if (newbest) {
|
|
|
|
tmp_best_filter = mbmi->interp_filter;
|
|
|
|
tmp_best_rd = tmp_rd;
|
|
|
|
}
|
|
|
|
if ((newbest && cm->mcomp_filter_type == SWITCHABLE) ||
|
|
|
|
(mbmi->interp_filter == cm->mcomp_filter_type &&
|
|
|
|
cm->mcomp_filter_type != SWITCHABLE)) {
|
|
|
|
tmp_best_rdu = tmp_rd;
|
|
|
|
tmp_best_rate = rate;
|
|
|
|
tmp_best_ratey = rate_y;
|
|
|
|
tmp_best_distortion = distortion;
|
|
|
|
tmp_best_skippable = skippable;
|
|
|
|
vpx_memcpy(&tmp_best_mbmode, mbmi, sizeof(MB_MODE_INFO));
|
|
|
|
vpx_memcpy(&tmp_best_partition, x->partition_info,
|
|
|
|
sizeof(PARTITION_INFO));
|
|
|
|
for (i = 0; i < 16; i++) {
|
2013-04-26 16:19:43 +02:00
|
|
|
tmp_best_bmodes[i] = xd->mode_info_context->bmi[i];
|
2013-02-12 02:08:52 +01:00
|
|
|
}
|
|
|
|
pred_exists = 1;
|
|
|
|
}
|
|
|
|
} // switchable_filter_index loop
|
|
|
|
|
|
|
|
mbmi->interp_filter = (cm->mcomp_filter_type == SWITCHABLE ?
|
|
|
|
tmp_best_filter : cm->mcomp_filter_type);
|
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
|
|
|
if (!pred_exists) {
|
|
|
|
// Handles the special case when a filter that is not in the
|
|
|
|
// switchable list (bilinear, 6-tap) is indicated at the frame level
|
|
|
|
tmp_rd = rd_pick_best_mbsegmentation(cpi, x, &best_ref_mv,
|
|
|
|
second_ref, best_yrd, mdcounts,
|
|
|
|
&rate, &rate_y, &distortion,
|
|
|
|
&skippable,
|
|
|
|
(int)this_rd_thresh, seg_mvs,
|
|
|
|
txfm_cache);
|
|
|
|
} else {
|
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE) {
|
|
|
|
int rs = SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs
|
|
|
|
[vp9_get_pred_context(&cpi->common, xd,
|
|
|
|
PRED_SWITCHABLE_INTERP)]
|
|
|
|
[vp9_switchable_interp_map[mbmi->interp_filter]];
|
|
|
|
tmp_best_rdu -= RDCOST(x->rdmult, x->rddiv, rs, 0);
|
|
|
|
}
|
|
|
|
tmp_rd = tmp_best_rdu;
|
|
|
|
rate = tmp_best_rate;
|
|
|
|
rate_y = tmp_best_ratey;
|
|
|
|
distortion = tmp_best_distortion;
|
|
|
|
skippable = tmp_best_skippable;
|
|
|
|
vpx_memcpy(mbmi, &tmp_best_mbmode, sizeof(MB_MODE_INFO));
|
|
|
|
vpx_memcpy(x->partition_info, &tmp_best_partition,
|
|
|
|
sizeof(PARTITION_INFO));
|
|
|
|
for (i = 0; i < 16; i++) {
|
2013-04-26 16:19:43 +02:00
|
|
|
xd->mode_info_context->bmi[i] = tmp_best_bmodes[i];
|
2013-02-12 02:08:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-08 20:49:56 +02:00
|
|
|
rate2 += rate;
|
|
|
|
distortion2 += distortion;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-08 20:49:56 +02:00
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE)
|
|
|
|
rate2 += SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs
|
2012-10-29 14:44:18 +01:00
|
|
|
[vp9_get_pred_context(&cpi->common, xd, PRED_SWITCHABLE_INTERP)]
|
2013-02-12 02:08:52 +01:00
|
|
|
[vp9_switchable_interp_map[mbmi->interp_filter]];
|
2012-12-20 23:56:19 +01:00
|
|
|
|
2012-08-08 20:49:56 +02:00
|
|
|
// If even the 'Y' rd value of split is higher than best so far
|
|
|
|
// then dont bother looking at UV
|
|
|
|
if (tmp_rd < best_yrd) {
|
2012-10-17 23:32:17 +02:00
|
|
|
int uv_skippable;
|
|
|
|
|
2013-04-26 20:57:17 +02:00
|
|
|
vp9_build_inter_predictors_sbuv(&x->e_mbd, mi_row, mi_col,
|
2013-04-19 19:45:50 +02:00
|
|
|
BLOCK_SIZE_MB16X16);
|
2013-04-15 18:31:27 +02:00
|
|
|
|
2013-04-23 17:26:10 +02:00
|
|
|
vp9_subtract_sbuv(x, BLOCK_SIZE_MB16X16);
|
2013-04-15 18:31:27 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
super_block_uvrd_4x4(cm, x, &rate_uv, &distortion_uv,
|
|
|
|
&uv_skippable, BLOCK_SIZE_MB16X16);
|
2012-08-08 20:49:56 +02:00
|
|
|
rate2 += rate_uv;
|
|
|
|
distortion2 += distortion_uv;
|
2012-10-17 23:32:17 +02:00
|
|
|
skippable = skippable && uv_skippable;
|
2012-08-08 20:49:56 +02:00
|
|
|
} else {
|
2013-01-14 20:49:30 +01:00
|
|
|
this_rd = INT64_MAX;
|
2012-08-08 20:49:56 +02:00
|
|
|
disable_skip = 1;
|
|
|
|
}
|
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
if (!mode_excluded) {
|
|
|
|
if (is_comp_pred)
|
|
|
|
mode_excluded = cpi->common.comp_pred_mode == SINGLE_PREDICTION_ONLY;
|
|
|
|
else
|
|
|
|
mode_excluded = cpi->common.comp_pred_mode == COMP_PREDICTION_ONLY;
|
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-08 20:49:56 +02:00
|
|
|
compmode_cost =
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_COMP), is_comp_pred);
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->mode = this_mode;
|
2012-08-08 20:49:56 +02:00
|
|
|
}
|
2012-08-10 01:07:41 +02:00
|
|
|
else {
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if (mbmi->second_ref_frame == INTRA_FRAME) {
|
|
|
|
if (best_intra16_mode == DC_PRED - 1) continue;
|
|
|
|
mbmi->interintra_mode = best_intra16_mode;
|
|
|
|
#if SEPARATE_INTERINTRA_UV
|
|
|
|
mbmi->interintra_uv_mode = best_intra16_uv_mode;
|
|
|
|
#else
|
|
|
|
mbmi->interintra_uv_mode = best_intra16_mode;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
2013-04-11 00:55:59 +02:00
|
|
|
this_rd = handle_inter_mode(cpi, x, BLOCK_SIZE_MB16X16,
|
2013-04-18 20:55:46 +02:00
|
|
|
mdcounts, txfm_cache,
|
2012-10-30 01:58:18 +01:00
|
|
|
&rate2, &distortion2, &skippable,
|
2012-11-07 15:50:25 +01:00
|
|
|
&compmode_cost,
|
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
&compmode_interintra_cost,
|
|
|
|
#endif
|
|
|
|
&rate_y, &distortion,
|
2012-10-30 01:58:18 +01:00
|
|
|
&rate_uv, &distortion_uv,
|
2013-02-07 19:09:05 +01:00
|
|
|
&mode_excluded, &disable_skip,
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
mode_index, &tmp_best_filter, frame_mv,
|
2013-04-26 20:57:17 +02:00
|
|
|
scaled_ref_frame, mi_row, mi_col);
|
2013-01-14 20:49:30 +01:00
|
|
|
if (this_rd == INT64_MAX)
|
2012-08-10 01:07:41 +02:00
|
|
|
continue;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if (cpi->common.use_interintra)
|
|
|
|
rate2 += compmode_interintra_cost;
|
|
|
|
#endif
|
|
|
|
|
2012-08-07 01:21:23 +02:00
|
|
|
if (cpi->common.comp_pred_mode == HYBRID_PREDICTION)
|
2012-07-14 00:21:29 +02:00
|
|
|
rate2 += compmode_cost;
|
|
|
|
|
|
|
|
// Estimate the reference frame signaling cost and add it
|
|
|
|
// to the rolling cost variable.
|
2012-08-10 15:12:43 +02:00
|
|
|
rate2 += ref_costs[mbmi->ref_frame];
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
if (!disable_skip) {
|
|
|
|
// Test for the condition where skip block will be activated
|
|
|
|
// because there are no non zero coefficients and make any
|
|
|
|
// necessary adjustment for rate. Ignore if skip is coded at
|
|
|
|
// segment level as the cost wont have been added in.
|
2013-04-09 19:54:19 +02:00
|
|
|
int mb_skip_allowed;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
// Is Mb level skip allowed (i.e. not coded at segment level).
|
|
|
|
mb_skip_allowed = !vp9_segfeature_active(xd, segment_id, SEG_LVL_SKIP);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
if (skippable) {
|
|
|
|
mbmi->mb_skip_coeff = 1;
|
2012-10-09 18:18:21 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
// Back out the coefficient coding costs
|
|
|
|
rate2 -= (rate_y + rate_uv);
|
|
|
|
// for best_yrd calculation
|
|
|
|
rate_uv = 0;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
if (mb_skip_allowed) {
|
|
|
|
int prob_skip_cost;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
// Cost the skip mb case
|
|
|
|
vp9_prob skip_prob =
|
|
|
|
vp9_get_pred_prob(cm, &x->e_mbd, PRED_MBSKIP);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
if (skip_prob) {
|
|
|
|
prob_skip_cost = vp9_cost_bit(skip_prob, 1);
|
2012-10-09 18:18:21 +02:00
|
|
|
rate2 += prob_skip_cost;
|
|
|
|
other_cost += prob_skip_cost;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2013-04-09 19:54:19 +02:00
|
|
|
} else {
|
|
|
|
// Add in the cost of the no skip flag.
|
|
|
|
mbmi->mb_skip_coeff = 0;
|
|
|
|
if (mb_skip_allowed) {
|
|
|
|
int prob_skip_cost = vp9_cost_bit(
|
|
|
|
vp9_get_pred_prob(cm, &x->e_mbd, PRED_MBSKIP), 0);
|
|
|
|
rate2 += prob_skip_cost;
|
|
|
|
other_cost += prob_skip_cost;
|
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// Calculate the final RD estimate for this mode.
|
|
|
|
this_rd = RDCOST(x->rdmult, x->rddiv, rate2, distortion2);
|
|
|
|
}
|
Dual 16x16 inter prediction.
This patch introduces the concept of dual inter16x16 prediction. A
16x16 inter-predicted macroblock can use 2 references instead of 1,
where both references use the same mvmode (new, near/est, zero). In the
case of newmv, this means that two MVs are coded instead of one. The
frame can be encoded in 3 ways: all MBs single-prediction, all MBs dual
prediction, or per-MB single/dual prediction selection ("hybrid"), in
which case a single bit is coded per-MB to indicate whether the MB uses
single or dual inter prediction.
In the future, we can (maybe?) get further gains by mixing this with
Adrian's 32x32 work, per-segment dual prediction settings, or adding
support for dual splitmv/8x8mv inter prediction.
Gain (on derf-set, CQ mode) is ~2.8% (SSIM) or ~3.6% (glb PSNR). Most
gain is at medium/high bitrates, but there's minor gains at low bitrates
also. Output was confirmed to match between encoder and decoder.
Note for optimization people: this patch introduces a 2nd version of
16x16/8x8 sixtap/bilin functions, which does an avg instead of a
store. They may want to look and make sure this is implemented to
their satisfaction so we can optimize it best in the future.
Change-ID: I59dc84b07cbb3ccf073ac0f756d03d294cb19281
2011-12-06 20:53:02 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// Keep record of best intra distortion
|
2012-08-10 15:12:43 +02:00
|
|
|
if ((mbmi->ref_frame == INTRA_FRAME) &&
|
2012-07-14 00:21:29 +02:00
|
|
|
(this_rd < best_intra_rd)) {
|
|
|
|
best_intra_rd = this_rd;
|
|
|
|
*returnintra = distortion2;
|
|
|
|
}
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if ((mbmi->ref_frame == INTRA_FRAME) &&
|
2012-11-30 16:29:43 +01:00
|
|
|
(this_mode <= TM_PRED) &&
|
2012-11-07 15:50:25 +01:00
|
|
|
(this_rd < best_intra16_rd)) {
|
|
|
|
best_intra16_rd = this_rd;
|
|
|
|
best_intra16_mode = this_mode;
|
2013-02-20 19:16:24 +01:00
|
|
|
#if SEPARATE_INTERINTRA_UV
|
2013-04-11 00:55:59 +02:00
|
|
|
best_intra16_uv_mode = uv_intra_mode[mbmi->txfm_size != TX_4X4];
|
2013-02-20 19:16:24 +01:00
|
|
|
#endif
|
2012-11-07 15:50:25 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-08-10 03:25:29 +02:00
|
|
|
if (!disable_skip && mbmi->ref_frame == INTRA_FRAME)
|
|
|
|
for (i = 0; i < NB_PREDICTION_TYPES; ++i)
|
|
|
|
best_pred_rd[i] = MIN(best_pred_rd[i], this_rd);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (this_rd < best_overall_rd) {
|
|
|
|
best_overall_rd = this_rd;
|
2013-02-12 02:08:52 +01:00
|
|
|
best_filter = tmp_best_filter;
|
2012-12-20 23:56:19 +01:00
|
|
|
best_mode = this_mode;
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
is_best_interintra = (mbmi->second_ref_frame == INTRA_FRAME);
|
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2013-04-23 00:42:41 +02:00
|
|
|
// Store the respective mode distortions for later use.
|
|
|
|
// Store the respective mode distortions for later use.
|
|
|
|
if (mode_distortions[this_mode] == -1
|
|
|
|
|| distortion2 < mode_distortions[this_mode]) {
|
|
|
|
mode_distortions[this_mode] = distortion2;
|
|
|
|
}
|
|
|
|
if (frame_distortions[mbmi->ref_frame] == -1 ||
|
|
|
|
distortion2 < frame_distortions[mbmi->ref_frame]) {
|
|
|
|
frame_distortions[mbmi->ref_frame] = distortion2;
|
|
|
|
}
|
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
// Did this mode help.. i.e. is it the new best mode
|
|
|
|
if (this_rd < best_rd || x->skip) {
|
|
|
|
if (!mode_excluded) {
|
|
|
|
/*
|
|
|
|
if (mbmi->second_ref_frame == INTRA_FRAME) {
|
|
|
|
printf("rd %d best %d bestintra16 %d\n", this_rd, best_rd, best_intra16_rd);
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
// Note index of best mode so far
|
|
|
|
best_mode_index = mode_index;
|
2012-06-26 01:23:58 +02:00
|
|
|
|
2013-04-11 21:16:35 +02:00
|
|
|
if (this_mode <= I4X4_PRED) {
|
2012-12-20 23:56:19 +01:00
|
|
|
if (mbmi->txfm_size != TX_4X4
|
2013-04-11 21:16:35 +02:00
|
|
|
&& this_mode != I4X4_PRED
|
2012-12-20 23:56:19 +01:00
|
|
|
&& this_mode != I8X8_PRED)
|
2013-04-11 00:55:59 +02:00
|
|
|
mbmi->uv_mode = uv_intra_mode[TX_8X8];
|
2012-12-20 23:56:19 +01:00
|
|
|
else
|
2013-04-11 00:55:59 +02:00
|
|
|
mbmi->uv_mode = uv_intra_mode[TX_4X4];
|
2012-12-20 23:56:19 +01:00
|
|
|
/* required for left and above block mv */
|
|
|
|
mbmi->mv[0].as_int = 0;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
other_cost += ref_costs[mbmi->ref_frame];
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
/* Calculate the final y RD estimate for this mode */
|
|
|
|
best_yrd = RDCOST(x->rdmult, x->rddiv, (rate2 - rate_uv - other_cost),
|
|
|
|
(distortion2 - distortion_uv));
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
*returnrate = rate2;
|
|
|
|
*returndistortion = distortion2;
|
|
|
|
best_rd = this_rd;
|
|
|
|
vpx_memcpy(&best_mbmode, mbmi, sizeof(MB_MODE_INFO));
|
|
|
|
vpx_memcpy(&best_partition, x->partition_info, sizeof(PARTITION_INFO));
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-04-11 21:16:35 +02:00
|
|
|
if ((this_mode == I4X4_PRED)
|
2012-12-20 23:56:19 +01:00
|
|
|
|| (this_mode == I8X8_PRED)
|
|
|
|
|| (this_mode == SPLITMV))
|
|
|
|
for (i = 0; i < 16; i++) {
|
2013-04-26 16:19:43 +02:00
|
|
|
best_bmodes[i] = xd->mode_info_context->bmi[i];
|
2012-12-20 23:56:19 +01:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2012-12-20 23:56:19 +01:00
|
|
|
|
|
|
|
// Testing this mode gave rise to an improvement in best error score.
|
|
|
|
// Lower threshold a bit for next time
|
|
|
|
cpi->rd_thresh_mult[mode_index] =
|
|
|
|
(cpi->rd_thresh_mult[mode_index] >= (MIN_THRESHMULT + 2)) ?
|
|
|
|
cpi->rd_thresh_mult[mode_index] - 2 : MIN_THRESHMULT;
|
|
|
|
cpi->rd_threshes[mode_index] =
|
|
|
|
(cpi->rd_baseline_thresh[mode_index] >> 7) *
|
|
|
|
cpi->rd_thresh_mult[mode_index];
|
|
|
|
} else {
|
2012-07-18 22:43:01 +02:00
|
|
|
// If the mode did not help improve the best error case then raise the
|
|
|
|
// threshold for testing that mode next time around.
|
2012-12-20 23:56:19 +01:00
|
|
|
cpi->rd_thresh_mult[mode_index] += 4;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
if (cpi->rd_thresh_mult[mode_index] > MAX_THRESHMULT)
|
|
|
|
cpi->rd_thresh_mult[mode_index] = MAX_THRESHMULT;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
cpi->rd_threshes[mode_index] = (cpi->rd_baseline_thresh[mode_index] >> 7)
|
|
|
|
* cpi->rd_thresh_mult[mode_index];
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
/* keep record of best compound/single-only prediction */
|
|
|
|
if (!disable_skip && mbmi->ref_frame != INTRA_FRAME) {
|
|
|
|
int64_t single_rd, hybrid_rd;
|
|
|
|
int single_rate, hybrid_rate;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
if (cpi->common.comp_pred_mode == HYBRID_PREDICTION) {
|
|
|
|
single_rate = rate2 - compmode_cost;
|
|
|
|
hybrid_rate = rate2;
|
|
|
|
} else {
|
|
|
|
single_rate = rate2;
|
|
|
|
hybrid_rate = rate2 + compmode_cost;
|
|
|
|
}
|
2011-12-01 01:25:00 +01:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
single_rd = RDCOST(x->rdmult, x->rddiv, single_rate, distortion2);
|
|
|
|
hybrid_rd = RDCOST(x->rdmult, x->rddiv, hybrid_rate, distortion2);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
if (mbmi->second_ref_frame <= INTRA_FRAME &&
|
|
|
|
single_rd < best_pred_rd[SINGLE_PREDICTION_ONLY]) {
|
|
|
|
best_pred_rd[SINGLE_PREDICTION_ONLY] = single_rd;
|
|
|
|
} else if (mbmi->second_ref_frame > INTRA_FRAME &&
|
|
|
|
single_rd < best_pred_rd[COMP_PREDICTION_ONLY]) {
|
|
|
|
best_pred_rd[COMP_PREDICTION_ONLY] = single_rd;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2012-12-20 23:56:19 +01:00
|
|
|
if (hybrid_rd < best_pred_rd[HYBRID_PREDICTION])
|
|
|
|
best_pred_rd[HYBRID_PREDICTION] = hybrid_rd;
|
|
|
|
}
|
2012-10-09 18:18:21 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
/* keep record of best txfm size */
|
2013-01-14 20:49:30 +01:00
|
|
|
if (!mode_excluded && this_rd != INT64_MAX) {
|
2012-12-20 23:56:19 +01:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
int64_t adj_rd;
|
2013-04-11 21:16:35 +02:00
|
|
|
if (this_mode != I4X4_PRED) {
|
2012-12-20 23:56:19 +01:00
|
|
|
const int64_t txfm_mode_diff =
|
|
|
|
txfm_cache[i] - txfm_cache[cm->txfm_mode];
|
|
|
|
adj_rd = this_rd + txfm_mode_diff;
|
|
|
|
} else {
|
|
|
|
adj_rd = this_rd;
|
2012-10-09 18:18:21 +02:00
|
|
|
}
|
2012-12-20 23:56:19 +01:00
|
|
|
if (adj_rd < best_txfm_rd[i])
|
|
|
|
best_txfm_rd[i] = adj_rd;
|
2012-10-09 18:18:21 +02:00
|
|
|
}
|
2011-05-19 21:03:36 +02:00
|
|
|
}
|
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
if (x->skip && !mode_excluded)
|
2012-07-14 00:21:29 +02:00
|
|
|
break;
|
2013-02-12 02:08:52 +01:00
|
|
|
}
|
2012-12-20 23:56:19 +01:00
|
|
|
|
|
|
|
assert((cm->mcomp_filter_type == SWITCHABLE) ||
|
|
|
|
(cm->mcomp_filter_type == best_mbmode.interp_filter) ||
|
2013-04-11 21:16:35 +02:00
|
|
|
(best_mbmode.mode <= I4X4_PRED));
|
2012-04-07 01:38:34 +02:00
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
++cpi->interintra_select_count[is_best_interintra];
|
2012-07-14 00:21:29 +02:00
|
|
|
#endif
|
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
// Accumulate filter usage stats
|
|
|
|
// TODO(agrange): Use RD criteria to select interpolation filter mode.
|
2013-04-24 21:14:58 +02:00
|
|
|
if (is_inter_mode(best_mode))
|
2012-12-20 23:56:19 +01:00
|
|
|
++cpi->best_switchable_interp_count[vp9_switchable_interp_map[best_filter]];
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// Reduce the activation RD thresholds for the best choice mode
|
2012-07-18 22:43:01 +02:00
|
|
|
if ((cpi->rd_baseline_thresh[best_mode_index] > 0) &&
|
|
|
|
(cpi->rd_baseline_thresh[best_mode_index] < (INT_MAX >> 2))) {
|
2012-07-14 00:21:29 +02:00
|
|
|
int best_adjustment = (cpi->rd_thresh_mult[best_mode_index] >> 2);
|
|
|
|
|
2012-07-18 22:43:01 +02:00
|
|
|
cpi->rd_thresh_mult[best_mode_index] =
|
|
|
|
(cpi->rd_thresh_mult[best_mode_index] >=
|
|
|
|
(MIN_THRESHMULT + best_adjustment)) ?
|
|
|
|
cpi->rd_thresh_mult[best_mode_index] - best_adjustment : MIN_THRESHMULT;
|
|
|
|
cpi->rd_threshes[best_mode_index] =
|
|
|
|
(cpi->rd_baseline_thresh[best_mode_index] >> 7) *
|
|
|
|
cpi->rd_thresh_mult[best_mode_index];
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2013-01-28 16:22:53 +01:00
|
|
|
// This code forces Altref,0,0 and skip for the frame that overlays a
|
2012-07-14 00:21:29 +02:00
|
|
|
// an alrtef unless Altref is filtered. However, this is unsafe if
|
2013-01-28 16:22:53 +01:00
|
|
|
// segment level coding of ref frame is enabled for this
|
2012-07-14 00:21:29 +02:00
|
|
|
// segment.
|
2012-10-30 06:15:27 +01:00
|
|
|
if (!vp9_segfeature_active(xd, segment_id, SEG_LVL_REF_FRAME) &&
|
2012-07-14 00:21:29 +02:00
|
|
|
cpi->is_src_frame_alt_ref &&
|
|
|
|
(cpi->oxcf.arnr_max_frames == 0) &&
|
|
|
|
(best_mbmode.mode != ZEROMV || best_mbmode.ref_frame != ALTREF_FRAME)) {
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->mode = ZEROMV;
|
2013-01-04 21:05:40 +01:00
|
|
|
if (cm->txfm_mode <= ALLOW_8X8)
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = cm->txfm_mode;
|
|
|
|
else
|
|
|
|
mbmi->txfm_size = TX_16X16;
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->ref_frame = ALTREF_FRAME;
|
2012-08-10 01:07:41 +02:00
|
|
|
mbmi->mv[0].as_int = 0;
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->uv_mode = DC_PRED;
|
2013-04-09 19:54:19 +02:00
|
|
|
mbmi->mb_skip_coeff = 1;
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->partitioning = 0;
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
set_scale_factors(xd, mbmi->ref_frame, mbmi->second_ref_frame,
|
|
|
|
scale_factor);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-10 03:25:29 +02:00
|
|
|
vpx_memset(best_pred_diff, 0, sizeof(best_pred_diff));
|
2012-10-09 18:18:21 +02:00
|
|
|
vpx_memset(best_txfm_diff, 0, sizeof(best_txfm_diff));
|
2012-08-10 03:25:29 +02:00
|
|
|
goto end;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// macroblock modes
|
2012-08-10 15:12:43 +02:00
|
|
|
vpx_memcpy(mbmi, &best_mbmode, sizeof(MB_MODE_INFO));
|
2013-04-11 21:16:35 +02:00
|
|
|
if (best_mbmode.mode == I4X4_PRED) {
|
2012-07-14 00:21:29 +02:00
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
xd->mode_info_context->bmi[i].as_mode = best_bmodes[i].as_mode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-07 01:21:23 +02:00
|
|
|
if (best_mbmode.mode == I8X8_PRED)
|
2012-07-14 00:21:29 +02:00
|
|
|
set_i8x8_block_modes(x, mode8x8);
|
|
|
|
|
|
|
|
if (best_mbmode.mode == SPLITMV) {
|
|
|
|
for (i = 0; i < 16; i++)
|
2013-02-09 04:46:36 +01:00
|
|
|
xd->mode_info_context->bmi[i].as_mv[0].as_int =
|
|
|
|
best_bmodes[i].as_mv[0].as_int;
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame > 0)
|
2012-07-14 00:21:29 +02:00
|
|
|
for (i = 0; i < 16; i++)
|
2013-02-09 04:46:36 +01:00
|
|
|
xd->mode_info_context->bmi[i].as_mv[1].as_int =
|
|
|
|
best_bmodes[i].as_mv[1].as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
vpx_memcpy(x->partition_info, &best_partition, sizeof(PARTITION_INFO));
|
|
|
|
|
2012-08-10 01:07:41 +02:00
|
|
|
mbmi->mv[0].as_int = x->partition_info->bmi[15].mv.as_int;
|
|
|
|
mbmi->mv[1].as_int = x->partition_info->bmi[15].second_mv.as_int;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
2012-08-10 03:25:29 +02:00
|
|
|
for (i = 0; i < NB_PREDICTION_TYPES; ++i) {
|
2013-01-14 20:49:30 +01:00
|
|
|
if (best_pred_rd[i] == INT64_MAX)
|
2012-08-10 03:25:29 +02:00
|
|
|
best_pred_diff[i] = INT_MIN;
|
|
|
|
else
|
|
|
|
best_pred_diff[i] = best_rd - best_pred_rd[i];
|
|
|
|
}
|
|
|
|
|
2012-10-09 18:18:21 +02:00
|
|
|
if (!x->skip) {
|
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
2013-01-14 20:49:30 +01:00
|
|
|
if (best_txfm_rd[i] == INT64_MAX)
|
2013-02-16 00:55:31 +01:00
|
|
|
best_txfm_diff[i] = 0;
|
2012-10-09 18:18:21 +02:00
|
|
|
else
|
|
|
|
best_txfm_diff[i] = best_rd - best_txfm_rd[i];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
vpx_memset(best_txfm_diff, 0, sizeof(best_txfm_diff));
|
|
|
|
}
|
|
|
|
|
2012-08-10 03:25:29 +02:00
|
|
|
end:
|
2013-04-23 00:42:41 +02:00
|
|
|
|
|
|
|
// Flag all modes that have a distortion thats > 2x the best we found at
|
|
|
|
// this level.
|
|
|
|
for (mode_index = 0; mode_index < MB_MODE_COUNT; ++mode_index) {
|
|
|
|
if (mode_index == NEARESTMV || mode_index == NEARMV || mode_index == NEWMV
|
|
|
|
|| mode_index == SPLITMV)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (mode_distortions[mode_index] > 2 * *returndistortion) {
|
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].modes_with_high_error |= (1
|
|
|
|
<< mode_index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flag all ref frames that have a distortion thats > 2x the best we found at
|
|
|
|
// this level.
|
|
|
|
for (ref_frame = INTRA_FRAME; ref_frame <= ALTREF_FRAME; ref_frame++) {
|
|
|
|
if (frame_distortions[ref_frame] > 2 * *returndistortion) {
|
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].frames_with_high_error |= (1
|
|
|
|
<< ref_frame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
set_scale_factors(xd, mbmi->ref_frame, mbmi->second_ref_frame,
|
|
|
|
scale_factor);
|
2013-01-06 03:20:25 +01:00
|
|
|
store_coding_context(x, &x->mb_context[xd->sb_index][xd->mb_index],
|
|
|
|
best_mode_index, &best_partition,
|
|
|
|
&mbmi->ref_mvs[mbmi->ref_frame][0],
|
|
|
|
&mbmi->ref_mvs[mbmi->second_ref_frame < 0 ? 0 :
|
|
|
|
mbmi->second_ref_frame][0],
|
|
|
|
best_pred_diff, best_txfm_diff);
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif // !CONFIG_SB8X8
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
void vp9_rd_pick_intra_mode_sb(VP9_COMP *cpi, MACROBLOCK *x,
|
|
|
|
int *returnrate, int *returndist,
|
2013-04-11 18:33:49 +02:00
|
|
|
BLOCK_SIZE_TYPE bsize,
|
|
|
|
PICK_MODE_CONTEXT *ctx) {
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-08-29 19:43:20 +02:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
2013-01-15 03:03:34 +01:00
|
|
|
int rate_y = 0, rate_uv;
|
|
|
|
int rate_y_tokenonly = 0, rate_uv_tokenonly;
|
|
|
|
int dist_y = 0, dist_uv;
|
|
|
|
int y_skip = 0, uv_skip;
|
2013-02-16 00:55:31 +01:00
|
|
|
int64_t txfm_cache[NB_TXFM_MODES], err;
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
MB_PREDICTION_MODE mode;
|
|
|
|
TX_SIZE txfm_size;
|
|
|
|
int rate4x4_y, rate4x4_y_tokenonly, dist4x4_y;
|
|
|
|
int64_t err4x4 = INT64_MAX;
|
|
|
|
#endif
|
2013-02-16 00:55:31 +01:00
|
|
|
int i;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-15 21:50:32 +02:00
|
|
|
ctx->skip = 0;
|
2013-02-20 19:16:24 +01:00
|
|
|
xd->mode_info_context->mbmi.mode = DC_PRED;
|
2013-02-16 00:55:31 +01:00
|
|
|
err = rd_pick_intra_sby_mode(cpi, x, &rate_y, &rate_y_tokenonly,
|
2013-04-11 00:55:59 +02:00
|
|
|
&dist_y, &y_skip, bsize, txfm_cache);
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
mode = xd->mode_info_context->mbmi.mode;
|
|
|
|
txfm_size = xd->mode_info_context->mbmi.txfm_size;
|
|
|
|
#endif
|
2013-01-14 22:56:52 +01:00
|
|
|
rd_pick_intra_sbuv_mode(cpi, x, &rate_uv, &rate_uv_tokenonly,
|
2013-04-11 00:55:59 +02:00
|
|
|
&dist_uv, &uv_skip, bsize);
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
if (bsize == BLOCK_SIZE_SB8X8)
|
|
|
|
err4x4 = rd_pick_intra4x4mby_modes(cpi, x, &rate4x4_y,
|
|
|
|
&rate4x4_y_tokenonly,
|
|
|
|
&dist4x4_y, err);
|
|
|
|
#endif
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
if (y_skip && uv_skip) {
|
2012-08-29 19:43:20 +02:00
|
|
|
*returnrate = rate_y + rate_uv - rate_y_tokenonly - rate_uv_tokenonly +
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_MBSKIP), 1);
|
2012-08-29 19:43:20 +02:00
|
|
|
*returndist = dist_y + (dist_uv >> 2);
|
2013-04-11 18:33:49 +02:00
|
|
|
memset(ctx->txfm_rd_diff, 0,
|
|
|
|
sizeof(x->sb32_context[xd->sb_index].txfm_rd_diff));
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
xd->mode_info_context->mbmi.mode = mode;
|
|
|
|
xd->mode_info_context->mbmi.txfm_size = txfm_size;
|
|
|
|
} else if (bsize == BLOCK_SIZE_SB8X8 && err4x4 < err) {
|
|
|
|
*returnrate = rate4x4_y + rate_uv +
|
|
|
|
vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_MBSKIP), 0);
|
|
|
|
*returndist = dist4x4_y + (dist_uv >> 2);
|
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
ctx->txfm_rd_diff[i] = MIN(err4x4, err - txfm_cache[i]);
|
|
|
|
}
|
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_4X4;
|
|
|
|
#endif
|
2013-01-06 03:20:25 +01:00
|
|
|
} else {
|
2013-04-09 19:54:19 +02:00
|
|
|
*returnrate = rate_y + rate_uv +
|
|
|
|
vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_MBSKIP), 0);
|
2013-01-06 03:20:25 +01:00
|
|
|
*returndist = dist_y + (dist_uv >> 2);
|
2013-02-16 00:55:31 +01:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
ctx->txfm_rd_diff[i] = MIN(err4x4, err - txfm_cache[i]);
|
|
|
|
#else
|
2013-04-11 18:33:49 +02:00
|
|
|
ctx->txfm_rd_diff[i] = err - txfm_cache[i];
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif
|
2013-02-16 00:55:31 +01:00
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
xd->mode_info_context->mbmi.txfm_size = txfm_size;
|
|
|
|
xd->mode_info_context->mbmi.mode = mode;
|
|
|
|
#endif
|
2013-01-06 03:20:25 +01:00
|
|
|
}
|
2013-04-11 18:33:49 +02:00
|
|
|
|
|
|
|
vpx_memcpy(&ctx->mic, xd->mode_info_context, sizeof(MODE_INFO));
|
2013-01-06 03:20:25 +01:00
|
|
|
}
|
2011-06-08 18:05:05 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#if !CONFIG_SB8X8
|
2012-10-31 01:53:32 +01:00
|
|
|
void vp9_rd_pick_intra_mode(VP9_COMP *cpi, MACROBLOCK *x,
|
2012-08-20 23:43:34 +02:00
|
|
|
int *returnrate, int *returndist) {
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 00:21:29 +02:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
2012-08-10 15:12:43 +02:00
|
|
|
MB_MODE_INFO * mbmi = &x->e_mbd.mode_info_context->mbmi;
|
2012-08-02 19:07:33 +02:00
|
|
|
int64_t error4x4, error16x16;
|
2013-04-11 00:55:59 +02:00
|
|
|
int rate4x4, rate16x16 = 0, rateuv[2];
|
|
|
|
int dist4x4 = 0, dist16x16 = 0, distuv[2];
|
2012-08-07 01:21:23 +02:00
|
|
|
int rate;
|
|
|
|
int rate4x4_tokenonly = 0;
|
|
|
|
int rate16x16_tokenonly = 0;
|
2013-04-11 00:55:59 +02:00
|
|
|
int rateuv_tokenonly[2];
|
2012-08-02 19:07:33 +02:00
|
|
|
int64_t error8x8;
|
|
|
|
int rate8x8_tokenonly=0;
|
2012-08-07 01:21:23 +02:00
|
|
|
int rate8x8, dist8x8;
|
|
|
|
int mode16x16;
|
2013-01-14 23:37:53 +01:00
|
|
|
int mode8x8[4];
|
2012-08-20 23:43:34 +02:00
|
|
|
int dist;
|
2013-04-11 00:55:59 +02:00
|
|
|
int modeuv[2], uv_intra_skippable[2];
|
2012-11-30 16:29:43 +01:00
|
|
|
int y_intra16x16_skippable = 0;
|
2013-03-07 19:45:25 +01:00
|
|
|
int64_t txfm_cache[2][NB_TXFM_MODES];
|
|
|
|
TX_SIZE txfm_size_16x16, txfm_size_8x8;
|
2012-10-09 18:18:21 +02:00
|
|
|
int i;
|
2012-06-25 21:26:09 +02:00
|
|
|
|
2013-04-15 21:50:32 +02:00
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].skip = 0;
|
2012-08-10 19:00:18 +02:00
|
|
|
mbmi->ref_frame = INTRA_FRAME;
|
2013-02-20 19:16:24 +01:00
|
|
|
mbmi->mode = DC_PRED;
|
2013-04-11 00:55:59 +02:00
|
|
|
for (i = 0; i <= TX_8X8; i++) {
|
|
|
|
mbmi->txfm_size = i;
|
|
|
|
rd_pick_intra_sbuv_mode(cpi, x, &rateuv[i], &rateuv_tokenonly[i],
|
|
|
|
&distuv[i], &uv_intra_skippable[i],
|
|
|
|
BLOCK_SIZE_MB16X16);
|
|
|
|
modeuv[i] = mbmi->uv_mode;
|
2012-08-29 19:43:20 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-10 19:00:18 +02:00
|
|
|
// current macroblock under rate-distortion optimization test loop
|
2013-04-11 00:55:59 +02:00
|
|
|
error16x16 = rd_pick_intra_sby_mode(cpi, x, &rate16x16,
|
|
|
|
&rate16x16_tokenonly, &dist16x16,
|
|
|
|
&y_intra16x16_skippable,
|
|
|
|
BLOCK_SIZE_MB16X16, txfm_cache[1]);
|
2012-08-10 15:12:43 +02:00
|
|
|
mode16x16 = mbmi->mode;
|
2012-10-09 18:18:21 +02:00
|
|
|
txfm_size_16x16 = mbmi->txfm_size;
|
2013-04-09 19:54:19 +02:00
|
|
|
if (y_intra16x16_skippable &&
|
2013-04-11 00:55:59 +02:00
|
|
|
((cm->txfm_mode == ONLY_4X4 && uv_intra_skippable[TX_4X4]) ||
|
|
|
|
(cm->txfm_mode != ONLY_4X4 && uv_intra_skippable[TX_8X8]))) {
|
2013-03-07 19:45:25 +01:00
|
|
|
error16x16 -= RDCOST(x->rdmult, x->rddiv, rate16x16_tokenonly, 0);
|
|
|
|
rate16x16 -= rate16x16_tokenonly;
|
|
|
|
}
|
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
txfm_cache[0][i] = error16x16 - txfm_cache[1][cm->txfm_mode] +
|
|
|
|
txfm_cache[1][i];
|
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-03-07 19:45:25 +01:00
|
|
|
error8x8 = rd_pick_intra8x8mby_modes_and_txsz(cpi, x, &rate8x8,
|
|
|
|
&rate8x8_tokenonly,
|
|
|
|
&dist8x8, mode8x8,
|
|
|
|
error16x16, txfm_cache[1]);
|
|
|
|
txfm_size_8x8 = mbmi->txfm_size;
|
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
int64_t tmp_rd = error8x8 - txfm_cache[1][cm->txfm_mode] + txfm_cache[1][i];
|
|
|
|
if (tmp_rd < txfm_cache[0][i])
|
|
|
|
txfm_cache[0][i] = tmp_rd;
|
|
|
|
}
|
Improved coding using 8x8 transform
In summary, this commit encompasses a series of changes in attempt to
improve the 8x8 transform based coding to help overall compression
quality, please refer to the detailed commit history below for what
are the rationale underly the series of changes:
a. A frame level flag to indicate if 8x8 transform is used at all.
b. 8x8 transform is not used for key frames and small image size.
c. On inter coded frame, macroblocks using modes B_PRED, SPLIT_MV
and I8X8_PRED are forced to using 4x4 transform based coding, the
rest uses 8x8 transform based coding.
d. Encoder and decoder has the same assumption on the relationship
between prediction modes and transform size, therefore no signaling
is encoded in bitstream.
e. Mode decision process now calculate the rate and distortion scores
using their respective transforms.
Overall test results:
1. HD set
http://www.corp.google.com/~yaowu/no_crawl/t8x8/HD_t8x8_20120206.html
(avg psnr: 3.09% glb psnr: 3.22%, ssim: 3.90%)
2. Cif set:
http://www.corp.google.com/~yaowu/no_crawl/t8x8/cif_t8x8_20120206.html
(avg psnr: -0.03%, glb psnr: -0.02%, ssim: -0.04%)
It should be noted here, as 8x8 transform coding itself is disabled
for cif size clips, the 0.03% loss is purely from the 1 bit/frame
flag overhead on if 8x8 transform is used or not for the frame.
---patch history for future reference---
Patch 1:
this commit tries to select transform size based on macroblock
prediction mode. If the size of a prediction mode is 16x16, then
the macroblock is forced to use 8x8 transform. If the prediction
mode is B_PRED, SPLITMV or I8X8_PRED, then the macroblock is forced
to use 4x4 transform. Tests on the following HD clips showed mixed
results: (all hd clips only used first 100 frames in the test)
http://www.corp.google.com/~yaowu/no_crawl/t8x8/hdmodebased8x8.html
http://www.corp.google.com/~yaowu/no_crawl/t8x8/hdmodebased8x8_log.html
while the results are mixed and overall negative, it is interesting to
see 8x8 helped a few of the clips.
Patch 2:
this patch tries to hard-wire selection of transform size based on
prediction modes without using segmentation to signal the transform size.
encoder and decoder both takes the same assumption that all macroblocks
use 8x8 transform except when prediciton mode is B_PRED, I8X8_PRED or
SPLITMV. Test results are as follows:
http://www.corp.google.com/~yaowu/no_crawl/t8x8/cifmodebase8x8_0125.html
http://www.corp.google.com/~yaowu/no_crawl/t8x8/hdmodebased8x8_0125log.html
Interestingly, by removing the overhead or coding the segmentation, the
results on this limited HD set have turn positive on average.
Patch 3:
this patch disabled the usage of 8x8 transform on key frames, and kept the
logic from patch 2 for inter frames only. test results on HD set turned
decidedly positive with 8x8 transform enabled on inter frame with 16x16
prediction modes: (avg psnr: .81% glb psnr: .82 ssim: .55%)
http://www.corp.google.com/~yaowu/no_crawl/t8x8/hdintermode8x8_0125.html
results on cif set still negative overall
Patch 4:
continued from last patch, but now in mode decision process, the rate and
distortion estimates are computed based on 8x8 transform results for MBs
with modes associated with 8x8 transform. This patch also fixed a problem
related to segment based eob coding when 8x8 transform is used. The patch
significantly improved the results on HD clips:
http://www.corp.google.com/~yaowu/no_crawl/t8x8/hd8x8RDintermode.html
(avg psnr: 2.70% glb psnr: 2.76% ssim: 3.34%)
results on cif also improved, though they are still negative compared to
baseline that uses 4x4 transform only:
http://www.corp.google.com/~yaowu/no_crawl/t8x8/cif8x8RDintermode.html
(avg psnr: -.78% glb psnr: -.86% ssim: -.19%)
Patch 5:
This patch does 3 things:
a. a bunch of decoder bug fixes, encodings and decodings were verified
to have matched recon buffer on a number of encodes on cif size mobile and
hd version of _pedestrian.
b. the patch further improved the rate distortion calculation of MBS that
use 8x8 transform. This provided some further gain on compression.
c. the patch also got the experimental work SEG_LVL_EOB to work with 8x8
transformed macroblock, test results indicates it improves the cif set
but hurt the HD set slightly.
Tests results on HD clips:
http://www.corp.google.com/~yaowu/no_crawl/t8x8/HD_t8x8_20120201.html
(avg psnr: 3.19% glb psnr: 3.30% ssim: 3.93%)
Test results on cif clips:
http://www.corp.google.com/~yaowu/no_crawl/t8x8/cif_t8x8_20120201.html
(avg psnr: -.47% glb psnr: -.51% ssim: +.28%)
Patch 6:
Added a frame level flag to indicate if 8x8 transform is allowed at all.
temporarily the decision is based on frame size, can be optimized later
one. This get the cif results to basically unchanged, with one bit per
frame overhead on both cif and hd clips.
Patch 8:
Rebase and Merge to head by PGW.
Fixed some suspect 4s that look like hey should be 64s in regard
to segmented EOB. Perhaps #defines would be bette.
Bulit and tested without T8x8 enabled and produces unchanged
output.
Patch 9:
Corrected misalligned code/decode of "txfm_mode" bit.
Limited testing for correct encode and decode with
T8x8 configured on derf clips.
Change-Id: I156e1405d25f81579d579dff8ab9af53944ec49c
2012-02-10 01:12:23 +01:00
|
|
|
|
2013-02-15 23:45:20 +01:00
|
|
|
mbmi->txfm_size = TX_4X4;
|
2012-08-07 01:21:23 +02:00
|
|
|
error4x4 = rd_pick_intra4x4mby_modes(cpi, x,
|
|
|
|
&rate4x4, &rate4x4_tokenonly,
|
2013-01-25 01:28:53 +01:00
|
|
|
&dist4x4, error16x16);
|
2013-03-07 19:45:25 +01:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
if (error4x4 < txfm_cache[0][i])
|
|
|
|
txfm_cache[0][i] = error4x4;
|
|
|
|
}
|
2011-06-08 18:05:05 +02:00
|
|
|
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->mb_skip_coeff = 0;
|
2013-04-09 19:54:19 +02:00
|
|
|
if (y_intra16x16_skippable &&
|
2013-04-11 00:55:59 +02:00
|
|
|
((cm->txfm_mode == ONLY_4X4 && uv_intra_skippable[TX_4X4]) ||
|
|
|
|
(cm->txfm_mode != ONLY_4X4 && uv_intra_skippable[TX_8X8]))) {
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->mb_skip_coeff = 1;
|
2012-10-16 21:41:56 +02:00
|
|
|
mbmi->mode = mode16x16;
|
2013-04-11 00:55:59 +02:00
|
|
|
mbmi->uv_mode = modeuv[cm->txfm_mode != ONLY_4X4];
|
2013-03-07 19:45:25 +01:00
|
|
|
rate = rate16x16 + vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_MBSKIP), 1);
|
|
|
|
dist = dist16x16;
|
2013-04-11 00:55:59 +02:00
|
|
|
rate += rateuv[cm->txfm_mode != ONLY_4X4] -
|
|
|
|
rateuv_tokenonly[cm->txfm_mode != ONLY_4X4];
|
|
|
|
dist += (distuv[cm->txfm_mode != ONLY_4X4] >> 2);
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = txfm_size_16x16;
|
2012-08-29 19:43:20 +02:00
|
|
|
} else if (error8x8 > error16x16) {
|
2012-07-14 00:21:29 +02:00
|
|
|
if (error4x4 < error16x16) {
|
2013-04-11 00:55:59 +02:00
|
|
|
rate = rateuv[TX_4X4] + rate4x4;
|
2013-04-11 21:16:35 +02:00
|
|
|
mbmi->mode = I4X4_PRED;
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = TX_4X4;
|
2013-04-11 00:55:59 +02:00
|
|
|
dist = dist4x4 + (distuv[TX_4X4] >> 2);
|
|
|
|
mbmi->uv_mode = modeuv[TX_4X4];
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = txfm_size_16x16;
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->mode = mode16x16;
|
2013-04-11 00:55:59 +02:00
|
|
|
rate = rate16x16 + rateuv[mbmi->txfm_size != TX_4X4];
|
|
|
|
dist = dist16x16 + (distuv[mbmi->txfm_size != TX_4X4] >> 2);
|
|
|
|
mbmi->uv_mode = modeuv[mbmi->txfm_size != TX_4X4];
|
2011-08-05 01:30:27 +02:00
|
|
|
}
|
2013-04-09 19:54:19 +02:00
|
|
|
rate += vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_MBSKIP), 0);
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
|
|
|
if (error4x4 < error8x8) {
|
2013-04-11 00:55:59 +02:00
|
|
|
rate = rateuv[TX_4X4] + rate4x4;
|
2013-04-11 21:16:35 +02:00
|
|
|
mbmi->mode = I4X4_PRED;
|
2012-10-09 18:18:21 +02:00
|
|
|
mbmi->txfm_size = TX_4X4;
|
2013-04-11 00:55:59 +02:00
|
|
|
dist = dist4x4 + (distuv[TX_4X4] >> 2);
|
|
|
|
mbmi->uv_mode = modeuv[TX_4X4];
|
2012-07-14 00:21:29 +02:00
|
|
|
} else {
|
2012-08-10 15:12:43 +02:00
|
|
|
mbmi->mode = I8X8_PRED;
|
2013-03-07 19:45:25 +01:00
|
|
|
mbmi->txfm_size = txfm_size_8x8;
|
2012-07-14 00:21:29 +02:00
|
|
|
set_i8x8_block_modes(x, mode8x8);
|
2013-04-11 00:55:59 +02:00
|
|
|
rate = rate8x8 + rateuv[TX_4X4];
|
|
|
|
dist = dist8x8 + (distuv[TX_4X4] >> 2);
|
2011-08-05 01:30:27 +02:00
|
|
|
}
|
2013-04-09 19:54:19 +02:00
|
|
|
rate += vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_MBSKIP), 0);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-03-07 19:45:25 +01:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].txfm_rd_diff[i] =
|
|
|
|
txfm_cache[0][cm->txfm_mode] - txfm_cache[0][i];
|
|
|
|
}
|
|
|
|
|
2012-08-29 19:43:20 +02:00
|
|
|
*returnrate = rate;
|
|
|
|
*returndist = dist;
|
2012-04-07 01:38:34 +02:00
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif
|
2012-04-07 01:38:34 +02:00
|
|
|
|
2013-04-11 00:55:59 +02:00
|
|
|
int64_t vp9_rd_pick_inter_mode_sb(VP9_COMP *cpi, MACROBLOCK *x,
|
2013-04-26 20:57:17 +02:00
|
|
|
int mi_row, int mi_col,
|
2013-04-11 00:55:59 +02:00
|
|
|
int *returnrate,
|
|
|
|
int *returndistortion,
|
2013-04-11 18:33:49 +02:00
|
|
|
BLOCK_SIZE_TYPE bsize,
|
|
|
|
PICK_MODE_CONTEXT *ctx) {
|
2013-04-13 19:01:46 +02:00
|
|
|
const enum BlockSize block_size = y_bsizet_to_block_size(bsize);
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-08-20 23:43:34 +02:00
|
|
|
MACROBLOCKD *xd = &x->e_mbd;
|
2012-10-30 01:58:18 +01:00
|
|
|
MB_MODE_INFO *mbmi = &xd->mode_info_context->mbmi;
|
2012-08-20 23:43:34 +02:00
|
|
|
MB_PREDICTION_MODE this_mode;
|
2012-12-20 23:56:19 +01:00
|
|
|
MB_PREDICTION_MODE best_mode = DC_PRED;
|
2013-04-20 04:16:14 +02:00
|
|
|
MV_REFERENCE_FRAME ref_frame, second_ref;
|
2012-08-20 23:43:34 +02:00
|
|
|
unsigned char segment_id = xd->mode_info_context->mbmi.segment_id;
|
2012-11-08 20:03:00 +01:00
|
|
|
int comp_pred, i;
|
2012-10-30 01:58:18 +01:00
|
|
|
int_mv frame_mv[MB_MODE_COUNT][MAX_REF_FRAMES];
|
2012-08-20 23:43:34 +02:00
|
|
|
int frame_mdcounts[4][4];
|
2013-02-07 19:09:05 +01:00
|
|
|
YV12_BUFFER_CONFIG yv12_mb[4];
|
2012-10-31 22:40:53 +01:00
|
|
|
static const int flag_list[4] = { 0, VP9_LAST_FLAG, VP9_GOLD_FLAG,
|
|
|
|
VP9_ALT_FLAG };
|
2013-01-15 22:49:44 +01:00
|
|
|
int idx_list[4] = {0,
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
cpi->lst_fb_idx,
|
|
|
|
cpi->gld_fb_idx,
|
|
|
|
cpi->alt_fb_idx};
|
2012-08-20 23:43:34 +02:00
|
|
|
int mdcounts[4];
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_rd = INT64_MAX;
|
2012-11-08 20:03:00 +01:00
|
|
|
int64_t best_txfm_rd[NB_TXFM_MODES];
|
|
|
|
int64_t best_txfm_diff[NB_TXFM_MODES];
|
|
|
|
int64_t best_pred_diff[NB_PREDICTION_TYPES];
|
|
|
|
int64_t best_pred_rd[NB_PREDICTION_TYPES];
|
2012-08-20 23:43:34 +02:00
|
|
|
MB_MODE_INFO best_mbmode;
|
2013-04-19 17:04:57 +02:00
|
|
|
int j;
|
2012-11-30 16:29:43 +01:00
|
|
|
int mode_index, best_mode_index = 0;
|
2012-08-20 23:43:34 +02:00
|
|
|
unsigned int ref_costs[MAX_REF_FRAMES];
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
int is_best_interintra = 0;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_intra16_rd = INT64_MAX;
|
2013-02-20 19:16:24 +01:00
|
|
|
int best_intra16_mode = DC_PRED;
|
|
|
|
#if SEPARATE_INTERINTRA_UV
|
|
|
|
int best_intra16_uv_mode = DC_PRED;
|
|
|
|
#endif
|
2012-11-07 15:50:25 +01:00
|
|
|
#endif
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t best_overall_rd = INT64_MAX;
|
2012-12-20 23:56:19 +01:00
|
|
|
INTERPOLATIONFILTERTYPE best_filter = SWITCHABLE;
|
2013-02-12 02:08:52 +01:00
|
|
|
INTERPOLATIONFILTERTYPE tmp_best_filter = SWITCHABLE;
|
2013-04-11 00:55:59 +02:00
|
|
|
int rate_uv_intra[TX_SIZE_MAX_SB], rate_uv_tokenonly[TX_SIZE_MAX_SB];
|
|
|
|
int dist_uv[TX_SIZE_MAX_SB], skip_uv[TX_SIZE_MAX_SB];
|
|
|
|
MB_PREDICTION_MODE mode_uv[TX_SIZE_MAX_SB];
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
struct scale_factors scale_factor[4];
|
2013-04-19 17:04:57 +02:00
|
|
|
unsigned int ref_frame_mask = 0;
|
|
|
|
unsigned int mode_mask = 0;
|
2013-04-23 00:42:41 +02:00
|
|
|
int64_t mode_distortions[MB_MODE_COUNT] = {-1};
|
|
|
|
int64_t frame_distortions[MAX_REF_FRAMES] = {-1};
|
2013-05-01 01:13:20 +02:00
|
|
|
int intra_cost_penalty = 20 * vp9_dc_quant(cpi->common.base_qindex,
|
|
|
|
cpi->common.y_dc_delta_q);
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
int_mv seg_mvs[4][MAX_REF_FRAMES - 1];
|
|
|
|
#endif
|
2013-04-23 00:42:41 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0; j < MAX_REF_FRAMES - 1; j++)
|
|
|
|
seg_mvs[i][j].as_int = INVALID_MV;
|
|
|
|
}
|
|
|
|
#endif
|
2013-04-23 00:42:41 +02:00
|
|
|
// Everywhere the flag is set the error is much higher than its neighbors.
|
|
|
|
ctx->frames_with_high_error = 0;
|
|
|
|
ctx->modes_with_high_error = 0;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
|
|
|
xd->mode_info_context->mbmi.segment_id = segment_id;
|
2012-10-31 22:40:53 +01:00
|
|
|
estimate_ref_frame_costs(cpi, segment_id, ref_costs);
|
2012-10-30 01:58:18 +01:00
|
|
|
vpx_memset(&best_mbmode, 0, sizeof(best_mbmode));
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-11-08 20:03:00 +01:00
|
|
|
for (i = 0; i < NB_PREDICTION_TYPES; ++i)
|
2013-01-14 20:49:30 +01:00
|
|
|
best_pred_rd[i] = INT64_MAX;
|
2012-11-08 20:03:00 +01:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++)
|
2013-01-14 20:49:30 +01:00
|
|
|
best_txfm_rd[i] = INT64_MAX;
|
2012-11-08 20:03:00 +01:00
|
|
|
|
2013-04-23 00:42:41 +02:00
|
|
|
// Create a mask set to 1 for each frame used by a smaller resolution.
|
2013-04-19 17:04:57 +02:00
|
|
|
if (cpi->Speed > 0) {
|
|
|
|
switch (block_size) {
|
|
|
|
case BLOCK_64X64:
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
for (j = 0; j < 4; j++) {
|
2013-04-23 00:42:41 +02:00
|
|
|
ref_frame_mask |= x->mb_context[i][j].frames_with_high_error;
|
|
|
|
mode_mask |= x->mb_context[i][j].modes_with_high_error;
|
2013-04-19 17:04:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = 0; i < 4; i++) {
|
2013-04-23 00:42:41 +02:00
|
|
|
ref_frame_mask |= x->sb32_context[i].frames_with_high_error;
|
|
|
|
mode_mask |= x->sb32_context[i].modes_with_high_error;
|
2013-04-19 17:04:57 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BLOCK_32X32:
|
|
|
|
for (i = 0; i < 4; i++) {
|
2013-04-23 00:42:41 +02:00
|
|
|
ref_frame_mask |=
|
|
|
|
x->mb_context[xd->sb_index][i].frames_with_high_error;
|
|
|
|
mode_mask |= x->mb_context[xd->sb_index][i].modes_with_high_error;
|
2013-04-19 17:04:57 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2013-04-19 18:48:56 +02:00
|
|
|
// Until we handle all block sizes set it to present;
|
2013-04-23 00:42:41 +02:00
|
|
|
ref_frame_mask = 0;
|
|
|
|
mode_mask = 0;
|
2013-04-19 17:04:57 +02:00
|
|
|
break;
|
|
|
|
}
|
2013-04-23 00:42:41 +02:00
|
|
|
ref_frame_mask = ~ref_frame_mask;
|
|
|
|
mode_mask = ~mode_mask;
|
2013-04-19 17:04:57 +02:00
|
|
|
}
|
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
for (ref_frame = LAST_FRAME; ref_frame <= ALTREF_FRAME; ref_frame++) {
|
|
|
|
if (cpi->ref_frame_flags & flag_list[ref_frame]) {
|
2013-01-06 03:20:25 +01:00
|
|
|
setup_buffer_inter(cpi, x, idx_list[ref_frame], ref_frame, block_size,
|
2013-04-26 20:57:17 +02:00
|
|
|
mi_row, mi_col, frame_mv[NEARESTMV], frame_mv[NEARMV],
|
2013-04-19 17:04:57 +02:00
|
|
|
frame_mdcounts, yv12_mb, scale_factor);
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
2012-10-30 01:58:18 +01:00
|
|
|
frame_mv[NEWMV][ref_frame].as_int = INVALID_MV;
|
|
|
|
frame_mv[ZEROMV][ref_frame].as_int = 0;
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
2013-04-19 17:04:57 +02:00
|
|
|
if (cpi->Speed == 0
|
|
|
|
|| (cpi->Speed > 0 && (ref_frame_mask & (1 << INTRA_FRAME)))) {
|
|
|
|
mbmi->mode = DC_PRED;
|
|
|
|
for (i = 0; i <= ((bsize < BLOCK_SIZE_SB64X64) ? TX_16X16 : TX_32X32);
|
|
|
|
i++) {
|
|
|
|
mbmi->txfm_size = i;
|
|
|
|
rd_pick_intra_sbuv_mode(cpi, x, &rate_uv_intra[i], &rate_uv_tokenonly[i],
|
|
|
|
&dist_uv[i], &skip_uv[i], bsize);
|
|
|
|
mode_uv[i] = mbmi->uv_mode;
|
|
|
|
}
|
32x32 transform for superblocks.
This adds Debargha's DCT/DWT hybrid and a regular 32x32 DCT, and adds
code all over the place to wrap that in the bitstream/encoder/decoder/RD.
Some implementation notes (these probably need careful review):
- token range is extended by 1 bit, since the value range out of this
transform is [-16384,16383].
- the coefficients coming out of the FDCT are manually scaled back by
1 bit, or else they won't fit in int16_t (they are 17 bits). Because
of this, the RD error scoring does not right-shift the MSE score by
two (unlike for 4x4/8x8/16x16).
- to compensate for this loss in precision, the quantizer is halved
also. This is currently a little hacky.
- FDCT and IDCT is double-only right now. Needs a fixed-point impl.
- There are no default probabilities for the 32x32 transform yet; I'm
simply using the 16x16 luma ones. A future commit will add newly
generated probabilities for all transforms.
- No ADST version. I don't think we'll add one for this level; if an
ADST is desired, transform-size selection can scale back to 16x16
or lower, and use an ADST at that level.
Additional notes specific to Debargha's DWT/DCT hybrid:
- coefficient scale is different for the top/left 16x16 (DCT-over-DWT)
block than for the rest (DWT pixel differences) of the block. Therefore,
RD error scoring isn't easily scalable between coefficient and pixel
domain. Thus, unfortunately, we need to compute the RD distortion in
the pixel domain until we figure out how to scale these appropriately.
Change-Id: I00386f20f35d7fabb19aba94c8162f8aee64ef2b
2012-12-07 23:45:05 +01:00
|
|
|
}
|
2012-11-16 00:50:07 +01:00
|
|
|
|
2013-02-12 02:08:52 +01:00
|
|
|
for (mode_index = 0; mode_index < MAX_MODES; ++mode_index) {
|
2012-11-16 00:50:07 +01:00
|
|
|
int mode_excluded = 0;
|
2013-01-14 20:49:30 +01:00
|
|
|
int64_t this_rd = INT64_MAX;
|
2012-08-20 23:43:34 +02:00
|
|
|
int disable_skip = 0;
|
|
|
|
int other_cost = 0;
|
|
|
|
int compmode_cost = 0;
|
2012-10-30 01:58:18 +01:00
|
|
|
int rate2 = 0, rate_y = 0, rate_uv = 0;
|
|
|
|
int distortion2 = 0, distortion_y = 0, distortion_uv = 0;
|
|
|
|
int skippable;
|
|
|
|
int64_t txfm_cache[NB_TXFM_MODES];
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
int compmode_interintra_cost = 0;
|
|
|
|
#endif
|
2012-08-20 23:43:34 +02:00
|
|
|
|
|
|
|
// Test best rd so far against threshold for trying this mode.
|
2012-11-24 17:19:04 +01:00
|
|
|
if (best_rd <= cpi->rd_threshes[mode_index] ||
|
|
|
|
cpi->rd_threshes[mode_index] == INT_MAX) {
|
2012-08-20 23:43:34 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-01-30 06:42:46 +01:00
|
|
|
x->skip = 0;
|
2012-10-31 01:12:12 +01:00
|
|
|
this_mode = vp9_mode_order[mode_index].mode;
|
|
|
|
ref_frame = vp9_mode_order[mode_index].ref_frame;
|
2013-04-19 17:04:57 +02:00
|
|
|
if (!(ref_frame == INTRA_FRAME
|
|
|
|
|| (cpi->ref_frame_flags & flag_list[ref_frame]))) {
|
2012-11-16 00:14:38 +01:00
|
|
|
continue;
|
|
|
|
}
|
2013-04-19 17:04:57 +02:00
|
|
|
if (cpi->Speed > 0) {
|
|
|
|
if (!(ref_frame_mask & (1 << ref_frame))) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-04-23 00:42:41 +02:00
|
|
|
if (!(mode_mask & (1 << this_mode))) {
|
|
|
|
continue;
|
|
|
|
}
|
2013-04-19 17:04:57 +02:00
|
|
|
if (vp9_mode_order[mode_index].second_ref_frame != NONE
|
|
|
|
&& !(ref_frame_mask
|
|
|
|
& (1 << vp9_mode_order[mode_index].second_ref_frame))) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-30 01:58:18 +01:00
|
|
|
mbmi->ref_frame = ref_frame;
|
2012-11-30 20:46:20 +01:00
|
|
|
mbmi->second_ref_frame = vp9_mode_order[mode_index].second_ref_frame;
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
set_scale_factors(xd, mbmi->ref_frame, mbmi->second_ref_frame,
|
|
|
|
scale_factor);
|
2012-11-30 20:46:20 +01:00
|
|
|
comp_pred = mbmi->second_ref_frame > INTRA_FRAME;
|
2012-10-30 01:58:18 +01:00
|
|
|
mbmi->mode = this_mode;
|
|
|
|
mbmi->uv_mode = DC_PRED;
|
2012-11-30 20:46:20 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
mbmi->interintra_mode = (MB_PREDICTION_MODE)(DC_PRED - 1);
|
|
|
|
mbmi->interintra_uv_mode = (MB_PREDICTION_MODE)(DC_PRED - 1);
|
2012-08-20 23:43:34 +02:00
|
|
|
#endif
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
// Evaluate all sub-pel filters irrespective of whether we can use
|
|
|
|
// them for this frame.
|
2013-02-12 02:08:52 +01:00
|
|
|
mbmi->interp_filter = cm->mcomp_filter_type;
|
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-11-30 20:46:20 +01:00
|
|
|
// if (!(cpi->ref_frame_flags & flag_list[ref_frame]))
|
|
|
|
// continue;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
if (
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
bsize != BLOCK_SIZE_SB8X8 &&
|
|
|
|
(this_mode == I4X4_PRED || this_mode == SPLITMV)
|
|
|
|
#else
|
2013-04-11 21:16:35 +02:00
|
|
|
this_mode == I4X4_PRED ||
|
2013-05-01 01:13:20 +02:00
|
|
|
this_mode == I8X8_PRED ||
|
|
|
|
this_mode == SPLITMV
|
|
|
|
#endif
|
|
|
|
)
|
2012-08-20 23:43:34 +02:00
|
|
|
continue;
|
2012-11-30 20:46:20 +01:00
|
|
|
// if (vp9_mode_order[mode_index].second_ref_frame == INTRA_FRAME)
|
|
|
|
// continue;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
|
|
|
if (comp_pred) {
|
|
|
|
if (ref_frame == ALTREF_FRAME) {
|
|
|
|
second_ref = LAST_FRAME;
|
|
|
|
} else {
|
|
|
|
second_ref = ref_frame + 1;
|
|
|
|
}
|
|
|
|
if (!(cpi->ref_frame_flags & flag_list[second_ref]))
|
|
|
|
continue;
|
2012-10-30 01:58:18 +01:00
|
|
|
mbmi->second_ref_frame = second_ref;
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
set_scale_factors(xd, mbmi->ref_frame, mbmi->second_ref_frame,
|
|
|
|
scale_factor);
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
mode_excluded =
|
|
|
|
mode_excluded ?
|
|
|
|
mode_excluded : cm->comp_pred_mode == SINGLE_PREDICTION_ONLY;
|
2012-08-20 23:43:34 +02:00
|
|
|
} else {
|
2012-11-30 20:46:20 +01:00
|
|
|
// mbmi->second_ref_frame = vp9_mode_order[mode_index].second_ref_frame;
|
|
|
|
if (ref_frame != INTRA_FRAME) {
|
|
|
|
if (mbmi->second_ref_frame != INTRA_FRAME)
|
2012-12-20 23:56:19 +01:00
|
|
|
mode_excluded =
|
|
|
|
mode_excluded ?
|
|
|
|
mode_excluded : cm->comp_pred_mode == COMP_PREDICTION_ONLY;
|
2012-11-30 20:46:20 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
else
|
2012-12-20 23:56:19 +01:00
|
|
|
mode_excluded = mode_excluded ? mode_excluded : !cm->use_interintra;
|
2012-11-30 20:46:20 +01:00
|
|
|
#endif
|
|
|
|
}
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
|
2013-04-20 04:16:14 +02:00
|
|
|
setup_pre_planes(xd, &yv12_mb[ref_frame],
|
|
|
|
comp_pred ? &yv12_mb[second_ref] : NULL, 0, 0, NULL, NULL);
|
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
vpx_memcpy(mdcounts, frame_mdcounts[ref_frame], sizeof(mdcounts));
|
|
|
|
|
|
|
|
// If the segment reference frame feature is enabled....
|
|
|
|
// then do nothing if the current ref frame is not allowed..
|
2012-10-30 06:15:27 +01:00
|
|
|
if (vp9_segfeature_active(xd, segment_id, SEG_LVL_REF_FRAME) &&
|
|
|
|
!vp9_check_segref(xd, segment_id, ref_frame)) {
|
2012-08-20 23:43:34 +02:00
|
|
|
continue;
|
2013-01-28 16:22:53 +01:00
|
|
|
// If the segment skip feature is enabled....
|
2012-08-20 23:43:34 +02:00
|
|
|
// then do nothing if the current mode is not allowed..
|
2013-01-28 16:22:53 +01:00
|
|
|
} else if (vp9_segfeature_active(xd, segment_id, SEG_LVL_SKIP) &&
|
|
|
|
(this_mode != ZEROMV)) {
|
2012-08-20 23:43:34 +02:00
|
|
|
continue;
|
2013-01-28 16:22:53 +01:00
|
|
|
// Disable this drop out case if the ref frame
|
2012-08-20 23:43:34 +02:00
|
|
|
// segment level feature is enabled for this segment. This is to
|
|
|
|
// prevent the possibility that we end up unable to pick any mode.
|
2013-01-28 16:22:53 +01:00
|
|
|
} else if (!vp9_segfeature_active(xd, segment_id, SEG_LVL_REF_FRAME)) {
|
2012-08-20 23:43:34 +02:00
|
|
|
// Only consider ZEROMV/ALTREF_FRAME for alt ref frame,
|
|
|
|
// unless ARNR filtering is enabled in which case we want
|
|
|
|
// an unfiltered alternative
|
|
|
|
if (cpi->is_src_frame_alt_ref && (cpi->oxcf.arnr_max_frames == 0)) {
|
|
|
|
if (this_mode != ZEROMV || ref_frame != ALTREF_FRAME) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
if (this_mode == I4X4_PRED) {
|
|
|
|
int rate;
|
|
|
|
|
|
|
|
// Note the rate value returned here includes the cost of coding
|
|
|
|
// the I4X4_PRED mode : x->mbmode_cost[xd->frame_type][I4X4_PRED];
|
|
|
|
assert(bsize == BLOCK_SIZE_SB8X8);
|
|
|
|
mbmi->txfm_size = TX_4X4;
|
|
|
|
rd_pick_intra4x4mby_modes(cpi, x, &rate, &rate_y,
|
|
|
|
&distortion_y, INT64_MAX);
|
|
|
|
rate2 += rate;
|
|
|
|
rate2 += intra_cost_penalty;
|
|
|
|
distortion2 += distortion_y;
|
|
|
|
|
|
|
|
rate2 += rate_uv_intra[TX_4X4];
|
|
|
|
rate_uv = rate_uv_intra[TX_4X4];
|
|
|
|
distortion2 += dist_uv[TX_4X4];
|
|
|
|
distortion_uv = dist_uv[TX_4X4];
|
|
|
|
mbmi->uv_mode = mode_uv[TX_4X4];
|
|
|
|
} else
|
|
|
|
#endif
|
2012-11-16 00:50:07 +01:00
|
|
|
if (ref_frame == INTRA_FRAME) {
|
2013-04-11 00:55:59 +02:00
|
|
|
TX_SIZE uv_tx;
|
2013-04-11 21:12:11 +02:00
|
|
|
vp9_build_intra_predictors_sby_s(xd, bsize);
|
2013-04-11 00:55:59 +02:00
|
|
|
super_block_yrd(cpi, x, &rate_y, &distortion_y, &skippable,
|
|
|
|
bsize, txfm_cache);
|
|
|
|
|
|
|
|
uv_tx = mbmi->txfm_size;
|
|
|
|
if (bsize < BLOCK_SIZE_SB32X32 && uv_tx == TX_16X16)
|
|
|
|
uv_tx = TX_8X8;
|
|
|
|
else if (bsize < BLOCK_SIZE_SB64X64 && uv_tx == TX_32X32)
|
|
|
|
uv_tx = TX_16X16;
|
|
|
|
|
|
|
|
rate_uv = rate_uv_intra[uv_tx];
|
|
|
|
distortion_uv = dist_uv[uv_tx];
|
|
|
|
skippable = skippable && skip_uv[uv_tx];
|
|
|
|
mbmi->uv_mode = mode_uv[uv_tx];
|
2012-11-16 00:50:07 +01:00
|
|
|
|
|
|
|
rate2 = rate_y + x->mbmode_cost[cm->frame_type][mbmi->mode] + rate_uv;
|
2013-05-01 01:13:20 +02:00
|
|
|
if (mbmi->mode != DC_PRED && mbmi->mode != TM_PRED)
|
|
|
|
rate2 += intra_cost_penalty;
|
2012-11-16 00:50:07 +01:00
|
|
|
distortion2 = distortion_y + distortion_uv;
|
2013-05-01 01:13:20 +02:00
|
|
|
#if CONFIG_SB8X8
|
|
|
|
} else if (this_mode == SPLITMV) {
|
|
|
|
const int is_comp_pred = mbmi->second_ref_frame > 0;
|
|
|
|
int rate, distortion;
|
|
|
|
int64_t this_rd_thresh;
|
|
|
|
int64_t tmp_rd, tmp_best_rd = INT64_MAX, tmp_best_rdu = INT64_MAX;
|
|
|
|
int tmp_best_rate = INT_MAX, tmp_best_ratey = INT_MAX;
|
|
|
|
int tmp_best_distortion = INT_MAX, tmp_best_skippable = 0;
|
|
|
|
int switchable_filter_index;
|
|
|
|
int_mv *second_ref = is_comp_pred ?
|
|
|
|
&mbmi->ref_mvs[mbmi->second_ref_frame][0] : NULL;
|
|
|
|
union b_mode_info tmp_best_bmodes[16];
|
|
|
|
MB_MODE_INFO tmp_best_mbmode;
|
|
|
|
PARTITION_INFO tmp_best_partition;
|
|
|
|
int pred_exists = 0;
|
|
|
|
int uv_skippable;
|
|
|
|
|
|
|
|
this_rd_thresh = (mbmi->ref_frame == LAST_FRAME) ?
|
|
|
|
cpi->rd_threshes[THR_NEWMV] : cpi->rd_threshes[THR_NEWA];
|
|
|
|
this_rd_thresh = (mbmi->ref_frame == GOLDEN_FRAME) ?
|
|
|
|
cpi->rd_threshes[THR_NEWG] : this_rd_thresh;
|
|
|
|
xd->mode_info_context->mbmi.txfm_size = TX_4X4;
|
|
|
|
|
|
|
|
for (switchable_filter_index = 0;
|
|
|
|
switchable_filter_index < VP9_SWITCHABLE_FILTERS;
|
|
|
|
++switchable_filter_index) {
|
|
|
|
int newbest;
|
|
|
|
mbmi->interp_filter =
|
|
|
|
vp9_switchable_interp[switchable_filter_index];
|
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
|
|
|
|
|
|
|
tmp_rd = rd_pick_best_mbsegmentation(cpi, x,
|
|
|
|
&mbmi->ref_mvs[mbmi->ref_frame][0],
|
|
|
|
second_ref, INT64_MAX, mdcounts,
|
|
|
|
&rate, &rate_y, &distortion,
|
|
|
|
&skippable,
|
|
|
|
(int)this_rd_thresh, seg_mvs);
|
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE) {
|
|
|
|
int rs = SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs
|
|
|
|
[vp9_get_pred_context(&cpi->common, xd,
|
|
|
|
PRED_SWITCHABLE_INTERP)]
|
|
|
|
[vp9_switchable_interp_map[mbmi->interp_filter]];
|
|
|
|
tmp_rd += RDCOST(x->rdmult, x->rddiv, rs, 0);
|
|
|
|
}
|
|
|
|
newbest = (tmp_rd < tmp_best_rd);
|
|
|
|
if (newbest) {
|
|
|
|
tmp_best_filter = mbmi->interp_filter;
|
|
|
|
tmp_best_rd = tmp_rd;
|
|
|
|
}
|
|
|
|
if ((newbest && cm->mcomp_filter_type == SWITCHABLE) ||
|
|
|
|
(mbmi->interp_filter == cm->mcomp_filter_type &&
|
|
|
|
cm->mcomp_filter_type != SWITCHABLE)) {
|
|
|
|
tmp_best_rdu = tmp_rd;
|
|
|
|
tmp_best_rate = rate;
|
|
|
|
tmp_best_ratey = rate_y;
|
|
|
|
tmp_best_distortion = distortion;
|
|
|
|
tmp_best_skippable = skippable;
|
|
|
|
vpx_memcpy(&tmp_best_mbmode, mbmi, sizeof(MB_MODE_INFO));
|
|
|
|
vpx_memcpy(&tmp_best_partition, x->partition_info,
|
|
|
|
sizeof(PARTITION_INFO));
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
tmp_best_bmodes[i] = xd->mode_info_context->bmi[i];
|
|
|
|
}
|
|
|
|
pred_exists = 1;
|
|
|
|
}
|
|
|
|
} // switchable_filter_index loop
|
|
|
|
|
|
|
|
mbmi->interp_filter = (cm->mcomp_filter_type == SWITCHABLE ?
|
|
|
|
tmp_best_filter : cm->mcomp_filter_type);
|
|
|
|
vp9_setup_interp_filters(xd, mbmi->interp_filter, &cpi->common);
|
|
|
|
if (!pred_exists) {
|
|
|
|
// Handles the special case when a filter that is not in the
|
|
|
|
// switchable list (bilinear, 6-tap) is indicated at the frame level
|
|
|
|
tmp_rd = rd_pick_best_mbsegmentation(cpi, x,
|
|
|
|
&mbmi->ref_mvs[mbmi->ref_frame][0],
|
|
|
|
second_ref, INT64_MAX, mdcounts,
|
|
|
|
&rate, &rate_y, &distortion,
|
|
|
|
&skippable,
|
|
|
|
(int)this_rd_thresh, seg_mvs);
|
|
|
|
} else {
|
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE) {
|
|
|
|
int rs = SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs
|
|
|
|
[vp9_get_pred_context(&cpi->common, xd,
|
|
|
|
PRED_SWITCHABLE_INTERP)]
|
|
|
|
[vp9_switchable_interp_map[mbmi->interp_filter]];
|
|
|
|
tmp_best_rdu -= RDCOST(x->rdmult, x->rddiv, rs, 0);
|
|
|
|
}
|
|
|
|
tmp_rd = tmp_best_rdu;
|
|
|
|
rate = tmp_best_rate;
|
|
|
|
rate_y = tmp_best_ratey;
|
|
|
|
distortion = tmp_best_distortion;
|
|
|
|
skippable = tmp_best_skippable;
|
|
|
|
vpx_memcpy(mbmi, &tmp_best_mbmode, sizeof(MB_MODE_INFO));
|
|
|
|
vpx_memcpy(x->partition_info, &tmp_best_partition,
|
|
|
|
sizeof(PARTITION_INFO));
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
xd->mode_info_context->bmi[i] = tmp_best_bmodes[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rate2 += rate;
|
|
|
|
distortion2 += distortion;
|
|
|
|
|
|
|
|
if (cpi->common.mcomp_filter_type == SWITCHABLE)
|
|
|
|
rate2 += SWITCHABLE_INTERP_RATE_FACTOR * x->switchable_interp_costs
|
|
|
|
[vp9_get_pred_context(&cpi->common, xd, PRED_SWITCHABLE_INTERP)]
|
|
|
|
[vp9_switchable_interp_map[mbmi->interp_filter]];
|
|
|
|
|
|
|
|
// If even the 'Y' rd value of split is higher than best so far
|
|
|
|
// then dont bother looking at UV
|
|
|
|
vp9_build_inter_predictors_sbuv(&x->e_mbd, mi_row, mi_col,
|
|
|
|
bsize);
|
|
|
|
vp9_subtract_sbuv(x, bsize);
|
|
|
|
super_block_uvrd_4x4(cm, x, &rate_uv, &distortion_uv,
|
|
|
|
&uv_skippable, bsize);
|
|
|
|
rate2 += rate_uv;
|
|
|
|
distortion2 += distortion_uv;
|
|
|
|
skippable = skippable && uv_skippable;
|
|
|
|
|
|
|
|
if (!mode_excluded) {
|
|
|
|
if (is_comp_pred)
|
|
|
|
mode_excluded = cpi->common.comp_pred_mode == SINGLE_PREDICTION_ONLY;
|
|
|
|
else
|
|
|
|
mode_excluded = cpi->common.comp_pred_mode == COMP_PREDICTION_ONLY;
|
|
|
|
}
|
|
|
|
|
|
|
|
compmode_cost =
|
|
|
|
vp9_cost_bit(vp9_get_pred_prob(cm, xd, PRED_COMP), is_comp_pred);
|
|
|
|
mbmi->mode = this_mode;
|
|
|
|
#endif
|
2012-11-16 00:50:07 +01:00
|
|
|
} else {
|
2013-02-20 21:34:31 +01:00
|
|
|
YV12_BUFFER_CONFIG *scaled_ref_frame = NULL;
|
|
|
|
int fb;
|
|
|
|
|
|
|
|
if (mbmi->ref_frame == LAST_FRAME) {
|
|
|
|
fb = cpi->lst_fb_idx;
|
|
|
|
} else if (mbmi->ref_frame == GOLDEN_FRAME) {
|
|
|
|
fb = cpi->gld_fb_idx;
|
|
|
|
} else {
|
|
|
|
fb = cpi->alt_fb_idx;
|
|
|
|
}
|
|
|
|
|
2013-02-22 20:22:03 +01:00
|
|
|
if (cpi->scaled_ref_idx[fb] != cm->ref_frame_map[fb])
|
2013-02-20 21:34:31 +01:00
|
|
|
scaled_ref_frame = &cm->yv12_fb[cpi->scaled_ref_idx[fb]];
|
|
|
|
|
2012-11-30 20:46:20 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if (mbmi->second_ref_frame == INTRA_FRAME) {
|
|
|
|
if (best_intra16_mode == DC_PRED - 1) continue;
|
|
|
|
mbmi->interintra_mode = best_intra16_mode;
|
|
|
|
#if SEPARATE_INTERINTRA_UV
|
|
|
|
mbmi->interintra_uv_mode = best_intra16_uv_mode;
|
|
|
|
#else
|
|
|
|
mbmi->interintra_uv_mode = best_intra16_mode;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
2013-04-11 00:55:59 +02:00
|
|
|
this_rd = handle_inter_mode(cpi, x, bsize,
|
2013-04-18 20:55:46 +02:00
|
|
|
mdcounts, txfm_cache,
|
2012-11-16 00:50:07 +01:00
|
|
|
&rate2, &distortion2, &skippable,
|
|
|
|
&compmode_cost,
|
2012-11-07 15:50:25 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
2012-11-16 00:50:07 +01:00
|
|
|
&compmode_interintra_cost,
|
2012-11-07 15:50:25 +01:00
|
|
|
#endif
|
2012-11-16 00:50:07 +01:00
|
|
|
&rate_y, &distortion_y,
|
|
|
|
&rate_uv, &distortion_uv,
|
2013-02-07 19:09:05 +01:00
|
|
|
&mode_excluded, &disable_skip,
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
mode_index, &tmp_best_filter, frame_mv,
|
2013-04-26 20:57:17 +02:00
|
|
|
scaled_ref_frame, mi_row, mi_col);
|
2013-01-14 20:49:30 +01:00
|
|
|
if (this_rd == INT64_MAX)
|
2012-11-16 00:50:07 +01:00
|
|
|
continue;
|
|
|
|
}
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-11-30 20:46:20 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if (cpi->common.use_interintra) {
|
|
|
|
rate2 += compmode_interintra_cost;
|
|
|
|
}
|
|
|
|
#endif
|
2012-08-20 23:43:34 +02:00
|
|
|
if (cpi->common.comp_pred_mode == HYBRID_PREDICTION) {
|
|
|
|
rate2 += compmode_cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Estimate the reference frame signaling cost and add it
|
|
|
|
// to the rolling cost variable.
|
|
|
|
rate2 += ref_costs[xd->mode_info_context->mbmi.ref_frame];
|
|
|
|
|
|
|
|
if (!disable_skip) {
|
|
|
|
// Test for the condition where skip block will be activated
|
|
|
|
// because there are no non zero coefficients and make any
|
|
|
|
// necessary adjustment for rate. Ignore if skip is coded at
|
|
|
|
// segment level as the cost wont have been added in.
|
2013-04-09 19:54:19 +02:00
|
|
|
int mb_skip_allowed;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
// Is Mb level skip allowed (i.e. not coded at segment level).
|
|
|
|
mb_skip_allowed = !vp9_segfeature_active(xd, segment_id, SEG_LVL_SKIP);
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
if (skippable) {
|
|
|
|
// Back out the coefficient coding costs
|
|
|
|
rate2 -= (rate_y + rate_uv);
|
|
|
|
// for best_yrd calculation
|
|
|
|
rate_uv = 0;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
if (mb_skip_allowed) {
|
|
|
|
int prob_skip_cost;
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
// Cost the skip mb case
|
|
|
|
vp9_prob skip_prob =
|
|
|
|
vp9_get_pred_prob(cm, xd, PRED_MBSKIP);
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2013-04-09 19:54:19 +02:00
|
|
|
if (skip_prob) {
|
|
|
|
prob_skip_cost = vp9_cost_bit(skip_prob, 1);
|
|
|
|
rate2 += prob_skip_cost;
|
|
|
|
other_cost += prob_skip_cost;
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
}
|
2013-04-09 19:54:19 +02:00
|
|
|
} else if (mb_skip_allowed) {
|
2012-08-20 23:43:34 +02:00
|
|
|
// Add in the cost of the no skip flag.
|
2013-04-09 19:54:19 +02:00
|
|
|
int prob_skip_cost = vp9_cost_bit(vp9_get_pred_prob(cm, xd,
|
|
|
|
PRED_MBSKIP), 0);
|
|
|
|
rate2 += prob_skip_cost;
|
|
|
|
other_cost += prob_skip_cost;
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate the final RD estimate for this mode.
|
|
|
|
this_rd = RDCOST(x->rdmult, x->rddiv, rate2, distortion2);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
// Keep record of best intra distortion
|
|
|
|
if ((xd->mode_info_context->mbmi.ref_frame == INTRA_FRAME) &&
|
|
|
|
(this_rd < best_intra_rd)) {
|
|
|
|
best_intra_rd = this_rd;
|
|
|
|
*returnintra = distortion2;
|
|
|
|
}
|
|
|
|
#endif
|
2012-11-30 20:46:20 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
if ((mbmi->ref_frame == INTRA_FRAME) &&
|
|
|
|
(this_mode <= TM_PRED) &&
|
|
|
|
(this_rd < best_intra16_rd)) {
|
|
|
|
best_intra16_rd = this_rd;
|
|
|
|
best_intra16_mode = this_mode;
|
2013-02-20 19:16:24 +01:00
|
|
|
#if SEPARATE_INTERINTRA_UV
|
2012-11-30 20:46:20 +01:00
|
|
|
best_intra16_uv_mode = (mbmi->txfm_size != TX_4X4 ?
|
|
|
|
mode_uv_8x8 : mode_uv_4x4);
|
2013-02-20 19:16:24 +01:00
|
|
|
#endif
|
2012-11-30 20:46:20 +01:00
|
|
|
}
|
|
|
|
#endif
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-11-08 20:03:00 +01:00
|
|
|
if (!disable_skip && mbmi->ref_frame == INTRA_FRAME)
|
|
|
|
for (i = 0; i < NB_PREDICTION_TYPES; ++i)
|
|
|
|
best_pred_rd[i] = MIN(best_pred_rd[i], this_rd);
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-11-30 20:46:20 +01:00
|
|
|
if (this_rd < best_overall_rd) {
|
|
|
|
best_overall_rd = this_rd;
|
2013-02-12 02:08:52 +01:00
|
|
|
best_filter = tmp_best_filter;
|
2012-12-20 23:56:19 +01:00
|
|
|
best_mode = this_mode;
|
2012-11-30 20:46:20 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
is_best_interintra = (mbmi->second_ref_frame == INTRA_FRAME);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-04-23 00:42:41 +02:00
|
|
|
// Store the respective mode distortions for later use.
|
|
|
|
if (mode_distortions[this_mode] == -1
|
|
|
|
|| distortion2 < mode_distortions[this_mode]) {
|
|
|
|
mode_distortions[this_mode] = distortion2;
|
|
|
|
}
|
|
|
|
if (frame_distortions[mbmi->ref_frame] == -1
|
|
|
|
|| distortion2 < frame_distortions[mbmi->ref_frame]) {
|
|
|
|
frame_distortions[mbmi->ref_frame] = distortion2;
|
|
|
|
}
|
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
// Did this mode help.. i.e. is it the new best mode
|
|
|
|
if (this_rd < best_rd || x->skip) {
|
|
|
|
if (!mode_excluded) {
|
|
|
|
// Note index of best mode so far
|
|
|
|
best_mode_index = mode_index;
|
|
|
|
|
2013-04-11 21:16:35 +02:00
|
|
|
if (this_mode <= I4X4_PRED) {
|
2012-08-20 23:43:34 +02:00
|
|
|
/* required for left and above block mv */
|
2012-11-16 00:50:07 +01:00
|
|
|
mbmi->mv[0].as_int = 0;
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
other_cost += ref_costs[xd->mode_info_context->mbmi.ref_frame];
|
|
|
|
*returnrate = rate2;
|
|
|
|
*returndistortion = distortion2;
|
|
|
|
best_rd = this_rd;
|
2012-10-30 01:58:18 +01:00
|
|
|
vpx_memcpy(&best_mbmode, mbmi, sizeof(MB_MODE_INFO));
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
#if 0
|
2012-12-20 23:56:19 +01:00
|
|
|
// Testing this mode gave rise to an improvement in best error score.
|
|
|
|
// Lower threshold a bit for next time
|
|
|
|
cpi->rd_thresh_mult[mode_index] =
|
|
|
|
(cpi->rd_thresh_mult[mode_index] >= (MIN_THRESHMULT + 2)) ?
|
|
|
|
cpi->rd_thresh_mult[mode_index] - 2 : MIN_THRESHMULT;
|
|
|
|
cpi->rd_threshes[mode_index] =
|
|
|
|
(cpi->rd_baseline_thresh[mode_index] >> 7)
|
|
|
|
* cpi->rd_thresh_mult[mode_index];
|
2012-08-20 23:43:34 +02:00
|
|
|
#endif
|
2012-12-20 23:56:19 +01:00
|
|
|
} else {
|
|
|
|
// If the mode did not help improve the best error case then
|
|
|
|
// raise the threshold for testing that mode next time around.
|
2012-08-20 23:43:34 +02:00
|
|
|
#if 0
|
|
|
|
cpi->rd_thresh_mult[mode_index] += 4;
|
|
|
|
|
|
|
|
if (cpi->rd_thresh_mult[mode_index] > MAX_THRESHMULT)
|
|
|
|
cpi->rd_thresh_mult[mode_index] = MAX_THRESHMULT;
|
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
cpi->rd_threshes[mode_index] =
|
|
|
|
(cpi->rd_baseline_thresh[mode_index] >> 7)
|
|
|
|
* cpi->rd_thresh_mult[mode_index];
|
2012-08-20 23:43:34 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* keep record of best compound/single-only prediction */
|
2012-10-30 01:58:18 +01:00
|
|
|
if (!disable_skip && mbmi->ref_frame != INTRA_FRAME) {
|
2012-08-20 23:43:34 +02:00
|
|
|
int single_rd, hybrid_rd, single_rate, hybrid_rate;
|
|
|
|
|
|
|
|
if (cpi->common.comp_pred_mode == HYBRID_PREDICTION) {
|
|
|
|
single_rate = rate2 - compmode_cost;
|
|
|
|
hybrid_rate = rate2;
|
|
|
|
} else {
|
|
|
|
single_rate = rate2;
|
|
|
|
hybrid_rate = rate2 + compmode_cost;
|
|
|
|
}
|
|
|
|
|
|
|
|
single_rd = RDCOST(x->rdmult, x->rddiv, single_rate, distortion2);
|
|
|
|
hybrid_rd = RDCOST(x->rdmult, x->rddiv, hybrid_rate, distortion2);
|
|
|
|
|
2012-11-07 15:50:25 +01:00
|
|
|
if (mbmi->second_ref_frame <= INTRA_FRAME &&
|
2012-11-08 20:03:00 +01:00
|
|
|
single_rd < best_pred_rd[SINGLE_PREDICTION_ONLY]) {
|
|
|
|
best_pred_rd[SINGLE_PREDICTION_ONLY] = single_rd;
|
2012-11-07 15:50:25 +01:00
|
|
|
} else if (mbmi->second_ref_frame > INTRA_FRAME &&
|
2012-11-08 20:03:00 +01:00
|
|
|
single_rd < best_pred_rd[COMP_PREDICTION_ONLY]) {
|
|
|
|
best_pred_rd[COMP_PREDICTION_ONLY] = single_rd;
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
2012-11-08 20:03:00 +01:00
|
|
|
if (hybrid_rd < best_pred_rd[HYBRID_PREDICTION])
|
|
|
|
best_pred_rd[HYBRID_PREDICTION] = hybrid_rd;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* keep record of best txfm size */
|
2013-05-01 01:13:20 +02:00
|
|
|
if (bsize < BLOCK_SIZE_SB32X32) {
|
|
|
|
if (bsize < BLOCK_SIZE_MB16X16) {
|
|
|
|
if (this_mode == SPLITMV || this_mode == I4X4_PRED)
|
|
|
|
txfm_cache[ALLOW_8X8] = txfm_cache[ONLY_4X4];
|
|
|
|
txfm_cache[ALLOW_16X16] = txfm_cache[ALLOW_8X8];
|
|
|
|
}
|
|
|
|
txfm_cache[ALLOW_32X32] = txfm_cache[ALLOW_16X16];
|
|
|
|
}
|
2013-01-14 20:49:30 +01:00
|
|
|
if (!mode_excluded && this_rd != INT64_MAX) {
|
2012-11-08 20:03:00 +01:00
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
|
|
|
int64_t adj_rd;
|
2013-04-11 21:16:35 +02:00
|
|
|
if (this_mode != I4X4_PRED) {
|
2012-11-08 20:03:00 +01:00
|
|
|
adj_rd = this_rd + txfm_cache[i] - txfm_cache[cm->txfm_mode];
|
|
|
|
} else {
|
|
|
|
adj_rd = this_rd;
|
|
|
|
}
|
|
|
|
if (adj_rd < best_txfm_rd[i])
|
|
|
|
best_txfm_rd[i] = adj_rd;
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (x->skip && !mode_excluded)
|
|
|
|
break;
|
|
|
|
}
|
2013-04-23 00:42:41 +02:00
|
|
|
// Flag all modes that have a distortion thats > 2x the best we found at
|
|
|
|
// this level.
|
|
|
|
for (mode_index = 0; mode_index < MB_MODE_COUNT; ++mode_index) {
|
|
|
|
if (mode_index == NEARESTMV || mode_index == NEARMV || mode_index == NEWMV
|
|
|
|
|| mode_index == SPLITMV)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (mode_distortions[mode_index] > 2 * *returndistortion) {
|
|
|
|
ctx->modes_with_high_error |= (1 << mode_index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flag all ref frames that have a distortion thats > 2x the best we found at
|
|
|
|
// this level.
|
|
|
|
for (ref_frame = INTRA_FRAME; ref_frame <= ALTREF_FRAME; ref_frame++) {
|
|
|
|
if (frame_distortions[ref_frame] > 2 * *returndistortion) {
|
|
|
|
ctx->frames_with_high_error |= (1 << ref_frame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
assert((cm->mcomp_filter_type == SWITCHABLE) ||
|
|
|
|
(cm->mcomp_filter_type == best_mbmode.interp_filter) ||
|
2013-04-11 21:16:35 +02:00
|
|
|
(best_mbmode.mode <= I4X4_PRED));
|
2012-12-20 23:56:19 +01:00
|
|
|
|
2012-11-30 20:46:20 +01:00
|
|
|
#if CONFIG_COMP_INTERINTRA_PRED
|
|
|
|
++cpi->interintra_select_count[is_best_interintra];
|
|
|
|
// if (is_best_interintra) printf("best_interintra\n");
|
|
|
|
#endif
|
|
|
|
|
2012-12-20 23:56:19 +01:00
|
|
|
// Accumulate filter usage stats
|
|
|
|
// TODO(agrange): Use RD criteria to select interpolation filter mode.
|
2013-04-24 21:14:58 +02:00
|
|
|
if (is_inter_mode(best_mode))
|
2012-12-20 23:56:19 +01:00
|
|
|
++cpi->best_switchable_interp_count[vp9_switchable_interp_map[best_filter]];
|
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
// TODO(rbultje) integrate with RD thresholding
|
|
|
|
#if 0
|
|
|
|
// Reduce the activation RD thresholds for the best choice mode
|
|
|
|
if ((cpi->rd_baseline_thresh[best_mode_index] > 0) &&
|
|
|
|
(cpi->rd_baseline_thresh[best_mode_index] < (INT_MAX >> 2))) {
|
|
|
|
int best_adjustment = (cpi->rd_thresh_mult[best_mode_index] >> 2);
|
|
|
|
|
|
|
|
cpi->rd_thresh_mult[best_mode_index] =
|
|
|
|
(cpi->rd_thresh_mult[best_mode_index] >= (MIN_THRESHMULT + best_adjustment)) ?
|
|
|
|
cpi->rd_thresh_mult[best_mode_index] - best_adjustment : MIN_THRESHMULT;
|
|
|
|
cpi->rd_threshes[best_mode_index] =
|
|
|
|
(cpi->rd_baseline_thresh[best_mode_index] >> 7) * cpi->rd_thresh_mult[best_mode_index];
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// This code forces Altref,0,0 and skip for the frame that overlays a
|
|
|
|
// an alrtef unless Altref is filtered. However, this is unsafe if
|
2013-01-28 16:22:53 +01:00
|
|
|
// segment level coding of ref frame is enabled for this segment.
|
2012-10-30 06:15:27 +01:00
|
|
|
if (!vp9_segfeature_active(xd, segment_id, SEG_LVL_REF_FRAME) &&
|
2012-08-20 23:43:34 +02:00
|
|
|
cpi->is_src_frame_alt_ref &&
|
|
|
|
(cpi->oxcf.arnr_max_frames == 0) &&
|
|
|
|
(best_mbmode.mode != ZEROMV || best_mbmode.ref_frame != ALTREF_FRAME)) {
|
2012-10-30 01:58:18 +01:00
|
|
|
mbmi->mode = ZEROMV;
|
|
|
|
mbmi->ref_frame = ALTREF_FRAME;
|
2013-05-01 01:13:20 +02:00
|
|
|
mbmi->second_ref_frame = NONE;
|
2012-10-30 01:58:18 +01:00
|
|
|
mbmi->mv[0].as_int = 0;
|
|
|
|
mbmi->uv_mode = DC_PRED;
|
2013-04-09 19:54:19 +02:00
|
|
|
mbmi->mb_skip_coeff = 1;
|
2013-05-01 01:13:20 +02:00
|
|
|
#if !CONFIG_SB8X8
|
2012-10-30 01:58:18 +01:00
|
|
|
mbmi->partitioning = 0;
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif
|
|
|
|
if (cm->txfm_mode == TX_MODE_SELECT) {
|
|
|
|
if (bsize >= BLOCK_SIZE_SB32X32)
|
|
|
|
mbmi->txfm_size = TX_32X32;
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
else if (bsize >= BLOCK_SIZE_MB16X16)
|
|
|
|
#else
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
mbmi->txfm_size = TX_16X16;
|
|
|
|
#if CONFIG_SB8X8
|
|
|
|
else
|
|
|
|
mbmi->txfm_size = TX_8X8;
|
|
|
|
#endif
|
|
|
|
}
|
2012-08-20 23:43:34 +02:00
|
|
|
|
2012-11-08 20:03:00 +01:00
|
|
|
vpx_memset(best_txfm_diff, 0, sizeof(best_txfm_diff));
|
|
|
|
vpx_memset(best_pred_diff, 0, sizeof(best_pred_diff));
|
|
|
|
goto end;
|
2012-08-20 23:43:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// macroblock modes
|
2012-10-30 01:58:18 +01:00
|
|
|
vpx_memcpy(mbmi, &best_mbmode, sizeof(MB_MODE_INFO));
|
2012-11-08 20:03:00 +01:00
|
|
|
|
|
|
|
for (i = 0; i < NB_PREDICTION_TYPES; ++i) {
|
2013-01-14 20:49:30 +01:00
|
|
|
if (best_pred_rd[i] == INT64_MAX)
|
2012-11-08 20:03:00 +01:00
|
|
|
best_pred_diff[i] = INT_MIN;
|
|
|
|
else
|
|
|
|
best_pred_diff[i] = best_rd - best_pred_rd[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!x->skip) {
|
|
|
|
for (i = 0; i < NB_TXFM_MODES; i++) {
|
2013-01-14 20:49:30 +01:00
|
|
|
if (best_txfm_rd[i] == INT64_MAX)
|
2013-02-16 00:55:31 +01:00
|
|
|
best_txfm_diff[i] = 0;
|
2012-11-08 20:03:00 +01:00
|
|
|
else
|
|
|
|
best_txfm_diff[i] = best_rd - best_txfm_rd[i];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
vpx_memset(best_txfm_diff, 0, sizeof(best_txfm_diff));
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 05:55:14 +01:00
|
|
|
set_scale_factors(xd, mbmi->ref_frame, mbmi->second_ref_frame,
|
|
|
|
scale_factor);
|
2013-04-11 18:33:49 +02:00
|
|
|
store_coding_context(x, ctx, best_mode_index, NULL,
|
|
|
|
&mbmi->ref_mvs[mbmi->ref_frame][0],
|
|
|
|
&mbmi->ref_mvs[mbmi->second_ref_frame < 0 ? 0 :
|
|
|
|
mbmi->second_ref_frame][0],
|
|
|
|
best_pred_diff, best_txfm_diff);
|
2012-08-20 23:43:34 +02:00
|
|
|
|
|
|
|
return best_rd;
|
|
|
|
}
|
2013-01-06 03:20:25 +01:00
|
|
|
|
2013-05-01 01:13:20 +02:00
|
|
|
#if !CONFIG_SB8X8
|
2012-10-31 01:53:32 +01:00
|
|
|
void vp9_pick_mode_inter_macroblock(VP9_COMP *cpi, MACROBLOCK *x,
|
2013-04-26 20:57:17 +02:00
|
|
|
int mi_row, int mi_col,
|
2012-10-30 22:25:33 +01:00
|
|
|
int *totalrate, int *totaldist) {
|
2012-07-14 00:21:29 +02:00
|
|
|
MACROBLOCKD *const xd = &x->e_mbd;
|
2012-08-10 15:12:43 +02:00
|
|
|
MB_MODE_INFO * mbmi = &x->e_mbd.mode_info_context->mbmi;
|
2012-08-10 03:25:29 +02:00
|
|
|
int rate, distortion;
|
2012-08-02 20:01:48 +02:00
|
|
|
int64_t intra_error = 0;
|
2012-08-10 15:12:43 +02:00
|
|
|
unsigned char *segment_id = &mbmi->segment_id;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-30 01:07:17 +02:00
|
|
|
x->encode_breakout = xd->segmentation_enabled ?
|
|
|
|
cpi->segment_encode_breakout[*segment_id] :
|
|
|
|
cpi->oxcf.encode_breakout;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// if (cpi->sf.RD)
|
|
|
|
// For now this codebase is limited to a single rd encode path
|
|
|
|
{
|
|
|
|
int zbin_mode_boost_enabled = cpi->zbin_mode_boost_enabled;
|
|
|
|
|
2013-04-26 20:57:17 +02:00
|
|
|
rd_pick_inter_mode(cpi, x, mi_row, mi_col, &rate,
|
2012-11-02 19:22:57 +01:00
|
|
|
&distortion, &intra_error);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
/* restore cpi->zbin_mode_boost_enabled */
|
|
|
|
cpi->zbin_mode_boost_enabled = zbin_mode_boost_enabled;
|
|
|
|
}
|
|
|
|
// else
|
|
|
|
// The non rd encode path has been deleted from this code base
|
|
|
|
// to simplify development
|
2012-10-31 22:40:53 +01:00
|
|
|
// vp9_pick_inter_mode
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
// Store metrics so they can be added in to totals if this mode is picked
|
2013-01-06 03:20:25 +01:00
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].distortion = distortion;
|
|
|
|
x->mb_context[xd->sb_index][xd->mb_index].intra_error = intra_error;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2012-08-20 23:43:34 +02:00
|
|
|
*totalrate = rate;
|
|
|
|
*totaldist = distortion;
|
2011-06-08 18:05:05 +02:00
|
|
|
}
|
2013-05-01 01:13:20 +02:00
|
|
|
#endif
|