Index: source/libvpx/vp9/encoder/vp9_pickmode.c |
=================================================================== |
--- source/libvpx/vp9/encoder/vp9_pickmode.c (revision 281795) |
+++ source/libvpx/vp9/encoder/vp9_pickmode.c (working copy) |
@@ -23,9 +23,89 @@ |
#include "vp9/common/vp9_reconintra.h" |
#include "vp9/encoder/vp9_encoder.h" |
+#include "vp9/encoder/vp9_pickmode.h" |
#include "vp9/encoder/vp9_ratectrl.h" |
-#include "vp9/encoder/vp9_rdopt.h" |
+#include "vp9/encoder/vp9_rd.h" |
+static int mv_refs_rt(const VP9_COMMON *cm, const MACROBLOCKD *xd, |
+ const TileInfo *const tile, |
+ MODE_INFO *mi, MV_REFERENCE_FRAME ref_frame, |
+ int_mv *mv_ref_list, |
+ int mi_row, int mi_col) { |
+ const int *ref_sign_bias = cm->ref_frame_sign_bias; |
+ int i, refmv_count = 0; |
+ |
+ const POSITION *const mv_ref_search = mv_ref_blocks[mi->mbmi.sb_type]; |
+ |
+ int different_ref_found = 0; |
+ int context_counter = 0; |
+ int const_motion = 0; |
+ |
+ // Blank the reference vector list |
+ vpx_memset(mv_ref_list, 0, sizeof(*mv_ref_list) * MAX_MV_REF_CANDIDATES); |
+ |
+ // The nearest 2 blocks are treated differently |
+ // if the size < 8x8 we get the mv from the bmi substructure, |
+ // and we also need to keep a mode count. |
+ for (i = 0; i < 2; ++i) { |
+ const POSITION *const mv_ref = &mv_ref_search[i]; |
+ if (is_inside(tile, mi_col, mi_row, cm->mi_rows, mv_ref)) { |
+ const MODE_INFO *const candidate_mi = xd->mi[mv_ref->col + mv_ref->row * |
+ xd->mi_stride]; |
+ const MB_MODE_INFO *const candidate = &candidate_mi->mbmi; |
+ // Keep counts for entropy encoding. |
+ context_counter += mode_2_counter[candidate->mode]; |
+ different_ref_found = 1; |
+ |
+ if (candidate->ref_frame[0] == ref_frame) |
+ ADD_MV_REF_LIST(get_sub_block_mv(candidate_mi, 0, mv_ref->col, -1)); |
+ } |
+ } |
+ |
+ const_motion = 1; |
+ |
+ // Check the rest of the neighbors in much the same way |
+ // as before except we don't need to keep track of sub blocks or |
+ // mode counts. |
+ for (; i < MVREF_NEIGHBOURS && !refmv_count; ++i) { |
+ const POSITION *const mv_ref = &mv_ref_search[i]; |
+ if (is_inside(tile, mi_col, mi_row, cm->mi_rows, mv_ref)) { |
+ const MB_MODE_INFO *const candidate = &xd->mi[mv_ref->col + mv_ref->row * |
+ xd->mi_stride]->mbmi; |
+ different_ref_found = 1; |
+ |
+ if (candidate->ref_frame[0] == ref_frame) |
+ ADD_MV_REF_LIST(candidate->mv[0]); |
+ } |
+ } |
+ |
+ // Since we couldn't find 2 mvs from the same reference frame |
+ // go back through the neighbors and find motion vectors from |
+ // different reference frames. |
+ if (different_ref_found && !refmv_count) { |
+ for (i = 0; i < MVREF_NEIGHBOURS; ++i) { |
+ const POSITION *mv_ref = &mv_ref_search[i]; |
+ if (is_inside(tile, mi_col, mi_row, cm->mi_rows, mv_ref)) { |
+ const MB_MODE_INFO *const candidate = &xd->mi[mv_ref->col + mv_ref->row |
+ * xd->mi_stride]->mbmi; |
+ |
+ // If the candidate is INTRA we don't want to consider its mv. |
+ IF_DIFF_REF_FRAME_ADD_MV(candidate); |
+ } |
+ } |
+ } |
+ |
+ Done: |
+ |
+ mi->mbmi.mode_context[ref_frame] = counter_to_context[context_counter]; |
+ |
+ // Clamp vectors |
+ for (i = 0; i < MAX_MV_REF_CANDIDATES; ++i) |
+ clamp_mv_ref(&mv_ref_list[i].as_mv, xd); |
+ |
+ return const_motion; |
+} |
+ |
static void full_pixel_motion_search(VP9_COMP *cpi, MACROBLOCK *x, |
BLOCK_SIZE bsize, int mi_row, int mi_col, |
int_mv *tmp_mv, int *rate_mv) { |
@@ -61,7 +141,7 @@ |
// TODO(jingning) exploiting adaptive motion search control in non-RD |
// mode decision too. |
- step_param = 6; |
+ step_param = cpi->sf.mv.fullpel_search_step_param; |
for (i = LAST_FRAME; i <= LAST_FRAME && cpi->common.show_frame; ++i) { |
if ((x->pred_mv_sad[ref] >> 3) > x->pred_mv_sad[i]) { |
@@ -172,6 +252,17 @@ |
else |
x->skip_txfm = 0; |
+ if (cpi->common.tx_mode == TX_MODE_SELECT) { |
+ if (sse > (var << 2)) |
+ xd->mi[0]->mbmi.tx_size = MIN(max_txsize_lookup[bsize], |
+ tx_mode_to_biggest_tx_size[cpi->common.tx_mode]); |
+ else |
+ xd->mi[0]->mbmi.tx_size = TX_8X8; |
+ } else { |
+ xd->mi[0]->mbmi.tx_size = MIN(max_txsize_lookup[bsize], |
+ tx_mode_to_biggest_tx_size[cpi->common.tx_mode]); |
+ } |
+ |
vp9_model_rd_from_var_lapndz(sse - var, 1 << num_pels_log2_lookup[bsize], |
dc_quant >> 3, &rate, &dist); |
*out_rate_sum = rate >> 1; |
@@ -183,6 +274,109 @@ |
*out_dist_sum += dist << 4; |
} |
+static int get_pred_buffer(PRED_BUFFER *p, int len) { |
+ int i; |
+ |
+ for (i = 0; i < len; i++) { |
+ if (!p[i].in_use) { |
+ p[i].in_use = 1; |
+ return i; |
+ } |
+ } |
+ return -1; |
+} |
+ |
+static void free_pred_buffer(PRED_BUFFER *p) { |
+ p->in_use = 0; |
+} |
+ |
+static void encode_breakout_test(VP9_COMP *cpi, MACROBLOCK *x, |
+ BLOCK_SIZE bsize, int mi_row, int mi_col, |
+ MV_REFERENCE_FRAME ref_frame, |
+ PREDICTION_MODE this_mode, |
+ unsigned int var_y, unsigned int sse_y, |
+ struct buf_2d yv12_mb[][MAX_MB_PLANE], |
+ int *rate, int64_t *dist) { |
+ MACROBLOCKD *xd = &x->e_mbd; |
+ MB_MODE_INFO *mbmi = &xd->mi[0]->mbmi; |
+ |
+ const BLOCK_SIZE uv_size = get_plane_block_size(bsize, &xd->plane[1]); |
+ unsigned int var = var_y, sse = sse_y; |
+ // Skipping threshold for ac. |
+ unsigned int thresh_ac; |
+ // Skipping threshold for dc. |
+ unsigned int thresh_dc; |
+ if (x->encode_breakout > 0) { |
+ // Set a maximum for threshold to avoid big PSNR loss in low bit rate |
+ // case. Use extreme low threshold for static frames to limit |
+ // skipping. |
+ const unsigned int max_thresh = 36000; |
+ // The encode_breakout input |
+ const unsigned int min_thresh = |
+ MIN(((unsigned int)x->encode_breakout << 4), max_thresh); |
+ |
+ // Calculate threshold according to dequant value. |
+ thresh_ac = (xd->plane[0].dequant[1] * xd->plane[0].dequant[1]) / 9; |
+ thresh_ac = clamp(thresh_ac, min_thresh, max_thresh); |
+ |
+ // Adjust ac threshold according to partition size. |
+ thresh_ac >>= |
+ 8 - (b_width_log2_lookup[bsize] + b_height_log2_lookup[bsize]); |
+ |
+ thresh_dc = (xd->plane[0].dequant[0] * xd->plane[0].dequant[0] >> 6); |
+ } else { |
+ thresh_ac = 0; |
+ thresh_dc = 0; |
+ } |
+ |
+ // Y skipping condition checking for ac and dc. |
+ if (var <= thresh_ac && (sse - var) <= thresh_dc) { |
+ unsigned int sse_u, sse_v; |
+ unsigned int var_u, var_v; |
+ |
+ // Skip UV prediction unless breakout is zero (lossless) to save |
+ // computation with low impact on the result |
+ if (x->encode_breakout == 0) { |
+ xd->plane[1].pre[0] = yv12_mb[ref_frame][1]; |
+ xd->plane[2].pre[0] = yv12_mb[ref_frame][2]; |
+ vp9_build_inter_predictors_sbuv(xd, mi_row, mi_col, bsize); |
+ } |
+ |
+ var_u = cpi->fn_ptr[uv_size].vf(x->plane[1].src.buf, |
+ x->plane[1].src.stride, |
+ xd->plane[1].dst.buf, |
+ xd->plane[1].dst.stride, &sse_u); |
+ |
+ // U skipping condition checking |
+ if ((var_u * 4 <= thresh_ac) && (sse_u - var_u <= thresh_dc)) { |
+ var_v = cpi->fn_ptr[uv_size].vf(x->plane[2].src.buf, |
+ x->plane[2].src.stride, |
+ xd->plane[2].dst.buf, |
+ xd->plane[2].dst.stride, &sse_v); |
+ |
+ // V skipping condition checking |
+ if ((var_v * 4 <= thresh_ac) && (sse_v - var_v <= thresh_dc)) { |
+ x->skip = 1; |
+ |
+ // The cost of skip bit needs to be added. |
+ *rate = cpi->inter_mode_cost[mbmi->mode_context[ref_frame]] |
+ [INTER_OFFSET(this_mode)]; |
+ |
+ // More on this part of rate |
+ // rate += vp9_cost_bit(vp9_get_skip_prob(cm, xd), 1); |
+ |
+ // Scaling factor for SSE from spatial domain to frequency |
+ // domain is 16. Adjust distortion accordingly. |
+ // TODO(yunqingwang): In this function, only y-plane dist is |
+ // calculated. |
+ *dist = (sse << 4); // + ((sse_u + sse_v) << 4); |
+ |
+ // *disable_skip = 1; |
+ } |
+ } |
+ } |
+} |
+ |
// TODO(jingning) placeholder for inter-frame non-RD mode decision. |
// this needs various further optimizations. to be continued.. |
int64_t vp9_pick_inter_mode(VP9_COMP *cpi, MACROBLOCK *x, |
@@ -197,6 +391,8 @@ |
struct macroblockd_plane *const pd = &xd->plane[0]; |
PREDICTION_MODE this_mode, best_mode = ZEROMV; |
MV_REFERENCE_FRAME ref_frame, best_ref_frame = LAST_FRAME; |
+ TX_SIZE best_tx_size = MIN(max_txsize_lookup[bsize], |
+ tx_mode_to_biggest_tx_size[cpi->common.tx_mode]); |
INTERP_FILTER best_pred_filter = EIGHTTAP; |
int_mv frame_mv[MB_MODE_COUNT][MAX_REF_FRAMES]; |
struct buf_2d yv12_mb[4][MAX_MB_PLANE]; |
@@ -224,11 +420,43 @@ |
const int *const rd_thresh_freq_fact = cpi->rd.thresh_freq_fact[bsize]; |
// Mode index conversion form THR_MODES to PREDICTION_MODE for a ref frame. |
int mode_idx[MB_MODE_COUNT] = {0}; |
- INTERP_FILTER filter_ref = SWITCHABLE; |
+ INTERP_FILTER filter_ref = cm->interp_filter; |
int bsl = mi_width_log2_lookup[bsize]; |
- const int pred_filter_search = (((mi_row + mi_col) >> bsl) + |
- get_chessboard_index(cm)) % 2; |
+ const int pred_filter_search = cm->interp_filter == SWITCHABLE ? |
+ (((mi_row + mi_col) >> bsl) + get_chessboard_index(cm)) % 2 : 0; |
+ int const_motion[MAX_REF_FRAMES] = { 0 }; |
+ // For speed 6, the result of interp filter is reused later in actual encoding |
+ // process. |
+ int bh = num_4x4_blocks_high_lookup[bsize] << 2; |
+ int bw = num_4x4_blocks_wide_lookup[bsize] << 2; |
+ int pixels_in_block = bh * bw; |
+ // tmp[3] points to dst buffer, and the other 3 point to allocated buffers. |
+ PRED_BUFFER tmp[4]; |
+ DECLARE_ALIGNED_ARRAY(16, uint8_t, pred_buf, 3 * 64 * 64); |
+ struct buf_2d orig_dst = pd->dst; |
+ PRED_BUFFER *best_pred = NULL; |
+ PRED_BUFFER *this_mode_pred = NULL; |
+ int i; |
+ |
+#if CONFIG_DENOISING |
+ if (cpi->oxcf.noise_sensitivity > 0) { |
+ vp9_denoiser_reset_frame_stats(&cpi->denoiser); |
+ } |
+#endif |
+ |
+ if (cpi->sf.reuse_inter_pred_sby) { |
+ for (i = 0; i < 3; i++) { |
+ tmp[i].data = &pred_buf[pixels_in_block * i]; |
+ tmp[i].stride = bw; |
+ tmp[i].in_use = 0; |
+ } |
+ |
+ tmp[3].data = pd->dst.buf; |
+ tmp[3].stride = pd->dst.stride; |
+ tmp[3].in_use = 0; |
+ } |
+ |
x->skip_encode = cpi->sf.skip_encode_frame && x->q_index < QIDX_SKIP_THRESH; |
x->skip = 0; |
@@ -241,18 +469,36 @@ |
mbmi->ref_frame[0] = NONE; |
mbmi->ref_frame[1] = NONE; |
mbmi->tx_size = MIN(max_txsize_lookup[bsize], |
- tx_mode_to_biggest_tx_size[cpi->common.tx_mode]); |
- mbmi->interp_filter = cpi->common.interp_filter == SWITCHABLE ? |
- EIGHTTAP : cpi->common.interp_filter; |
+ tx_mode_to_biggest_tx_size[cm->tx_mode]); |
+ mbmi->interp_filter = cm->interp_filter == SWITCHABLE ? |
+ EIGHTTAP : cm->interp_filter; |
mbmi->skip = 0; |
mbmi->segment_id = segment_id; |
for (ref_frame = LAST_FRAME; ref_frame <= LAST_FRAME ; ++ref_frame) { |
x->pred_mv_sad[ref_frame] = INT_MAX; |
if (cpi->ref_frame_flags & flag_list[ref_frame]) { |
- vp9_setup_buffer_inter(cpi, x, tile, |
- ref_frame, bsize, mi_row, mi_col, |
- frame_mv[NEARESTMV], frame_mv[NEARMV], yv12_mb); |
+ const YV12_BUFFER_CONFIG *yv12 = get_ref_frame_buffer(cpi, ref_frame); |
+ int_mv *const candidates = mbmi->ref_mvs[ref_frame]; |
+ const struct scale_factors *const sf = &cm->frame_refs[ref_frame - 1].sf; |
+ vp9_setup_pred_block(xd, yv12_mb[ref_frame], yv12, mi_row, mi_col, |
+ sf, sf); |
+ |
+ if (cm->coding_use_prev_mi) |
+ vp9_find_mv_refs(cm, xd, tile, xd->mi[0], ref_frame, |
+ candidates, mi_row, mi_col); |
+ else |
+ const_motion[ref_frame] = mv_refs_rt(cm, xd, tile, xd->mi[0], |
+ ref_frame, candidates, |
+ mi_row, mi_col); |
+ |
+ vp9_find_best_ref_mvs(xd, cm->allow_high_precision_mv, candidates, |
+ &frame_mv[NEARESTMV][ref_frame], |
+ &frame_mv[NEARMV][ref_frame]); |
+ |
+ if (!vp9_is_scaled(sf) && bsize >= BLOCK_8X8) |
+ vp9_mv_pred(cpi, x, yv12_mb[ref_frame][0].buf, yv12->y_stride, |
+ ref_frame, bsize); |
} |
frame_mv[NEWMV][ref_frame].as_int = INVALID_MV; |
frame_mv[ZEROMV][ref_frame].as_int = 0; |
@@ -286,6 +532,10 @@ |
for (this_mode = NEARESTMV; this_mode <= NEWMV; ++this_mode) { |
int rate_mv = 0; |
+ if (const_motion[ref_frame] && |
+ (this_mode == NEARMV || this_mode == ZEROMV)) |
+ continue; |
+ |
if (!(cpi->sf.inter_mode_mask[bsize] & (1 << this_mode))) |
continue; |
@@ -324,6 +574,16 @@ |
// Search for the best prediction filter type, when the resulting |
// motion vector is at sub-pixel accuracy level for luma component, i.e., |
// the last three bits are all zeros. |
+ if (cpi->sf.reuse_inter_pred_sby) { |
+ if (this_mode == NEARESTMV) { |
+ this_mode_pred = &tmp[3]; |
+ } else { |
+ this_mode_pred = &tmp[get_pred_buffer(tmp, 3)]; |
+ pd->dst.buf = this_mode_pred->data; |
+ pd->dst.stride = bw; |
+ } |
+ } |
+ |
if ((this_mode == NEWMV || filter_ref == SWITCHABLE) && |
pred_filter_search && |
((mbmi->mv[0].as_mv.row & 0x07) != 0 || |
@@ -332,8 +592,10 @@ |
int64_t pf_dist[3]; |
unsigned int pf_var[3]; |
unsigned int pf_sse[3]; |
+ TX_SIZE pf_tx_size[3]; |
int64_t best_cost = INT64_MAX; |
INTERP_FILTER best_filter = SWITCHABLE, filter; |
+ PRED_BUFFER *current_pred = this_mode_pred; |
for (filter = EIGHTTAP; filter <= EIGHTTAP_SHARP; ++filter) { |
int64_t cost; |
@@ -344,14 +606,32 @@ |
cost = RDCOST(x->rdmult, x->rddiv, |
vp9_get_switchable_rate(cpi) + pf_rate[filter], |
pf_dist[filter]); |
+ pf_tx_size[filter] = mbmi->tx_size; |
if (cost < best_cost) { |
- best_filter = filter; |
- best_cost = cost; |
- skip_txfm = x->skip_txfm; |
+ best_filter = filter; |
+ best_cost = cost; |
+ skip_txfm = x->skip_txfm; |
+ |
+ if (cpi->sf.reuse_inter_pred_sby) { |
+ if (this_mode_pred != current_pred) { |
+ free_pred_buffer(this_mode_pred); |
+ this_mode_pred = current_pred; |
+ } |
+ |
+ if (filter < EIGHTTAP_SHARP) { |
+ current_pred = &tmp[get_pred_buffer(tmp, 3)]; |
+ pd->dst.buf = current_pred->data; |
+ pd->dst.stride = bw; |
+ } |
+ } |
} |
} |
+ if (cpi->sf.reuse_inter_pred_sby && this_mode_pred != current_pred) |
+ free_pred_buffer(current_pred); |
+ |
mbmi->interp_filter = best_filter; |
+ mbmi->tx_size = pf_tx_size[mbmi->interp_filter]; |
rate = pf_rate[mbmi->interp_filter]; |
dist = pf_dist[mbmi->interp_filter]; |
var_y = pf_var[mbmi->interp_filter]; |
@@ -370,77 +650,19 @@ |
// Skipping checking: test to see if this block can be reconstructed by |
// prediction only. |
- if (!x->in_active_map) { |
- x->skip = 1; |
- } else if (cpi->allow_encode_breakout && x->encode_breakout) { |
- const BLOCK_SIZE uv_size = get_plane_block_size(bsize, &xd->plane[1]); |
- unsigned int var = var_y, sse = sse_y; |
- // Skipping threshold for ac. |
- unsigned int thresh_ac; |
- // Skipping threshold for dc. |
- unsigned int thresh_dc; |
- // Set a maximum for threshold to avoid big PSNR loss in low bit rate |
- // case. Use extreme low threshold for static frames to limit skipping. |
- const unsigned int max_thresh = 36000; |
- // The encode_breakout input |
- const unsigned int min_thresh = |
- MIN(((unsigned int)x->encode_breakout << 4), max_thresh); |
- |
- // Calculate threshold according to dequant value. |
- thresh_ac = (xd->plane[0].dequant[1] * xd->plane[0].dequant[1]) / 9; |
- thresh_ac = clamp(thresh_ac, min_thresh, max_thresh); |
- |
- // Adjust ac threshold according to partition size. |
- thresh_ac >>= 8 - (b_width_log2_lookup[bsize] + |
- b_height_log2_lookup[bsize]); |
- |
- thresh_dc = (xd->plane[0].dequant[0] * xd->plane[0].dequant[0] >> 6); |
- |
- // Y skipping condition checking for ac and dc. |
- if (var <= thresh_ac && (sse - var) <= thresh_dc) { |
- unsigned int sse_u, sse_v; |
- unsigned int var_u, var_v; |
- |
- // Skip u v prediction for less calculation, that won't affect |
- // result much. |
- var_u = cpi->fn_ptr[uv_size].vf(x->plane[1].src.buf, |
- x->plane[1].src.stride, |
- xd->plane[1].dst.buf, |
- xd->plane[1].dst.stride, &sse_u); |
- |
- // U skipping condition checking |
- if ((var_u * 4 <= thresh_ac) && (sse_u - var_u <= thresh_dc)) { |
- var_v = cpi->fn_ptr[uv_size].vf(x->plane[2].src.buf, |
- x->plane[2].src.stride, |
- xd->plane[2].dst.buf, |
- xd->plane[2].dst.stride, &sse_v); |
- |
- // V skipping condition checking |
- if ((var_v * 4 <= thresh_ac) && (sse_v - var_v <= thresh_dc)) { |
- x->skip = 1; |
- |
- // The cost of skip bit needs to be added. |
- rate = rate_mv; |
- rate += cpi->inter_mode_cost[mbmi->mode_context[ref_frame]] |
- [INTER_OFFSET(this_mode)]; |
- |
- // More on this part of rate |
- // rate += vp9_cost_bit(vp9_get_skip_prob(cm, xd), 1); |
- |
- // Scaling factor for SSE from spatial domain to frequency |
- // domain is 16. Adjust distortion accordingly. |
- // TODO(yunqingwang): In this function, only y-plane dist is |
- // calculated. |
- dist = (sse << 4); // + ((sse_u + sse_v) << 4); |
- this_rd = RDCOST(x->rdmult, x->rddiv, rate, dist); |
- // *disable_skip = 1; |
- } |
- } |
+ if (cpi->allow_encode_breakout) { |
+ encode_breakout_test(cpi, x, bsize, mi_row, mi_col, ref_frame, |
+ this_mode, var_y, sse_y, yv12_mb, &rate, &dist); |
+ if (x->skip) { |
+ rate += rate_mv; |
+ this_rd = RDCOST(x->rdmult, x->rddiv, rate, dist); |
} |
} |
#if CONFIG_DENOISING |
- vp9_denoiser_update_frame_stats(); |
+ if (cpi->oxcf.noise_sensitivity > 0) { |
+ vp9_denoiser_update_frame_stats(&cpi->denoiser, mbmi, sse_y, this_mode); |
+ } |
#endif |
if (this_rd < best_rd || x->skip) { |
@@ -449,8 +671,19 @@ |
*returndistortion = dist; |
best_mode = this_mode; |
best_pred_filter = mbmi->interp_filter; |
+ best_tx_size = mbmi->tx_size; |
best_ref_frame = ref_frame; |
skip_txfm = x->skip_txfm; |
+ |
+ if (cpi->sf.reuse_inter_pred_sby) { |
+ if (best_pred != NULL) |
+ free_pred_buffer(best_pred); |
+ |
+ best_pred = this_mode_pred; |
+ } |
+ } else { |
+ if (cpi->sf.reuse_inter_pred_sby) |
+ free_pred_buffer(this_mode_pred); |
} |
if (x->skip) |
@@ -458,11 +691,25 @@ |
} |
} |
+ // If best prediction is not in dst buf, then copy the prediction block from |
+ // temp buf to dst buf. |
+ if (cpi->sf.reuse_inter_pred_sby && best_pred->data != orig_dst.buf) { |
+ uint8_t *copy_from, *copy_to; |
- mbmi->mode = best_mode; |
+ pd->dst = orig_dst; |
+ copy_to = pd->dst.buf; |
+ |
+ copy_from = best_pred->data; |
+ |
+ vp9_convolve_copy(copy_from, bw, copy_to, pd->dst.stride, NULL, 0, NULL, 0, |
+ bw, bh); |
+ } |
+ |
+ mbmi->mode = best_mode; |
mbmi->interp_filter = best_pred_filter; |
- mbmi->ref_frame[0] = best_ref_frame; |
- mbmi->mv[0].as_int = frame_mv[best_mode][best_ref_frame].as_int; |
+ mbmi->tx_size = best_tx_size; |
+ mbmi->ref_frame[0] = best_ref_frame; |
+ mbmi->mv[0].as_int = frame_mv[best_mode][best_ref_frame].as_int; |
xd->mi[0]->bmi[0].as_mv[0].as_int = mbmi->mv[0].as_int; |
x->skip_txfm = skip_txfm; |
@@ -470,22 +717,57 @@ |
// threshold. |
if (!x->skip && best_rd > inter_mode_thresh && |
bsize <= cpi->sf.max_intra_bsize) { |
+ int i, j; |
+ const int width = num_4x4_blocks_wide_lookup[bsize]; |
+ const int height = num_4x4_blocks_high_lookup[bsize]; |
+ |
+ int rate2 = 0; |
+ int64_t dist2 = 0; |
+ const int dst_stride = pd->dst.stride; |
+ const int src_stride = p->src.stride; |
+ int block_idx = 0; |
+ |
+ TX_SIZE tmp_tx_size = MIN(max_txsize_lookup[bsize], |
+ tx_mode_to_biggest_tx_size[cpi->common.tx_mode]); |
+ const int step = 1 << tmp_tx_size; |
+ |
for (this_mode = DC_PRED; this_mode <= DC_PRED; ++this_mode) { |
- vp9_predict_intra_block(xd, 0, b_width_log2(bsize), |
- mbmi->tx_size, this_mode, |
- &p->src.buf[0], p->src.stride, |
- &pd->dst.buf[0], pd->dst.stride, 0, 0, 0); |
+ if (cpi->sf.reuse_inter_pred_sby) { |
+ pd->dst.buf = tmp[0].data; |
+ pd->dst.stride = bw; |
+ } |
- model_rd_for_sb_y(cpi, bsize, x, xd, &rate, &dist, &var_y, &sse_y); |
+ for (j = 0; j < height; j += step) { |
+ for (i = 0; i < width; i += step) { |
+ vp9_predict_intra_block(xd, block_idx, b_width_log2(bsize), |
+ tmp_tx_size, this_mode, |
+ &p->src.buf[4 * (j * dst_stride + i)], |
+ src_stride, |
+ &pd->dst.buf[4 * (j * dst_stride + i)], |
+ dst_stride, i, j, 0); |
+ model_rd_for_sb_y(cpi, bsize, x, xd, &rate, &dist, &var_y, &sse_y); |
+ rate2 += rate; |
+ dist2 += dist; |
+ ++block_idx; |
+ } |
+ } |
+ |
+ rate = rate2; |
+ dist = dist2; |
+ |
rate += cpi->mbmode_cost[this_mode]; |
rate += intra_cost_penalty; |
this_rd = RDCOST(x->rdmult, x->rddiv, rate, dist); |
+ if (cpi->sf.reuse_inter_pred_sby) |
+ pd->dst = orig_dst; |
+ |
if (this_rd + intra_mode_cost < best_rd) { |
best_rd = this_rd; |
*returnrate = rate; |
*returndistortion = dist; |
mbmi->mode = this_mode; |
+ mbmi->tx_size = tmp_tx_size; |
mbmi->ref_frame[0] = INTRA_FRAME; |
mbmi->uv_mode = this_mode; |
mbmi->mv[0].as_int = INVALID_MV; |
@@ -494,8 +776,11 @@ |
} |
} |
} |
+ |
#if CONFIG_DENOISING |
- vp9_denoiser_denoise(&cpi->denoiser, x, mi_row, mi_col, bsize); |
+ if (cpi->oxcf.noise_sensitivity > 0) { |
+ vp9_denoiser_denoise(&cpi->denoiser, x, mi_row, mi_col, bsize); |
+ } |
#endif |
return INT64_MAX; |