1*fb1b10abSAndroid Build Coastguard Worker /*
2*fb1b10abSAndroid Build Coastguard Worker * Copyright (c) 2010 The WebM project authors. All Rights Reserved.
3*fb1b10abSAndroid Build Coastguard Worker *
4*fb1b10abSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*fb1b10abSAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*fb1b10abSAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*fb1b10abSAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*fb1b10abSAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*fb1b10abSAndroid Build Coastguard Worker */
10*fb1b10abSAndroid Build Coastguard Worker
11*fb1b10abSAndroid Build Coastguard Worker #include <assert.h>
12*fb1b10abSAndroid Build Coastguard Worker #include <stdio.h>
13*fb1b10abSAndroid Build Coastguard Worker #include <math.h>
14*fb1b10abSAndroid Build Coastguard Worker #include <limits.h>
15*fb1b10abSAndroid Build Coastguard Worker #include <assert.h>
16*fb1b10abSAndroid Build Coastguard Worker #include "vpx_config.h"
17*fb1b10abSAndroid Build Coastguard Worker #include "vp8_rtcd.h"
18*fb1b10abSAndroid Build Coastguard Worker #include "./vpx_dsp_rtcd.h"
19*fb1b10abSAndroid Build Coastguard Worker #include "encodeframe.h"
20*fb1b10abSAndroid Build Coastguard Worker #include "tokenize.h"
21*fb1b10abSAndroid Build Coastguard Worker #include "treewriter.h"
22*fb1b10abSAndroid Build Coastguard Worker #include "onyx_int.h"
23*fb1b10abSAndroid Build Coastguard Worker #include "modecosts.h"
24*fb1b10abSAndroid Build Coastguard Worker #include "encodeintra.h"
25*fb1b10abSAndroid Build Coastguard Worker #include "pickinter.h"
26*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/common.h"
27*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/entropymode.h"
28*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/reconinter.h"
29*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/reconintra.h"
30*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/reconintra4x4.h"
31*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/findnearmv.h"
32*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/quant_common.h"
33*fb1b10abSAndroid Build Coastguard Worker #include "encodemb.h"
34*fb1b10abSAndroid Build Coastguard Worker #include "vp8/encoder/quantize.h"
35*fb1b10abSAndroid Build Coastguard Worker #include "vpx_dsp/variance.h"
36*fb1b10abSAndroid Build Coastguard Worker #include "vpx_ports/system_state.h"
37*fb1b10abSAndroid Build Coastguard Worker #include "mcomp.h"
38*fb1b10abSAndroid Build Coastguard Worker #include "rdopt.h"
39*fb1b10abSAndroid Build Coastguard Worker #include "vpx_mem/vpx_mem.h"
40*fb1b10abSAndroid Build Coastguard Worker #include "vp8/common/systemdependent.h"
41*fb1b10abSAndroid Build Coastguard Worker #if CONFIG_TEMPORAL_DENOISING
42*fb1b10abSAndroid Build Coastguard Worker #include "denoising.h"
43*fb1b10abSAndroid Build Coastguard Worker #endif
44*fb1b10abSAndroid Build Coastguard Worker extern void vp8_update_zbin_extra(VP8_COMP *cpi, MACROBLOCK *x);
45*fb1b10abSAndroid Build Coastguard Worker
46*fb1b10abSAndroid Build Coastguard Worker #define MAXF(a, b) (((a) > (b)) ? (a) : (b))
47*fb1b10abSAndroid Build Coastguard Worker
48*fb1b10abSAndroid Build Coastguard Worker typedef struct rate_distortion_struct {
49*fb1b10abSAndroid Build Coastguard Worker int rate2;
50*fb1b10abSAndroid Build Coastguard Worker int rate_y;
51*fb1b10abSAndroid Build Coastguard Worker int rate_uv;
52*fb1b10abSAndroid Build Coastguard Worker int distortion2;
53*fb1b10abSAndroid Build Coastguard Worker int distortion_uv;
54*fb1b10abSAndroid Build Coastguard Worker } RATE_DISTORTION;
55*fb1b10abSAndroid Build Coastguard Worker
56*fb1b10abSAndroid Build Coastguard Worker typedef struct best_mode_struct {
57*fb1b10abSAndroid Build Coastguard Worker int yrd;
58*fb1b10abSAndroid Build Coastguard Worker int rd;
59*fb1b10abSAndroid Build Coastguard Worker int intra_rd;
60*fb1b10abSAndroid Build Coastguard Worker MB_MODE_INFO mbmode;
61*fb1b10abSAndroid Build Coastguard Worker union b_mode_info bmodes[16];
62*fb1b10abSAndroid Build Coastguard Worker PARTITION_INFO partition;
63*fb1b10abSAndroid Build Coastguard Worker } BEST_MODE;
64*fb1b10abSAndroid Build Coastguard Worker
65*fb1b10abSAndroid Build Coastguard Worker static const int auto_speed_thresh[17] = { 1000, 200, 150, 130, 150, 125,
66*fb1b10abSAndroid Build Coastguard Worker 120, 115, 115, 115, 115, 115,
67*fb1b10abSAndroid Build Coastguard Worker 115, 115, 115, 115, 105 };
68*fb1b10abSAndroid Build Coastguard Worker
69*fb1b10abSAndroid Build Coastguard Worker const MB_PREDICTION_MODE vp8_mode_order[MAX_MODES] = {
70*fb1b10abSAndroid Build Coastguard Worker ZEROMV, DC_PRED,
71*fb1b10abSAndroid Build Coastguard Worker
72*fb1b10abSAndroid Build Coastguard Worker NEARESTMV, NEARMV,
73*fb1b10abSAndroid Build Coastguard Worker
74*fb1b10abSAndroid Build Coastguard Worker ZEROMV, NEARESTMV,
75*fb1b10abSAndroid Build Coastguard Worker
76*fb1b10abSAndroid Build Coastguard Worker ZEROMV, NEARESTMV,
77*fb1b10abSAndroid Build Coastguard Worker
78*fb1b10abSAndroid Build Coastguard Worker NEARMV, NEARMV,
79*fb1b10abSAndroid Build Coastguard Worker
80*fb1b10abSAndroid Build Coastguard Worker V_PRED, H_PRED, TM_PRED,
81*fb1b10abSAndroid Build Coastguard Worker
82*fb1b10abSAndroid Build Coastguard Worker NEWMV, NEWMV, NEWMV,
83*fb1b10abSAndroid Build Coastguard Worker
84*fb1b10abSAndroid Build Coastguard Worker SPLITMV, SPLITMV, SPLITMV,
85*fb1b10abSAndroid Build Coastguard Worker
86*fb1b10abSAndroid Build Coastguard Worker B_PRED,
87*fb1b10abSAndroid Build Coastguard Worker };
88*fb1b10abSAndroid Build Coastguard Worker
89*fb1b10abSAndroid Build Coastguard Worker /* This table determines the search order in reference frame priority order,
90*fb1b10abSAndroid Build Coastguard Worker * which may not necessarily match INTRA,LAST,GOLDEN,ARF
91*fb1b10abSAndroid Build Coastguard Worker */
92*fb1b10abSAndroid Build Coastguard Worker const int vp8_ref_frame_order[MAX_MODES] = {
93*fb1b10abSAndroid Build Coastguard Worker 1, 0,
94*fb1b10abSAndroid Build Coastguard Worker
95*fb1b10abSAndroid Build Coastguard Worker 1, 1,
96*fb1b10abSAndroid Build Coastguard Worker
97*fb1b10abSAndroid Build Coastguard Worker 2, 2,
98*fb1b10abSAndroid Build Coastguard Worker
99*fb1b10abSAndroid Build Coastguard Worker 3, 3,
100*fb1b10abSAndroid Build Coastguard Worker
101*fb1b10abSAndroid Build Coastguard Worker 2, 3,
102*fb1b10abSAndroid Build Coastguard Worker
103*fb1b10abSAndroid Build Coastguard Worker 0, 0, 0,
104*fb1b10abSAndroid Build Coastguard Worker
105*fb1b10abSAndroid Build Coastguard Worker 1, 2, 3,
106*fb1b10abSAndroid Build Coastguard Worker
107*fb1b10abSAndroid Build Coastguard Worker 1, 2, 3,
108*fb1b10abSAndroid Build Coastguard Worker
109*fb1b10abSAndroid Build Coastguard Worker 0,
110*fb1b10abSAndroid Build Coastguard Worker };
111*fb1b10abSAndroid Build Coastguard Worker
fill_token_costs(int c[BLOCK_TYPES][COEF_BANDS][PREV_COEF_CONTEXTS][MAX_ENTROPY_TOKENS],const vp8_prob p[BLOCK_TYPES][COEF_BANDS][PREV_COEF_CONTEXTS][ENTROPY_NODES])112*fb1b10abSAndroid Build Coastguard Worker static void fill_token_costs(
113*fb1b10abSAndroid Build Coastguard Worker int c[BLOCK_TYPES][COEF_BANDS][PREV_COEF_CONTEXTS][MAX_ENTROPY_TOKENS],
114*fb1b10abSAndroid Build Coastguard Worker const vp8_prob p[BLOCK_TYPES][COEF_BANDS][PREV_COEF_CONTEXTS]
115*fb1b10abSAndroid Build Coastguard Worker [ENTROPY_NODES]) {
116*fb1b10abSAndroid Build Coastguard Worker int i, j, k;
117*fb1b10abSAndroid Build Coastguard Worker
118*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < BLOCK_TYPES; ++i) {
119*fb1b10abSAndroid Build Coastguard Worker for (j = 0; j < COEF_BANDS; ++j) {
120*fb1b10abSAndroid Build Coastguard Worker for (k = 0; k < PREV_COEF_CONTEXTS; ++k) {
121*fb1b10abSAndroid Build Coastguard Worker /* check for pt=0 and band > 1 if block type 0
122*fb1b10abSAndroid Build Coastguard Worker * and 0 if blocktype 1
123*fb1b10abSAndroid Build Coastguard Worker */
124*fb1b10abSAndroid Build Coastguard Worker if (k == 0 && j > (i == 0)) {
125*fb1b10abSAndroid Build Coastguard Worker vp8_cost_tokens2(c[i][j][k], p[i][j][k], vp8_coef_tree, 2);
126*fb1b10abSAndroid Build Coastguard Worker } else {
127*fb1b10abSAndroid Build Coastguard Worker vp8_cost_tokens(c[i][j][k], p[i][j][k], vp8_coef_tree);
128*fb1b10abSAndroid Build Coastguard Worker }
129*fb1b10abSAndroid Build Coastguard Worker }
130*fb1b10abSAndroid Build Coastguard Worker }
131*fb1b10abSAndroid Build Coastguard Worker }
132*fb1b10abSAndroid Build Coastguard Worker }
133*fb1b10abSAndroid Build Coastguard Worker
134*fb1b10abSAndroid Build Coastguard Worker static const int rd_iifactor[32] = { 4, 4, 3, 2, 1, 0, 0, 0, 0, 0, 0,
135*fb1b10abSAndroid Build Coastguard Worker 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
136*fb1b10abSAndroid Build Coastguard Worker 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
137*fb1b10abSAndroid Build Coastguard Worker
138*fb1b10abSAndroid Build Coastguard Worker /* values are now correlated to quantizer */
139*fb1b10abSAndroid Build Coastguard Worker static const int sad_per_bit16lut[QINDEX_RANGE] = {
140*fb1b10abSAndroid Build Coastguard Worker 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3,
141*fb1b10abSAndroid Build Coastguard Worker 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4, 4,
142*fb1b10abSAndroid Build Coastguard Worker 4, 4, 4, 4, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 6, 6, 6,
143*fb1b10abSAndroid Build Coastguard Worker 6, 6, 6, 6, 6, 6, 6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
144*fb1b10abSAndroid Build Coastguard Worker 7, 7, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 9, 9, 9, 9, 9,
145*fb1b10abSAndroid Build Coastguard Worker 9, 9, 9, 9, 9, 9, 9, 10, 10, 10, 10, 10, 10, 10, 10, 11, 11, 11, 11,
146*fb1b10abSAndroid Build Coastguard Worker 11, 11, 12, 12, 12, 12, 12, 12, 13, 13, 13, 13, 14, 14
147*fb1b10abSAndroid Build Coastguard Worker };
148*fb1b10abSAndroid Build Coastguard Worker static const int sad_per_bit4lut[QINDEX_RANGE] = {
149*fb1b10abSAndroid Build Coastguard Worker 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
150*fb1b10abSAndroid Build Coastguard Worker 3, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 5, 5, 5, 5, 5, 5, 6, 6,
151*fb1b10abSAndroid Build Coastguard Worker 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 7, 7,
152*fb1b10abSAndroid Build Coastguard Worker 7, 7, 7, 7, 8, 8, 8, 8, 8, 9, 9, 9, 9, 9, 9, 10, 10, 10, 10,
153*fb1b10abSAndroid Build Coastguard Worker 10, 10, 10, 10, 11, 11, 11, 11, 11, 11, 11, 11, 12, 12, 12, 12, 12, 12, 12,
154*fb1b10abSAndroid Build Coastguard Worker 12, 13, 13, 13, 13, 13, 13, 13, 14, 14, 14, 14, 14, 15, 15, 15, 15, 16, 16,
155*fb1b10abSAndroid Build Coastguard Worker 16, 16, 17, 17, 17, 18, 18, 18, 19, 19, 19, 20, 20, 20,
156*fb1b10abSAndroid Build Coastguard Worker };
157*fb1b10abSAndroid Build Coastguard Worker
vp8cx_initialize_me_consts(VP8_COMP * cpi,int QIndex)158*fb1b10abSAndroid Build Coastguard Worker void vp8cx_initialize_me_consts(VP8_COMP *cpi, int QIndex) {
159*fb1b10abSAndroid Build Coastguard Worker cpi->mb.sadperbit16 = sad_per_bit16lut[QIndex];
160*fb1b10abSAndroid Build Coastguard Worker cpi->mb.sadperbit4 = sad_per_bit4lut[QIndex];
161*fb1b10abSAndroid Build Coastguard Worker }
162*fb1b10abSAndroid Build Coastguard Worker
vp8_initialize_rd_consts(VP8_COMP * cpi,MACROBLOCK * x,int Qvalue)163*fb1b10abSAndroid Build Coastguard Worker void vp8_initialize_rd_consts(VP8_COMP *cpi, MACROBLOCK *x, int Qvalue) {
164*fb1b10abSAndroid Build Coastguard Worker int q;
165*fb1b10abSAndroid Build Coastguard Worker int i;
166*fb1b10abSAndroid Build Coastguard Worker double capped_q = (Qvalue < 160) ? (double)Qvalue : 160.0;
167*fb1b10abSAndroid Build Coastguard Worker double rdconst = 2.80;
168*fb1b10abSAndroid Build Coastguard Worker
169*fb1b10abSAndroid Build Coastguard Worker vpx_clear_system_state();
170*fb1b10abSAndroid Build Coastguard Worker
171*fb1b10abSAndroid Build Coastguard Worker /* Further tests required to see if optimum is different
172*fb1b10abSAndroid Build Coastguard Worker * for key frames, golden frames and arf frames.
173*fb1b10abSAndroid Build Coastguard Worker */
174*fb1b10abSAndroid Build Coastguard Worker cpi->RDMULT = (int)(rdconst * (capped_q * capped_q));
175*fb1b10abSAndroid Build Coastguard Worker
176*fb1b10abSAndroid Build Coastguard Worker /* Extend rate multiplier along side quantizer zbin increases */
177*fb1b10abSAndroid Build Coastguard Worker if (cpi->mb.zbin_over_quant > 0) {
178*fb1b10abSAndroid Build Coastguard Worker double oq_factor;
179*fb1b10abSAndroid Build Coastguard Worker double modq;
180*fb1b10abSAndroid Build Coastguard Worker
181*fb1b10abSAndroid Build Coastguard Worker /* Experimental code using the same basic equation as used for Q above
182*fb1b10abSAndroid Build Coastguard Worker * The units of cpi->mb.zbin_over_quant are 1/128 of Q bin size
183*fb1b10abSAndroid Build Coastguard Worker */
184*fb1b10abSAndroid Build Coastguard Worker oq_factor = 1.0 + ((double)0.0015625 * cpi->mb.zbin_over_quant);
185*fb1b10abSAndroid Build Coastguard Worker modq = (int)((double)capped_q * oq_factor);
186*fb1b10abSAndroid Build Coastguard Worker cpi->RDMULT = (int)(rdconst * (modq * modq));
187*fb1b10abSAndroid Build Coastguard Worker }
188*fb1b10abSAndroid Build Coastguard Worker
189*fb1b10abSAndroid Build Coastguard Worker if (cpi->pass == 2 && (cpi->common.frame_type != KEY_FRAME)) {
190*fb1b10abSAndroid Build Coastguard Worker if (cpi->twopass.next_iiratio > 31) {
191*fb1b10abSAndroid Build Coastguard Worker cpi->RDMULT += (cpi->RDMULT * rd_iifactor[31]) >> 4;
192*fb1b10abSAndroid Build Coastguard Worker } else {
193*fb1b10abSAndroid Build Coastguard Worker cpi->RDMULT +=
194*fb1b10abSAndroid Build Coastguard Worker (cpi->RDMULT * rd_iifactor[cpi->twopass.next_iiratio]) >> 4;
195*fb1b10abSAndroid Build Coastguard Worker }
196*fb1b10abSAndroid Build Coastguard Worker }
197*fb1b10abSAndroid Build Coastguard Worker
198*fb1b10abSAndroid Build Coastguard Worker cpi->mb.errorperbit = (cpi->RDMULT / 110);
199*fb1b10abSAndroid Build Coastguard Worker cpi->mb.errorperbit += (cpi->mb.errorperbit == 0);
200*fb1b10abSAndroid Build Coastguard Worker
201*fb1b10abSAndroid Build Coastguard Worker vp8_set_speed_features(cpi);
202*fb1b10abSAndroid Build Coastguard Worker
203*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < MAX_MODES; ++i) {
204*fb1b10abSAndroid Build Coastguard Worker x->mode_test_hit_counts[i] = 0;
205*fb1b10abSAndroid Build Coastguard Worker }
206*fb1b10abSAndroid Build Coastguard Worker
207*fb1b10abSAndroid Build Coastguard Worker q = (int)pow(Qvalue, 1.25);
208*fb1b10abSAndroid Build Coastguard Worker
209*fb1b10abSAndroid Build Coastguard Worker if (q < 8) q = 8;
210*fb1b10abSAndroid Build Coastguard Worker
211*fb1b10abSAndroid Build Coastguard Worker if (cpi->RDMULT > 1000) {
212*fb1b10abSAndroid Build Coastguard Worker cpi->RDDIV = 1;
213*fb1b10abSAndroid Build Coastguard Worker cpi->RDMULT /= 100;
214*fb1b10abSAndroid Build Coastguard Worker
215*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < MAX_MODES; ++i) {
216*fb1b10abSAndroid Build Coastguard Worker if (cpi->sf.thresh_mult[i] < INT_MAX) {
217*fb1b10abSAndroid Build Coastguard Worker x->rd_threshes[i] = cpi->sf.thresh_mult[i] * q / 100;
218*fb1b10abSAndroid Build Coastguard Worker } else {
219*fb1b10abSAndroid Build Coastguard Worker x->rd_threshes[i] = INT_MAX;
220*fb1b10abSAndroid Build Coastguard Worker }
221*fb1b10abSAndroid Build Coastguard Worker
222*fb1b10abSAndroid Build Coastguard Worker cpi->rd_baseline_thresh[i] = x->rd_threshes[i];
223*fb1b10abSAndroid Build Coastguard Worker }
224*fb1b10abSAndroid Build Coastguard Worker } else {
225*fb1b10abSAndroid Build Coastguard Worker cpi->RDDIV = 100;
226*fb1b10abSAndroid Build Coastguard Worker
227*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < MAX_MODES; ++i) {
228*fb1b10abSAndroid Build Coastguard Worker if (cpi->sf.thresh_mult[i] < (INT_MAX / q)) {
229*fb1b10abSAndroid Build Coastguard Worker x->rd_threshes[i] = cpi->sf.thresh_mult[i] * q;
230*fb1b10abSAndroid Build Coastguard Worker } else {
231*fb1b10abSAndroid Build Coastguard Worker x->rd_threshes[i] = INT_MAX;
232*fb1b10abSAndroid Build Coastguard Worker }
233*fb1b10abSAndroid Build Coastguard Worker
234*fb1b10abSAndroid Build Coastguard Worker cpi->rd_baseline_thresh[i] = x->rd_threshes[i];
235*fb1b10abSAndroid Build Coastguard Worker }
236*fb1b10abSAndroid Build Coastguard Worker }
237*fb1b10abSAndroid Build Coastguard Worker
238*fb1b10abSAndroid Build Coastguard Worker {
239*fb1b10abSAndroid Build Coastguard Worker /* build token cost array for the type of frame we have now */
240*fb1b10abSAndroid Build Coastguard Worker FRAME_CONTEXT *l = &cpi->lfc_n;
241*fb1b10abSAndroid Build Coastguard Worker
242*fb1b10abSAndroid Build Coastguard Worker if (cpi->common.refresh_alt_ref_frame) {
243*fb1b10abSAndroid Build Coastguard Worker l = &cpi->lfc_a;
244*fb1b10abSAndroid Build Coastguard Worker } else if (cpi->common.refresh_golden_frame) {
245*fb1b10abSAndroid Build Coastguard Worker l = &cpi->lfc_g;
246*fb1b10abSAndroid Build Coastguard Worker }
247*fb1b10abSAndroid Build Coastguard Worker
248*fb1b10abSAndroid Build Coastguard Worker fill_token_costs(cpi->mb.token_costs,
249*fb1b10abSAndroid Build Coastguard Worker (const vp8_prob(*)[8][3][11])l->coef_probs);
250*fb1b10abSAndroid Build Coastguard Worker /*
251*fb1b10abSAndroid Build Coastguard Worker fill_token_costs(
252*fb1b10abSAndroid Build Coastguard Worker cpi->mb.token_costs,
253*fb1b10abSAndroid Build Coastguard Worker (const vp8_prob( *)[8][3][11]) cpi->common.fc.coef_probs);
254*fb1b10abSAndroid Build Coastguard Worker */
255*fb1b10abSAndroid Build Coastguard Worker
256*fb1b10abSAndroid Build Coastguard Worker /* TODO make these mode costs depend on last,alt or gold too. (jbb) */
257*fb1b10abSAndroid Build Coastguard Worker vp8_init_mode_costs(cpi);
258*fb1b10abSAndroid Build Coastguard Worker }
259*fb1b10abSAndroid Build Coastguard Worker }
260*fb1b10abSAndroid Build Coastguard Worker
vp8_auto_select_speed(VP8_COMP * cpi)261*fb1b10abSAndroid Build Coastguard Worker void vp8_auto_select_speed(VP8_COMP *cpi) {
262*fb1b10abSAndroid Build Coastguard Worker int milliseconds_for_compress = (int)(1000000 / cpi->framerate);
263*fb1b10abSAndroid Build Coastguard Worker
264*fb1b10abSAndroid Build Coastguard Worker milliseconds_for_compress =
265*fb1b10abSAndroid Build Coastguard Worker milliseconds_for_compress * (16 - cpi->oxcf.cpu_used) / 16;
266*fb1b10abSAndroid Build Coastguard Worker
267*fb1b10abSAndroid Build Coastguard Worker #if 0
268*fb1b10abSAndroid Build Coastguard Worker
269*fb1b10abSAndroid Build Coastguard Worker if (0)
270*fb1b10abSAndroid Build Coastguard Worker {
271*fb1b10abSAndroid Build Coastguard Worker FILE *f;
272*fb1b10abSAndroid Build Coastguard Worker
273*fb1b10abSAndroid Build Coastguard Worker f = fopen("speed.stt", "a");
274*fb1b10abSAndroid Build Coastguard Worker fprintf(f, " %8ld %10ld %10ld %10ld\n",
275*fb1b10abSAndroid Build Coastguard Worker cpi->common.current_video_frame, cpi->Speed, milliseconds_for_compress, cpi->avg_pick_mode_time);
276*fb1b10abSAndroid Build Coastguard Worker fclose(f);
277*fb1b10abSAndroid Build Coastguard Worker }
278*fb1b10abSAndroid Build Coastguard Worker
279*fb1b10abSAndroid Build Coastguard Worker #endif
280*fb1b10abSAndroid Build Coastguard Worker
281*fb1b10abSAndroid Build Coastguard Worker if (cpi->avg_pick_mode_time < milliseconds_for_compress &&
282*fb1b10abSAndroid Build Coastguard Worker (cpi->avg_encode_time - cpi->avg_pick_mode_time) <
283*fb1b10abSAndroid Build Coastguard Worker milliseconds_for_compress) {
284*fb1b10abSAndroid Build Coastguard Worker if (cpi->avg_pick_mode_time == 0) {
285*fb1b10abSAndroid Build Coastguard Worker cpi->Speed = 4;
286*fb1b10abSAndroid Build Coastguard Worker } else {
287*fb1b10abSAndroid Build Coastguard Worker if (milliseconds_for_compress * 100 < cpi->avg_encode_time * 95) {
288*fb1b10abSAndroid Build Coastguard Worker cpi->Speed += 2;
289*fb1b10abSAndroid Build Coastguard Worker cpi->avg_pick_mode_time = 0;
290*fb1b10abSAndroid Build Coastguard Worker cpi->avg_encode_time = 0;
291*fb1b10abSAndroid Build Coastguard Worker
292*fb1b10abSAndroid Build Coastguard Worker if (cpi->Speed > 16) {
293*fb1b10abSAndroid Build Coastguard Worker cpi->Speed = 16;
294*fb1b10abSAndroid Build Coastguard Worker }
295*fb1b10abSAndroid Build Coastguard Worker }
296*fb1b10abSAndroid Build Coastguard Worker
297*fb1b10abSAndroid Build Coastguard Worker if (milliseconds_for_compress * 100 >
298*fb1b10abSAndroid Build Coastguard Worker cpi->avg_encode_time * auto_speed_thresh[cpi->Speed]) {
299*fb1b10abSAndroid Build Coastguard Worker cpi->Speed -= 1;
300*fb1b10abSAndroid Build Coastguard Worker cpi->avg_pick_mode_time = 0;
301*fb1b10abSAndroid Build Coastguard Worker cpi->avg_encode_time = 0;
302*fb1b10abSAndroid Build Coastguard Worker
303*fb1b10abSAndroid Build Coastguard Worker /* In real-time mode, cpi->speed is in [4, 16]. */
304*fb1b10abSAndroid Build Coastguard Worker if (cpi->Speed < 4) {
305*fb1b10abSAndroid Build Coastguard Worker cpi->Speed = 4;
306*fb1b10abSAndroid Build Coastguard Worker }
307*fb1b10abSAndroid Build Coastguard Worker }
308*fb1b10abSAndroid Build Coastguard Worker }
309*fb1b10abSAndroid Build Coastguard Worker } else {
310*fb1b10abSAndroid Build Coastguard Worker cpi->Speed += 4;
311*fb1b10abSAndroid Build Coastguard Worker
312*fb1b10abSAndroid Build Coastguard Worker if (cpi->Speed > 16) cpi->Speed = 16;
313*fb1b10abSAndroid Build Coastguard Worker
314*fb1b10abSAndroid Build Coastguard Worker cpi->avg_pick_mode_time = 0;
315*fb1b10abSAndroid Build Coastguard Worker cpi->avg_encode_time = 0;
316*fb1b10abSAndroid Build Coastguard Worker }
317*fb1b10abSAndroid Build Coastguard Worker }
318*fb1b10abSAndroid Build Coastguard Worker
vp8_block_error_c(short * coeff,short * dqcoeff)319*fb1b10abSAndroid Build Coastguard Worker int vp8_block_error_c(short *coeff, short *dqcoeff) {
320*fb1b10abSAndroid Build Coastguard Worker int i;
321*fb1b10abSAndroid Build Coastguard Worker int error = 0;
322*fb1b10abSAndroid Build Coastguard Worker
323*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
324*fb1b10abSAndroid Build Coastguard Worker int this_diff = coeff[i] - dqcoeff[i];
325*fb1b10abSAndroid Build Coastguard Worker error += this_diff * this_diff;
326*fb1b10abSAndroid Build Coastguard Worker }
327*fb1b10abSAndroid Build Coastguard Worker
328*fb1b10abSAndroid Build Coastguard Worker return error;
329*fb1b10abSAndroid Build Coastguard Worker }
330*fb1b10abSAndroid Build Coastguard Worker
vp8_mbblock_error_c(MACROBLOCK * mb,int dc)331*fb1b10abSAndroid Build Coastguard Worker int vp8_mbblock_error_c(MACROBLOCK *mb, int dc) {
332*fb1b10abSAndroid Build Coastguard Worker BLOCK *be;
333*fb1b10abSAndroid Build Coastguard Worker BLOCKD *bd;
334*fb1b10abSAndroid Build Coastguard Worker int i, j;
335*fb1b10abSAndroid Build Coastguard Worker int berror, error = 0;
336*fb1b10abSAndroid Build Coastguard Worker
337*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
338*fb1b10abSAndroid Build Coastguard Worker be = &mb->block[i];
339*fb1b10abSAndroid Build Coastguard Worker bd = &mb->e_mbd.block[i];
340*fb1b10abSAndroid Build Coastguard Worker
341*fb1b10abSAndroid Build Coastguard Worker berror = 0;
342*fb1b10abSAndroid Build Coastguard Worker
343*fb1b10abSAndroid Build Coastguard Worker for (j = dc; j < 16; ++j) {
344*fb1b10abSAndroid Build Coastguard Worker int this_diff = be->coeff[j] - bd->dqcoeff[j];
345*fb1b10abSAndroid Build Coastguard Worker berror += this_diff * this_diff;
346*fb1b10abSAndroid Build Coastguard Worker }
347*fb1b10abSAndroid Build Coastguard Worker
348*fb1b10abSAndroid Build Coastguard Worker error += berror;
349*fb1b10abSAndroid Build Coastguard Worker }
350*fb1b10abSAndroid Build Coastguard Worker
351*fb1b10abSAndroid Build Coastguard Worker return error;
352*fb1b10abSAndroid Build Coastguard Worker }
353*fb1b10abSAndroid Build Coastguard Worker
vp8_mbuverror_c(MACROBLOCK * mb)354*fb1b10abSAndroid Build Coastguard Worker int vp8_mbuverror_c(MACROBLOCK *mb) {
355*fb1b10abSAndroid Build Coastguard Worker BLOCK *be;
356*fb1b10abSAndroid Build Coastguard Worker BLOCKD *bd;
357*fb1b10abSAndroid Build Coastguard Worker
358*fb1b10abSAndroid Build Coastguard Worker int i;
359*fb1b10abSAndroid Build Coastguard Worker int error = 0;
360*fb1b10abSAndroid Build Coastguard Worker
361*fb1b10abSAndroid Build Coastguard Worker for (i = 16; i < 24; ++i) {
362*fb1b10abSAndroid Build Coastguard Worker be = &mb->block[i];
363*fb1b10abSAndroid Build Coastguard Worker bd = &mb->e_mbd.block[i];
364*fb1b10abSAndroid Build Coastguard Worker
365*fb1b10abSAndroid Build Coastguard Worker error += vp8_block_error_c(be->coeff, bd->dqcoeff);
366*fb1b10abSAndroid Build Coastguard Worker }
367*fb1b10abSAndroid Build Coastguard Worker
368*fb1b10abSAndroid Build Coastguard Worker return error;
369*fb1b10abSAndroid Build Coastguard Worker }
370*fb1b10abSAndroid Build Coastguard Worker
VP8_UVSSE(MACROBLOCK * x)371*fb1b10abSAndroid Build Coastguard Worker int VP8_UVSSE(MACROBLOCK *x) {
372*fb1b10abSAndroid Build Coastguard Worker unsigned char *uptr, *vptr;
373*fb1b10abSAndroid Build Coastguard Worker unsigned char *upred_ptr = (*(x->block[16].base_src) + x->block[16].src);
374*fb1b10abSAndroid Build Coastguard Worker unsigned char *vpred_ptr = (*(x->block[20].base_src) + x->block[20].src);
375*fb1b10abSAndroid Build Coastguard Worker int uv_stride = x->block[16].src_stride;
376*fb1b10abSAndroid Build Coastguard Worker
377*fb1b10abSAndroid Build Coastguard Worker unsigned int sse1 = 0;
378*fb1b10abSAndroid Build Coastguard Worker unsigned int sse2 = 0;
379*fb1b10abSAndroid Build Coastguard Worker int mv_row = x->e_mbd.mode_info_context->mbmi.mv.as_mv.row;
380*fb1b10abSAndroid Build Coastguard Worker int mv_col = x->e_mbd.mode_info_context->mbmi.mv.as_mv.col;
381*fb1b10abSAndroid Build Coastguard Worker int offset;
382*fb1b10abSAndroid Build Coastguard Worker int pre_stride = x->e_mbd.pre.uv_stride;
383*fb1b10abSAndroid Build Coastguard Worker
384*fb1b10abSAndroid Build Coastguard Worker if (mv_row < 0) {
385*fb1b10abSAndroid Build Coastguard Worker mv_row -= 1;
386*fb1b10abSAndroid Build Coastguard Worker } else {
387*fb1b10abSAndroid Build Coastguard Worker mv_row += 1;
388*fb1b10abSAndroid Build Coastguard Worker }
389*fb1b10abSAndroid Build Coastguard Worker
390*fb1b10abSAndroid Build Coastguard Worker if (mv_col < 0) {
391*fb1b10abSAndroid Build Coastguard Worker mv_col -= 1;
392*fb1b10abSAndroid Build Coastguard Worker } else {
393*fb1b10abSAndroid Build Coastguard Worker mv_col += 1;
394*fb1b10abSAndroid Build Coastguard Worker }
395*fb1b10abSAndroid Build Coastguard Worker
396*fb1b10abSAndroid Build Coastguard Worker mv_row /= 2;
397*fb1b10abSAndroid Build Coastguard Worker mv_col /= 2;
398*fb1b10abSAndroid Build Coastguard Worker
399*fb1b10abSAndroid Build Coastguard Worker offset = (mv_row >> 3) * pre_stride + (mv_col >> 3);
400*fb1b10abSAndroid Build Coastguard Worker uptr = x->e_mbd.pre.u_buffer + offset;
401*fb1b10abSAndroid Build Coastguard Worker vptr = x->e_mbd.pre.v_buffer + offset;
402*fb1b10abSAndroid Build Coastguard Worker
403*fb1b10abSAndroid Build Coastguard Worker if ((mv_row | mv_col) & 7) {
404*fb1b10abSAndroid Build Coastguard Worker vpx_sub_pixel_variance8x8(uptr, pre_stride, mv_col & 7, mv_row & 7,
405*fb1b10abSAndroid Build Coastguard Worker upred_ptr, uv_stride, &sse2);
406*fb1b10abSAndroid Build Coastguard Worker vpx_sub_pixel_variance8x8(vptr, pre_stride, mv_col & 7, mv_row & 7,
407*fb1b10abSAndroid Build Coastguard Worker vpred_ptr, uv_stride, &sse1);
408*fb1b10abSAndroid Build Coastguard Worker sse2 += sse1;
409*fb1b10abSAndroid Build Coastguard Worker } else {
410*fb1b10abSAndroid Build Coastguard Worker vpx_variance8x8(uptr, pre_stride, upred_ptr, uv_stride, &sse2);
411*fb1b10abSAndroid Build Coastguard Worker vpx_variance8x8(vptr, pre_stride, vpred_ptr, uv_stride, &sse1);
412*fb1b10abSAndroid Build Coastguard Worker sse2 += sse1;
413*fb1b10abSAndroid Build Coastguard Worker }
414*fb1b10abSAndroid Build Coastguard Worker return sse2;
415*fb1b10abSAndroid Build Coastguard Worker }
416*fb1b10abSAndroid Build Coastguard Worker
cost_coeffs(MACROBLOCK * mb,BLOCKD * b,int type,ENTROPY_CONTEXT * a,ENTROPY_CONTEXT * l)417*fb1b10abSAndroid Build Coastguard Worker static int cost_coeffs(MACROBLOCK *mb, BLOCKD *b, int type, ENTROPY_CONTEXT *a,
418*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *l) {
419*fb1b10abSAndroid Build Coastguard Worker int c = !type; /* start at coef 0, unless Y with Y2 */
420*fb1b10abSAndroid Build Coastguard Worker int eob = (int)(*b->eob);
421*fb1b10abSAndroid Build Coastguard Worker int pt; /* surrounding block/prev coef predictor */
422*fb1b10abSAndroid Build Coastguard Worker int cost = 0;
423*fb1b10abSAndroid Build Coastguard Worker short *qcoeff_ptr = b->qcoeff;
424*fb1b10abSAndroid Build Coastguard Worker
425*fb1b10abSAndroid Build Coastguard Worker VP8_COMBINEENTROPYCONTEXTS(pt, *a, *l);
426*fb1b10abSAndroid Build Coastguard Worker
427*fb1b10abSAndroid Build Coastguard Worker assert(eob <= 16);
428*fb1b10abSAndroid Build Coastguard Worker for (; c < eob; ++c) {
429*fb1b10abSAndroid Build Coastguard Worker const int v = qcoeff_ptr[vp8_default_zig_zag1d[c]];
430*fb1b10abSAndroid Build Coastguard Worker const int t = vp8_dct_value_tokens_ptr[v].Token;
431*fb1b10abSAndroid Build Coastguard Worker cost += mb->token_costs[type][vp8_coef_bands[c]][pt][t];
432*fb1b10abSAndroid Build Coastguard Worker cost += vp8_dct_value_cost_ptr[v];
433*fb1b10abSAndroid Build Coastguard Worker pt = vp8_prev_token_class[t];
434*fb1b10abSAndroid Build Coastguard Worker }
435*fb1b10abSAndroid Build Coastguard Worker
436*fb1b10abSAndroid Build Coastguard Worker if (c < 16) {
437*fb1b10abSAndroid Build Coastguard Worker cost += mb->token_costs[type][vp8_coef_bands[c]][pt][DCT_EOB_TOKEN];
438*fb1b10abSAndroid Build Coastguard Worker }
439*fb1b10abSAndroid Build Coastguard Worker
440*fb1b10abSAndroid Build Coastguard Worker pt = (c != !type); /* is eob first coefficient; */
441*fb1b10abSAndroid Build Coastguard Worker *a = *l = pt;
442*fb1b10abSAndroid Build Coastguard Worker
443*fb1b10abSAndroid Build Coastguard Worker return cost;
444*fb1b10abSAndroid Build Coastguard Worker }
445*fb1b10abSAndroid Build Coastguard Worker
vp8_rdcost_mby(MACROBLOCK * mb)446*fb1b10abSAndroid Build Coastguard Worker static int vp8_rdcost_mby(MACROBLOCK *mb) {
447*fb1b10abSAndroid Build Coastguard Worker int cost = 0;
448*fb1b10abSAndroid Build Coastguard Worker int b;
449*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *x = &mb->e_mbd;
450*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT_PLANES t_above, t_left;
451*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *ta;
452*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *tl;
453*fb1b10abSAndroid Build Coastguard Worker
454*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_above, mb->e_mbd.above_context, sizeof(ENTROPY_CONTEXT_PLANES));
455*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_left, mb->e_mbd.left_context, sizeof(ENTROPY_CONTEXT_PLANES));
456*fb1b10abSAndroid Build Coastguard Worker
457*fb1b10abSAndroid Build Coastguard Worker ta = (ENTROPY_CONTEXT *)&t_above;
458*fb1b10abSAndroid Build Coastguard Worker tl = (ENTROPY_CONTEXT *)&t_left;
459*fb1b10abSAndroid Build Coastguard Worker
460*fb1b10abSAndroid Build Coastguard Worker for (b = 0; b < 16; ++b) {
461*fb1b10abSAndroid Build Coastguard Worker cost += cost_coeffs(mb, x->block + b, PLANE_TYPE_Y_NO_DC,
462*fb1b10abSAndroid Build Coastguard Worker ta + vp8_block2above[b], tl + vp8_block2left[b]);
463*fb1b10abSAndroid Build Coastguard Worker }
464*fb1b10abSAndroid Build Coastguard Worker
465*fb1b10abSAndroid Build Coastguard Worker cost += cost_coeffs(mb, x->block + 24, PLANE_TYPE_Y2,
466*fb1b10abSAndroid Build Coastguard Worker ta + vp8_block2above[24], tl + vp8_block2left[24]);
467*fb1b10abSAndroid Build Coastguard Worker
468*fb1b10abSAndroid Build Coastguard Worker return cost;
469*fb1b10abSAndroid Build Coastguard Worker }
470*fb1b10abSAndroid Build Coastguard Worker
macro_block_yrd(MACROBLOCK * mb,int * Rate,int * Distortion)471*fb1b10abSAndroid Build Coastguard Worker static void macro_block_yrd(MACROBLOCK *mb, int *Rate, int *Distortion) {
472*fb1b10abSAndroid Build Coastguard Worker int b;
473*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *const x = &mb->e_mbd;
474*fb1b10abSAndroid Build Coastguard Worker BLOCK *const mb_y2 = mb->block + 24;
475*fb1b10abSAndroid Build Coastguard Worker BLOCKD *const x_y2 = x->block + 24;
476*fb1b10abSAndroid Build Coastguard Worker short *Y2DCPtr = mb_y2->src_diff;
477*fb1b10abSAndroid Build Coastguard Worker BLOCK *beptr;
478*fb1b10abSAndroid Build Coastguard Worker int d;
479*fb1b10abSAndroid Build Coastguard Worker
480*fb1b10abSAndroid Build Coastguard Worker vp8_subtract_mby(mb->src_diff, *(mb->block[0].base_src),
481*fb1b10abSAndroid Build Coastguard Worker mb->block[0].src_stride, mb->e_mbd.predictor, 16);
482*fb1b10abSAndroid Build Coastguard Worker
483*fb1b10abSAndroid Build Coastguard Worker /* Fdct and building the 2nd order block */
484*fb1b10abSAndroid Build Coastguard Worker for (beptr = mb->block; beptr < mb->block + 16; beptr += 2) {
485*fb1b10abSAndroid Build Coastguard Worker mb->short_fdct8x4(beptr->src_diff, beptr->coeff, 32);
486*fb1b10abSAndroid Build Coastguard Worker *Y2DCPtr++ = beptr->coeff[0];
487*fb1b10abSAndroid Build Coastguard Worker *Y2DCPtr++ = beptr->coeff[16];
488*fb1b10abSAndroid Build Coastguard Worker }
489*fb1b10abSAndroid Build Coastguard Worker
490*fb1b10abSAndroid Build Coastguard Worker /* 2nd order fdct */
491*fb1b10abSAndroid Build Coastguard Worker mb->short_walsh4x4(mb_y2->src_diff, mb_y2->coeff, 8);
492*fb1b10abSAndroid Build Coastguard Worker
493*fb1b10abSAndroid Build Coastguard Worker /* Quantization */
494*fb1b10abSAndroid Build Coastguard Worker for (b = 0; b < 16; ++b) {
495*fb1b10abSAndroid Build Coastguard Worker mb->quantize_b(&mb->block[b], &mb->e_mbd.block[b]);
496*fb1b10abSAndroid Build Coastguard Worker }
497*fb1b10abSAndroid Build Coastguard Worker
498*fb1b10abSAndroid Build Coastguard Worker /* DC predication and Quantization of 2nd Order block */
499*fb1b10abSAndroid Build Coastguard Worker mb->quantize_b(mb_y2, x_y2);
500*fb1b10abSAndroid Build Coastguard Worker
501*fb1b10abSAndroid Build Coastguard Worker /* Distortion */
502*fb1b10abSAndroid Build Coastguard Worker d = vp8_mbblock_error(mb, 1) << 2;
503*fb1b10abSAndroid Build Coastguard Worker d += vp8_block_error(mb_y2->coeff, x_y2->dqcoeff);
504*fb1b10abSAndroid Build Coastguard Worker
505*fb1b10abSAndroid Build Coastguard Worker *Distortion = (d >> 4);
506*fb1b10abSAndroid Build Coastguard Worker
507*fb1b10abSAndroid Build Coastguard Worker /* rate */
508*fb1b10abSAndroid Build Coastguard Worker *Rate = vp8_rdcost_mby(mb);
509*fb1b10abSAndroid Build Coastguard Worker }
510*fb1b10abSAndroid Build Coastguard Worker
copy_predictor(unsigned char * dst,const unsigned char * predictor)511*fb1b10abSAndroid Build Coastguard Worker static void copy_predictor(unsigned char *dst, const unsigned char *predictor) {
512*fb1b10abSAndroid Build Coastguard Worker const unsigned int *p = (const unsigned int *)predictor;
513*fb1b10abSAndroid Build Coastguard Worker unsigned int *d = (unsigned int *)dst;
514*fb1b10abSAndroid Build Coastguard Worker d[0] = p[0];
515*fb1b10abSAndroid Build Coastguard Worker d[4] = p[4];
516*fb1b10abSAndroid Build Coastguard Worker d[8] = p[8];
517*fb1b10abSAndroid Build Coastguard Worker d[12] = p[12];
518*fb1b10abSAndroid Build Coastguard Worker }
rd_pick_intra4x4block(MACROBLOCK * x,BLOCK * be,BLOCKD * b,B_PREDICTION_MODE * best_mode,const int * bmode_costs,ENTROPY_CONTEXT * a,ENTROPY_CONTEXT * l,int * bestrate,int * bestratey,int * bestdistortion)519*fb1b10abSAndroid Build Coastguard Worker static int rd_pick_intra4x4block(MACROBLOCK *x, BLOCK *be, BLOCKD *b,
520*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE *best_mode,
521*fb1b10abSAndroid Build Coastguard Worker const int *bmode_costs, ENTROPY_CONTEXT *a,
522*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *l,
523*fb1b10abSAndroid Build Coastguard Worker
524*fb1b10abSAndroid Build Coastguard Worker int *bestrate, int *bestratey,
525*fb1b10abSAndroid Build Coastguard Worker int *bestdistortion) {
526*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE mode;
527*fb1b10abSAndroid Build Coastguard Worker int best_rd = INT_MAX;
528*fb1b10abSAndroid Build Coastguard Worker int rate = 0;
529*fb1b10abSAndroid Build Coastguard Worker int distortion;
530*fb1b10abSAndroid Build Coastguard Worker
531*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT ta = *a, tempa = *a;
532*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT tl = *l, templ = *l;
533*fb1b10abSAndroid Build Coastguard Worker /*
534*fb1b10abSAndroid Build Coastguard Worker * The predictor buffer is a 2d buffer with a stride of 16. Create
535*fb1b10abSAndroid Build Coastguard Worker * a temp buffer that meets the stride requirements, but we are only
536*fb1b10abSAndroid Build Coastguard Worker * interested in the left 4x4 block
537*fb1b10abSAndroid Build Coastguard Worker * */
538*fb1b10abSAndroid Build Coastguard Worker DECLARE_ALIGNED(16, unsigned char, best_predictor[16 * 4]);
539*fb1b10abSAndroid Build Coastguard Worker DECLARE_ALIGNED(16, short, best_dqcoeff[16]);
540*fb1b10abSAndroid Build Coastguard Worker int dst_stride = x->e_mbd.dst.y_stride;
541*fb1b10abSAndroid Build Coastguard Worker unsigned char *dst = x->e_mbd.dst.y_buffer + b->offset;
542*fb1b10abSAndroid Build Coastguard Worker
543*fb1b10abSAndroid Build Coastguard Worker unsigned char *Above = dst - dst_stride;
544*fb1b10abSAndroid Build Coastguard Worker unsigned char *yleft = dst - 1;
545*fb1b10abSAndroid Build Coastguard Worker unsigned char top_left = Above[-1];
546*fb1b10abSAndroid Build Coastguard Worker
547*fb1b10abSAndroid Build Coastguard Worker for (mode = B_DC_PRED; mode <= B_HU_PRED; ++mode) {
548*fb1b10abSAndroid Build Coastguard Worker int this_rd;
549*fb1b10abSAndroid Build Coastguard Worker int ratey;
550*fb1b10abSAndroid Build Coastguard Worker
551*fb1b10abSAndroid Build Coastguard Worker rate = bmode_costs[mode];
552*fb1b10abSAndroid Build Coastguard Worker
553*fb1b10abSAndroid Build Coastguard Worker vp8_intra4x4_predict(Above, yleft, dst_stride, mode, b->predictor, 16,
554*fb1b10abSAndroid Build Coastguard Worker top_left);
555*fb1b10abSAndroid Build Coastguard Worker vp8_subtract_b(be, b, 16);
556*fb1b10abSAndroid Build Coastguard Worker x->short_fdct4x4(be->src_diff, be->coeff, 32);
557*fb1b10abSAndroid Build Coastguard Worker x->quantize_b(be, b);
558*fb1b10abSAndroid Build Coastguard Worker
559*fb1b10abSAndroid Build Coastguard Worker tempa = ta;
560*fb1b10abSAndroid Build Coastguard Worker templ = tl;
561*fb1b10abSAndroid Build Coastguard Worker
562*fb1b10abSAndroid Build Coastguard Worker ratey = cost_coeffs(x, b, PLANE_TYPE_Y_WITH_DC, &tempa, &templ);
563*fb1b10abSAndroid Build Coastguard Worker rate += ratey;
564*fb1b10abSAndroid Build Coastguard Worker distortion = vp8_block_error(be->coeff, b->dqcoeff) >> 2;
565*fb1b10abSAndroid Build Coastguard Worker
566*fb1b10abSAndroid Build Coastguard Worker this_rd = RDCOST(x->rdmult, x->rddiv, rate, distortion);
567*fb1b10abSAndroid Build Coastguard Worker
568*fb1b10abSAndroid Build Coastguard Worker if (this_rd < best_rd) {
569*fb1b10abSAndroid Build Coastguard Worker *bestrate = rate;
570*fb1b10abSAndroid Build Coastguard Worker *bestratey = ratey;
571*fb1b10abSAndroid Build Coastguard Worker *bestdistortion = distortion;
572*fb1b10abSAndroid Build Coastguard Worker best_rd = this_rd;
573*fb1b10abSAndroid Build Coastguard Worker *best_mode = mode;
574*fb1b10abSAndroid Build Coastguard Worker *a = tempa;
575*fb1b10abSAndroid Build Coastguard Worker *l = templ;
576*fb1b10abSAndroid Build Coastguard Worker copy_predictor(best_predictor, b->predictor);
577*fb1b10abSAndroid Build Coastguard Worker memcpy(best_dqcoeff, b->dqcoeff, 32);
578*fb1b10abSAndroid Build Coastguard Worker }
579*fb1b10abSAndroid Build Coastguard Worker }
580*fb1b10abSAndroid Build Coastguard Worker b->bmi.as_mode = *best_mode;
581*fb1b10abSAndroid Build Coastguard Worker
582*fb1b10abSAndroid Build Coastguard Worker vp8_short_idct4x4llm(best_dqcoeff, best_predictor, 16, dst, dst_stride);
583*fb1b10abSAndroid Build Coastguard Worker
584*fb1b10abSAndroid Build Coastguard Worker return best_rd;
585*fb1b10abSAndroid Build Coastguard Worker }
586*fb1b10abSAndroid Build Coastguard Worker
rd_pick_intra4x4mby_modes(MACROBLOCK * mb,int * Rate,int * rate_y,int * Distortion,int best_rd)587*fb1b10abSAndroid Build Coastguard Worker static int rd_pick_intra4x4mby_modes(MACROBLOCK *mb, int *Rate, int *rate_y,
588*fb1b10abSAndroid Build Coastguard Worker int *Distortion, int best_rd) {
589*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *const xd = &mb->e_mbd;
590*fb1b10abSAndroid Build Coastguard Worker int i;
591*fb1b10abSAndroid Build Coastguard Worker int cost = mb->mbmode_cost[xd->frame_type][B_PRED];
592*fb1b10abSAndroid Build Coastguard Worker int distortion = 0;
593*fb1b10abSAndroid Build Coastguard Worker int tot_rate_y = 0;
594*fb1b10abSAndroid Build Coastguard Worker int64_t total_rd = 0;
595*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT_PLANES t_above, t_left;
596*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *ta;
597*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *tl;
598*fb1b10abSAndroid Build Coastguard Worker const int *bmode_costs;
599*fb1b10abSAndroid Build Coastguard Worker
600*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_above, mb->e_mbd.above_context, sizeof(ENTROPY_CONTEXT_PLANES));
601*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_left, mb->e_mbd.left_context, sizeof(ENTROPY_CONTEXT_PLANES));
602*fb1b10abSAndroid Build Coastguard Worker
603*fb1b10abSAndroid Build Coastguard Worker ta = (ENTROPY_CONTEXT *)&t_above;
604*fb1b10abSAndroid Build Coastguard Worker tl = (ENTROPY_CONTEXT *)&t_left;
605*fb1b10abSAndroid Build Coastguard Worker
606*fb1b10abSAndroid Build Coastguard Worker intra_prediction_down_copy(xd, xd->dst.y_buffer - xd->dst.y_stride + 16);
607*fb1b10abSAndroid Build Coastguard Worker
608*fb1b10abSAndroid Build Coastguard Worker bmode_costs = mb->inter_bmode_costs;
609*fb1b10abSAndroid Build Coastguard Worker
610*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
611*fb1b10abSAndroid Build Coastguard Worker MODE_INFO *const mic = xd->mode_info_context;
612*fb1b10abSAndroid Build Coastguard Worker const int mis = xd->mode_info_stride;
613*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE best_mode = B_MODE_COUNT;
614*fb1b10abSAndroid Build Coastguard Worker int r = 0, ry = 0, d = 0;
615*fb1b10abSAndroid Build Coastguard Worker
616*fb1b10abSAndroid Build Coastguard Worker if (mb->e_mbd.frame_type == KEY_FRAME) {
617*fb1b10abSAndroid Build Coastguard Worker const B_PREDICTION_MODE A = above_block_mode(mic, i, mis);
618*fb1b10abSAndroid Build Coastguard Worker const B_PREDICTION_MODE L = left_block_mode(mic, i);
619*fb1b10abSAndroid Build Coastguard Worker
620*fb1b10abSAndroid Build Coastguard Worker bmode_costs = mb->bmode_costs[A][L];
621*fb1b10abSAndroid Build Coastguard Worker }
622*fb1b10abSAndroid Build Coastguard Worker
623*fb1b10abSAndroid Build Coastguard Worker total_rd += rd_pick_intra4x4block(
624*fb1b10abSAndroid Build Coastguard Worker mb, mb->block + i, xd->block + i, &best_mode, bmode_costs,
625*fb1b10abSAndroid Build Coastguard Worker ta + vp8_block2above[i], tl + vp8_block2left[i], &r, &ry, &d);
626*fb1b10abSAndroid Build Coastguard Worker
627*fb1b10abSAndroid Build Coastguard Worker cost += r;
628*fb1b10abSAndroid Build Coastguard Worker distortion += d;
629*fb1b10abSAndroid Build Coastguard Worker tot_rate_y += ry;
630*fb1b10abSAndroid Build Coastguard Worker
631*fb1b10abSAndroid Build Coastguard Worker assert(best_mode != B_MODE_COUNT);
632*fb1b10abSAndroid Build Coastguard Worker mic->bmi[i].as_mode = best_mode;
633*fb1b10abSAndroid Build Coastguard Worker
634*fb1b10abSAndroid Build Coastguard Worker if (total_rd >= (int64_t)best_rd) break;
635*fb1b10abSAndroid Build Coastguard Worker }
636*fb1b10abSAndroid Build Coastguard Worker
637*fb1b10abSAndroid Build Coastguard Worker if (total_rd >= (int64_t)best_rd) return INT_MAX;
638*fb1b10abSAndroid Build Coastguard Worker
639*fb1b10abSAndroid Build Coastguard Worker *Rate = cost;
640*fb1b10abSAndroid Build Coastguard Worker *rate_y = tot_rate_y;
641*fb1b10abSAndroid Build Coastguard Worker *Distortion = distortion;
642*fb1b10abSAndroid Build Coastguard Worker
643*fb1b10abSAndroid Build Coastguard Worker return RDCOST(mb->rdmult, mb->rddiv, cost, distortion);
644*fb1b10abSAndroid Build Coastguard Worker }
645*fb1b10abSAndroid Build Coastguard Worker
rd_pick_intra16x16mby_mode(MACROBLOCK * x,int * Rate,int * rate_y,int * Distortion)646*fb1b10abSAndroid Build Coastguard Worker static int rd_pick_intra16x16mby_mode(MACROBLOCK *x, int *Rate, int *rate_y,
647*fb1b10abSAndroid Build Coastguard Worker int *Distortion) {
648*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE mode;
649*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE mode_selected = MB_MODE_COUNT;
650*fb1b10abSAndroid Build Coastguard Worker int rate, ratey;
651*fb1b10abSAndroid Build Coastguard Worker int distortion;
652*fb1b10abSAndroid Build Coastguard Worker int best_rd = INT_MAX;
653*fb1b10abSAndroid Build Coastguard Worker int this_rd;
654*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *xd = &x->e_mbd;
655*fb1b10abSAndroid Build Coastguard Worker
656*fb1b10abSAndroid Build Coastguard Worker /* Y Search for 16x16 intra prediction mode */
657*fb1b10abSAndroid Build Coastguard Worker for (mode = DC_PRED; mode <= TM_PRED; ++mode) {
658*fb1b10abSAndroid Build Coastguard Worker xd->mode_info_context->mbmi.mode = mode;
659*fb1b10abSAndroid Build Coastguard Worker
660*fb1b10abSAndroid Build Coastguard Worker vp8_build_intra_predictors_mby_s(xd, xd->dst.y_buffer - xd->dst.y_stride,
661*fb1b10abSAndroid Build Coastguard Worker xd->dst.y_buffer - 1, xd->dst.y_stride,
662*fb1b10abSAndroid Build Coastguard Worker xd->predictor, 16);
663*fb1b10abSAndroid Build Coastguard Worker
664*fb1b10abSAndroid Build Coastguard Worker macro_block_yrd(x, &ratey, &distortion);
665*fb1b10abSAndroid Build Coastguard Worker rate = ratey +
666*fb1b10abSAndroid Build Coastguard Worker x->mbmode_cost[xd->frame_type][xd->mode_info_context->mbmi.mode];
667*fb1b10abSAndroid Build Coastguard Worker
668*fb1b10abSAndroid Build Coastguard Worker this_rd = RDCOST(x->rdmult, x->rddiv, rate, distortion);
669*fb1b10abSAndroid Build Coastguard Worker
670*fb1b10abSAndroid Build Coastguard Worker if (this_rd < best_rd) {
671*fb1b10abSAndroid Build Coastguard Worker mode_selected = mode;
672*fb1b10abSAndroid Build Coastguard Worker best_rd = this_rd;
673*fb1b10abSAndroid Build Coastguard Worker *Rate = rate;
674*fb1b10abSAndroid Build Coastguard Worker *rate_y = ratey;
675*fb1b10abSAndroid Build Coastguard Worker *Distortion = distortion;
676*fb1b10abSAndroid Build Coastguard Worker }
677*fb1b10abSAndroid Build Coastguard Worker }
678*fb1b10abSAndroid Build Coastguard Worker
679*fb1b10abSAndroid Build Coastguard Worker assert(mode_selected != MB_MODE_COUNT);
680*fb1b10abSAndroid Build Coastguard Worker xd->mode_info_context->mbmi.mode = mode_selected;
681*fb1b10abSAndroid Build Coastguard Worker return best_rd;
682*fb1b10abSAndroid Build Coastguard Worker }
683*fb1b10abSAndroid Build Coastguard Worker
rd_cost_mbuv(MACROBLOCK * mb)684*fb1b10abSAndroid Build Coastguard Worker static int rd_cost_mbuv(MACROBLOCK *mb) {
685*fb1b10abSAndroid Build Coastguard Worker int b;
686*fb1b10abSAndroid Build Coastguard Worker int cost = 0;
687*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *x = &mb->e_mbd;
688*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT_PLANES t_above, t_left;
689*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *ta;
690*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *tl;
691*fb1b10abSAndroid Build Coastguard Worker
692*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_above, mb->e_mbd.above_context, sizeof(ENTROPY_CONTEXT_PLANES));
693*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_left, mb->e_mbd.left_context, sizeof(ENTROPY_CONTEXT_PLANES));
694*fb1b10abSAndroid Build Coastguard Worker
695*fb1b10abSAndroid Build Coastguard Worker ta = (ENTROPY_CONTEXT *)&t_above;
696*fb1b10abSAndroid Build Coastguard Worker tl = (ENTROPY_CONTEXT *)&t_left;
697*fb1b10abSAndroid Build Coastguard Worker
698*fb1b10abSAndroid Build Coastguard Worker for (b = 16; b < 24; ++b) {
699*fb1b10abSAndroid Build Coastguard Worker cost += cost_coeffs(mb, x->block + b, PLANE_TYPE_UV,
700*fb1b10abSAndroid Build Coastguard Worker ta + vp8_block2above[b], tl + vp8_block2left[b]);
701*fb1b10abSAndroid Build Coastguard Worker }
702*fb1b10abSAndroid Build Coastguard Worker
703*fb1b10abSAndroid Build Coastguard Worker return cost;
704*fb1b10abSAndroid Build Coastguard Worker }
705*fb1b10abSAndroid Build Coastguard Worker
rd_inter16x16_uv(VP8_COMP * cpi,MACROBLOCK * x,int * rate,int * distortion,int fullpixel)706*fb1b10abSAndroid Build Coastguard Worker static int rd_inter16x16_uv(VP8_COMP *cpi, MACROBLOCK *x, int *rate,
707*fb1b10abSAndroid Build Coastguard Worker int *distortion, int fullpixel) {
708*fb1b10abSAndroid Build Coastguard Worker (void)cpi;
709*fb1b10abSAndroid Build Coastguard Worker (void)fullpixel;
710*fb1b10abSAndroid Build Coastguard Worker
711*fb1b10abSAndroid Build Coastguard Worker vp8_build_inter16x16_predictors_mbuv(&x->e_mbd);
712*fb1b10abSAndroid Build Coastguard Worker vp8_subtract_mbuv(x->src_diff, x->src.u_buffer, x->src.v_buffer,
713*fb1b10abSAndroid Build Coastguard Worker x->src.uv_stride, &x->e_mbd.predictor[256],
714*fb1b10abSAndroid Build Coastguard Worker &x->e_mbd.predictor[320], 8);
715*fb1b10abSAndroid Build Coastguard Worker
716*fb1b10abSAndroid Build Coastguard Worker vp8_transform_mbuv(x);
717*fb1b10abSAndroid Build Coastguard Worker vp8_quantize_mbuv(x);
718*fb1b10abSAndroid Build Coastguard Worker
719*fb1b10abSAndroid Build Coastguard Worker *rate = rd_cost_mbuv(x);
720*fb1b10abSAndroid Build Coastguard Worker *distortion = vp8_mbuverror(x) / 4;
721*fb1b10abSAndroid Build Coastguard Worker
722*fb1b10abSAndroid Build Coastguard Worker return RDCOST(x->rdmult, x->rddiv, *rate, *distortion);
723*fb1b10abSAndroid Build Coastguard Worker }
724*fb1b10abSAndroid Build Coastguard Worker
rd_inter4x4_uv(VP8_COMP * cpi,MACROBLOCK * x,int * rate,int * distortion,int fullpixel)725*fb1b10abSAndroid Build Coastguard Worker static int rd_inter4x4_uv(VP8_COMP *cpi, MACROBLOCK *x, int *rate,
726*fb1b10abSAndroid Build Coastguard Worker int *distortion, int fullpixel) {
727*fb1b10abSAndroid Build Coastguard Worker (void)cpi;
728*fb1b10abSAndroid Build Coastguard Worker (void)fullpixel;
729*fb1b10abSAndroid Build Coastguard Worker
730*fb1b10abSAndroid Build Coastguard Worker vp8_build_inter4x4_predictors_mbuv(&x->e_mbd);
731*fb1b10abSAndroid Build Coastguard Worker vp8_subtract_mbuv(x->src_diff, x->src.u_buffer, x->src.v_buffer,
732*fb1b10abSAndroid Build Coastguard Worker x->src.uv_stride, &x->e_mbd.predictor[256],
733*fb1b10abSAndroid Build Coastguard Worker &x->e_mbd.predictor[320], 8);
734*fb1b10abSAndroid Build Coastguard Worker
735*fb1b10abSAndroid Build Coastguard Worker vp8_transform_mbuv(x);
736*fb1b10abSAndroid Build Coastguard Worker vp8_quantize_mbuv(x);
737*fb1b10abSAndroid Build Coastguard Worker
738*fb1b10abSAndroid Build Coastguard Worker *rate = rd_cost_mbuv(x);
739*fb1b10abSAndroid Build Coastguard Worker *distortion = vp8_mbuverror(x) / 4;
740*fb1b10abSAndroid Build Coastguard Worker
741*fb1b10abSAndroid Build Coastguard Worker return RDCOST(x->rdmult, x->rddiv, *rate, *distortion);
742*fb1b10abSAndroid Build Coastguard Worker }
743*fb1b10abSAndroid Build Coastguard Worker
rd_pick_intra_mbuv_mode(MACROBLOCK * x,int * rate,int * rate_tokenonly,int * distortion)744*fb1b10abSAndroid Build Coastguard Worker static void rd_pick_intra_mbuv_mode(MACROBLOCK *x, int *rate,
745*fb1b10abSAndroid Build Coastguard Worker int *rate_tokenonly, int *distortion) {
746*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE mode;
747*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE mode_selected = MB_MODE_COUNT;
748*fb1b10abSAndroid Build Coastguard Worker int best_rd = INT_MAX;
749*fb1b10abSAndroid Build Coastguard Worker int d = 0, r = 0;
750*fb1b10abSAndroid Build Coastguard Worker int rate_to;
751*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *xd = &x->e_mbd;
752*fb1b10abSAndroid Build Coastguard Worker
753*fb1b10abSAndroid Build Coastguard Worker for (mode = DC_PRED; mode <= TM_PRED; ++mode) {
754*fb1b10abSAndroid Build Coastguard Worker int this_rate;
755*fb1b10abSAndroid Build Coastguard Worker int this_distortion;
756*fb1b10abSAndroid Build Coastguard Worker int this_rd;
757*fb1b10abSAndroid Build Coastguard Worker
758*fb1b10abSAndroid Build Coastguard Worker xd->mode_info_context->mbmi.uv_mode = mode;
759*fb1b10abSAndroid Build Coastguard Worker
760*fb1b10abSAndroid Build Coastguard Worker vp8_build_intra_predictors_mbuv_s(
761*fb1b10abSAndroid Build Coastguard Worker xd, xd->dst.u_buffer - xd->dst.uv_stride,
762*fb1b10abSAndroid Build Coastguard Worker xd->dst.v_buffer - xd->dst.uv_stride, xd->dst.u_buffer - 1,
763*fb1b10abSAndroid Build Coastguard Worker xd->dst.v_buffer - 1, xd->dst.uv_stride, &xd->predictor[256],
764*fb1b10abSAndroid Build Coastguard Worker &xd->predictor[320], 8);
765*fb1b10abSAndroid Build Coastguard Worker
766*fb1b10abSAndroid Build Coastguard Worker vp8_subtract_mbuv(x->src_diff, x->src.u_buffer, x->src.v_buffer,
767*fb1b10abSAndroid Build Coastguard Worker x->src.uv_stride, &xd->predictor[256],
768*fb1b10abSAndroid Build Coastguard Worker &xd->predictor[320], 8);
769*fb1b10abSAndroid Build Coastguard Worker vp8_transform_mbuv(x);
770*fb1b10abSAndroid Build Coastguard Worker vp8_quantize_mbuv(x);
771*fb1b10abSAndroid Build Coastguard Worker
772*fb1b10abSAndroid Build Coastguard Worker rate_to = rd_cost_mbuv(x);
773*fb1b10abSAndroid Build Coastguard Worker this_rate =
774*fb1b10abSAndroid Build Coastguard Worker rate_to + x->intra_uv_mode_cost[xd->frame_type]
775*fb1b10abSAndroid Build Coastguard Worker [xd->mode_info_context->mbmi.uv_mode];
776*fb1b10abSAndroid Build Coastguard Worker
777*fb1b10abSAndroid Build Coastguard Worker this_distortion = vp8_mbuverror(x) / 4;
778*fb1b10abSAndroid Build Coastguard Worker
779*fb1b10abSAndroid Build Coastguard Worker this_rd = RDCOST(x->rdmult, x->rddiv, this_rate, this_distortion);
780*fb1b10abSAndroid Build Coastguard Worker
781*fb1b10abSAndroid Build Coastguard Worker if (this_rd < best_rd) {
782*fb1b10abSAndroid Build Coastguard Worker best_rd = this_rd;
783*fb1b10abSAndroid Build Coastguard Worker d = this_distortion;
784*fb1b10abSAndroid Build Coastguard Worker r = this_rate;
785*fb1b10abSAndroid Build Coastguard Worker *rate_tokenonly = rate_to;
786*fb1b10abSAndroid Build Coastguard Worker mode_selected = mode;
787*fb1b10abSAndroid Build Coastguard Worker }
788*fb1b10abSAndroid Build Coastguard Worker }
789*fb1b10abSAndroid Build Coastguard Worker
790*fb1b10abSAndroid Build Coastguard Worker *rate = r;
791*fb1b10abSAndroid Build Coastguard Worker *distortion = d;
792*fb1b10abSAndroid Build Coastguard Worker
793*fb1b10abSAndroid Build Coastguard Worker assert(mode_selected != MB_MODE_COUNT);
794*fb1b10abSAndroid Build Coastguard Worker xd->mode_info_context->mbmi.uv_mode = mode_selected;
795*fb1b10abSAndroid Build Coastguard Worker }
796*fb1b10abSAndroid Build Coastguard Worker
vp8_cost_mv_ref(MB_PREDICTION_MODE m,const int near_mv_ref_ct[4])797*fb1b10abSAndroid Build Coastguard Worker int vp8_cost_mv_ref(MB_PREDICTION_MODE m, const int near_mv_ref_ct[4]) {
798*fb1b10abSAndroid Build Coastguard Worker vp8_prob p[VP8_MVREFS - 1];
799*fb1b10abSAndroid Build Coastguard Worker assert(NEARESTMV <= m && m <= SPLITMV);
800*fb1b10abSAndroid Build Coastguard Worker vp8_mv_ref_probs(p, near_mv_ref_ct);
801*fb1b10abSAndroid Build Coastguard Worker return vp8_cost_token(vp8_mv_ref_tree, p,
802*fb1b10abSAndroid Build Coastguard Worker vp8_mv_ref_encoding_array + (m - NEARESTMV));
803*fb1b10abSAndroid Build Coastguard Worker }
804*fb1b10abSAndroid Build Coastguard Worker
vp8_set_mbmode_and_mvs(MACROBLOCK * x,MB_PREDICTION_MODE mb,int_mv * mv)805*fb1b10abSAndroid Build Coastguard Worker void vp8_set_mbmode_and_mvs(MACROBLOCK *x, MB_PREDICTION_MODE mb, int_mv *mv) {
806*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mode = mb;
807*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mv.as_int = mv->as_int;
808*fb1b10abSAndroid Build Coastguard Worker }
809*fb1b10abSAndroid Build Coastguard Worker
labels2mode(MACROBLOCK * x,int const * labelings,int which_label,B_PREDICTION_MODE this_mode,int_mv * this_mv,int_mv * best_ref_mv,int * mvcost[2])810*fb1b10abSAndroid Build Coastguard Worker static int labels2mode(MACROBLOCK *x, int const *labelings, int which_label,
811*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE this_mode, int_mv *this_mv,
812*fb1b10abSAndroid Build Coastguard Worker int_mv *best_ref_mv, int *mvcost[2]) {
813*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *const xd = &x->e_mbd;
814*fb1b10abSAndroid Build Coastguard Worker MODE_INFO *const mic = xd->mode_info_context;
815*fb1b10abSAndroid Build Coastguard Worker const int mis = xd->mode_info_stride;
816*fb1b10abSAndroid Build Coastguard Worker
817*fb1b10abSAndroid Build Coastguard Worker int cost = 0;
818*fb1b10abSAndroid Build Coastguard Worker int thismvcost = 0;
819*fb1b10abSAndroid Build Coastguard Worker
820*fb1b10abSAndroid Build Coastguard Worker /* We have to be careful retrieving previously-encoded motion vectors.
821*fb1b10abSAndroid Build Coastguard Worker Ones from this macroblock have to be pulled from the BLOCKD array
822*fb1b10abSAndroid Build Coastguard Worker as they have not yet made it to the bmi array in our MB_MODE_INFO. */
823*fb1b10abSAndroid Build Coastguard Worker
824*fb1b10abSAndroid Build Coastguard Worker int i = 0;
825*fb1b10abSAndroid Build Coastguard Worker
826*fb1b10abSAndroid Build Coastguard Worker do {
827*fb1b10abSAndroid Build Coastguard Worker BLOCKD *const d = xd->block + i;
828*fb1b10abSAndroid Build Coastguard Worker const int row = i >> 2, col = i & 3;
829*fb1b10abSAndroid Build Coastguard Worker
830*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE m;
831*fb1b10abSAndroid Build Coastguard Worker
832*fb1b10abSAndroid Build Coastguard Worker if (labelings[i] != which_label) continue;
833*fb1b10abSAndroid Build Coastguard Worker
834*fb1b10abSAndroid Build Coastguard Worker if (col && labelings[i] == labelings[i - 1]) {
835*fb1b10abSAndroid Build Coastguard Worker m = LEFT4X4;
836*fb1b10abSAndroid Build Coastguard Worker } else if (row && labelings[i] == labelings[i - 4]) {
837*fb1b10abSAndroid Build Coastguard Worker m = ABOVE4X4;
838*fb1b10abSAndroid Build Coastguard Worker } else {
839*fb1b10abSAndroid Build Coastguard Worker /* the only time we should do costing for new motion vector
840*fb1b10abSAndroid Build Coastguard Worker * or mode is when we are on a new label (jbb May 08, 2007)
841*fb1b10abSAndroid Build Coastguard Worker */
842*fb1b10abSAndroid Build Coastguard Worker switch (m = this_mode) {
843*fb1b10abSAndroid Build Coastguard Worker case NEW4X4:
844*fb1b10abSAndroid Build Coastguard Worker thismvcost = vp8_mv_bit_cost(this_mv, best_ref_mv, mvcost, 102);
845*fb1b10abSAndroid Build Coastguard Worker break;
846*fb1b10abSAndroid Build Coastguard Worker case LEFT4X4:
847*fb1b10abSAndroid Build Coastguard Worker this_mv->as_int = col ? d[-1].bmi.mv.as_int : left_block_mv(mic, i);
848*fb1b10abSAndroid Build Coastguard Worker break;
849*fb1b10abSAndroid Build Coastguard Worker case ABOVE4X4:
850*fb1b10abSAndroid Build Coastguard Worker this_mv->as_int =
851*fb1b10abSAndroid Build Coastguard Worker row ? d[-4].bmi.mv.as_int : above_block_mv(mic, i, mis);
852*fb1b10abSAndroid Build Coastguard Worker break;
853*fb1b10abSAndroid Build Coastguard Worker case ZERO4X4: this_mv->as_int = 0; break;
854*fb1b10abSAndroid Build Coastguard Worker default: break;
855*fb1b10abSAndroid Build Coastguard Worker }
856*fb1b10abSAndroid Build Coastguard Worker
857*fb1b10abSAndroid Build Coastguard Worker if (m == ABOVE4X4) { /* replace above with left if same */
858*fb1b10abSAndroid Build Coastguard Worker int_mv left_mv;
859*fb1b10abSAndroid Build Coastguard Worker
860*fb1b10abSAndroid Build Coastguard Worker left_mv.as_int = col ? d[-1].bmi.mv.as_int : left_block_mv(mic, i);
861*fb1b10abSAndroid Build Coastguard Worker
862*fb1b10abSAndroid Build Coastguard Worker if (left_mv.as_int == this_mv->as_int) m = LEFT4X4;
863*fb1b10abSAndroid Build Coastguard Worker }
864*fb1b10abSAndroid Build Coastguard Worker
865*fb1b10abSAndroid Build Coastguard Worker cost = x->inter_bmode_costs[m];
866*fb1b10abSAndroid Build Coastguard Worker }
867*fb1b10abSAndroid Build Coastguard Worker
868*fb1b10abSAndroid Build Coastguard Worker d->bmi.mv.as_int = this_mv->as_int;
869*fb1b10abSAndroid Build Coastguard Worker
870*fb1b10abSAndroid Build Coastguard Worker x->partition_info->bmi[i].mode = m;
871*fb1b10abSAndroid Build Coastguard Worker x->partition_info->bmi[i].mv.as_int = this_mv->as_int;
872*fb1b10abSAndroid Build Coastguard Worker
873*fb1b10abSAndroid Build Coastguard Worker } while (++i < 16);
874*fb1b10abSAndroid Build Coastguard Worker
875*fb1b10abSAndroid Build Coastguard Worker cost += thismvcost;
876*fb1b10abSAndroid Build Coastguard Worker return cost;
877*fb1b10abSAndroid Build Coastguard Worker }
878*fb1b10abSAndroid Build Coastguard Worker
rdcost_mbsegment_y(MACROBLOCK * mb,const int * labels,int which_label,ENTROPY_CONTEXT * ta,ENTROPY_CONTEXT * tl)879*fb1b10abSAndroid Build Coastguard Worker static int rdcost_mbsegment_y(MACROBLOCK *mb, const int *labels,
880*fb1b10abSAndroid Build Coastguard Worker int which_label, ENTROPY_CONTEXT *ta,
881*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *tl) {
882*fb1b10abSAndroid Build Coastguard Worker int cost = 0;
883*fb1b10abSAndroid Build Coastguard Worker int b;
884*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *x = &mb->e_mbd;
885*fb1b10abSAndroid Build Coastguard Worker
886*fb1b10abSAndroid Build Coastguard Worker for (b = 0; b < 16; ++b) {
887*fb1b10abSAndroid Build Coastguard Worker if (labels[b] == which_label) {
888*fb1b10abSAndroid Build Coastguard Worker cost += cost_coeffs(mb, x->block + b, PLANE_TYPE_Y_WITH_DC,
889*fb1b10abSAndroid Build Coastguard Worker ta + vp8_block2above[b], tl + vp8_block2left[b]);
890*fb1b10abSAndroid Build Coastguard Worker }
891*fb1b10abSAndroid Build Coastguard Worker }
892*fb1b10abSAndroid Build Coastguard Worker
893*fb1b10abSAndroid Build Coastguard Worker return cost;
894*fb1b10abSAndroid Build Coastguard Worker }
vp8_encode_inter_mb_segment(MACROBLOCK * x,int const * labels,int which_label)895*fb1b10abSAndroid Build Coastguard Worker static unsigned int vp8_encode_inter_mb_segment(MACROBLOCK *x,
896*fb1b10abSAndroid Build Coastguard Worker int const *labels,
897*fb1b10abSAndroid Build Coastguard Worker int which_label) {
898*fb1b10abSAndroid Build Coastguard Worker int i;
899*fb1b10abSAndroid Build Coastguard Worker unsigned int distortion = 0;
900*fb1b10abSAndroid Build Coastguard Worker int pre_stride = x->e_mbd.pre.y_stride;
901*fb1b10abSAndroid Build Coastguard Worker unsigned char *base_pre = x->e_mbd.pre.y_buffer;
902*fb1b10abSAndroid Build Coastguard Worker
903*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
904*fb1b10abSAndroid Build Coastguard Worker if (labels[i] == which_label) {
905*fb1b10abSAndroid Build Coastguard Worker BLOCKD *bd = &x->e_mbd.block[i];
906*fb1b10abSAndroid Build Coastguard Worker BLOCK *be = &x->block[i];
907*fb1b10abSAndroid Build Coastguard Worker
908*fb1b10abSAndroid Build Coastguard Worker vp8_build_inter_predictors_b(bd, 16, base_pre, pre_stride,
909*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.subpixel_predict);
910*fb1b10abSAndroid Build Coastguard Worker vp8_subtract_b(be, bd, 16);
911*fb1b10abSAndroid Build Coastguard Worker x->short_fdct4x4(be->src_diff, be->coeff, 32);
912*fb1b10abSAndroid Build Coastguard Worker x->quantize_b(be, bd);
913*fb1b10abSAndroid Build Coastguard Worker
914*fb1b10abSAndroid Build Coastguard Worker distortion += vp8_block_error(be->coeff, bd->dqcoeff);
915*fb1b10abSAndroid Build Coastguard Worker }
916*fb1b10abSAndroid Build Coastguard Worker }
917*fb1b10abSAndroid Build Coastguard Worker
918*fb1b10abSAndroid Build Coastguard Worker return distortion;
919*fb1b10abSAndroid Build Coastguard Worker }
920*fb1b10abSAndroid Build Coastguard Worker
921*fb1b10abSAndroid Build Coastguard Worker static const unsigned int segmentation_to_sseshift[4] = { 3, 3, 2, 0 };
922*fb1b10abSAndroid Build Coastguard Worker
923*fb1b10abSAndroid Build Coastguard Worker typedef struct {
924*fb1b10abSAndroid Build Coastguard Worker int_mv *ref_mv;
925*fb1b10abSAndroid Build Coastguard Worker int_mv mvp;
926*fb1b10abSAndroid Build Coastguard Worker
927*fb1b10abSAndroid Build Coastguard Worker int segment_rd;
928*fb1b10abSAndroid Build Coastguard Worker int segment_num;
929*fb1b10abSAndroid Build Coastguard Worker int r;
930*fb1b10abSAndroid Build Coastguard Worker int d;
931*fb1b10abSAndroid Build Coastguard Worker int segment_yrate;
932*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE modes[16];
933*fb1b10abSAndroid Build Coastguard Worker int_mv mvs[16];
934*fb1b10abSAndroid Build Coastguard Worker unsigned char eobs[16];
935*fb1b10abSAndroid Build Coastguard Worker
936*fb1b10abSAndroid Build Coastguard Worker int mvthresh;
937*fb1b10abSAndroid Build Coastguard Worker int *mdcounts;
938*fb1b10abSAndroid Build Coastguard Worker
939*fb1b10abSAndroid Build Coastguard Worker int_mv sv_mvp[4]; /* save 4 mvp from 8x8 */
940*fb1b10abSAndroid Build Coastguard Worker int sv_istep[2]; /* save 2 initial step_param for 16x8/8x16 */
941*fb1b10abSAndroid Build Coastguard Worker
942*fb1b10abSAndroid Build Coastguard Worker } BEST_SEG_INFO;
943*fb1b10abSAndroid Build Coastguard Worker
rd_check_segment(VP8_COMP * cpi,MACROBLOCK * x,BEST_SEG_INFO * bsi,unsigned int segmentation)944*fb1b10abSAndroid Build Coastguard Worker static void rd_check_segment(VP8_COMP *cpi, MACROBLOCK *x, BEST_SEG_INFO *bsi,
945*fb1b10abSAndroid Build Coastguard Worker unsigned int segmentation) {
946*fb1b10abSAndroid Build Coastguard Worker int i;
947*fb1b10abSAndroid Build Coastguard Worker int const *labels;
948*fb1b10abSAndroid Build Coastguard Worker int br = 0;
949*fb1b10abSAndroid Build Coastguard Worker int bd = 0;
950*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE this_mode;
951*fb1b10abSAndroid Build Coastguard Worker
952*fb1b10abSAndroid Build Coastguard Worker int label_count;
953*fb1b10abSAndroid Build Coastguard Worker int this_segment_rd = 0;
954*fb1b10abSAndroid Build Coastguard Worker int label_mv_thresh;
955*fb1b10abSAndroid Build Coastguard Worker int rate = 0;
956*fb1b10abSAndroid Build Coastguard Worker int sbr = 0;
957*fb1b10abSAndroid Build Coastguard Worker int sbd = 0;
958*fb1b10abSAndroid Build Coastguard Worker int segmentyrate = 0;
959*fb1b10abSAndroid Build Coastguard Worker
960*fb1b10abSAndroid Build Coastguard Worker vp8_variance_fn_ptr_t *v_fn_ptr;
961*fb1b10abSAndroid Build Coastguard Worker
962*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT_PLANES t_above, t_left;
963*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT_PLANES t_above_b, t_left_b;
964*fb1b10abSAndroid Build Coastguard Worker
965*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_above, x->e_mbd.above_context, sizeof(ENTROPY_CONTEXT_PLANES));
966*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_left, x->e_mbd.left_context, sizeof(ENTROPY_CONTEXT_PLANES));
967*fb1b10abSAndroid Build Coastguard Worker
968*fb1b10abSAndroid Build Coastguard Worker vp8_zero(t_above_b);
969*fb1b10abSAndroid Build Coastguard Worker vp8_zero(t_left_b);
970*fb1b10abSAndroid Build Coastguard Worker
971*fb1b10abSAndroid Build Coastguard Worker br = 0;
972*fb1b10abSAndroid Build Coastguard Worker bd = 0;
973*fb1b10abSAndroid Build Coastguard Worker
974*fb1b10abSAndroid Build Coastguard Worker v_fn_ptr = &cpi->fn_ptr[segmentation];
975*fb1b10abSAndroid Build Coastguard Worker labels = vp8_mbsplits[segmentation];
976*fb1b10abSAndroid Build Coastguard Worker label_count = vp8_mbsplit_count[segmentation];
977*fb1b10abSAndroid Build Coastguard Worker
978*fb1b10abSAndroid Build Coastguard Worker /* 64 makes this threshold really big effectively making it so that we
979*fb1b10abSAndroid Build Coastguard Worker * very rarely check mvs on segments. setting this to 1 would make mv
980*fb1b10abSAndroid Build Coastguard Worker * thresh roughly equal to what it is for macroblocks
981*fb1b10abSAndroid Build Coastguard Worker */
982*fb1b10abSAndroid Build Coastguard Worker label_mv_thresh = 1 * bsi->mvthresh / label_count;
983*fb1b10abSAndroid Build Coastguard Worker
984*fb1b10abSAndroid Build Coastguard Worker /* Segmentation method overheads */
985*fb1b10abSAndroid Build Coastguard Worker rate = vp8_cost_token(vp8_mbsplit_tree, vp8_mbsplit_probs,
986*fb1b10abSAndroid Build Coastguard Worker vp8_mbsplit_encodings + segmentation);
987*fb1b10abSAndroid Build Coastguard Worker rate += vp8_cost_mv_ref(SPLITMV, bsi->mdcounts);
988*fb1b10abSAndroid Build Coastguard Worker this_segment_rd += RDCOST(x->rdmult, x->rddiv, rate, 0);
989*fb1b10abSAndroid Build Coastguard Worker br += rate;
990*fb1b10abSAndroid Build Coastguard Worker
991*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < label_count; ++i) {
992*fb1b10abSAndroid Build Coastguard Worker int_mv mode_mv[B_MODE_COUNT] = { { 0 }, { 0 } };
993*fb1b10abSAndroid Build Coastguard Worker int best_label_rd = INT_MAX;
994*fb1b10abSAndroid Build Coastguard Worker B_PREDICTION_MODE mode_selected = ZERO4X4;
995*fb1b10abSAndroid Build Coastguard Worker int bestlabelyrate = 0;
996*fb1b10abSAndroid Build Coastguard Worker
997*fb1b10abSAndroid Build Coastguard Worker /* search for the best motion vector on this segment */
998*fb1b10abSAndroid Build Coastguard Worker for (this_mode = LEFT4X4; this_mode <= NEW4X4; ++this_mode) {
999*fb1b10abSAndroid Build Coastguard Worker int this_rd;
1000*fb1b10abSAndroid Build Coastguard Worker int distortion;
1001*fb1b10abSAndroid Build Coastguard Worker int labelyrate;
1002*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT_PLANES t_above_s, t_left_s;
1003*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *ta_s;
1004*fb1b10abSAndroid Build Coastguard Worker ENTROPY_CONTEXT *tl_s;
1005*fb1b10abSAndroid Build Coastguard Worker
1006*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_above_s, &t_above, sizeof(ENTROPY_CONTEXT_PLANES));
1007*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_left_s, &t_left, sizeof(ENTROPY_CONTEXT_PLANES));
1008*fb1b10abSAndroid Build Coastguard Worker
1009*fb1b10abSAndroid Build Coastguard Worker ta_s = (ENTROPY_CONTEXT *)&t_above_s;
1010*fb1b10abSAndroid Build Coastguard Worker tl_s = (ENTROPY_CONTEXT *)&t_left_s;
1011*fb1b10abSAndroid Build Coastguard Worker
1012*fb1b10abSAndroid Build Coastguard Worker if (this_mode == NEW4X4) {
1013*fb1b10abSAndroid Build Coastguard Worker int sseshift;
1014*fb1b10abSAndroid Build Coastguard Worker int num00;
1015*fb1b10abSAndroid Build Coastguard Worker int step_param = 0;
1016*fb1b10abSAndroid Build Coastguard Worker int further_steps;
1017*fb1b10abSAndroid Build Coastguard Worker int n;
1018*fb1b10abSAndroid Build Coastguard Worker int thissme;
1019*fb1b10abSAndroid Build Coastguard Worker int bestsme = INT_MAX;
1020*fb1b10abSAndroid Build Coastguard Worker int_mv temp_mv;
1021*fb1b10abSAndroid Build Coastguard Worker BLOCK *c;
1022*fb1b10abSAndroid Build Coastguard Worker BLOCKD *e;
1023*fb1b10abSAndroid Build Coastguard Worker
1024*fb1b10abSAndroid Build Coastguard Worker /* Is the best so far sufficiently good that we can't justify
1025*fb1b10abSAndroid Build Coastguard Worker * doing a new motion search.
1026*fb1b10abSAndroid Build Coastguard Worker */
1027*fb1b10abSAndroid Build Coastguard Worker if (best_label_rd < label_mv_thresh) break;
1028*fb1b10abSAndroid Build Coastguard Worker
1029*fb1b10abSAndroid Build Coastguard Worker if (cpi->compressor_speed) {
1030*fb1b10abSAndroid Build Coastguard Worker if (segmentation == BLOCK_8X16 || segmentation == BLOCK_16X8) {
1031*fb1b10abSAndroid Build Coastguard Worker bsi->mvp.as_int = bsi->sv_mvp[i].as_int;
1032*fb1b10abSAndroid Build Coastguard Worker if (i == 1 && segmentation == BLOCK_16X8) {
1033*fb1b10abSAndroid Build Coastguard Worker bsi->mvp.as_int = bsi->sv_mvp[2].as_int;
1034*fb1b10abSAndroid Build Coastguard Worker }
1035*fb1b10abSAndroid Build Coastguard Worker
1036*fb1b10abSAndroid Build Coastguard Worker step_param = bsi->sv_istep[i];
1037*fb1b10abSAndroid Build Coastguard Worker }
1038*fb1b10abSAndroid Build Coastguard Worker
1039*fb1b10abSAndroid Build Coastguard Worker /* use previous block's result as next block's MV
1040*fb1b10abSAndroid Build Coastguard Worker * predictor.
1041*fb1b10abSAndroid Build Coastguard Worker */
1042*fb1b10abSAndroid Build Coastguard Worker if (segmentation == BLOCK_4X4 && i > 0) {
1043*fb1b10abSAndroid Build Coastguard Worker bsi->mvp.as_int = x->e_mbd.block[i - 1].bmi.mv.as_int;
1044*fb1b10abSAndroid Build Coastguard Worker if (i == 4 || i == 8 || i == 12) {
1045*fb1b10abSAndroid Build Coastguard Worker bsi->mvp.as_int = x->e_mbd.block[i - 4].bmi.mv.as_int;
1046*fb1b10abSAndroid Build Coastguard Worker }
1047*fb1b10abSAndroid Build Coastguard Worker step_param = 2;
1048*fb1b10abSAndroid Build Coastguard Worker }
1049*fb1b10abSAndroid Build Coastguard Worker }
1050*fb1b10abSAndroid Build Coastguard Worker
1051*fb1b10abSAndroid Build Coastguard Worker further_steps = (MAX_MVSEARCH_STEPS - 1) - step_param;
1052*fb1b10abSAndroid Build Coastguard Worker
1053*fb1b10abSAndroid Build Coastguard Worker {
1054*fb1b10abSAndroid Build Coastguard Worker int sadpb = x->sadperbit4;
1055*fb1b10abSAndroid Build Coastguard Worker int_mv mvp_full;
1056*fb1b10abSAndroid Build Coastguard Worker
1057*fb1b10abSAndroid Build Coastguard Worker mvp_full.as_mv.row = bsi->mvp.as_mv.row >> 3;
1058*fb1b10abSAndroid Build Coastguard Worker mvp_full.as_mv.col = bsi->mvp.as_mv.col >> 3;
1059*fb1b10abSAndroid Build Coastguard Worker
1060*fb1b10abSAndroid Build Coastguard Worker /* find first label */
1061*fb1b10abSAndroid Build Coastguard Worker n = vp8_mbsplit_offset[segmentation][i];
1062*fb1b10abSAndroid Build Coastguard Worker
1063*fb1b10abSAndroid Build Coastguard Worker c = &x->block[n];
1064*fb1b10abSAndroid Build Coastguard Worker e = &x->e_mbd.block[n];
1065*fb1b10abSAndroid Build Coastguard Worker
1066*fb1b10abSAndroid Build Coastguard Worker {
1067*fb1b10abSAndroid Build Coastguard Worker bestsme = cpi->diamond_search_sad(
1068*fb1b10abSAndroid Build Coastguard Worker x, c, e, &mvp_full, &mode_mv[NEW4X4], step_param, sadpb, &num00,
1069*fb1b10abSAndroid Build Coastguard Worker v_fn_ptr, x->mvcost, bsi->ref_mv);
1070*fb1b10abSAndroid Build Coastguard Worker
1071*fb1b10abSAndroid Build Coastguard Worker n = num00;
1072*fb1b10abSAndroid Build Coastguard Worker num00 = 0;
1073*fb1b10abSAndroid Build Coastguard Worker
1074*fb1b10abSAndroid Build Coastguard Worker while (n < further_steps) {
1075*fb1b10abSAndroid Build Coastguard Worker n++;
1076*fb1b10abSAndroid Build Coastguard Worker
1077*fb1b10abSAndroid Build Coastguard Worker if (num00) {
1078*fb1b10abSAndroid Build Coastguard Worker num00--;
1079*fb1b10abSAndroid Build Coastguard Worker } else {
1080*fb1b10abSAndroid Build Coastguard Worker thissme = cpi->diamond_search_sad(
1081*fb1b10abSAndroid Build Coastguard Worker x, c, e, &mvp_full, &temp_mv, step_param + n, sadpb, &num00,
1082*fb1b10abSAndroid Build Coastguard Worker v_fn_ptr, x->mvcost, bsi->ref_mv);
1083*fb1b10abSAndroid Build Coastguard Worker
1084*fb1b10abSAndroid Build Coastguard Worker if (thissme < bestsme) {
1085*fb1b10abSAndroid Build Coastguard Worker bestsme = thissme;
1086*fb1b10abSAndroid Build Coastguard Worker mode_mv[NEW4X4].as_int = temp_mv.as_int;
1087*fb1b10abSAndroid Build Coastguard Worker }
1088*fb1b10abSAndroid Build Coastguard Worker }
1089*fb1b10abSAndroid Build Coastguard Worker }
1090*fb1b10abSAndroid Build Coastguard Worker }
1091*fb1b10abSAndroid Build Coastguard Worker
1092*fb1b10abSAndroid Build Coastguard Worker sseshift = segmentation_to_sseshift[segmentation];
1093*fb1b10abSAndroid Build Coastguard Worker
1094*fb1b10abSAndroid Build Coastguard Worker /* Should we do a full search (best quality only) */
1095*fb1b10abSAndroid Build Coastguard Worker if ((cpi->compressor_speed == 0) && (bestsme >> sseshift) > 4000) {
1096*fb1b10abSAndroid Build Coastguard Worker /* Check if mvp_full is within the range. */
1097*fb1b10abSAndroid Build Coastguard Worker vp8_clamp_mv(&mvp_full, x->mv_col_min, x->mv_col_max, x->mv_row_min,
1098*fb1b10abSAndroid Build Coastguard Worker x->mv_row_max);
1099*fb1b10abSAndroid Build Coastguard Worker
1100*fb1b10abSAndroid Build Coastguard Worker thissme = vp8_full_search_sad(x, c, e, &mvp_full, sadpb, 16,
1101*fb1b10abSAndroid Build Coastguard Worker v_fn_ptr, x->mvcost, bsi->ref_mv);
1102*fb1b10abSAndroid Build Coastguard Worker
1103*fb1b10abSAndroid Build Coastguard Worker if (thissme < bestsme) {
1104*fb1b10abSAndroid Build Coastguard Worker bestsme = thissme;
1105*fb1b10abSAndroid Build Coastguard Worker mode_mv[NEW4X4].as_int = e->bmi.mv.as_int;
1106*fb1b10abSAndroid Build Coastguard Worker } else {
1107*fb1b10abSAndroid Build Coastguard Worker /* The full search result is actually worse so
1108*fb1b10abSAndroid Build Coastguard Worker * re-instate the previous best vector
1109*fb1b10abSAndroid Build Coastguard Worker */
1110*fb1b10abSAndroid Build Coastguard Worker e->bmi.mv.as_int = mode_mv[NEW4X4].as_int;
1111*fb1b10abSAndroid Build Coastguard Worker }
1112*fb1b10abSAndroid Build Coastguard Worker }
1113*fb1b10abSAndroid Build Coastguard Worker }
1114*fb1b10abSAndroid Build Coastguard Worker
1115*fb1b10abSAndroid Build Coastguard Worker if (bestsme < INT_MAX) {
1116*fb1b10abSAndroid Build Coastguard Worker int disto;
1117*fb1b10abSAndroid Build Coastguard Worker unsigned int sse;
1118*fb1b10abSAndroid Build Coastguard Worker cpi->find_fractional_mv_step(x, c, e, &mode_mv[NEW4X4], bsi->ref_mv,
1119*fb1b10abSAndroid Build Coastguard Worker x->errorperbit, v_fn_ptr, x->mvcost,
1120*fb1b10abSAndroid Build Coastguard Worker &disto, &sse);
1121*fb1b10abSAndroid Build Coastguard Worker }
1122*fb1b10abSAndroid Build Coastguard Worker } /* NEW4X4 */
1123*fb1b10abSAndroid Build Coastguard Worker
1124*fb1b10abSAndroid Build Coastguard Worker rate = labels2mode(x, labels, i, this_mode, &mode_mv[this_mode],
1125*fb1b10abSAndroid Build Coastguard Worker bsi->ref_mv, x->mvcost);
1126*fb1b10abSAndroid Build Coastguard Worker
1127*fb1b10abSAndroid Build Coastguard Worker /* Trap vectors that reach beyond the UMV borders */
1128*fb1b10abSAndroid Build Coastguard Worker if (((mode_mv[this_mode].as_mv.row >> 3) < x->mv_row_min) ||
1129*fb1b10abSAndroid Build Coastguard Worker ((mode_mv[this_mode].as_mv.row >> 3) > x->mv_row_max) ||
1130*fb1b10abSAndroid Build Coastguard Worker ((mode_mv[this_mode].as_mv.col >> 3) < x->mv_col_min) ||
1131*fb1b10abSAndroid Build Coastguard Worker ((mode_mv[this_mode].as_mv.col >> 3) > x->mv_col_max)) {
1132*fb1b10abSAndroid Build Coastguard Worker continue;
1133*fb1b10abSAndroid Build Coastguard Worker }
1134*fb1b10abSAndroid Build Coastguard Worker
1135*fb1b10abSAndroid Build Coastguard Worker distortion = vp8_encode_inter_mb_segment(x, labels, i) / 4;
1136*fb1b10abSAndroid Build Coastguard Worker
1137*fb1b10abSAndroid Build Coastguard Worker labelyrate = rdcost_mbsegment_y(x, labels, i, ta_s, tl_s);
1138*fb1b10abSAndroid Build Coastguard Worker rate += labelyrate;
1139*fb1b10abSAndroid Build Coastguard Worker
1140*fb1b10abSAndroid Build Coastguard Worker this_rd = RDCOST(x->rdmult, x->rddiv, rate, distortion);
1141*fb1b10abSAndroid Build Coastguard Worker
1142*fb1b10abSAndroid Build Coastguard Worker if (this_rd < best_label_rd) {
1143*fb1b10abSAndroid Build Coastguard Worker sbr = rate;
1144*fb1b10abSAndroid Build Coastguard Worker sbd = distortion;
1145*fb1b10abSAndroid Build Coastguard Worker bestlabelyrate = labelyrate;
1146*fb1b10abSAndroid Build Coastguard Worker mode_selected = this_mode;
1147*fb1b10abSAndroid Build Coastguard Worker best_label_rd = this_rd;
1148*fb1b10abSAndroid Build Coastguard Worker
1149*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_above_b, &t_above_s, sizeof(ENTROPY_CONTEXT_PLANES));
1150*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_left_b, &t_left_s, sizeof(ENTROPY_CONTEXT_PLANES));
1151*fb1b10abSAndroid Build Coastguard Worker }
1152*fb1b10abSAndroid Build Coastguard Worker } /*for each 4x4 mode*/
1153*fb1b10abSAndroid Build Coastguard Worker
1154*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_above, &t_above_b, sizeof(ENTROPY_CONTEXT_PLANES));
1155*fb1b10abSAndroid Build Coastguard Worker memcpy(&t_left, &t_left_b, sizeof(ENTROPY_CONTEXT_PLANES));
1156*fb1b10abSAndroid Build Coastguard Worker
1157*fb1b10abSAndroid Build Coastguard Worker labels2mode(x, labels, i, mode_selected, &mode_mv[mode_selected],
1158*fb1b10abSAndroid Build Coastguard Worker bsi->ref_mv, x->mvcost);
1159*fb1b10abSAndroid Build Coastguard Worker
1160*fb1b10abSAndroid Build Coastguard Worker br += sbr;
1161*fb1b10abSAndroid Build Coastguard Worker bd += sbd;
1162*fb1b10abSAndroid Build Coastguard Worker segmentyrate += bestlabelyrate;
1163*fb1b10abSAndroid Build Coastguard Worker this_segment_rd += best_label_rd;
1164*fb1b10abSAndroid Build Coastguard Worker
1165*fb1b10abSAndroid Build Coastguard Worker if (this_segment_rd >= bsi->segment_rd) break;
1166*fb1b10abSAndroid Build Coastguard Worker
1167*fb1b10abSAndroid Build Coastguard Worker } /* for each label */
1168*fb1b10abSAndroid Build Coastguard Worker
1169*fb1b10abSAndroid Build Coastguard Worker if (this_segment_rd < bsi->segment_rd) {
1170*fb1b10abSAndroid Build Coastguard Worker bsi->r = br;
1171*fb1b10abSAndroid Build Coastguard Worker bsi->d = bd;
1172*fb1b10abSAndroid Build Coastguard Worker bsi->segment_yrate = segmentyrate;
1173*fb1b10abSAndroid Build Coastguard Worker bsi->segment_rd = this_segment_rd;
1174*fb1b10abSAndroid Build Coastguard Worker bsi->segment_num = segmentation;
1175*fb1b10abSAndroid Build Coastguard Worker
1176*fb1b10abSAndroid Build Coastguard Worker /* store everything needed to come back to this!! */
1177*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
1178*fb1b10abSAndroid Build Coastguard Worker bsi->mvs[i].as_mv = x->partition_info->bmi[i].mv.as_mv;
1179*fb1b10abSAndroid Build Coastguard Worker bsi->modes[i] = x->partition_info->bmi[i].mode;
1180*fb1b10abSAndroid Build Coastguard Worker bsi->eobs[i] = x->e_mbd.eobs[i];
1181*fb1b10abSAndroid Build Coastguard Worker }
1182*fb1b10abSAndroid Build Coastguard Worker }
1183*fb1b10abSAndroid Build Coastguard Worker }
1184*fb1b10abSAndroid Build Coastguard Worker
vp8_cal_step_param(int sr,int * sp)1185*fb1b10abSAndroid Build Coastguard Worker static void vp8_cal_step_param(int sr, int *sp) {
1186*fb1b10abSAndroid Build Coastguard Worker int step = 0;
1187*fb1b10abSAndroid Build Coastguard Worker
1188*fb1b10abSAndroid Build Coastguard Worker if (sr > MAX_FIRST_STEP) {
1189*fb1b10abSAndroid Build Coastguard Worker sr = MAX_FIRST_STEP;
1190*fb1b10abSAndroid Build Coastguard Worker } else if (sr < 1) {
1191*fb1b10abSAndroid Build Coastguard Worker sr = 1;
1192*fb1b10abSAndroid Build Coastguard Worker }
1193*fb1b10abSAndroid Build Coastguard Worker
1194*fb1b10abSAndroid Build Coastguard Worker while (sr >>= 1) step++;
1195*fb1b10abSAndroid Build Coastguard Worker
1196*fb1b10abSAndroid Build Coastguard Worker *sp = MAX_MVSEARCH_STEPS - 1 - step;
1197*fb1b10abSAndroid Build Coastguard Worker }
1198*fb1b10abSAndroid Build Coastguard Worker
vp8_rd_pick_best_mbsegmentation(VP8_COMP * cpi,MACROBLOCK * x,int_mv * best_ref_mv,int best_rd,int * mdcounts,int * returntotrate,int * returnyrate,int * returndistortion,int mvthresh)1199*fb1b10abSAndroid Build Coastguard Worker static int vp8_rd_pick_best_mbsegmentation(VP8_COMP *cpi, MACROBLOCK *x,
1200*fb1b10abSAndroid Build Coastguard Worker int_mv *best_ref_mv, int best_rd,
1201*fb1b10abSAndroid Build Coastguard Worker int *mdcounts, int *returntotrate,
1202*fb1b10abSAndroid Build Coastguard Worker int *returnyrate,
1203*fb1b10abSAndroid Build Coastguard Worker int *returndistortion,
1204*fb1b10abSAndroid Build Coastguard Worker int mvthresh) {
1205*fb1b10abSAndroid Build Coastguard Worker int i;
1206*fb1b10abSAndroid Build Coastguard Worker BEST_SEG_INFO bsi;
1207*fb1b10abSAndroid Build Coastguard Worker
1208*fb1b10abSAndroid Build Coastguard Worker memset(&bsi, 0, sizeof(bsi));
1209*fb1b10abSAndroid Build Coastguard Worker
1210*fb1b10abSAndroid Build Coastguard Worker bsi.segment_rd = best_rd;
1211*fb1b10abSAndroid Build Coastguard Worker bsi.ref_mv = best_ref_mv;
1212*fb1b10abSAndroid Build Coastguard Worker bsi.mvp.as_int = best_ref_mv->as_int;
1213*fb1b10abSAndroid Build Coastguard Worker bsi.mvthresh = mvthresh;
1214*fb1b10abSAndroid Build Coastguard Worker bsi.mdcounts = mdcounts;
1215*fb1b10abSAndroid Build Coastguard Worker
1216*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
1217*fb1b10abSAndroid Build Coastguard Worker bsi.modes[i] = ZERO4X4;
1218*fb1b10abSAndroid Build Coastguard Worker }
1219*fb1b10abSAndroid Build Coastguard Worker
1220*fb1b10abSAndroid Build Coastguard Worker if (cpi->compressor_speed == 0) {
1221*fb1b10abSAndroid Build Coastguard Worker /* for now, we will keep the original segmentation order
1222*fb1b10abSAndroid Build Coastguard Worker when in best quality mode */
1223*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_16X8);
1224*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_8X16);
1225*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_8X8);
1226*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_4X4);
1227*fb1b10abSAndroid Build Coastguard Worker } else {
1228*fb1b10abSAndroid Build Coastguard Worker int sr;
1229*fb1b10abSAndroid Build Coastguard Worker
1230*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_8X8);
1231*fb1b10abSAndroid Build Coastguard Worker
1232*fb1b10abSAndroid Build Coastguard Worker if (bsi.segment_rd < best_rd) {
1233*fb1b10abSAndroid Build Coastguard Worker int col_min = ((best_ref_mv->as_mv.col + 7) >> 3) - MAX_FULL_PEL_VAL;
1234*fb1b10abSAndroid Build Coastguard Worker int row_min = ((best_ref_mv->as_mv.row + 7) >> 3) - MAX_FULL_PEL_VAL;
1235*fb1b10abSAndroid Build Coastguard Worker int col_max = (best_ref_mv->as_mv.col >> 3) + MAX_FULL_PEL_VAL;
1236*fb1b10abSAndroid Build Coastguard Worker int row_max = (best_ref_mv->as_mv.row >> 3) + MAX_FULL_PEL_VAL;
1237*fb1b10abSAndroid Build Coastguard Worker
1238*fb1b10abSAndroid Build Coastguard Worker int tmp_col_min = x->mv_col_min;
1239*fb1b10abSAndroid Build Coastguard Worker int tmp_col_max = x->mv_col_max;
1240*fb1b10abSAndroid Build Coastguard Worker int tmp_row_min = x->mv_row_min;
1241*fb1b10abSAndroid Build Coastguard Worker int tmp_row_max = x->mv_row_max;
1242*fb1b10abSAndroid Build Coastguard Worker
1243*fb1b10abSAndroid Build Coastguard Worker /* Get intersection of UMV window and valid MV window to reduce # of
1244*fb1b10abSAndroid Build Coastguard Worker * checks in diamond search. */
1245*fb1b10abSAndroid Build Coastguard Worker if (x->mv_col_min < col_min) x->mv_col_min = col_min;
1246*fb1b10abSAndroid Build Coastguard Worker if (x->mv_col_max > col_max) x->mv_col_max = col_max;
1247*fb1b10abSAndroid Build Coastguard Worker if (x->mv_row_min < row_min) x->mv_row_min = row_min;
1248*fb1b10abSAndroid Build Coastguard Worker if (x->mv_row_max > row_max) x->mv_row_max = row_max;
1249*fb1b10abSAndroid Build Coastguard Worker
1250*fb1b10abSAndroid Build Coastguard Worker /* Get 8x8 result */
1251*fb1b10abSAndroid Build Coastguard Worker bsi.sv_mvp[0].as_int = bsi.mvs[0].as_int;
1252*fb1b10abSAndroid Build Coastguard Worker bsi.sv_mvp[1].as_int = bsi.mvs[2].as_int;
1253*fb1b10abSAndroid Build Coastguard Worker bsi.sv_mvp[2].as_int = bsi.mvs[8].as_int;
1254*fb1b10abSAndroid Build Coastguard Worker bsi.sv_mvp[3].as_int = bsi.mvs[10].as_int;
1255*fb1b10abSAndroid Build Coastguard Worker
1256*fb1b10abSAndroid Build Coastguard Worker /* Use 8x8 result as 16x8/8x16's predictor MV. Adjust search range
1257*fb1b10abSAndroid Build Coastguard Worker * according to the closeness of 2 MV. */
1258*fb1b10abSAndroid Build Coastguard Worker /* block 8X16 */
1259*fb1b10abSAndroid Build Coastguard Worker {
1260*fb1b10abSAndroid Build Coastguard Worker sr =
1261*fb1b10abSAndroid Build Coastguard Worker MAXF((abs(bsi.sv_mvp[0].as_mv.row - bsi.sv_mvp[2].as_mv.row)) >> 3,
1262*fb1b10abSAndroid Build Coastguard Worker (abs(bsi.sv_mvp[0].as_mv.col - bsi.sv_mvp[2].as_mv.col)) >> 3);
1263*fb1b10abSAndroid Build Coastguard Worker vp8_cal_step_param(sr, &bsi.sv_istep[0]);
1264*fb1b10abSAndroid Build Coastguard Worker
1265*fb1b10abSAndroid Build Coastguard Worker sr =
1266*fb1b10abSAndroid Build Coastguard Worker MAXF((abs(bsi.sv_mvp[1].as_mv.row - bsi.sv_mvp[3].as_mv.row)) >> 3,
1267*fb1b10abSAndroid Build Coastguard Worker (abs(bsi.sv_mvp[1].as_mv.col - bsi.sv_mvp[3].as_mv.col)) >> 3);
1268*fb1b10abSAndroid Build Coastguard Worker vp8_cal_step_param(sr, &bsi.sv_istep[1]);
1269*fb1b10abSAndroid Build Coastguard Worker
1270*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_8X16);
1271*fb1b10abSAndroid Build Coastguard Worker }
1272*fb1b10abSAndroid Build Coastguard Worker
1273*fb1b10abSAndroid Build Coastguard Worker /* block 16X8 */
1274*fb1b10abSAndroid Build Coastguard Worker {
1275*fb1b10abSAndroid Build Coastguard Worker sr =
1276*fb1b10abSAndroid Build Coastguard Worker MAXF((abs(bsi.sv_mvp[0].as_mv.row - bsi.sv_mvp[1].as_mv.row)) >> 3,
1277*fb1b10abSAndroid Build Coastguard Worker (abs(bsi.sv_mvp[0].as_mv.col - bsi.sv_mvp[1].as_mv.col)) >> 3);
1278*fb1b10abSAndroid Build Coastguard Worker vp8_cal_step_param(sr, &bsi.sv_istep[0]);
1279*fb1b10abSAndroid Build Coastguard Worker
1280*fb1b10abSAndroid Build Coastguard Worker sr =
1281*fb1b10abSAndroid Build Coastguard Worker MAXF((abs(bsi.sv_mvp[2].as_mv.row - bsi.sv_mvp[3].as_mv.row)) >> 3,
1282*fb1b10abSAndroid Build Coastguard Worker (abs(bsi.sv_mvp[2].as_mv.col - bsi.sv_mvp[3].as_mv.col)) >> 3);
1283*fb1b10abSAndroid Build Coastguard Worker vp8_cal_step_param(sr, &bsi.sv_istep[1]);
1284*fb1b10abSAndroid Build Coastguard Worker
1285*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_16X8);
1286*fb1b10abSAndroid Build Coastguard Worker }
1287*fb1b10abSAndroid Build Coastguard Worker
1288*fb1b10abSAndroid Build Coastguard Worker /* If 8x8 is better than 16x8/8x16, then do 4x4 search */
1289*fb1b10abSAndroid Build Coastguard Worker /* Not skip 4x4 if speed=0 (good quality) */
1290*fb1b10abSAndroid Build Coastguard Worker if (cpi->sf.no_skip_block4x4_search || bsi.segment_num == BLOCK_8X8)
1291*fb1b10abSAndroid Build Coastguard Worker /* || (sv_segment_rd8x8-bsi.segment_rd) < sv_segment_rd8x8>>5) */
1292*fb1b10abSAndroid Build Coastguard Worker {
1293*fb1b10abSAndroid Build Coastguard Worker bsi.mvp.as_int = bsi.sv_mvp[0].as_int;
1294*fb1b10abSAndroid Build Coastguard Worker rd_check_segment(cpi, x, &bsi, BLOCK_4X4);
1295*fb1b10abSAndroid Build Coastguard Worker }
1296*fb1b10abSAndroid Build Coastguard Worker
1297*fb1b10abSAndroid Build Coastguard Worker /* restore UMV window */
1298*fb1b10abSAndroid Build Coastguard Worker x->mv_col_min = tmp_col_min;
1299*fb1b10abSAndroid Build Coastguard Worker x->mv_col_max = tmp_col_max;
1300*fb1b10abSAndroid Build Coastguard Worker x->mv_row_min = tmp_row_min;
1301*fb1b10abSAndroid Build Coastguard Worker x->mv_row_max = tmp_row_max;
1302*fb1b10abSAndroid Build Coastguard Worker }
1303*fb1b10abSAndroid Build Coastguard Worker }
1304*fb1b10abSAndroid Build Coastguard Worker
1305*fb1b10abSAndroid Build Coastguard Worker /* set it to the best */
1306*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
1307*fb1b10abSAndroid Build Coastguard Worker BLOCKD *bd = &x->e_mbd.block[i];
1308*fb1b10abSAndroid Build Coastguard Worker
1309*fb1b10abSAndroid Build Coastguard Worker bd->bmi.mv.as_int = bsi.mvs[i].as_int;
1310*fb1b10abSAndroid Build Coastguard Worker *bd->eob = bsi.eobs[i];
1311*fb1b10abSAndroid Build Coastguard Worker }
1312*fb1b10abSAndroid Build Coastguard Worker
1313*fb1b10abSAndroid Build Coastguard Worker *returntotrate = bsi.r;
1314*fb1b10abSAndroid Build Coastguard Worker *returndistortion = bsi.d;
1315*fb1b10abSAndroid Build Coastguard Worker *returnyrate = bsi.segment_yrate;
1316*fb1b10abSAndroid Build Coastguard Worker
1317*fb1b10abSAndroid Build Coastguard Worker /* save partitions */
1318*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.partitioning = bsi.segment_num;
1319*fb1b10abSAndroid Build Coastguard Worker x->partition_info->count = vp8_mbsplit_count[bsi.segment_num];
1320*fb1b10abSAndroid Build Coastguard Worker
1321*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < x->partition_info->count; ++i) {
1322*fb1b10abSAndroid Build Coastguard Worker int j;
1323*fb1b10abSAndroid Build Coastguard Worker
1324*fb1b10abSAndroid Build Coastguard Worker j = vp8_mbsplit_offset[bsi.segment_num][i];
1325*fb1b10abSAndroid Build Coastguard Worker
1326*fb1b10abSAndroid Build Coastguard Worker x->partition_info->bmi[i].mode = bsi.modes[j];
1327*fb1b10abSAndroid Build Coastguard Worker x->partition_info->bmi[i].mv.as_mv = bsi.mvs[j].as_mv;
1328*fb1b10abSAndroid Build Coastguard Worker }
1329*fb1b10abSAndroid Build Coastguard Worker /*
1330*fb1b10abSAndroid Build Coastguard Worker * used to set x->e_mbd.mode_info_context->mbmi.mv.as_int
1331*fb1b10abSAndroid Build Coastguard Worker */
1332*fb1b10abSAndroid Build Coastguard Worker x->partition_info->bmi[15].mv.as_int = bsi.mvs[15].as_int;
1333*fb1b10abSAndroid Build Coastguard Worker
1334*fb1b10abSAndroid Build Coastguard Worker return bsi.segment_rd;
1335*fb1b10abSAndroid Build Coastguard Worker }
1336*fb1b10abSAndroid Build Coastguard Worker
1337*fb1b10abSAndroid Build Coastguard Worker /* The improved MV prediction */
vp8_mv_pred(VP8_COMP * cpi,MACROBLOCKD * xd,const MODE_INFO * here,int_mv * mvp,int refframe,int * ref_frame_sign_bias,int * sr,int near_sadidx[])1338*fb1b10abSAndroid Build Coastguard Worker void vp8_mv_pred(VP8_COMP *cpi, MACROBLOCKD *xd, const MODE_INFO *here,
1339*fb1b10abSAndroid Build Coastguard Worker int_mv *mvp, int refframe, int *ref_frame_sign_bias, int *sr,
1340*fb1b10abSAndroid Build Coastguard Worker int near_sadidx[]) {
1341*fb1b10abSAndroid Build Coastguard Worker const MODE_INFO *above = here - xd->mode_info_stride;
1342*fb1b10abSAndroid Build Coastguard Worker const MODE_INFO *left = here - 1;
1343*fb1b10abSAndroid Build Coastguard Worker const MODE_INFO *aboveleft = above - 1;
1344*fb1b10abSAndroid Build Coastguard Worker int_mv near_mvs[8];
1345*fb1b10abSAndroid Build Coastguard Worker int near_ref[8];
1346*fb1b10abSAndroid Build Coastguard Worker int_mv mv;
1347*fb1b10abSAndroid Build Coastguard Worker int vcnt = 0;
1348*fb1b10abSAndroid Build Coastguard Worker int find = 0;
1349*fb1b10abSAndroid Build Coastguard Worker int mb_offset;
1350*fb1b10abSAndroid Build Coastguard Worker
1351*fb1b10abSAndroid Build Coastguard Worker int mvx[8];
1352*fb1b10abSAndroid Build Coastguard Worker int mvy[8];
1353*fb1b10abSAndroid Build Coastguard Worker int i;
1354*fb1b10abSAndroid Build Coastguard Worker
1355*fb1b10abSAndroid Build Coastguard Worker mv.as_int = 0;
1356*fb1b10abSAndroid Build Coastguard Worker
1357*fb1b10abSAndroid Build Coastguard Worker if (here->mbmi.ref_frame != INTRA_FRAME) {
1358*fb1b10abSAndroid Build Coastguard Worker near_mvs[0].as_int = near_mvs[1].as_int = near_mvs[2].as_int =
1359*fb1b10abSAndroid Build Coastguard Worker near_mvs[3].as_int = near_mvs[4].as_int = near_mvs[5].as_int =
1360*fb1b10abSAndroid Build Coastguard Worker near_mvs[6].as_int = near_mvs[7].as_int = 0;
1361*fb1b10abSAndroid Build Coastguard Worker near_ref[0] = near_ref[1] = near_ref[2] = near_ref[3] = near_ref[4] =
1362*fb1b10abSAndroid Build Coastguard Worker near_ref[5] = near_ref[6] = near_ref[7] = 0;
1363*fb1b10abSAndroid Build Coastguard Worker
1364*fb1b10abSAndroid Build Coastguard Worker /* read in 3 nearby block's MVs from current frame as prediction
1365*fb1b10abSAndroid Build Coastguard Worker * candidates.
1366*fb1b10abSAndroid Build Coastguard Worker */
1367*fb1b10abSAndroid Build Coastguard Worker if (above->mbmi.ref_frame != INTRA_FRAME) {
1368*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int = above->mbmi.mv.as_int;
1369*fb1b10abSAndroid Build Coastguard Worker mv_bias(ref_frame_sign_bias[above->mbmi.ref_frame], refframe,
1370*fb1b10abSAndroid Build Coastguard Worker &near_mvs[vcnt], ref_frame_sign_bias);
1371*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] = above->mbmi.ref_frame;
1372*fb1b10abSAndroid Build Coastguard Worker }
1373*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1374*fb1b10abSAndroid Build Coastguard Worker if (left->mbmi.ref_frame != INTRA_FRAME) {
1375*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int = left->mbmi.mv.as_int;
1376*fb1b10abSAndroid Build Coastguard Worker mv_bias(ref_frame_sign_bias[left->mbmi.ref_frame], refframe,
1377*fb1b10abSAndroid Build Coastguard Worker &near_mvs[vcnt], ref_frame_sign_bias);
1378*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] = left->mbmi.ref_frame;
1379*fb1b10abSAndroid Build Coastguard Worker }
1380*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1381*fb1b10abSAndroid Build Coastguard Worker if (aboveleft->mbmi.ref_frame != INTRA_FRAME) {
1382*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int = aboveleft->mbmi.mv.as_int;
1383*fb1b10abSAndroid Build Coastguard Worker mv_bias(ref_frame_sign_bias[aboveleft->mbmi.ref_frame], refframe,
1384*fb1b10abSAndroid Build Coastguard Worker &near_mvs[vcnt], ref_frame_sign_bias);
1385*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] = aboveleft->mbmi.ref_frame;
1386*fb1b10abSAndroid Build Coastguard Worker }
1387*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1388*fb1b10abSAndroid Build Coastguard Worker
1389*fb1b10abSAndroid Build Coastguard Worker /* read in 5 nearby block's MVs from last frame. */
1390*fb1b10abSAndroid Build Coastguard Worker if (cpi->common.last_frame_type != KEY_FRAME) {
1391*fb1b10abSAndroid Build Coastguard Worker mb_offset = (-xd->mb_to_top_edge / 128 + 1) * (xd->mode_info_stride + 1) +
1392*fb1b10abSAndroid Build Coastguard Worker (-xd->mb_to_left_edge / 128 + 1);
1393*fb1b10abSAndroid Build Coastguard Worker
1394*fb1b10abSAndroid Build Coastguard Worker /* current in last frame */
1395*fb1b10abSAndroid Build Coastguard Worker if (cpi->lf_ref_frame[mb_offset] != INTRA_FRAME) {
1396*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int = cpi->lfmv[mb_offset].as_int;
1397*fb1b10abSAndroid Build Coastguard Worker mv_bias(cpi->lf_ref_frame_sign_bias[mb_offset], refframe,
1398*fb1b10abSAndroid Build Coastguard Worker &near_mvs[vcnt], ref_frame_sign_bias);
1399*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] = cpi->lf_ref_frame[mb_offset];
1400*fb1b10abSAndroid Build Coastguard Worker }
1401*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1402*fb1b10abSAndroid Build Coastguard Worker
1403*fb1b10abSAndroid Build Coastguard Worker /* above in last frame */
1404*fb1b10abSAndroid Build Coastguard Worker if (cpi->lf_ref_frame[mb_offset - xd->mode_info_stride - 1] !=
1405*fb1b10abSAndroid Build Coastguard Worker INTRA_FRAME) {
1406*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int =
1407*fb1b10abSAndroid Build Coastguard Worker cpi->lfmv[mb_offset - xd->mode_info_stride - 1].as_int;
1408*fb1b10abSAndroid Build Coastguard Worker mv_bias(
1409*fb1b10abSAndroid Build Coastguard Worker cpi->lf_ref_frame_sign_bias[mb_offset - xd->mode_info_stride - 1],
1410*fb1b10abSAndroid Build Coastguard Worker refframe, &near_mvs[vcnt], ref_frame_sign_bias);
1411*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] =
1412*fb1b10abSAndroid Build Coastguard Worker cpi->lf_ref_frame[mb_offset - xd->mode_info_stride - 1];
1413*fb1b10abSAndroid Build Coastguard Worker }
1414*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1415*fb1b10abSAndroid Build Coastguard Worker
1416*fb1b10abSAndroid Build Coastguard Worker /* left in last frame */
1417*fb1b10abSAndroid Build Coastguard Worker if (cpi->lf_ref_frame[mb_offset - 1] != INTRA_FRAME) {
1418*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int = cpi->lfmv[mb_offset - 1].as_int;
1419*fb1b10abSAndroid Build Coastguard Worker mv_bias(cpi->lf_ref_frame_sign_bias[mb_offset - 1], refframe,
1420*fb1b10abSAndroid Build Coastguard Worker &near_mvs[vcnt], ref_frame_sign_bias);
1421*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] = cpi->lf_ref_frame[mb_offset - 1];
1422*fb1b10abSAndroid Build Coastguard Worker }
1423*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1424*fb1b10abSAndroid Build Coastguard Worker
1425*fb1b10abSAndroid Build Coastguard Worker /* right in last frame */
1426*fb1b10abSAndroid Build Coastguard Worker if (cpi->lf_ref_frame[mb_offset + 1] != INTRA_FRAME) {
1427*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int = cpi->lfmv[mb_offset + 1].as_int;
1428*fb1b10abSAndroid Build Coastguard Worker mv_bias(cpi->lf_ref_frame_sign_bias[mb_offset + 1], refframe,
1429*fb1b10abSAndroid Build Coastguard Worker &near_mvs[vcnt], ref_frame_sign_bias);
1430*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] = cpi->lf_ref_frame[mb_offset + 1];
1431*fb1b10abSAndroid Build Coastguard Worker }
1432*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1433*fb1b10abSAndroid Build Coastguard Worker
1434*fb1b10abSAndroid Build Coastguard Worker /* below in last frame */
1435*fb1b10abSAndroid Build Coastguard Worker if (cpi->lf_ref_frame[mb_offset + xd->mode_info_stride + 1] !=
1436*fb1b10abSAndroid Build Coastguard Worker INTRA_FRAME) {
1437*fb1b10abSAndroid Build Coastguard Worker near_mvs[vcnt].as_int =
1438*fb1b10abSAndroid Build Coastguard Worker cpi->lfmv[mb_offset + xd->mode_info_stride + 1].as_int;
1439*fb1b10abSAndroid Build Coastguard Worker mv_bias(
1440*fb1b10abSAndroid Build Coastguard Worker cpi->lf_ref_frame_sign_bias[mb_offset + xd->mode_info_stride + 1],
1441*fb1b10abSAndroid Build Coastguard Worker refframe, &near_mvs[vcnt], ref_frame_sign_bias);
1442*fb1b10abSAndroid Build Coastguard Worker near_ref[vcnt] =
1443*fb1b10abSAndroid Build Coastguard Worker cpi->lf_ref_frame[mb_offset + xd->mode_info_stride + 1];
1444*fb1b10abSAndroid Build Coastguard Worker }
1445*fb1b10abSAndroid Build Coastguard Worker vcnt++;
1446*fb1b10abSAndroid Build Coastguard Worker }
1447*fb1b10abSAndroid Build Coastguard Worker
1448*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < vcnt; ++i) {
1449*fb1b10abSAndroid Build Coastguard Worker if (near_ref[near_sadidx[i]] != INTRA_FRAME) {
1450*fb1b10abSAndroid Build Coastguard Worker if (here->mbmi.ref_frame == near_ref[near_sadidx[i]]) {
1451*fb1b10abSAndroid Build Coastguard Worker mv.as_int = near_mvs[near_sadidx[i]].as_int;
1452*fb1b10abSAndroid Build Coastguard Worker find = 1;
1453*fb1b10abSAndroid Build Coastguard Worker if (i < 3) {
1454*fb1b10abSAndroid Build Coastguard Worker *sr = 3;
1455*fb1b10abSAndroid Build Coastguard Worker } else {
1456*fb1b10abSAndroid Build Coastguard Worker *sr = 2;
1457*fb1b10abSAndroid Build Coastguard Worker }
1458*fb1b10abSAndroid Build Coastguard Worker break;
1459*fb1b10abSAndroid Build Coastguard Worker }
1460*fb1b10abSAndroid Build Coastguard Worker }
1461*fb1b10abSAndroid Build Coastguard Worker }
1462*fb1b10abSAndroid Build Coastguard Worker
1463*fb1b10abSAndroid Build Coastguard Worker if (!find) {
1464*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < vcnt; ++i) {
1465*fb1b10abSAndroid Build Coastguard Worker mvx[i] = near_mvs[i].as_mv.row;
1466*fb1b10abSAndroid Build Coastguard Worker mvy[i] = near_mvs[i].as_mv.col;
1467*fb1b10abSAndroid Build Coastguard Worker }
1468*fb1b10abSAndroid Build Coastguard Worker
1469*fb1b10abSAndroid Build Coastguard Worker insertsortmv(mvx, vcnt);
1470*fb1b10abSAndroid Build Coastguard Worker insertsortmv(mvy, vcnt);
1471*fb1b10abSAndroid Build Coastguard Worker mv.as_mv.row = mvx[vcnt / 2];
1472*fb1b10abSAndroid Build Coastguard Worker mv.as_mv.col = mvy[vcnt / 2];
1473*fb1b10abSAndroid Build Coastguard Worker
1474*fb1b10abSAndroid Build Coastguard Worker /* sr is set to 0 to allow calling function to decide the search
1475*fb1b10abSAndroid Build Coastguard Worker * range.
1476*fb1b10abSAndroid Build Coastguard Worker */
1477*fb1b10abSAndroid Build Coastguard Worker *sr = 0;
1478*fb1b10abSAndroid Build Coastguard Worker }
1479*fb1b10abSAndroid Build Coastguard Worker }
1480*fb1b10abSAndroid Build Coastguard Worker
1481*fb1b10abSAndroid Build Coastguard Worker /* Set up return values */
1482*fb1b10abSAndroid Build Coastguard Worker mvp->as_int = mv.as_int;
1483*fb1b10abSAndroid Build Coastguard Worker vp8_clamp_mv2(mvp, xd);
1484*fb1b10abSAndroid Build Coastguard Worker }
1485*fb1b10abSAndroid Build Coastguard Worker
vp8_cal_sad(VP8_COMP * cpi,MACROBLOCKD * xd,MACROBLOCK * x,int recon_yoffset,int near_sadidx[])1486*fb1b10abSAndroid Build Coastguard Worker void vp8_cal_sad(VP8_COMP *cpi, MACROBLOCKD *xd, MACROBLOCK *x,
1487*fb1b10abSAndroid Build Coastguard Worker int recon_yoffset, int near_sadidx[]) {
1488*fb1b10abSAndroid Build Coastguard Worker /* near_sad indexes:
1489*fb1b10abSAndroid Build Coastguard Worker * 0-cf above, 1-cf left, 2-cf aboveleft,
1490*fb1b10abSAndroid Build Coastguard Worker * 3-lf current, 4-lf above, 5-lf left, 6-lf right, 7-lf below
1491*fb1b10abSAndroid Build Coastguard Worker */
1492*fb1b10abSAndroid Build Coastguard Worker int near_sad[8] = { 0 };
1493*fb1b10abSAndroid Build Coastguard Worker BLOCK *b = &x->block[0];
1494*fb1b10abSAndroid Build Coastguard Worker unsigned char *src_y_ptr = *(b->base_src);
1495*fb1b10abSAndroid Build Coastguard Worker
1496*fb1b10abSAndroid Build Coastguard Worker /* calculate sad for current frame 3 nearby MBs. */
1497*fb1b10abSAndroid Build Coastguard Worker if (xd->mb_to_top_edge == 0 && xd->mb_to_left_edge == 0) {
1498*fb1b10abSAndroid Build Coastguard Worker near_sad[0] = near_sad[1] = near_sad[2] = INT_MAX;
1499*fb1b10abSAndroid Build Coastguard Worker } else if (xd->mb_to_top_edge ==
1500*fb1b10abSAndroid Build Coastguard Worker 0) { /* only has left MB for sad calculation. */
1501*fb1b10abSAndroid Build Coastguard Worker near_sad[0] = near_sad[2] = INT_MAX;
1502*fb1b10abSAndroid Build Coastguard Worker near_sad[1] = cpi->fn_ptr[BLOCK_16X16].sdf(
1503*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, xd->dst.y_buffer - 16, xd->dst.y_stride);
1504*fb1b10abSAndroid Build Coastguard Worker } else if (xd->mb_to_left_edge ==
1505*fb1b10abSAndroid Build Coastguard Worker 0) { /* only has left MB for sad calculation. */
1506*fb1b10abSAndroid Build Coastguard Worker near_sad[1] = near_sad[2] = INT_MAX;
1507*fb1b10abSAndroid Build Coastguard Worker near_sad[0] = cpi->fn_ptr[BLOCK_16X16].sdf(
1508*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, xd->dst.y_buffer - xd->dst.y_stride * 16,
1509*fb1b10abSAndroid Build Coastguard Worker xd->dst.y_stride);
1510*fb1b10abSAndroid Build Coastguard Worker } else {
1511*fb1b10abSAndroid Build Coastguard Worker near_sad[0] = cpi->fn_ptr[BLOCK_16X16].sdf(
1512*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, xd->dst.y_buffer - xd->dst.y_stride * 16,
1513*fb1b10abSAndroid Build Coastguard Worker xd->dst.y_stride);
1514*fb1b10abSAndroid Build Coastguard Worker near_sad[1] = cpi->fn_ptr[BLOCK_16X16].sdf(
1515*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, xd->dst.y_buffer - 16, xd->dst.y_stride);
1516*fb1b10abSAndroid Build Coastguard Worker near_sad[2] = cpi->fn_ptr[BLOCK_16X16].sdf(
1517*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, xd->dst.y_buffer - xd->dst.y_stride * 16 - 16,
1518*fb1b10abSAndroid Build Coastguard Worker xd->dst.y_stride);
1519*fb1b10abSAndroid Build Coastguard Worker }
1520*fb1b10abSAndroid Build Coastguard Worker
1521*fb1b10abSAndroid Build Coastguard Worker if (cpi->common.last_frame_type != KEY_FRAME) {
1522*fb1b10abSAndroid Build Coastguard Worker /* calculate sad for last frame 5 nearby MBs. */
1523*fb1b10abSAndroid Build Coastguard Worker unsigned char *pre_y_buffer =
1524*fb1b10abSAndroid Build Coastguard Worker cpi->common.yv12_fb[cpi->common.lst_fb_idx].y_buffer + recon_yoffset;
1525*fb1b10abSAndroid Build Coastguard Worker int pre_y_stride = cpi->common.yv12_fb[cpi->common.lst_fb_idx].y_stride;
1526*fb1b10abSAndroid Build Coastguard Worker
1527*fb1b10abSAndroid Build Coastguard Worker if (xd->mb_to_top_edge == 0) near_sad[4] = INT_MAX;
1528*fb1b10abSAndroid Build Coastguard Worker if (xd->mb_to_left_edge == 0) near_sad[5] = INT_MAX;
1529*fb1b10abSAndroid Build Coastguard Worker if (xd->mb_to_right_edge == 0) near_sad[6] = INT_MAX;
1530*fb1b10abSAndroid Build Coastguard Worker if (xd->mb_to_bottom_edge == 0) near_sad[7] = INT_MAX;
1531*fb1b10abSAndroid Build Coastguard Worker
1532*fb1b10abSAndroid Build Coastguard Worker if (near_sad[4] != INT_MAX) {
1533*fb1b10abSAndroid Build Coastguard Worker near_sad[4] = cpi->fn_ptr[BLOCK_16X16].sdf(
1534*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, pre_y_buffer - pre_y_stride * 16,
1535*fb1b10abSAndroid Build Coastguard Worker pre_y_stride);
1536*fb1b10abSAndroid Build Coastguard Worker }
1537*fb1b10abSAndroid Build Coastguard Worker if (near_sad[5] != INT_MAX) {
1538*fb1b10abSAndroid Build Coastguard Worker near_sad[5] = cpi->fn_ptr[BLOCK_16X16].sdf(
1539*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, pre_y_buffer - 16, pre_y_stride);
1540*fb1b10abSAndroid Build Coastguard Worker }
1541*fb1b10abSAndroid Build Coastguard Worker near_sad[3] = cpi->fn_ptr[BLOCK_16X16].sdf(src_y_ptr, b->src_stride,
1542*fb1b10abSAndroid Build Coastguard Worker pre_y_buffer, pre_y_stride);
1543*fb1b10abSAndroid Build Coastguard Worker if (near_sad[6] != INT_MAX) {
1544*fb1b10abSAndroid Build Coastguard Worker near_sad[6] = cpi->fn_ptr[BLOCK_16X16].sdf(
1545*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, pre_y_buffer + 16, pre_y_stride);
1546*fb1b10abSAndroid Build Coastguard Worker }
1547*fb1b10abSAndroid Build Coastguard Worker if (near_sad[7] != INT_MAX) {
1548*fb1b10abSAndroid Build Coastguard Worker near_sad[7] = cpi->fn_ptr[BLOCK_16X16].sdf(
1549*fb1b10abSAndroid Build Coastguard Worker src_y_ptr, b->src_stride, pre_y_buffer + pre_y_stride * 16,
1550*fb1b10abSAndroid Build Coastguard Worker pre_y_stride);
1551*fb1b10abSAndroid Build Coastguard Worker }
1552*fb1b10abSAndroid Build Coastguard Worker }
1553*fb1b10abSAndroid Build Coastguard Worker
1554*fb1b10abSAndroid Build Coastguard Worker if (cpi->common.last_frame_type != KEY_FRAME) {
1555*fb1b10abSAndroid Build Coastguard Worker insertsortsad(near_sad, near_sadidx, 8);
1556*fb1b10abSAndroid Build Coastguard Worker } else {
1557*fb1b10abSAndroid Build Coastguard Worker insertsortsad(near_sad, near_sadidx, 3);
1558*fb1b10abSAndroid Build Coastguard Worker }
1559*fb1b10abSAndroid Build Coastguard Worker }
1560*fb1b10abSAndroid Build Coastguard Worker
rd_update_mvcount(MACROBLOCK * x,int_mv * best_ref_mv)1561*fb1b10abSAndroid Build Coastguard Worker static void rd_update_mvcount(MACROBLOCK *x, int_mv *best_ref_mv) {
1562*fb1b10abSAndroid Build Coastguard Worker if (x->e_mbd.mode_info_context->mbmi.mode == SPLITMV) {
1563*fb1b10abSAndroid Build Coastguard Worker int i;
1564*fb1b10abSAndroid Build Coastguard Worker
1565*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < x->partition_info->count; ++i) {
1566*fb1b10abSAndroid Build Coastguard Worker if (x->partition_info->bmi[i].mode == NEW4X4) {
1567*fb1b10abSAndroid Build Coastguard Worker const int row_val = ((x->partition_info->bmi[i].mv.as_mv.row -
1568*fb1b10abSAndroid Build Coastguard Worker best_ref_mv->as_mv.row) >>
1569*fb1b10abSAndroid Build Coastguard Worker 1);
1570*fb1b10abSAndroid Build Coastguard Worker const int row_idx = mv_max + row_val;
1571*fb1b10abSAndroid Build Coastguard Worker const int col_val = ((x->partition_info->bmi[i].mv.as_mv.col -
1572*fb1b10abSAndroid Build Coastguard Worker best_ref_mv->as_mv.col) >>
1573*fb1b10abSAndroid Build Coastguard Worker 1);
1574*fb1b10abSAndroid Build Coastguard Worker const int col_idx = mv_max + col_val;
1575*fb1b10abSAndroid Build Coastguard Worker if (row_idx >= 0 && row_idx < MVvals && col_idx >= 0 &&
1576*fb1b10abSAndroid Build Coastguard Worker col_idx < MVvals) {
1577*fb1b10abSAndroid Build Coastguard Worker x->MVcount[0][row_idx]++;
1578*fb1b10abSAndroid Build Coastguard Worker x->MVcount[1][col_idx]++;
1579*fb1b10abSAndroid Build Coastguard Worker }
1580*fb1b10abSAndroid Build Coastguard Worker }
1581*fb1b10abSAndroid Build Coastguard Worker }
1582*fb1b10abSAndroid Build Coastguard Worker } else if (x->e_mbd.mode_info_context->mbmi.mode == NEWMV) {
1583*fb1b10abSAndroid Build Coastguard Worker const int row_val = ((x->e_mbd.mode_info_context->mbmi.mv.as_mv.row -
1584*fb1b10abSAndroid Build Coastguard Worker best_ref_mv->as_mv.row) >>
1585*fb1b10abSAndroid Build Coastguard Worker 1);
1586*fb1b10abSAndroid Build Coastguard Worker const int row_idx = mv_max + row_val;
1587*fb1b10abSAndroid Build Coastguard Worker const int col_val = ((x->e_mbd.mode_info_context->mbmi.mv.as_mv.col -
1588*fb1b10abSAndroid Build Coastguard Worker best_ref_mv->as_mv.col) >>
1589*fb1b10abSAndroid Build Coastguard Worker 1);
1590*fb1b10abSAndroid Build Coastguard Worker const int col_idx = mv_max + col_val;
1591*fb1b10abSAndroid Build Coastguard Worker if (row_idx >= 0 && row_idx < MVvals && col_idx >= 0 && col_idx < MVvals) {
1592*fb1b10abSAndroid Build Coastguard Worker x->MVcount[0][row_idx]++;
1593*fb1b10abSAndroid Build Coastguard Worker x->MVcount[1][col_idx]++;
1594*fb1b10abSAndroid Build Coastguard Worker }
1595*fb1b10abSAndroid Build Coastguard Worker }
1596*fb1b10abSAndroid Build Coastguard Worker }
1597*fb1b10abSAndroid Build Coastguard Worker
evaluate_inter_mode_rd(int mdcounts[4],RATE_DISTORTION * rd,int * disable_skip,VP8_COMP * cpi,MACROBLOCK * x)1598*fb1b10abSAndroid Build Coastguard Worker static int evaluate_inter_mode_rd(int mdcounts[4], RATE_DISTORTION *rd,
1599*fb1b10abSAndroid Build Coastguard Worker int *disable_skip, VP8_COMP *cpi,
1600*fb1b10abSAndroid Build Coastguard Worker MACROBLOCK *x) {
1601*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE this_mode = x->e_mbd.mode_info_context->mbmi.mode;
1602*fb1b10abSAndroid Build Coastguard Worker BLOCK *b = &x->block[0];
1603*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *xd = &x->e_mbd;
1604*fb1b10abSAndroid Build Coastguard Worker int distortion;
1605*fb1b10abSAndroid Build Coastguard Worker vp8_build_inter16x16_predictors_mby(&x->e_mbd, x->e_mbd.predictor, 16);
1606*fb1b10abSAndroid Build Coastguard Worker
1607*fb1b10abSAndroid Build Coastguard Worker if (cpi->active_map_enabled && x->active_ptr[0] == 0) {
1608*fb1b10abSAndroid Build Coastguard Worker x->skip = 1;
1609*fb1b10abSAndroid Build Coastguard Worker } else if (x->encode_breakout) {
1610*fb1b10abSAndroid Build Coastguard Worker unsigned int sse;
1611*fb1b10abSAndroid Build Coastguard Worker unsigned int var;
1612*fb1b10abSAndroid Build Coastguard Worker unsigned int threshold =
1613*fb1b10abSAndroid Build Coastguard Worker (xd->block[0].dequant[1] * xd->block[0].dequant[1] >> 4);
1614*fb1b10abSAndroid Build Coastguard Worker
1615*fb1b10abSAndroid Build Coastguard Worker if (threshold < x->encode_breakout) threshold = x->encode_breakout;
1616*fb1b10abSAndroid Build Coastguard Worker
1617*fb1b10abSAndroid Build Coastguard Worker var = vpx_variance16x16(*(b->base_src), b->src_stride, x->e_mbd.predictor,
1618*fb1b10abSAndroid Build Coastguard Worker 16, &sse);
1619*fb1b10abSAndroid Build Coastguard Worker
1620*fb1b10abSAndroid Build Coastguard Worker if (sse < threshold) {
1621*fb1b10abSAndroid Build Coastguard Worker unsigned int q2dc = xd->block[24].dequant[0];
1622*fb1b10abSAndroid Build Coastguard Worker /* If theres is no codeable 2nd order dc
1623*fb1b10abSAndroid Build Coastguard Worker or a very small uniform pixel change change */
1624*fb1b10abSAndroid Build Coastguard Worker if ((sse - var < q2dc * q2dc >> 4) || (sse / 2 > var && sse - var < 64)) {
1625*fb1b10abSAndroid Build Coastguard Worker /* Check u and v to make sure skip is ok */
1626*fb1b10abSAndroid Build Coastguard Worker unsigned int sse2 = VP8_UVSSE(x);
1627*fb1b10abSAndroid Build Coastguard Worker if (sse2 * 2 < threshold) {
1628*fb1b10abSAndroid Build Coastguard Worker x->skip = 1;
1629*fb1b10abSAndroid Build Coastguard Worker rd->distortion2 = sse + sse2;
1630*fb1b10abSAndroid Build Coastguard Worker rd->rate2 = 500;
1631*fb1b10abSAndroid Build Coastguard Worker
1632*fb1b10abSAndroid Build Coastguard Worker /* for best_yrd calculation */
1633*fb1b10abSAndroid Build Coastguard Worker rd->rate_uv = 0;
1634*fb1b10abSAndroid Build Coastguard Worker rd->distortion_uv = sse2;
1635*fb1b10abSAndroid Build Coastguard Worker
1636*fb1b10abSAndroid Build Coastguard Worker *disable_skip = 1;
1637*fb1b10abSAndroid Build Coastguard Worker return RDCOST(x->rdmult, x->rddiv, rd->rate2, rd->distortion2);
1638*fb1b10abSAndroid Build Coastguard Worker }
1639*fb1b10abSAndroid Build Coastguard Worker }
1640*fb1b10abSAndroid Build Coastguard Worker }
1641*fb1b10abSAndroid Build Coastguard Worker }
1642*fb1b10abSAndroid Build Coastguard Worker
1643*fb1b10abSAndroid Build Coastguard Worker /* Add in the Mv/mode cost */
1644*fb1b10abSAndroid Build Coastguard Worker rd->rate2 += vp8_cost_mv_ref(this_mode, mdcounts);
1645*fb1b10abSAndroid Build Coastguard Worker
1646*fb1b10abSAndroid Build Coastguard Worker /* Y cost and distortion */
1647*fb1b10abSAndroid Build Coastguard Worker macro_block_yrd(x, &rd->rate_y, &distortion);
1648*fb1b10abSAndroid Build Coastguard Worker rd->rate2 += rd->rate_y;
1649*fb1b10abSAndroid Build Coastguard Worker rd->distortion2 += distortion;
1650*fb1b10abSAndroid Build Coastguard Worker
1651*fb1b10abSAndroid Build Coastguard Worker /* UV cost and distortion */
1652*fb1b10abSAndroid Build Coastguard Worker rd_inter16x16_uv(cpi, x, &rd->rate_uv, &rd->distortion_uv,
1653*fb1b10abSAndroid Build Coastguard Worker cpi->common.full_pixel);
1654*fb1b10abSAndroid Build Coastguard Worker rd->rate2 += rd->rate_uv;
1655*fb1b10abSAndroid Build Coastguard Worker rd->distortion2 += rd->distortion_uv;
1656*fb1b10abSAndroid Build Coastguard Worker return INT_MAX;
1657*fb1b10abSAndroid Build Coastguard Worker }
1658*fb1b10abSAndroid Build Coastguard Worker
calculate_final_rd_costs(int this_rd,RATE_DISTORTION * rd,int * other_cost,int disable_skip,int uv_intra_tteob,int intra_rd_penalty,VP8_COMP * cpi,MACROBLOCK * x)1659*fb1b10abSAndroid Build Coastguard Worker static int calculate_final_rd_costs(int this_rd, RATE_DISTORTION *rd,
1660*fb1b10abSAndroid Build Coastguard Worker int *other_cost, int disable_skip,
1661*fb1b10abSAndroid Build Coastguard Worker int uv_intra_tteob, int intra_rd_penalty,
1662*fb1b10abSAndroid Build Coastguard Worker VP8_COMP *cpi, MACROBLOCK *x) {
1663*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE this_mode = x->e_mbd.mode_info_context->mbmi.mode;
1664*fb1b10abSAndroid Build Coastguard Worker
1665*fb1b10abSAndroid Build Coastguard Worker /* Where skip is allowable add in the default per mb cost for the no
1666*fb1b10abSAndroid Build Coastguard Worker * skip case. where we then decide to skip we have to delete this and
1667*fb1b10abSAndroid Build Coastguard Worker * replace it with the cost of signalling a skip
1668*fb1b10abSAndroid Build Coastguard Worker */
1669*fb1b10abSAndroid Build Coastguard Worker if (cpi->common.mb_no_coeff_skip) {
1670*fb1b10abSAndroid Build Coastguard Worker *other_cost += vp8_cost_bit(cpi->prob_skip_false, 0);
1671*fb1b10abSAndroid Build Coastguard Worker rd->rate2 += *other_cost;
1672*fb1b10abSAndroid Build Coastguard Worker }
1673*fb1b10abSAndroid Build Coastguard Worker
1674*fb1b10abSAndroid Build Coastguard Worker /* Estimate the reference frame signaling cost and add it
1675*fb1b10abSAndroid Build Coastguard Worker * to the rolling cost variable.
1676*fb1b10abSAndroid Build Coastguard Worker */
1677*fb1b10abSAndroid Build Coastguard Worker rd->rate2 += x->ref_frame_cost[x->e_mbd.mode_info_context->mbmi.ref_frame];
1678*fb1b10abSAndroid Build Coastguard Worker
1679*fb1b10abSAndroid Build Coastguard Worker if (!disable_skip) {
1680*fb1b10abSAndroid Build Coastguard Worker /* Test for the condition where skip block will be activated
1681*fb1b10abSAndroid Build Coastguard Worker * because there are no non zero coefficients and make any
1682*fb1b10abSAndroid Build Coastguard Worker * necessary adjustment for rate
1683*fb1b10abSAndroid Build Coastguard Worker */
1684*fb1b10abSAndroid Build Coastguard Worker if (cpi->common.mb_no_coeff_skip) {
1685*fb1b10abSAndroid Build Coastguard Worker int i;
1686*fb1b10abSAndroid Build Coastguard Worker int tteob;
1687*fb1b10abSAndroid Build Coastguard Worker int has_y2_block = (this_mode != SPLITMV && this_mode != B_PRED);
1688*fb1b10abSAndroid Build Coastguard Worker
1689*fb1b10abSAndroid Build Coastguard Worker tteob = 0;
1690*fb1b10abSAndroid Build Coastguard Worker if (has_y2_block) tteob += x->e_mbd.eobs[24];
1691*fb1b10abSAndroid Build Coastguard Worker
1692*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) tteob += (x->e_mbd.eobs[i] > has_y2_block);
1693*fb1b10abSAndroid Build Coastguard Worker
1694*fb1b10abSAndroid Build Coastguard Worker if (x->e_mbd.mode_info_context->mbmi.ref_frame) {
1695*fb1b10abSAndroid Build Coastguard Worker for (i = 16; i < 24; ++i) tteob += x->e_mbd.eobs[i];
1696*fb1b10abSAndroid Build Coastguard Worker } else {
1697*fb1b10abSAndroid Build Coastguard Worker tteob += uv_intra_tteob;
1698*fb1b10abSAndroid Build Coastguard Worker }
1699*fb1b10abSAndroid Build Coastguard Worker
1700*fb1b10abSAndroid Build Coastguard Worker if (tteob == 0) {
1701*fb1b10abSAndroid Build Coastguard Worker rd->rate2 -= (rd->rate_y + rd->rate_uv);
1702*fb1b10abSAndroid Build Coastguard Worker /* for best_yrd calculation */
1703*fb1b10abSAndroid Build Coastguard Worker rd->rate_uv = 0;
1704*fb1b10abSAndroid Build Coastguard Worker
1705*fb1b10abSAndroid Build Coastguard Worker /* Back out no skip flag costing and add in skip flag costing */
1706*fb1b10abSAndroid Build Coastguard Worker if (cpi->prob_skip_false) {
1707*fb1b10abSAndroid Build Coastguard Worker int prob_skip_cost;
1708*fb1b10abSAndroid Build Coastguard Worker
1709*fb1b10abSAndroid Build Coastguard Worker prob_skip_cost = vp8_cost_bit(cpi->prob_skip_false, 1);
1710*fb1b10abSAndroid Build Coastguard Worker prob_skip_cost -= (int)vp8_cost_bit(cpi->prob_skip_false, 0);
1711*fb1b10abSAndroid Build Coastguard Worker rd->rate2 += prob_skip_cost;
1712*fb1b10abSAndroid Build Coastguard Worker *other_cost += prob_skip_cost;
1713*fb1b10abSAndroid Build Coastguard Worker }
1714*fb1b10abSAndroid Build Coastguard Worker }
1715*fb1b10abSAndroid Build Coastguard Worker }
1716*fb1b10abSAndroid Build Coastguard Worker /* Calculate the final RD estimate for this mode */
1717*fb1b10abSAndroid Build Coastguard Worker this_rd = RDCOST(x->rdmult, x->rddiv, rd->rate2, rd->distortion2);
1718*fb1b10abSAndroid Build Coastguard Worker if (this_rd < INT_MAX &&
1719*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame == INTRA_FRAME) {
1720*fb1b10abSAndroid Build Coastguard Worker this_rd += intra_rd_penalty;
1721*fb1b10abSAndroid Build Coastguard Worker }
1722*fb1b10abSAndroid Build Coastguard Worker }
1723*fb1b10abSAndroid Build Coastguard Worker return this_rd;
1724*fb1b10abSAndroid Build Coastguard Worker }
1725*fb1b10abSAndroid Build Coastguard Worker
update_best_mode(BEST_MODE * best_mode,int this_rd,RATE_DISTORTION * rd,int other_cost,MACROBLOCK * x)1726*fb1b10abSAndroid Build Coastguard Worker static void update_best_mode(BEST_MODE *best_mode, int this_rd,
1727*fb1b10abSAndroid Build Coastguard Worker RATE_DISTORTION *rd, int other_cost,
1728*fb1b10abSAndroid Build Coastguard Worker MACROBLOCK *x) {
1729*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE this_mode = x->e_mbd.mode_info_context->mbmi.mode;
1730*fb1b10abSAndroid Build Coastguard Worker
1731*fb1b10abSAndroid Build Coastguard Worker other_cost += x->ref_frame_cost[x->e_mbd.mode_info_context->mbmi.ref_frame];
1732*fb1b10abSAndroid Build Coastguard Worker
1733*fb1b10abSAndroid Build Coastguard Worker /* Calculate the final y RD estimate for this mode */
1734*fb1b10abSAndroid Build Coastguard Worker best_mode->yrd =
1735*fb1b10abSAndroid Build Coastguard Worker RDCOST(x->rdmult, x->rddiv, (rd->rate2 - rd->rate_uv - other_cost),
1736*fb1b10abSAndroid Build Coastguard Worker (rd->distortion2 - rd->distortion_uv));
1737*fb1b10abSAndroid Build Coastguard Worker
1738*fb1b10abSAndroid Build Coastguard Worker best_mode->rd = this_rd;
1739*fb1b10abSAndroid Build Coastguard Worker memcpy(&best_mode->mbmode, &x->e_mbd.mode_info_context->mbmi,
1740*fb1b10abSAndroid Build Coastguard Worker sizeof(MB_MODE_INFO));
1741*fb1b10abSAndroid Build Coastguard Worker memcpy(&best_mode->partition, x->partition_info, sizeof(PARTITION_INFO));
1742*fb1b10abSAndroid Build Coastguard Worker
1743*fb1b10abSAndroid Build Coastguard Worker if ((this_mode == B_PRED) || (this_mode == SPLITMV)) {
1744*fb1b10abSAndroid Build Coastguard Worker int i;
1745*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
1746*fb1b10abSAndroid Build Coastguard Worker best_mode->bmodes[i] = x->e_mbd.block[i].bmi;
1747*fb1b10abSAndroid Build Coastguard Worker }
1748*fb1b10abSAndroid Build Coastguard Worker }
1749*fb1b10abSAndroid Build Coastguard Worker }
1750*fb1b10abSAndroid Build Coastguard Worker
vp8_rd_pick_inter_mode(VP8_COMP * cpi,MACROBLOCK * x,int recon_yoffset,int recon_uvoffset,int * returnrate,int * returndistortion,int * returnintra,int mb_row,int mb_col)1751*fb1b10abSAndroid Build Coastguard Worker void vp8_rd_pick_inter_mode(VP8_COMP *cpi, MACROBLOCK *x, int recon_yoffset,
1752*fb1b10abSAndroid Build Coastguard Worker int recon_uvoffset, int *returnrate,
1753*fb1b10abSAndroid Build Coastguard Worker int *returndistortion, int *returnintra, int mb_row,
1754*fb1b10abSAndroid Build Coastguard Worker int mb_col) {
1755*fb1b10abSAndroid Build Coastguard Worker BLOCK *b = &x->block[0];
1756*fb1b10abSAndroid Build Coastguard Worker BLOCKD *d = &x->e_mbd.block[0];
1757*fb1b10abSAndroid Build Coastguard Worker MACROBLOCKD *xd = &x->e_mbd;
1758*fb1b10abSAndroid Build Coastguard Worker int_mv best_ref_mv_sb[2];
1759*fb1b10abSAndroid Build Coastguard Worker int_mv mode_mv_sb[2][MB_MODE_COUNT];
1760*fb1b10abSAndroid Build Coastguard Worker int_mv best_ref_mv;
1761*fb1b10abSAndroid Build Coastguard Worker int_mv *mode_mv;
1762*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE this_mode;
1763*fb1b10abSAndroid Build Coastguard Worker int num00;
1764*fb1b10abSAndroid Build Coastguard Worker int best_mode_index = 0;
1765*fb1b10abSAndroid Build Coastguard Worker BEST_MODE best_mode;
1766*fb1b10abSAndroid Build Coastguard Worker
1767*fb1b10abSAndroid Build Coastguard Worker int i;
1768*fb1b10abSAndroid Build Coastguard Worker int mode_index;
1769*fb1b10abSAndroid Build Coastguard Worker int mdcounts[4];
1770*fb1b10abSAndroid Build Coastguard Worker int rate;
1771*fb1b10abSAndroid Build Coastguard Worker RATE_DISTORTION rd;
1772*fb1b10abSAndroid Build Coastguard Worker int uv_intra_rate, uv_intra_distortion, uv_intra_rate_tokenonly;
1773*fb1b10abSAndroid Build Coastguard Worker int uv_intra_tteob = 0;
1774*fb1b10abSAndroid Build Coastguard Worker int uv_intra_done = 0;
1775*fb1b10abSAndroid Build Coastguard Worker
1776*fb1b10abSAndroid Build Coastguard Worker MB_PREDICTION_MODE uv_intra_mode = 0;
1777*fb1b10abSAndroid Build Coastguard Worker int_mv mvp;
1778*fb1b10abSAndroid Build Coastguard Worker int near_sadidx[8] = { 0, 1, 2, 3, 4, 5, 6, 7 };
1779*fb1b10abSAndroid Build Coastguard Worker int saddone = 0;
1780*fb1b10abSAndroid Build Coastguard Worker /* search range got from mv_pred(). It uses step_param levels. (0-7) */
1781*fb1b10abSAndroid Build Coastguard Worker int sr = 0;
1782*fb1b10abSAndroid Build Coastguard Worker
1783*fb1b10abSAndroid Build Coastguard Worker unsigned char *plane[4][3] = { { 0, 0 } };
1784*fb1b10abSAndroid Build Coastguard Worker int ref_frame_map[4];
1785*fb1b10abSAndroid Build Coastguard Worker int sign_bias = 0;
1786*fb1b10abSAndroid Build Coastguard Worker
1787*fb1b10abSAndroid Build Coastguard Worker int intra_rd_penalty =
1788*fb1b10abSAndroid Build Coastguard Worker 10 * vp8_dc_quant(cpi->common.base_qindex, cpi->common.y1dc_delta_q);
1789*fb1b10abSAndroid Build Coastguard Worker
1790*fb1b10abSAndroid Build Coastguard Worker #if CONFIG_TEMPORAL_DENOISING
1791*fb1b10abSAndroid Build Coastguard Worker unsigned int zero_mv_sse = UINT_MAX, best_sse = UINT_MAX,
1792*fb1b10abSAndroid Build Coastguard Worker best_rd_sse = UINT_MAX;
1793*fb1b10abSAndroid Build Coastguard Worker #endif
1794*fb1b10abSAndroid Build Coastguard Worker
1795*fb1b10abSAndroid Build Coastguard Worker // _uv variables are not set consistantly before calling update_best_mode.
1796*fb1b10abSAndroid Build Coastguard Worker rd.rate_uv = 0;
1797*fb1b10abSAndroid Build Coastguard Worker rd.distortion_uv = 0;
1798*fb1b10abSAndroid Build Coastguard Worker
1799*fb1b10abSAndroid Build Coastguard Worker mode_mv = mode_mv_sb[sign_bias];
1800*fb1b10abSAndroid Build Coastguard Worker best_ref_mv.as_int = 0;
1801*fb1b10abSAndroid Build Coastguard Worker best_mode.rd = INT_MAX;
1802*fb1b10abSAndroid Build Coastguard Worker best_mode.yrd = INT_MAX;
1803*fb1b10abSAndroid Build Coastguard Worker best_mode.intra_rd = INT_MAX;
1804*fb1b10abSAndroid Build Coastguard Worker memset(mode_mv_sb, 0, sizeof(mode_mv_sb));
1805*fb1b10abSAndroid Build Coastguard Worker memset(&best_mode.mbmode, 0, sizeof(best_mode.mbmode));
1806*fb1b10abSAndroid Build Coastguard Worker memset(&best_mode.bmodes, 0, sizeof(best_mode.bmodes));
1807*fb1b10abSAndroid Build Coastguard Worker
1808*fb1b10abSAndroid Build Coastguard Worker /* Setup search priorities */
1809*fb1b10abSAndroid Build Coastguard Worker get_reference_search_order(cpi, ref_frame_map);
1810*fb1b10abSAndroid Build Coastguard Worker
1811*fb1b10abSAndroid Build Coastguard Worker /* Check to see if there is at least 1 valid reference frame that we need
1812*fb1b10abSAndroid Build Coastguard Worker * to calculate near_mvs.
1813*fb1b10abSAndroid Build Coastguard Worker */
1814*fb1b10abSAndroid Build Coastguard Worker if (ref_frame_map[1] > 0) {
1815*fb1b10abSAndroid Build Coastguard Worker sign_bias = vp8_find_near_mvs_bias(
1816*fb1b10abSAndroid Build Coastguard Worker &x->e_mbd, x->e_mbd.mode_info_context, mode_mv_sb, best_ref_mv_sb,
1817*fb1b10abSAndroid Build Coastguard Worker mdcounts, ref_frame_map[1], cpi->common.ref_frame_sign_bias);
1818*fb1b10abSAndroid Build Coastguard Worker
1819*fb1b10abSAndroid Build Coastguard Worker mode_mv = mode_mv_sb[sign_bias];
1820*fb1b10abSAndroid Build Coastguard Worker best_ref_mv.as_int = best_ref_mv_sb[sign_bias].as_int;
1821*fb1b10abSAndroid Build Coastguard Worker }
1822*fb1b10abSAndroid Build Coastguard Worker
1823*fb1b10abSAndroid Build Coastguard Worker get_predictor_pointers(cpi, plane, recon_yoffset, recon_uvoffset);
1824*fb1b10abSAndroid Build Coastguard Worker
1825*fb1b10abSAndroid Build Coastguard Worker *returnintra = INT_MAX;
1826*fb1b10abSAndroid Build Coastguard Worker /* Count of the number of MBs tested so far this frame */
1827*fb1b10abSAndroid Build Coastguard Worker x->mbs_tested_so_far++;
1828*fb1b10abSAndroid Build Coastguard Worker
1829*fb1b10abSAndroid Build Coastguard Worker x->skip = 0;
1830*fb1b10abSAndroid Build Coastguard Worker
1831*fb1b10abSAndroid Build Coastguard Worker for (mode_index = 0; mode_index < MAX_MODES; ++mode_index) {
1832*fb1b10abSAndroid Build Coastguard Worker int this_rd = INT_MAX;
1833*fb1b10abSAndroid Build Coastguard Worker int disable_skip = 0;
1834*fb1b10abSAndroid Build Coastguard Worker int other_cost = 0;
1835*fb1b10abSAndroid Build Coastguard Worker int this_ref_frame = ref_frame_map[vp8_ref_frame_order[mode_index]];
1836*fb1b10abSAndroid Build Coastguard Worker
1837*fb1b10abSAndroid Build Coastguard Worker /* Test best rd so far against threshold for trying this mode. */
1838*fb1b10abSAndroid Build Coastguard Worker if (best_mode.rd <= x->rd_threshes[mode_index]) continue;
1839*fb1b10abSAndroid Build Coastguard Worker
1840*fb1b10abSAndroid Build Coastguard Worker if (this_ref_frame < 0) continue;
1841*fb1b10abSAndroid Build Coastguard Worker
1842*fb1b10abSAndroid Build Coastguard Worker /* These variables hold are rolling total cost and distortion for
1843*fb1b10abSAndroid Build Coastguard Worker * this mode
1844*fb1b10abSAndroid Build Coastguard Worker */
1845*fb1b10abSAndroid Build Coastguard Worker rd.rate2 = 0;
1846*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 = 0;
1847*fb1b10abSAndroid Build Coastguard Worker
1848*fb1b10abSAndroid Build Coastguard Worker this_mode = vp8_mode_order[mode_index];
1849*fb1b10abSAndroid Build Coastguard Worker
1850*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mode = this_mode;
1851*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame = this_ref_frame;
1852*fb1b10abSAndroid Build Coastguard Worker
1853*fb1b10abSAndroid Build Coastguard Worker /* Only consider ZEROMV/ALTREF_FRAME for alt ref frame,
1854*fb1b10abSAndroid Build Coastguard Worker * unless ARNR filtering is enabled in which case we want
1855*fb1b10abSAndroid Build Coastguard Worker * an unfiltered alternative
1856*fb1b10abSAndroid Build Coastguard Worker */
1857*fb1b10abSAndroid Build Coastguard Worker if (cpi->is_src_frame_alt_ref && (cpi->oxcf.arnr_max_frames == 0)) {
1858*fb1b10abSAndroid Build Coastguard Worker if (this_mode != ZEROMV ||
1859*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame != ALTREF_FRAME) {
1860*fb1b10abSAndroid Build Coastguard Worker continue;
1861*fb1b10abSAndroid Build Coastguard Worker }
1862*fb1b10abSAndroid Build Coastguard Worker }
1863*fb1b10abSAndroid Build Coastguard Worker
1864*fb1b10abSAndroid Build Coastguard Worker /* everything but intra */
1865*fb1b10abSAndroid Build Coastguard Worker if (x->e_mbd.mode_info_context->mbmi.ref_frame) {
1866*fb1b10abSAndroid Build Coastguard Worker assert(plane[this_ref_frame][0] != NULL &&
1867*fb1b10abSAndroid Build Coastguard Worker plane[this_ref_frame][1] != NULL &&
1868*fb1b10abSAndroid Build Coastguard Worker plane[this_ref_frame][2] != NULL);
1869*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.pre.y_buffer = plane[this_ref_frame][0];
1870*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.pre.u_buffer = plane[this_ref_frame][1];
1871*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.pre.v_buffer = plane[this_ref_frame][2];
1872*fb1b10abSAndroid Build Coastguard Worker
1873*fb1b10abSAndroid Build Coastguard Worker if (sign_bias != cpi->common.ref_frame_sign_bias[this_ref_frame]) {
1874*fb1b10abSAndroid Build Coastguard Worker sign_bias = cpi->common.ref_frame_sign_bias[this_ref_frame];
1875*fb1b10abSAndroid Build Coastguard Worker mode_mv = mode_mv_sb[sign_bias];
1876*fb1b10abSAndroid Build Coastguard Worker best_ref_mv.as_int = best_ref_mv_sb[sign_bias].as_int;
1877*fb1b10abSAndroid Build Coastguard Worker }
1878*fb1b10abSAndroid Build Coastguard Worker }
1879*fb1b10abSAndroid Build Coastguard Worker
1880*fb1b10abSAndroid Build Coastguard Worker /* Check to see if the testing frequency for this mode is at its
1881*fb1b10abSAndroid Build Coastguard Worker * max If so then prevent it from being tested and increase the
1882*fb1b10abSAndroid Build Coastguard Worker * threshold for its testing
1883*fb1b10abSAndroid Build Coastguard Worker */
1884*fb1b10abSAndroid Build Coastguard Worker if (x->mode_test_hit_counts[mode_index] &&
1885*fb1b10abSAndroid Build Coastguard Worker (cpi->mode_check_freq[mode_index] > 1)) {
1886*fb1b10abSAndroid Build Coastguard Worker if (x->mbs_tested_so_far <= cpi->mode_check_freq[mode_index] *
1887*fb1b10abSAndroid Build Coastguard Worker x->mode_test_hit_counts[mode_index]) {
1888*fb1b10abSAndroid Build Coastguard Worker /* Increase the threshold for coding this mode to make it
1889*fb1b10abSAndroid Build Coastguard Worker * less likely to be chosen
1890*fb1b10abSAndroid Build Coastguard Worker */
1891*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[mode_index] += 4;
1892*fb1b10abSAndroid Build Coastguard Worker
1893*fb1b10abSAndroid Build Coastguard Worker if (x->rd_thresh_mult[mode_index] > MAX_THRESHMULT) {
1894*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[mode_index] = MAX_THRESHMULT;
1895*fb1b10abSAndroid Build Coastguard Worker }
1896*fb1b10abSAndroid Build Coastguard Worker
1897*fb1b10abSAndroid Build Coastguard Worker x->rd_threshes[mode_index] =
1898*fb1b10abSAndroid Build Coastguard Worker (cpi->rd_baseline_thresh[mode_index] >> 7) *
1899*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[mode_index];
1900*fb1b10abSAndroid Build Coastguard Worker
1901*fb1b10abSAndroid Build Coastguard Worker continue;
1902*fb1b10abSAndroid Build Coastguard Worker }
1903*fb1b10abSAndroid Build Coastguard Worker }
1904*fb1b10abSAndroid Build Coastguard Worker
1905*fb1b10abSAndroid Build Coastguard Worker /* We have now reached the point where we are going to test the
1906*fb1b10abSAndroid Build Coastguard Worker * current mode so increment the counter for the number of times
1907*fb1b10abSAndroid Build Coastguard Worker * it has been tested
1908*fb1b10abSAndroid Build Coastguard Worker */
1909*fb1b10abSAndroid Build Coastguard Worker x->mode_test_hit_counts[mode_index]++;
1910*fb1b10abSAndroid Build Coastguard Worker
1911*fb1b10abSAndroid Build Coastguard Worker /* Experimental code. Special case for gf and arf zeromv modes.
1912*fb1b10abSAndroid Build Coastguard Worker * Increase zbin size to supress noise
1913*fb1b10abSAndroid Build Coastguard Worker */
1914*fb1b10abSAndroid Build Coastguard Worker if (x->zbin_mode_boost_enabled) {
1915*fb1b10abSAndroid Build Coastguard Worker if (this_ref_frame == INTRA_FRAME) {
1916*fb1b10abSAndroid Build Coastguard Worker x->zbin_mode_boost = 0;
1917*fb1b10abSAndroid Build Coastguard Worker } else {
1918*fb1b10abSAndroid Build Coastguard Worker if (vp8_mode_order[mode_index] == ZEROMV) {
1919*fb1b10abSAndroid Build Coastguard Worker if (this_ref_frame != LAST_FRAME) {
1920*fb1b10abSAndroid Build Coastguard Worker x->zbin_mode_boost = GF_ZEROMV_ZBIN_BOOST;
1921*fb1b10abSAndroid Build Coastguard Worker } else {
1922*fb1b10abSAndroid Build Coastguard Worker x->zbin_mode_boost = LF_ZEROMV_ZBIN_BOOST;
1923*fb1b10abSAndroid Build Coastguard Worker }
1924*fb1b10abSAndroid Build Coastguard Worker } else if (vp8_mode_order[mode_index] == SPLITMV) {
1925*fb1b10abSAndroid Build Coastguard Worker x->zbin_mode_boost = 0;
1926*fb1b10abSAndroid Build Coastguard Worker } else {
1927*fb1b10abSAndroid Build Coastguard Worker x->zbin_mode_boost = MV_ZBIN_BOOST;
1928*fb1b10abSAndroid Build Coastguard Worker }
1929*fb1b10abSAndroid Build Coastguard Worker }
1930*fb1b10abSAndroid Build Coastguard Worker
1931*fb1b10abSAndroid Build Coastguard Worker vp8_update_zbin_extra(cpi, x);
1932*fb1b10abSAndroid Build Coastguard Worker }
1933*fb1b10abSAndroid Build Coastguard Worker
1934*fb1b10abSAndroid Build Coastguard Worker if (!uv_intra_done && this_ref_frame == INTRA_FRAME) {
1935*fb1b10abSAndroid Build Coastguard Worker rd_pick_intra_mbuv_mode(x, &uv_intra_rate, &uv_intra_rate_tokenonly,
1936*fb1b10abSAndroid Build Coastguard Worker &uv_intra_distortion);
1937*fb1b10abSAndroid Build Coastguard Worker uv_intra_mode = x->e_mbd.mode_info_context->mbmi.uv_mode;
1938*fb1b10abSAndroid Build Coastguard Worker
1939*fb1b10abSAndroid Build Coastguard Worker /*
1940*fb1b10abSAndroid Build Coastguard Worker * Total of the eobs is used later to further adjust rate2. Since uv
1941*fb1b10abSAndroid Build Coastguard Worker * block's intra eobs will be overwritten when we check inter modes,
1942*fb1b10abSAndroid Build Coastguard Worker * we need to save uv_intra_tteob here.
1943*fb1b10abSAndroid Build Coastguard Worker */
1944*fb1b10abSAndroid Build Coastguard Worker for (i = 16; i < 24; ++i) uv_intra_tteob += x->e_mbd.eobs[i];
1945*fb1b10abSAndroid Build Coastguard Worker
1946*fb1b10abSAndroid Build Coastguard Worker uv_intra_done = 1;
1947*fb1b10abSAndroid Build Coastguard Worker }
1948*fb1b10abSAndroid Build Coastguard Worker
1949*fb1b10abSAndroid Build Coastguard Worker switch (this_mode) {
1950*fb1b10abSAndroid Build Coastguard Worker case B_PRED: {
1951*fb1b10abSAndroid Build Coastguard Worker int tmp_rd;
1952*fb1b10abSAndroid Build Coastguard Worker
1953*fb1b10abSAndroid Build Coastguard Worker /* Note the rate value returned here includes the cost of
1954*fb1b10abSAndroid Build Coastguard Worker * coding the BPRED mode: x->mbmode_cost[x->e_mbd.frame_type][BPRED]
1955*fb1b10abSAndroid Build Coastguard Worker */
1956*fb1b10abSAndroid Build Coastguard Worker int distortion;
1957*fb1b10abSAndroid Build Coastguard Worker tmp_rd = rd_pick_intra4x4mby_modes(x, &rate, &rd.rate_y, &distortion,
1958*fb1b10abSAndroid Build Coastguard Worker best_mode.yrd);
1959*fb1b10abSAndroid Build Coastguard Worker rd.rate2 += rate;
1960*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 += distortion;
1961*fb1b10abSAndroid Build Coastguard Worker
1962*fb1b10abSAndroid Build Coastguard Worker if (tmp_rd < best_mode.yrd) {
1963*fb1b10abSAndroid Build Coastguard Worker assert(uv_intra_done);
1964*fb1b10abSAndroid Build Coastguard Worker rd.rate2 += uv_intra_rate;
1965*fb1b10abSAndroid Build Coastguard Worker rd.rate_uv = uv_intra_rate_tokenonly;
1966*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 += uv_intra_distortion;
1967*fb1b10abSAndroid Build Coastguard Worker rd.distortion_uv = uv_intra_distortion;
1968*fb1b10abSAndroid Build Coastguard Worker } else {
1969*fb1b10abSAndroid Build Coastguard Worker this_rd = INT_MAX;
1970*fb1b10abSAndroid Build Coastguard Worker disable_skip = 1;
1971*fb1b10abSAndroid Build Coastguard Worker }
1972*fb1b10abSAndroid Build Coastguard Worker break;
1973*fb1b10abSAndroid Build Coastguard Worker }
1974*fb1b10abSAndroid Build Coastguard Worker
1975*fb1b10abSAndroid Build Coastguard Worker case SPLITMV: {
1976*fb1b10abSAndroid Build Coastguard Worker int tmp_rd;
1977*fb1b10abSAndroid Build Coastguard Worker int this_rd_thresh;
1978*fb1b10abSAndroid Build Coastguard Worker int distortion;
1979*fb1b10abSAndroid Build Coastguard Worker
1980*fb1b10abSAndroid Build Coastguard Worker this_rd_thresh = (vp8_ref_frame_order[mode_index] == 1)
1981*fb1b10abSAndroid Build Coastguard Worker ? x->rd_threshes[THR_NEW1]
1982*fb1b10abSAndroid Build Coastguard Worker : x->rd_threshes[THR_NEW3];
1983*fb1b10abSAndroid Build Coastguard Worker this_rd_thresh = (vp8_ref_frame_order[mode_index] == 2)
1984*fb1b10abSAndroid Build Coastguard Worker ? x->rd_threshes[THR_NEW2]
1985*fb1b10abSAndroid Build Coastguard Worker : this_rd_thresh;
1986*fb1b10abSAndroid Build Coastguard Worker
1987*fb1b10abSAndroid Build Coastguard Worker tmp_rd = vp8_rd_pick_best_mbsegmentation(
1988*fb1b10abSAndroid Build Coastguard Worker cpi, x, &best_ref_mv, best_mode.yrd, mdcounts, &rate, &rd.rate_y,
1989*fb1b10abSAndroid Build Coastguard Worker &distortion, this_rd_thresh);
1990*fb1b10abSAndroid Build Coastguard Worker
1991*fb1b10abSAndroid Build Coastguard Worker rd.rate2 += rate;
1992*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 += distortion;
1993*fb1b10abSAndroid Build Coastguard Worker
1994*fb1b10abSAndroid Build Coastguard Worker /* If even the 'Y' rd value of split is higher than best so far
1995*fb1b10abSAndroid Build Coastguard Worker * then don't bother looking at UV
1996*fb1b10abSAndroid Build Coastguard Worker */
1997*fb1b10abSAndroid Build Coastguard Worker if (tmp_rd < best_mode.yrd) {
1998*fb1b10abSAndroid Build Coastguard Worker /* Now work out UV cost and add it in */
1999*fb1b10abSAndroid Build Coastguard Worker rd_inter4x4_uv(cpi, x, &rd.rate_uv, &rd.distortion_uv,
2000*fb1b10abSAndroid Build Coastguard Worker cpi->common.full_pixel);
2001*fb1b10abSAndroid Build Coastguard Worker rd.rate2 += rd.rate_uv;
2002*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 += rd.distortion_uv;
2003*fb1b10abSAndroid Build Coastguard Worker } else {
2004*fb1b10abSAndroid Build Coastguard Worker this_rd = INT_MAX;
2005*fb1b10abSAndroid Build Coastguard Worker disable_skip = 1;
2006*fb1b10abSAndroid Build Coastguard Worker }
2007*fb1b10abSAndroid Build Coastguard Worker break;
2008*fb1b10abSAndroid Build Coastguard Worker }
2009*fb1b10abSAndroid Build Coastguard Worker case DC_PRED:
2010*fb1b10abSAndroid Build Coastguard Worker case V_PRED:
2011*fb1b10abSAndroid Build Coastguard Worker case H_PRED:
2012*fb1b10abSAndroid Build Coastguard Worker case TM_PRED: {
2013*fb1b10abSAndroid Build Coastguard Worker int distortion;
2014*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame = INTRA_FRAME;
2015*fb1b10abSAndroid Build Coastguard Worker
2016*fb1b10abSAndroid Build Coastguard Worker vp8_build_intra_predictors_mby_s(
2017*fb1b10abSAndroid Build Coastguard Worker xd, xd->dst.y_buffer - xd->dst.y_stride, xd->dst.y_buffer - 1,
2018*fb1b10abSAndroid Build Coastguard Worker xd->dst.y_stride, xd->predictor, 16);
2019*fb1b10abSAndroid Build Coastguard Worker macro_block_yrd(x, &rd.rate_y, &distortion);
2020*fb1b10abSAndroid Build Coastguard Worker rd.rate2 += rd.rate_y;
2021*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 += distortion;
2022*fb1b10abSAndroid Build Coastguard Worker rd.rate2 += x->mbmode_cost[x->e_mbd.frame_type]
2023*fb1b10abSAndroid Build Coastguard Worker [x->e_mbd.mode_info_context->mbmi.mode];
2024*fb1b10abSAndroid Build Coastguard Worker assert(uv_intra_done);
2025*fb1b10abSAndroid Build Coastguard Worker rd.rate2 += uv_intra_rate;
2026*fb1b10abSAndroid Build Coastguard Worker rd.rate_uv = uv_intra_rate_tokenonly;
2027*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 += uv_intra_distortion;
2028*fb1b10abSAndroid Build Coastguard Worker rd.distortion_uv = uv_intra_distortion;
2029*fb1b10abSAndroid Build Coastguard Worker break;
2030*fb1b10abSAndroid Build Coastguard Worker }
2031*fb1b10abSAndroid Build Coastguard Worker
2032*fb1b10abSAndroid Build Coastguard Worker case NEWMV: {
2033*fb1b10abSAndroid Build Coastguard Worker int thissme;
2034*fb1b10abSAndroid Build Coastguard Worker int bestsme = INT_MAX;
2035*fb1b10abSAndroid Build Coastguard Worker int step_param = cpi->sf.first_step;
2036*fb1b10abSAndroid Build Coastguard Worker int further_steps;
2037*fb1b10abSAndroid Build Coastguard Worker int n;
2038*fb1b10abSAndroid Build Coastguard Worker /* If last step (1-away) of n-step search doesn't pick the center point
2039*fb1b10abSAndroid Build Coastguard Worker as the best match, we will do a final 1-away diamond refining search
2040*fb1b10abSAndroid Build Coastguard Worker */
2041*fb1b10abSAndroid Build Coastguard Worker int do_refine = 1;
2042*fb1b10abSAndroid Build Coastguard Worker
2043*fb1b10abSAndroid Build Coastguard Worker int sadpb = x->sadperbit16;
2044*fb1b10abSAndroid Build Coastguard Worker int_mv mvp_full;
2045*fb1b10abSAndroid Build Coastguard Worker
2046*fb1b10abSAndroid Build Coastguard Worker int col_min = ((best_ref_mv.as_mv.col + 7) >> 3) - MAX_FULL_PEL_VAL;
2047*fb1b10abSAndroid Build Coastguard Worker int row_min = ((best_ref_mv.as_mv.row + 7) >> 3) - MAX_FULL_PEL_VAL;
2048*fb1b10abSAndroid Build Coastguard Worker int col_max = (best_ref_mv.as_mv.col >> 3) + MAX_FULL_PEL_VAL;
2049*fb1b10abSAndroid Build Coastguard Worker int row_max = (best_ref_mv.as_mv.row >> 3) + MAX_FULL_PEL_VAL;
2050*fb1b10abSAndroid Build Coastguard Worker
2051*fb1b10abSAndroid Build Coastguard Worker int tmp_col_min = x->mv_col_min;
2052*fb1b10abSAndroid Build Coastguard Worker int tmp_col_max = x->mv_col_max;
2053*fb1b10abSAndroid Build Coastguard Worker int tmp_row_min = x->mv_row_min;
2054*fb1b10abSAndroid Build Coastguard Worker int tmp_row_max = x->mv_row_max;
2055*fb1b10abSAndroid Build Coastguard Worker
2056*fb1b10abSAndroid Build Coastguard Worker if (!saddone) {
2057*fb1b10abSAndroid Build Coastguard Worker vp8_cal_sad(cpi, xd, x, recon_yoffset, &near_sadidx[0]);
2058*fb1b10abSAndroid Build Coastguard Worker saddone = 1;
2059*fb1b10abSAndroid Build Coastguard Worker }
2060*fb1b10abSAndroid Build Coastguard Worker
2061*fb1b10abSAndroid Build Coastguard Worker vp8_mv_pred(cpi, &x->e_mbd, x->e_mbd.mode_info_context, &mvp,
2062*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame,
2063*fb1b10abSAndroid Build Coastguard Worker cpi->common.ref_frame_sign_bias, &sr, &near_sadidx[0]);
2064*fb1b10abSAndroid Build Coastguard Worker
2065*fb1b10abSAndroid Build Coastguard Worker mvp_full.as_mv.col = mvp.as_mv.col >> 3;
2066*fb1b10abSAndroid Build Coastguard Worker mvp_full.as_mv.row = mvp.as_mv.row >> 3;
2067*fb1b10abSAndroid Build Coastguard Worker
2068*fb1b10abSAndroid Build Coastguard Worker /* Get intersection of UMV window and valid MV window to
2069*fb1b10abSAndroid Build Coastguard Worker * reduce # of checks in diamond search.
2070*fb1b10abSAndroid Build Coastguard Worker */
2071*fb1b10abSAndroid Build Coastguard Worker if (x->mv_col_min < col_min) x->mv_col_min = col_min;
2072*fb1b10abSAndroid Build Coastguard Worker if (x->mv_col_max > col_max) x->mv_col_max = col_max;
2073*fb1b10abSAndroid Build Coastguard Worker if (x->mv_row_min < row_min) x->mv_row_min = row_min;
2074*fb1b10abSAndroid Build Coastguard Worker if (x->mv_row_max > row_max) x->mv_row_max = row_max;
2075*fb1b10abSAndroid Build Coastguard Worker
2076*fb1b10abSAndroid Build Coastguard Worker /* adjust search range according to sr from mv prediction */
2077*fb1b10abSAndroid Build Coastguard Worker if (sr > step_param) step_param = sr;
2078*fb1b10abSAndroid Build Coastguard Worker
2079*fb1b10abSAndroid Build Coastguard Worker /* Initial step/diamond search */
2080*fb1b10abSAndroid Build Coastguard Worker {
2081*fb1b10abSAndroid Build Coastguard Worker bestsme = cpi->diamond_search_sad(
2082*fb1b10abSAndroid Build Coastguard Worker x, b, d, &mvp_full, &d->bmi.mv, step_param, sadpb, &num00,
2083*fb1b10abSAndroid Build Coastguard Worker &cpi->fn_ptr[BLOCK_16X16], x->mvcost, &best_ref_mv);
2084*fb1b10abSAndroid Build Coastguard Worker mode_mv[NEWMV].as_int = d->bmi.mv.as_int;
2085*fb1b10abSAndroid Build Coastguard Worker
2086*fb1b10abSAndroid Build Coastguard Worker /* Further step/diamond searches as necessary */
2087*fb1b10abSAndroid Build Coastguard Worker further_steps = (cpi->sf.max_step_search_steps - 1) - step_param;
2088*fb1b10abSAndroid Build Coastguard Worker
2089*fb1b10abSAndroid Build Coastguard Worker n = num00;
2090*fb1b10abSAndroid Build Coastguard Worker num00 = 0;
2091*fb1b10abSAndroid Build Coastguard Worker
2092*fb1b10abSAndroid Build Coastguard Worker /* If there won't be more n-step search, check to see if refining
2093*fb1b10abSAndroid Build Coastguard Worker * search is needed. */
2094*fb1b10abSAndroid Build Coastguard Worker if (n > further_steps) do_refine = 0;
2095*fb1b10abSAndroid Build Coastguard Worker
2096*fb1b10abSAndroid Build Coastguard Worker while (n < further_steps) {
2097*fb1b10abSAndroid Build Coastguard Worker n++;
2098*fb1b10abSAndroid Build Coastguard Worker
2099*fb1b10abSAndroid Build Coastguard Worker if (num00) {
2100*fb1b10abSAndroid Build Coastguard Worker num00--;
2101*fb1b10abSAndroid Build Coastguard Worker } else {
2102*fb1b10abSAndroid Build Coastguard Worker thissme = cpi->diamond_search_sad(
2103*fb1b10abSAndroid Build Coastguard Worker x, b, d, &mvp_full, &d->bmi.mv, step_param + n, sadpb, &num00,
2104*fb1b10abSAndroid Build Coastguard Worker &cpi->fn_ptr[BLOCK_16X16], x->mvcost, &best_ref_mv);
2105*fb1b10abSAndroid Build Coastguard Worker
2106*fb1b10abSAndroid Build Coastguard Worker /* check to see if refining search is needed. */
2107*fb1b10abSAndroid Build Coastguard Worker if (num00 > (further_steps - n)) do_refine = 0;
2108*fb1b10abSAndroid Build Coastguard Worker
2109*fb1b10abSAndroid Build Coastguard Worker if (thissme < bestsme) {
2110*fb1b10abSAndroid Build Coastguard Worker bestsme = thissme;
2111*fb1b10abSAndroid Build Coastguard Worker mode_mv[NEWMV].as_int = d->bmi.mv.as_int;
2112*fb1b10abSAndroid Build Coastguard Worker } else {
2113*fb1b10abSAndroid Build Coastguard Worker d->bmi.mv.as_int = mode_mv[NEWMV].as_int;
2114*fb1b10abSAndroid Build Coastguard Worker }
2115*fb1b10abSAndroid Build Coastguard Worker }
2116*fb1b10abSAndroid Build Coastguard Worker }
2117*fb1b10abSAndroid Build Coastguard Worker }
2118*fb1b10abSAndroid Build Coastguard Worker
2119*fb1b10abSAndroid Build Coastguard Worker /* final 1-away diamond refining search */
2120*fb1b10abSAndroid Build Coastguard Worker if (do_refine == 1) {
2121*fb1b10abSAndroid Build Coastguard Worker int search_range;
2122*fb1b10abSAndroid Build Coastguard Worker
2123*fb1b10abSAndroid Build Coastguard Worker search_range = 8;
2124*fb1b10abSAndroid Build Coastguard Worker
2125*fb1b10abSAndroid Build Coastguard Worker thissme = cpi->refining_search_sad(
2126*fb1b10abSAndroid Build Coastguard Worker x, b, d, &d->bmi.mv, sadpb, search_range,
2127*fb1b10abSAndroid Build Coastguard Worker &cpi->fn_ptr[BLOCK_16X16], x->mvcost, &best_ref_mv);
2128*fb1b10abSAndroid Build Coastguard Worker
2129*fb1b10abSAndroid Build Coastguard Worker if (thissme < bestsme) {
2130*fb1b10abSAndroid Build Coastguard Worker bestsme = thissme;
2131*fb1b10abSAndroid Build Coastguard Worker mode_mv[NEWMV].as_int = d->bmi.mv.as_int;
2132*fb1b10abSAndroid Build Coastguard Worker } else {
2133*fb1b10abSAndroid Build Coastguard Worker d->bmi.mv.as_int = mode_mv[NEWMV].as_int;
2134*fb1b10abSAndroid Build Coastguard Worker }
2135*fb1b10abSAndroid Build Coastguard Worker }
2136*fb1b10abSAndroid Build Coastguard Worker
2137*fb1b10abSAndroid Build Coastguard Worker x->mv_col_min = tmp_col_min;
2138*fb1b10abSAndroid Build Coastguard Worker x->mv_col_max = tmp_col_max;
2139*fb1b10abSAndroid Build Coastguard Worker x->mv_row_min = tmp_row_min;
2140*fb1b10abSAndroid Build Coastguard Worker x->mv_row_max = tmp_row_max;
2141*fb1b10abSAndroid Build Coastguard Worker
2142*fb1b10abSAndroid Build Coastguard Worker if (bestsme < INT_MAX) {
2143*fb1b10abSAndroid Build Coastguard Worker int dis; /* TODO: use dis in distortion calculation later. */
2144*fb1b10abSAndroid Build Coastguard Worker unsigned int sse;
2145*fb1b10abSAndroid Build Coastguard Worker cpi->find_fractional_mv_step(
2146*fb1b10abSAndroid Build Coastguard Worker x, b, d, &d->bmi.mv, &best_ref_mv, x->errorperbit,
2147*fb1b10abSAndroid Build Coastguard Worker &cpi->fn_ptr[BLOCK_16X16], x->mvcost, &dis, &sse);
2148*fb1b10abSAndroid Build Coastguard Worker }
2149*fb1b10abSAndroid Build Coastguard Worker
2150*fb1b10abSAndroid Build Coastguard Worker mode_mv[NEWMV].as_int = d->bmi.mv.as_int;
2151*fb1b10abSAndroid Build Coastguard Worker
2152*fb1b10abSAndroid Build Coastguard Worker /* Add the new motion vector cost to our rolling cost variable */
2153*fb1b10abSAndroid Build Coastguard Worker rd.rate2 +=
2154*fb1b10abSAndroid Build Coastguard Worker vp8_mv_bit_cost(&mode_mv[NEWMV], &best_ref_mv, x->mvcost, 96);
2155*fb1b10abSAndroid Build Coastguard Worker }
2156*fb1b10abSAndroid Build Coastguard Worker // fall through
2157*fb1b10abSAndroid Build Coastguard Worker
2158*fb1b10abSAndroid Build Coastguard Worker case NEARESTMV:
2159*fb1b10abSAndroid Build Coastguard Worker case NEARMV:
2160*fb1b10abSAndroid Build Coastguard Worker /* Clip "next_nearest" so that it does not extend to far out
2161*fb1b10abSAndroid Build Coastguard Worker * of image
2162*fb1b10abSAndroid Build Coastguard Worker */
2163*fb1b10abSAndroid Build Coastguard Worker vp8_clamp_mv2(&mode_mv[this_mode], xd);
2164*fb1b10abSAndroid Build Coastguard Worker
2165*fb1b10abSAndroid Build Coastguard Worker /* Do not bother proceeding if the vector (from newmv, nearest
2166*fb1b10abSAndroid Build Coastguard Worker * or near) is 0,0 as this should then be coded using the zeromv
2167*fb1b10abSAndroid Build Coastguard Worker * mode.
2168*fb1b10abSAndroid Build Coastguard Worker */
2169*fb1b10abSAndroid Build Coastguard Worker if (((this_mode == NEARMV) || (this_mode == NEARESTMV)) &&
2170*fb1b10abSAndroid Build Coastguard Worker (mode_mv[this_mode].as_int == 0)) {
2171*fb1b10abSAndroid Build Coastguard Worker continue;
2172*fb1b10abSAndroid Build Coastguard Worker }
2173*fb1b10abSAndroid Build Coastguard Worker // fall through
2174*fb1b10abSAndroid Build Coastguard Worker
2175*fb1b10abSAndroid Build Coastguard Worker case ZEROMV:
2176*fb1b10abSAndroid Build Coastguard Worker
2177*fb1b10abSAndroid Build Coastguard Worker /* Trap vectors that reach beyond the UMV borders
2178*fb1b10abSAndroid Build Coastguard Worker * Note that ALL New MV, Nearest MV Near MV and Zero MV code
2179*fb1b10abSAndroid Build Coastguard Worker * drops through to this point because of the lack of break
2180*fb1b10abSAndroid Build Coastguard Worker * statements in the previous two cases.
2181*fb1b10abSAndroid Build Coastguard Worker */
2182*fb1b10abSAndroid Build Coastguard Worker if (((mode_mv[this_mode].as_mv.row >> 3) < x->mv_row_min) ||
2183*fb1b10abSAndroid Build Coastguard Worker ((mode_mv[this_mode].as_mv.row >> 3) > x->mv_row_max) ||
2184*fb1b10abSAndroid Build Coastguard Worker ((mode_mv[this_mode].as_mv.col >> 3) < x->mv_col_min) ||
2185*fb1b10abSAndroid Build Coastguard Worker ((mode_mv[this_mode].as_mv.col >> 3) > x->mv_col_max)) {
2186*fb1b10abSAndroid Build Coastguard Worker continue;
2187*fb1b10abSAndroid Build Coastguard Worker }
2188*fb1b10abSAndroid Build Coastguard Worker
2189*fb1b10abSAndroid Build Coastguard Worker vp8_set_mbmode_and_mvs(x, this_mode, &mode_mv[this_mode]);
2190*fb1b10abSAndroid Build Coastguard Worker this_rd = evaluate_inter_mode_rd(mdcounts, &rd, &disable_skip, cpi, x);
2191*fb1b10abSAndroid Build Coastguard Worker break;
2192*fb1b10abSAndroid Build Coastguard Worker
2193*fb1b10abSAndroid Build Coastguard Worker default: break;
2194*fb1b10abSAndroid Build Coastguard Worker }
2195*fb1b10abSAndroid Build Coastguard Worker
2196*fb1b10abSAndroid Build Coastguard Worker this_rd =
2197*fb1b10abSAndroid Build Coastguard Worker calculate_final_rd_costs(this_rd, &rd, &other_cost, disable_skip,
2198*fb1b10abSAndroid Build Coastguard Worker uv_intra_tteob, intra_rd_penalty, cpi, x);
2199*fb1b10abSAndroid Build Coastguard Worker
2200*fb1b10abSAndroid Build Coastguard Worker /* Keep record of best intra distortion */
2201*fb1b10abSAndroid Build Coastguard Worker if ((x->e_mbd.mode_info_context->mbmi.ref_frame == INTRA_FRAME) &&
2202*fb1b10abSAndroid Build Coastguard Worker (this_rd < best_mode.intra_rd)) {
2203*fb1b10abSAndroid Build Coastguard Worker best_mode.intra_rd = this_rd;
2204*fb1b10abSAndroid Build Coastguard Worker *returnintra = rd.distortion2;
2205*fb1b10abSAndroid Build Coastguard Worker }
2206*fb1b10abSAndroid Build Coastguard Worker #if CONFIG_TEMPORAL_DENOISING
2207*fb1b10abSAndroid Build Coastguard Worker if (cpi->oxcf.noise_sensitivity) {
2208*fb1b10abSAndroid Build Coastguard Worker unsigned int sse;
2209*fb1b10abSAndroid Build Coastguard Worker vp8_get_inter_mbpred_error(x, &cpi->fn_ptr[BLOCK_16X16], &sse,
2210*fb1b10abSAndroid Build Coastguard Worker mode_mv[this_mode]);
2211*fb1b10abSAndroid Build Coastguard Worker
2212*fb1b10abSAndroid Build Coastguard Worker if (sse < best_rd_sse) best_rd_sse = sse;
2213*fb1b10abSAndroid Build Coastguard Worker
2214*fb1b10abSAndroid Build Coastguard Worker /* Store for later use by denoiser. */
2215*fb1b10abSAndroid Build Coastguard Worker if (this_mode == ZEROMV && sse < zero_mv_sse) {
2216*fb1b10abSAndroid Build Coastguard Worker zero_mv_sse = sse;
2217*fb1b10abSAndroid Build Coastguard Worker x->best_zeromv_reference_frame =
2218*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame;
2219*fb1b10abSAndroid Build Coastguard Worker }
2220*fb1b10abSAndroid Build Coastguard Worker
2221*fb1b10abSAndroid Build Coastguard Worker /* Store the best NEWMV in x for later use in the denoiser. */
2222*fb1b10abSAndroid Build Coastguard Worker if (x->e_mbd.mode_info_context->mbmi.mode == NEWMV && sse < best_sse) {
2223*fb1b10abSAndroid Build Coastguard Worker best_sse = sse;
2224*fb1b10abSAndroid Build Coastguard Worker vp8_get_inter_mbpred_error(x, &cpi->fn_ptr[BLOCK_16X16], &best_sse,
2225*fb1b10abSAndroid Build Coastguard Worker mode_mv[this_mode]);
2226*fb1b10abSAndroid Build Coastguard Worker x->best_sse_inter_mode = NEWMV;
2227*fb1b10abSAndroid Build Coastguard Worker x->best_sse_mv = x->e_mbd.mode_info_context->mbmi.mv;
2228*fb1b10abSAndroid Build Coastguard Worker x->need_to_clamp_best_mvs =
2229*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.need_to_clamp_mvs;
2230*fb1b10abSAndroid Build Coastguard Worker x->best_reference_frame = x->e_mbd.mode_info_context->mbmi.ref_frame;
2231*fb1b10abSAndroid Build Coastguard Worker }
2232*fb1b10abSAndroid Build Coastguard Worker }
2233*fb1b10abSAndroid Build Coastguard Worker #endif
2234*fb1b10abSAndroid Build Coastguard Worker
2235*fb1b10abSAndroid Build Coastguard Worker /* Did this mode help.. i.i is it the new best mode */
2236*fb1b10abSAndroid Build Coastguard Worker if (this_rd < best_mode.rd || x->skip) {
2237*fb1b10abSAndroid Build Coastguard Worker /* Note index of best mode so far */
2238*fb1b10abSAndroid Build Coastguard Worker best_mode_index = mode_index;
2239*fb1b10abSAndroid Build Coastguard Worker *returnrate = rd.rate2;
2240*fb1b10abSAndroid Build Coastguard Worker *returndistortion = rd.distortion2;
2241*fb1b10abSAndroid Build Coastguard Worker if (this_mode <= B_PRED) {
2242*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.uv_mode = uv_intra_mode;
2243*fb1b10abSAndroid Build Coastguard Worker /* required for left and above block mv */
2244*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mv.as_int = 0;
2245*fb1b10abSAndroid Build Coastguard Worker }
2246*fb1b10abSAndroid Build Coastguard Worker update_best_mode(&best_mode, this_rd, &rd, other_cost, x);
2247*fb1b10abSAndroid Build Coastguard Worker
2248*fb1b10abSAndroid Build Coastguard Worker /* Testing this mode gave rise to an improvement in best error
2249*fb1b10abSAndroid Build Coastguard Worker * score. Lower threshold a bit for next time
2250*fb1b10abSAndroid Build Coastguard Worker */
2251*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[mode_index] =
2252*fb1b10abSAndroid Build Coastguard Worker (x->rd_thresh_mult[mode_index] >= (MIN_THRESHMULT + 2))
2253*fb1b10abSAndroid Build Coastguard Worker ? x->rd_thresh_mult[mode_index] - 2
2254*fb1b10abSAndroid Build Coastguard Worker : MIN_THRESHMULT;
2255*fb1b10abSAndroid Build Coastguard Worker }
2256*fb1b10abSAndroid Build Coastguard Worker
2257*fb1b10abSAndroid Build Coastguard Worker /* If the mode did not help improve the best error case then raise
2258*fb1b10abSAndroid Build Coastguard Worker * the threshold for testing that mode next time around.
2259*fb1b10abSAndroid Build Coastguard Worker */
2260*fb1b10abSAndroid Build Coastguard Worker else {
2261*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[mode_index] += 4;
2262*fb1b10abSAndroid Build Coastguard Worker
2263*fb1b10abSAndroid Build Coastguard Worker if (x->rd_thresh_mult[mode_index] > MAX_THRESHMULT) {
2264*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[mode_index] = MAX_THRESHMULT;
2265*fb1b10abSAndroid Build Coastguard Worker }
2266*fb1b10abSAndroid Build Coastguard Worker }
2267*fb1b10abSAndroid Build Coastguard Worker x->rd_threshes[mode_index] = (cpi->rd_baseline_thresh[mode_index] >> 7) *
2268*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[mode_index];
2269*fb1b10abSAndroid Build Coastguard Worker
2270*fb1b10abSAndroid Build Coastguard Worker if (x->skip) break;
2271*fb1b10abSAndroid Build Coastguard Worker }
2272*fb1b10abSAndroid Build Coastguard Worker
2273*fb1b10abSAndroid Build Coastguard Worker /* Reduce the activation RD thresholds for the best choice mode */
2274*fb1b10abSAndroid Build Coastguard Worker if ((cpi->rd_baseline_thresh[best_mode_index] > 0) &&
2275*fb1b10abSAndroid Build Coastguard Worker (cpi->rd_baseline_thresh[best_mode_index] < (INT_MAX >> 2))) {
2276*fb1b10abSAndroid Build Coastguard Worker int best_adjustment = (x->rd_thresh_mult[best_mode_index] >> 2);
2277*fb1b10abSAndroid Build Coastguard Worker
2278*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[best_mode_index] =
2279*fb1b10abSAndroid Build Coastguard Worker (x->rd_thresh_mult[best_mode_index] >=
2280*fb1b10abSAndroid Build Coastguard Worker (MIN_THRESHMULT + best_adjustment))
2281*fb1b10abSAndroid Build Coastguard Worker ? x->rd_thresh_mult[best_mode_index] - best_adjustment
2282*fb1b10abSAndroid Build Coastguard Worker : MIN_THRESHMULT;
2283*fb1b10abSAndroid Build Coastguard Worker x->rd_threshes[best_mode_index] =
2284*fb1b10abSAndroid Build Coastguard Worker (cpi->rd_baseline_thresh[best_mode_index] >> 7) *
2285*fb1b10abSAndroid Build Coastguard Worker x->rd_thresh_mult[best_mode_index];
2286*fb1b10abSAndroid Build Coastguard Worker }
2287*fb1b10abSAndroid Build Coastguard Worker
2288*fb1b10abSAndroid Build Coastguard Worker #if CONFIG_TEMPORAL_DENOISING
2289*fb1b10abSAndroid Build Coastguard Worker if (cpi->oxcf.noise_sensitivity) {
2290*fb1b10abSAndroid Build Coastguard Worker int block_index = mb_row * cpi->common.mb_cols + mb_col;
2291*fb1b10abSAndroid Build Coastguard Worker if (x->best_sse_inter_mode == DC_PRED) {
2292*fb1b10abSAndroid Build Coastguard Worker /* No best MV found. */
2293*fb1b10abSAndroid Build Coastguard Worker x->best_sse_inter_mode = best_mode.mbmode.mode;
2294*fb1b10abSAndroid Build Coastguard Worker x->best_sse_mv = best_mode.mbmode.mv;
2295*fb1b10abSAndroid Build Coastguard Worker x->need_to_clamp_best_mvs = best_mode.mbmode.need_to_clamp_mvs;
2296*fb1b10abSAndroid Build Coastguard Worker x->best_reference_frame = best_mode.mbmode.ref_frame;
2297*fb1b10abSAndroid Build Coastguard Worker best_sse = best_rd_sse;
2298*fb1b10abSAndroid Build Coastguard Worker }
2299*fb1b10abSAndroid Build Coastguard Worker vp8_denoiser_denoise_mb(&cpi->denoiser, x, best_sse, zero_mv_sse,
2300*fb1b10abSAndroid Build Coastguard Worker recon_yoffset, recon_uvoffset, &cpi->common.lf_info,
2301*fb1b10abSAndroid Build Coastguard Worker mb_row, mb_col, block_index, 0);
2302*fb1b10abSAndroid Build Coastguard Worker
2303*fb1b10abSAndroid Build Coastguard Worker /* Reevaluate ZEROMV after denoising. */
2304*fb1b10abSAndroid Build Coastguard Worker if (best_mode.mbmode.ref_frame == INTRA_FRAME &&
2305*fb1b10abSAndroid Build Coastguard Worker x->best_zeromv_reference_frame != INTRA_FRAME) {
2306*fb1b10abSAndroid Build Coastguard Worker int this_rd = INT_MAX;
2307*fb1b10abSAndroid Build Coastguard Worker int disable_skip = 0;
2308*fb1b10abSAndroid Build Coastguard Worker int other_cost = 0;
2309*fb1b10abSAndroid Build Coastguard Worker int this_ref_frame = x->best_zeromv_reference_frame;
2310*fb1b10abSAndroid Build Coastguard Worker rd.rate2 =
2311*fb1b10abSAndroid Build Coastguard Worker x->ref_frame_cost[this_ref_frame] + vp8_cost_mv_ref(ZEROMV, mdcounts);
2312*fb1b10abSAndroid Build Coastguard Worker rd.distortion2 = 0;
2313*fb1b10abSAndroid Build Coastguard Worker
2314*fb1b10abSAndroid Build Coastguard Worker /* set up the proper prediction buffers for the frame */
2315*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame = this_ref_frame;
2316*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.pre.y_buffer = plane[this_ref_frame][0];
2317*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.pre.u_buffer = plane[this_ref_frame][1];
2318*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.pre.v_buffer = plane[this_ref_frame][2];
2319*fb1b10abSAndroid Build Coastguard Worker
2320*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mode = ZEROMV;
2321*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.uv_mode = DC_PRED;
2322*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mv.as_int = 0;
2323*fb1b10abSAndroid Build Coastguard Worker
2324*fb1b10abSAndroid Build Coastguard Worker this_rd = evaluate_inter_mode_rd(mdcounts, &rd, &disable_skip, cpi, x);
2325*fb1b10abSAndroid Build Coastguard Worker this_rd =
2326*fb1b10abSAndroid Build Coastguard Worker calculate_final_rd_costs(this_rd, &rd, &other_cost, disable_skip,
2327*fb1b10abSAndroid Build Coastguard Worker uv_intra_tteob, intra_rd_penalty, cpi, x);
2328*fb1b10abSAndroid Build Coastguard Worker if (this_rd < best_mode.rd || x->skip) {
2329*fb1b10abSAndroid Build Coastguard Worker *returnrate = rd.rate2;
2330*fb1b10abSAndroid Build Coastguard Worker *returndistortion = rd.distortion2;
2331*fb1b10abSAndroid Build Coastguard Worker update_best_mode(&best_mode, this_rd, &rd, other_cost, x);
2332*fb1b10abSAndroid Build Coastguard Worker }
2333*fb1b10abSAndroid Build Coastguard Worker }
2334*fb1b10abSAndroid Build Coastguard Worker }
2335*fb1b10abSAndroid Build Coastguard Worker #endif
2336*fb1b10abSAndroid Build Coastguard Worker
2337*fb1b10abSAndroid Build Coastguard Worker if (cpi->is_src_frame_alt_ref &&
2338*fb1b10abSAndroid Build Coastguard Worker (best_mode.mbmode.mode != ZEROMV ||
2339*fb1b10abSAndroid Build Coastguard Worker best_mode.mbmode.ref_frame != ALTREF_FRAME)) {
2340*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mode = ZEROMV;
2341*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame = ALTREF_FRAME;
2342*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mv.as_int = 0;
2343*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.uv_mode = DC_PRED;
2344*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mb_skip_coeff =
2345*fb1b10abSAndroid Build Coastguard Worker (cpi->common.mb_no_coeff_skip);
2346*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.partitioning = 0;
2347*fb1b10abSAndroid Build Coastguard Worker return;
2348*fb1b10abSAndroid Build Coastguard Worker }
2349*fb1b10abSAndroid Build Coastguard Worker
2350*fb1b10abSAndroid Build Coastguard Worker /* macroblock modes */
2351*fb1b10abSAndroid Build Coastguard Worker memcpy(&x->e_mbd.mode_info_context->mbmi, &best_mode.mbmode,
2352*fb1b10abSAndroid Build Coastguard Worker sizeof(MB_MODE_INFO));
2353*fb1b10abSAndroid Build Coastguard Worker
2354*fb1b10abSAndroid Build Coastguard Worker if (best_mode.mbmode.mode == B_PRED) {
2355*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
2356*fb1b10abSAndroid Build Coastguard Worker xd->mode_info_context->bmi[i].as_mode = best_mode.bmodes[i].as_mode;
2357*fb1b10abSAndroid Build Coastguard Worker }
2358*fb1b10abSAndroid Build Coastguard Worker }
2359*fb1b10abSAndroid Build Coastguard Worker
2360*fb1b10abSAndroid Build Coastguard Worker if (best_mode.mbmode.mode == SPLITMV) {
2361*fb1b10abSAndroid Build Coastguard Worker for (i = 0; i < 16; ++i) {
2362*fb1b10abSAndroid Build Coastguard Worker xd->mode_info_context->bmi[i].mv.as_int = best_mode.bmodes[i].mv.as_int;
2363*fb1b10abSAndroid Build Coastguard Worker }
2364*fb1b10abSAndroid Build Coastguard Worker
2365*fb1b10abSAndroid Build Coastguard Worker memcpy(x->partition_info, &best_mode.partition, sizeof(PARTITION_INFO));
2366*fb1b10abSAndroid Build Coastguard Worker
2367*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mv.as_int =
2368*fb1b10abSAndroid Build Coastguard Worker x->partition_info->bmi[15].mv.as_int;
2369*fb1b10abSAndroid Build Coastguard Worker }
2370*fb1b10abSAndroid Build Coastguard Worker
2371*fb1b10abSAndroid Build Coastguard Worker if (sign_bias !=
2372*fb1b10abSAndroid Build Coastguard Worker cpi->common.ref_frame_sign_bias[xd->mode_info_context->mbmi.ref_frame]) {
2373*fb1b10abSAndroid Build Coastguard Worker best_ref_mv.as_int = best_ref_mv_sb[!sign_bias].as_int;
2374*fb1b10abSAndroid Build Coastguard Worker }
2375*fb1b10abSAndroid Build Coastguard Worker
2376*fb1b10abSAndroid Build Coastguard Worker rd_update_mvcount(x, &best_ref_mv);
2377*fb1b10abSAndroid Build Coastguard Worker }
2378*fb1b10abSAndroid Build Coastguard Worker
vp8_rd_pick_intra_mode(MACROBLOCK * x,int * rate)2379*fb1b10abSAndroid Build Coastguard Worker void vp8_rd_pick_intra_mode(MACROBLOCK *x, int *rate) {
2380*fb1b10abSAndroid Build Coastguard Worker int error4x4, error16x16;
2381*fb1b10abSAndroid Build Coastguard Worker int rate4x4, rate16x16 = 0, rateuv;
2382*fb1b10abSAndroid Build Coastguard Worker int dist4x4, dist16x16, distuv;
2383*fb1b10abSAndroid Build Coastguard Worker int rate_;
2384*fb1b10abSAndroid Build Coastguard Worker int rate4x4_tokenonly = 0;
2385*fb1b10abSAndroid Build Coastguard Worker int rate16x16_tokenonly = 0;
2386*fb1b10abSAndroid Build Coastguard Worker int rateuv_tokenonly = 0;
2387*fb1b10abSAndroid Build Coastguard Worker
2388*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.ref_frame = INTRA_FRAME;
2389*fb1b10abSAndroid Build Coastguard Worker
2390*fb1b10abSAndroid Build Coastguard Worker rd_pick_intra_mbuv_mode(x, &rateuv, &rateuv_tokenonly, &distuv);
2391*fb1b10abSAndroid Build Coastguard Worker rate_ = rateuv;
2392*fb1b10abSAndroid Build Coastguard Worker
2393*fb1b10abSAndroid Build Coastguard Worker error16x16 = rd_pick_intra16x16mby_mode(x, &rate16x16, &rate16x16_tokenonly,
2394*fb1b10abSAndroid Build Coastguard Worker &dist16x16);
2395*fb1b10abSAndroid Build Coastguard Worker
2396*fb1b10abSAndroid Build Coastguard Worker error4x4 = rd_pick_intra4x4mby_modes(x, &rate4x4, &rate4x4_tokenonly,
2397*fb1b10abSAndroid Build Coastguard Worker &dist4x4, error16x16);
2398*fb1b10abSAndroid Build Coastguard Worker
2399*fb1b10abSAndroid Build Coastguard Worker if (error4x4 < error16x16) {
2400*fb1b10abSAndroid Build Coastguard Worker x->e_mbd.mode_info_context->mbmi.mode = B_PRED;
2401*fb1b10abSAndroid Build Coastguard Worker rate_ += rate4x4;
2402*fb1b10abSAndroid Build Coastguard Worker } else {
2403*fb1b10abSAndroid Build Coastguard Worker rate_ += rate16x16;
2404*fb1b10abSAndroid Build Coastguard Worker }
2405*fb1b10abSAndroid Build Coastguard Worker
2406*fb1b10abSAndroid Build Coastguard Worker *rate = rate_;
2407*fb1b10abSAndroid Build Coastguard Worker }
2408