vp10: allow forward updates for keyframe y intra mode probabilities.
See issue 1040 point 5. Change-Id: I51a70b9eade39efba392a1457bd70a3c515525cb
This commit is contained in:
@@ -282,15 +282,6 @@ static INLINE void reset_skip_context(MACROBLOCKD *xd, BLOCK_SIZE bsize) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static INLINE const vpx_prob *get_y_mode_probs(const MODE_INFO *mi,
|
|
||||||
const MODE_INFO *above_mi,
|
|
||||||
const MODE_INFO *left_mi,
|
|
||||||
int block) {
|
|
||||||
const PREDICTION_MODE above = vp10_above_block_mode(mi, above_mi, block);
|
|
||||||
const PREDICTION_MODE left = vp10_left_block_mode(mi, left_mi, block);
|
|
||||||
return vp10_kf_y_mode_prob[above][left];
|
|
||||||
}
|
|
||||||
|
|
||||||
typedef void (*foreach_transformed_block_visitor)(int plane, int block,
|
typedef void (*foreach_transformed_block_visitor)(int plane, int block,
|
||||||
BLOCK_SIZE plane_bsize,
|
BLOCK_SIZE plane_bsize,
|
||||||
TX_SIZE tx_size,
|
TX_SIZE tx_size,
|
||||||
|
@@ -76,6 +76,7 @@ typedef struct frame_contexts {
|
|||||||
} FRAME_CONTEXT;
|
} FRAME_CONTEXT;
|
||||||
|
|
||||||
typedef struct FRAME_COUNTS {
|
typedef struct FRAME_COUNTS {
|
||||||
|
unsigned int kf_y_mode[INTRA_MODES][INTRA_MODES][INTRA_MODES];
|
||||||
unsigned int y_mode[BLOCK_SIZE_GROUPS][INTRA_MODES];
|
unsigned int y_mode[BLOCK_SIZE_GROUPS][INTRA_MODES];
|
||||||
unsigned int uv_mode[INTRA_MODES][INTRA_MODES];
|
unsigned int uv_mode[INTRA_MODES][INTRA_MODES];
|
||||||
unsigned int partition[PARTITION_CONTEXTS][PARTITION_TYPES];
|
unsigned int partition[PARTITION_CONTEXTS][PARTITION_TYPES];
|
||||||
|
@@ -302,6 +302,11 @@ typedef struct VP10Common {
|
|||||||
PARTITION_CONTEXT *above_seg_context;
|
PARTITION_CONTEXT *above_seg_context;
|
||||||
ENTROPY_CONTEXT *above_context;
|
ENTROPY_CONTEXT *above_context;
|
||||||
int above_context_alloc_cols;
|
int above_context_alloc_cols;
|
||||||
|
|
||||||
|
// scratch memory for intraonly/keyframe forward updates from default tables
|
||||||
|
// - this is intentionally not placed in FRAME_CONTEXT since it's reset upon
|
||||||
|
// each keyframe and not used afterwards
|
||||||
|
vpx_prob kf_y_prob[INTRA_MODES][INTRA_MODES][INTRA_MODES - 1];
|
||||||
} VP10_COMMON;
|
} VP10_COMMON;
|
||||||
|
|
||||||
// TODO(hkuang): Don't need to lock the whole pool after implementing atomic
|
// TODO(hkuang): Don't need to lock the whole pool after implementing atomic
|
||||||
@@ -443,6 +448,16 @@ static INLINE void set_mi_row_col(MACROBLOCKD *xd, const TileInfo *const tile,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static INLINE const vpx_prob *get_y_mode_probs(const VP10_COMMON *cm,
|
||||||
|
const MODE_INFO *mi,
|
||||||
|
const MODE_INFO *above_mi,
|
||||||
|
const MODE_INFO *left_mi,
|
||||||
|
int block) {
|
||||||
|
const PREDICTION_MODE above = vp10_above_block_mode(mi, above_mi, block);
|
||||||
|
const PREDICTION_MODE left = vp10_left_block_mode(mi, left_mi, block);
|
||||||
|
return cm->kf_y_prob[above][left];
|
||||||
|
}
|
||||||
|
|
||||||
static INLINE void update_partition_context(MACROBLOCKD *xd,
|
static INLINE void update_partition_context(MACROBLOCKD *xd,
|
||||||
int mi_row, int mi_col,
|
int mi_row, int mi_col,
|
||||||
BLOCK_SIZE subsize,
|
BLOCK_SIZE subsize,
|
||||||
|
@@ -2156,7 +2156,15 @@ static int read_compressed_header(VP10Decoder *pbi, const uint8_t *data,
|
|||||||
vp10_diff_update_prob(&r, &fc->partition_prob[j][i]);
|
vp10_diff_update_prob(&r, &fc->partition_prob[j][i]);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!frame_is_intra_only(cm)) {
|
if (frame_is_intra_only(cm)) {
|
||||||
|
vp10_copy(cm->kf_y_prob, vp10_kf_y_mode_prob);
|
||||||
|
#if CONFIG_MISC_FIXES
|
||||||
|
for (k = 0; k < INTRA_MODES; k++)
|
||||||
|
for (j = 0; j < INTRA_MODES; j++)
|
||||||
|
for (i = 0; i < INTRA_MODES - 1; ++i)
|
||||||
|
vp10_diff_update_prob(&r, &cm->kf_y_prob[k][j][i]);
|
||||||
|
#endif
|
||||||
|
} else {
|
||||||
nmv_context *const nmvc = &fc->nmvc;
|
nmv_context *const nmvc = &fc->nmvc;
|
||||||
|
|
||||||
read_inter_mode_probs(fc, &r);
|
read_inter_mode_probs(fc, &r);
|
||||||
|
@@ -305,24 +305,24 @@ static void read_intra_frame_mode_info(VP10_COMMON *const cm,
|
|||||||
case BLOCK_4X4:
|
case BLOCK_4X4:
|
||||||
for (i = 0; i < 4; ++i)
|
for (i = 0; i < 4; ++i)
|
||||||
mi->bmi[i].as_mode =
|
mi->bmi[i].as_mode =
|
||||||
read_intra_mode(r, get_y_mode_probs(mi, above_mi, left_mi, i));
|
read_intra_mode(r, get_y_mode_probs(cm, mi, above_mi, left_mi, i));
|
||||||
mbmi->mode = mi->bmi[3].as_mode;
|
mbmi->mode = mi->bmi[3].as_mode;
|
||||||
break;
|
break;
|
||||||
case BLOCK_4X8:
|
case BLOCK_4X8:
|
||||||
mi->bmi[0].as_mode = mi->bmi[2].as_mode =
|
mi->bmi[0].as_mode = mi->bmi[2].as_mode =
|
||||||
read_intra_mode(r, get_y_mode_probs(mi, above_mi, left_mi, 0));
|
read_intra_mode(r, get_y_mode_probs(cm, mi, above_mi, left_mi, 0));
|
||||||
mi->bmi[1].as_mode = mi->bmi[3].as_mode = mbmi->mode =
|
mi->bmi[1].as_mode = mi->bmi[3].as_mode = mbmi->mode =
|
||||||
read_intra_mode(r, get_y_mode_probs(mi, above_mi, left_mi, 1));
|
read_intra_mode(r, get_y_mode_probs(cm, mi, above_mi, left_mi, 1));
|
||||||
break;
|
break;
|
||||||
case BLOCK_8X4:
|
case BLOCK_8X4:
|
||||||
mi->bmi[0].as_mode = mi->bmi[1].as_mode =
|
mi->bmi[0].as_mode = mi->bmi[1].as_mode =
|
||||||
read_intra_mode(r, get_y_mode_probs(mi, above_mi, left_mi, 0));
|
read_intra_mode(r, get_y_mode_probs(cm, mi, above_mi, left_mi, 0));
|
||||||
mi->bmi[2].as_mode = mi->bmi[3].as_mode = mbmi->mode =
|
mi->bmi[2].as_mode = mi->bmi[3].as_mode = mbmi->mode =
|
||||||
read_intra_mode(r, get_y_mode_probs(mi, above_mi, left_mi, 2));
|
read_intra_mode(r, get_y_mode_probs(cm, mi, above_mi, left_mi, 2));
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
mbmi->mode = read_intra_mode(r,
|
mbmi->mode = read_intra_mode(r,
|
||||||
get_y_mode_probs(mi, above_mi, left_mi, 0));
|
get_y_mode_probs(cm, mi, above_mi, left_mi, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
mbmi->uv_mode = read_intra_mode_uv(cm, xd, r, mbmi->mode);
|
mbmi->uv_mode = read_intra_mode_uv(cm, xd, r, mbmi->mode);
|
||||||
|
@@ -455,7 +455,8 @@ static void write_mb_modes_kf(const VP10_COMMON *cm, const MACROBLOCKD *xd,
|
|||||||
write_selected_tx_size(cm, xd, w);
|
write_selected_tx_size(cm, xd, w);
|
||||||
|
|
||||||
if (bsize >= BLOCK_8X8) {
|
if (bsize >= BLOCK_8X8) {
|
||||||
write_intra_mode(w, mbmi->mode, get_y_mode_probs(mi, above_mi, left_mi, 0));
|
write_intra_mode(w, mbmi->mode,
|
||||||
|
get_y_mode_probs(cm, mi, above_mi, left_mi, 0));
|
||||||
} else {
|
} else {
|
||||||
const int num_4x4_w = num_4x4_blocks_wide_lookup[bsize];
|
const int num_4x4_w = num_4x4_blocks_wide_lookup[bsize];
|
||||||
const int num_4x4_h = num_4x4_blocks_high_lookup[bsize];
|
const int num_4x4_h = num_4x4_blocks_high_lookup[bsize];
|
||||||
@@ -465,7 +466,7 @@ static void write_mb_modes_kf(const VP10_COMMON *cm, const MACROBLOCKD *xd,
|
|||||||
for (idx = 0; idx < 2; idx += num_4x4_w) {
|
for (idx = 0; idx < 2; idx += num_4x4_w) {
|
||||||
const int block = idy * 2 + idx;
|
const int block = idy * 2 + idx;
|
||||||
write_intra_mode(w, mi->bmi[block].as_mode,
|
write_intra_mode(w, mi->bmi[block].as_mode,
|
||||||
get_y_mode_probs(mi, above_mi, left_mi, block));
|
get_y_mode_probs(cm, mi, above_mi, left_mi, block));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1348,6 +1349,9 @@ static size_t write_compressed_header(VP10_COMP *cpi, uint8_t *data) {
|
|||||||
FRAME_COUNTS *counts = cpi->td.counts;
|
FRAME_COUNTS *counts = cpi->td.counts;
|
||||||
vpx_writer header_bc;
|
vpx_writer header_bc;
|
||||||
int i;
|
int i;
|
||||||
|
#if CONFIG_MISC_FIXES
|
||||||
|
int j;
|
||||||
|
#endif
|
||||||
|
|
||||||
vpx_start_encode(&header_bc, data);
|
vpx_start_encode(&header_bc, data);
|
||||||
|
|
||||||
@@ -1373,7 +1377,15 @@ static size_t write_compressed_header(VP10_COMP *cpi, uint8_t *data) {
|
|||||||
counts->partition[i], PARTITION_TYPES, &header_bc);
|
counts->partition[i], PARTITION_TYPES, &header_bc);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!frame_is_intra_only(cm)) {
|
if (frame_is_intra_only(cm)) {
|
||||||
|
vp10_copy(cm->kf_y_prob, vp10_kf_y_mode_prob);
|
||||||
|
#if CONFIG_MISC_FIXES
|
||||||
|
for (i = 0; i < INTRA_MODES; ++i)
|
||||||
|
for (j = 0; j < INTRA_MODES; ++j)
|
||||||
|
prob_diff_update(vp10_intra_mode_tree, cm->kf_y_prob[i][j],
|
||||||
|
counts->kf_y_mode[i][j], INTRA_MODES, &header_bc);
|
||||||
|
#endif
|
||||||
|
} else {
|
||||||
for (i = 0; i < INTER_MODE_CONTEXTS; ++i)
|
for (i = 0; i < INTER_MODE_CONTEXTS; ++i)
|
||||||
prob_diff_update(vp10_inter_mode_tree, cm->fc->inter_mode_probs[i],
|
prob_diff_update(vp10_inter_mode_tree, cm->fc->inter_mode_probs[i],
|
||||||
counts->inter_mode[i], INTER_MODES, &header_bc);
|
counts->inter_mode[i], INTER_MODES, &header_bc);
|
||||||
|
@@ -2900,7 +2900,9 @@ void vp10_encode_frame(VP10_COMP *cpi) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sum_intra_stats(FRAME_COUNTS *counts, const MODE_INFO *mi) {
|
static void sum_intra_stats(FRAME_COUNTS *counts, const MODE_INFO *mi,
|
||||||
|
const MODE_INFO *above_mi, const MODE_INFO *left_mi,
|
||||||
|
const int intraonly) {
|
||||||
const PREDICTION_MODE y_mode = mi->mbmi.mode;
|
const PREDICTION_MODE y_mode = mi->mbmi.mode;
|
||||||
const PREDICTION_MODE uv_mode = mi->mbmi.uv_mode;
|
const PREDICTION_MODE uv_mode = mi->mbmi.uv_mode;
|
||||||
const BLOCK_SIZE bsize = mi->mbmi.sb_type;
|
const BLOCK_SIZE bsize = mi->mbmi.sb_type;
|
||||||
@@ -2910,11 +2912,26 @@ static void sum_intra_stats(FRAME_COUNTS *counts, const MODE_INFO *mi) {
|
|||||||
const int num_4x4_w = num_4x4_blocks_wide_lookup[bsize];
|
const int num_4x4_w = num_4x4_blocks_wide_lookup[bsize];
|
||||||
const int num_4x4_h = num_4x4_blocks_high_lookup[bsize];
|
const int num_4x4_h = num_4x4_blocks_high_lookup[bsize];
|
||||||
for (idy = 0; idy < 2; idy += num_4x4_h)
|
for (idy = 0; idy < 2; idy += num_4x4_h)
|
||||||
for (idx = 0; idx < 2; idx += num_4x4_w)
|
for (idx = 0; idx < 2; idx += num_4x4_w) {
|
||||||
++counts->y_mode[0][mi->bmi[idy * 2 + idx].as_mode];
|
const int bidx = idy * 2 + idx;
|
||||||
|
const PREDICTION_MODE bmode = mi->bmi[bidx].as_mode;
|
||||||
|
if (intraonly) {
|
||||||
|
const PREDICTION_MODE a = vp10_above_block_mode(mi, above_mi, bidx);
|
||||||
|
const PREDICTION_MODE l = vp10_left_block_mode(mi, left_mi, bidx);
|
||||||
|
++counts->kf_y_mode[a][l][bmode];
|
||||||
|
} else {
|
||||||
|
++counts->y_mode[0][bmode];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (intraonly) {
|
||||||
|
const PREDICTION_MODE above = vp10_above_block_mode(mi, above_mi, 0);
|
||||||
|
const PREDICTION_MODE left = vp10_left_block_mode(mi, left_mi, 0);
|
||||||
|
++counts->kf_y_mode[above][left][y_mode];
|
||||||
} else {
|
} else {
|
||||||
++counts->y_mode[size_group_lookup[bsize]][y_mode];
|
++counts->y_mode[size_group_lookup[bsize]][y_mode];
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
++counts->uv_mode[y_mode][uv_mode];
|
++counts->uv_mode[y_mode][uv_mode];
|
||||||
}
|
}
|
||||||
@@ -2953,7 +2970,8 @@ static void encode_superblock(VP10_COMP *cpi, ThreadData *td,
|
|||||||
for (plane = 0; plane < MAX_MB_PLANE; ++plane)
|
for (plane = 0; plane < MAX_MB_PLANE; ++plane)
|
||||||
vp10_encode_intra_block_plane(x, VPXMAX(bsize, BLOCK_8X8), plane);
|
vp10_encode_intra_block_plane(x, VPXMAX(bsize, BLOCK_8X8), plane);
|
||||||
if (output_enabled)
|
if (output_enabled)
|
||||||
sum_intra_stats(td->counts, mi);
|
sum_intra_stats(td->counts, mi, xd->above_mi, xd->left_mi,
|
||||||
|
frame_is_intra_only(cm));
|
||||||
|
|
||||||
if (bsize >= BLOCK_8X8 && output_enabled) {
|
if (bsize >= BLOCK_8X8 && output_enabled) {
|
||||||
if (mbmi->palette_mode_info.palette_size[0] > 0) {
|
if (mbmi->palette_mode_info.palette_size[0] > 0) {
|
||||||
|
Reference in New Issue
Block a user