xref: /aosp_15_r20/external/libvpx/vp8/encoder/rdopt.c (revision fb1b10ab9aebc7c7068eedab379b749d7e3900be)
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