2010-05-18 19:58:33 +04:00
|
|
|
/*
|
Speed feature to skip split partition based on var
Adds a speed feature to disable split partition search based on a
given threshold on the source variance. A tighter threshold derived
from the threshold provided is used to also disable horizontal and
vertical partitions.
Results on derfraw300:
threshold = 16, psnr = -0.057%, speedup ~1% (football)
threshold = 32, psnr = -0.150%, speedup ~4-5% (football)
threshold = 64, psnr = -0.570%, speedup ~10-12% (football)
Results on stdhdraw250:
threshold = 32, psnr = -0.18%, speedup is somewhat more than derf
because of a larger number of smoother blocks at higher resolution.
Based on these results, a threshold of 32 is chosen for speed 1,
and a threshold of 64 is chosen for speeds 2 and above.
Change-Id: If08912fb6c67fd4242d12a0d094783a99f52f6c6
2013-08-03 04:15:38 +04:00
|
|
|
* Copyright (c) 2010 The WebM project authors. All Rights Reserved.
|
2010-05-18 19:58:33 +04:00
|
|
|
*
|
2010-06-18 20:39:21 +04:00
|
|
|
* Use of this source code is governed by a BSD-style license
|
2010-06-05 00:19:40 +04:00
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
|
* tree. An additional intellectual property rights grant can be found
|
2010-06-18 20:39:21 +04:00
|
|
|
* in the file PATENTS. All contributing project authors may
|
2010-06-05 00:19:40 +04:00
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
2010-05-18 19:58:33 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
2011-08-02 00:42:14 +04:00
|
|
|
#include "vpx_config.h"
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 08:55:14 +04:00
|
|
|
#include "vp9/common/vp9_filter.h"
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_onyxc_int.h"
|
Convert subpixel filters to use convolve framework
Update the code to call the new convolution functions to do subpixel
prediction rather than the existing functions. Remove the old C and
assembly code, since it is unused. This causes a 50% performance
reduction on the decoder, but that will be resolved when the asm for
the new functions is available.
There is no consensus for whether 6-tap or 2-tap predictors will be
supported in the final codec, so these filters are implemented in
terms of the 8-tap code, so that quality testing of these modes
can continue. Implementing the lower complexity algorithms is a
simple exercise, should it be necessary.
This code produces slightly better results in the EIGHTTAP_SMOOTH
case, since the filter is now applied in only one direction when
the subpel motion is only in one direction. Like the previous code,
the filtering is skipped entirely on full-pel MVs. This combination
seems to give the best quality gains, but this may be indicative of a
bug in the encoder's filter selection, since the encoder could
achieve the result of skipping the filtering on full-pel by selecting
one of the other filters. This should be revisited.
Quality gains on derf positive on almost all clips. The only clip
that seemed to be hurt at all datarates was football
(-0.115% PSNR average, -0.587% min). Overall averages 0.375% PSNR,
0.347% SSIM.
Change-Id: I7d469716091b1d89b4b08adde5863999319d69ff
2013-01-29 04:59:03 +04:00
|
|
|
#include "vp9/common/vp9_reconinter.h"
|
2012-11-28 22:41:40 +04:00
|
|
|
#include "vp9/encoder/vp9_onyx_int.h"
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_systemdependent.h"
|
2012-11-28 22:41:40 +04:00
|
|
|
#include "vp9/encoder/vp9_quantize.h"
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_alloccommon.h"
|
2012-11-28 22:41:40 +04:00
|
|
|
#include "vp9/encoder/vp9_mcomp.h"
|
|
|
|
#include "vp9/encoder/vp9_firstpass.h"
|
|
|
|
#include "vp9/encoder/vp9_psnr.h"
|
2012-12-04 02:19:49 +04:00
|
|
|
#include "vpx_scale/vpx_scale.h"
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_extend.h"
|
2012-11-28 22:41:40 +04:00
|
|
|
#include "vp9/encoder/vp9_ratectrl.h"
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_quant_common.h"
|
2013-02-07 03:30:21 +04:00
|
|
|
#include "vp9/common/vp9_tile_common.h"
|
2012-11-28 22:41:40 +04:00
|
|
|
#include "vp9/encoder/vp9_segmentation.h"
|
2012-11-25 07:33:58 +04:00
|
|
|
#include "./vp9_rtcd.h"
|
2012-12-04 00:26:51 +04:00
|
|
|
#include "./vpx_scale_rtcd.h"
|
2013-09-04 21:02:08 +04:00
|
|
|
#if CONFIG_VP9_POSTPROC
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_postproc.h"
|
2011-08-02 00:42:14 +04:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
#include "vpx_mem/vpx_mem.h"
|
|
|
|
#include "vpx_ports/vpx_timer.h"
|
2011-10-05 14:26:00 +04:00
|
|
|
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_seg_common.h"
|
2012-11-28 22:41:40 +04:00
|
|
|
#include "vp9/encoder/vp9_mbgraph.h"
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/common/vp9_pred_common.h"
|
|
|
|
#include "vp9/encoder/vp9_rdopt.h"
|
2012-11-28 22:41:40 +04:00
|
|
|
#include "vp9/encoder/vp9_bitstream.h"
|
2012-11-28 01:59:17 +04:00
|
|
|
#include "vp9/encoder/vp9_picklpf.h"
|
|
|
|
#include "vp9/common/vp9_mvref_common.h"
|
2012-11-29 22:10:51 +04:00
|
|
|
#include "vp9/encoder/vp9_temporal_filter.h"
|
2012-08-24 18:44:01 +04:00
|
|
|
|
2010-05-18 19:58:33 +04:00
|
|
|
#include <math.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
|
|
extern void print_tree_update_probs();
|
2012-10-31 01:25:33 +04:00
|
|
|
|
2013-08-10 04:24:40 +04:00
|
|
|
static void set_default_lf_deltas(struct loopfilter *lf);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-09 02:14:01 +04:00
|
|
|
#define DEFAULT_INTERP_FILTER SWITCHABLE
|
|
|
|
|
2012-07-28 04:46:33 +04:00
|
|
|
#define SHARP_FILTER_QTHRESH 0 /* Q threshold for 8-tap sharp filter */
|
|
|
|
|
|
|
|
#define ALTREF_HIGH_PRECISION_MV 1 /* whether to use high precision mv
|
|
|
|
for altref computation */
|
|
|
|
#define HIGH_PRECISION_MV_QTHRESH 200 /* Q threshold for use of high precision
|
|
|
|
mv. Choose a very high value for
|
|
|
|
now so that HIGH_PRECISION is always
|
|
|
|
chosen */
|
2012-02-27 22:22:38 +04:00
|
|
|
|
2011-04-29 20:37:59 +04:00
|
|
|
#if CONFIG_INTERNAL_STATS
|
2010-05-18 19:58:33 +04:00
|
|
|
#include "math.h"
|
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
extern double vp9_calc_ssim(YV12_BUFFER_CONFIG *source,
|
2012-10-22 07:47:57 +04:00
|
|
|
YV12_BUFFER_CONFIG *dest, int lumamask,
|
|
|
|
double *weight);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2011-03-08 17:05:18 +03:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
extern double vp9_calc_ssimg(YV12_BUFFER_CONFIG *source,
|
2012-10-22 07:47:57 +04:00
|
|
|
YV12_BUFFER_CONFIG *dest, double *ssim_y,
|
|
|
|
double *ssim_u, double *ssim_v);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// #define OUTPUT_YUV_REC
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#ifdef OUTPUT_YUV_SRC
|
|
|
|
FILE *yuv_file;
|
|
|
|
#endif
|
2011-02-15 01:18:18 +03:00
|
|
|
#ifdef OUTPUT_YUV_REC
|
|
|
|
FILE *yuv_rec_file;
|
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
FILE *framepsnr;
|
2011-07-21 01:21:24 +04:00
|
|
|
FILE *kf_list;
|
2010-05-18 19:58:33 +04:00
|
|
|
FILE *keyfile;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef ENTROPY_STATS
|
2013-08-23 05:40:34 +04:00
|
|
|
extern int intra_mode_stats[INTRA_MODES]
|
|
|
|
[INTRA_MODES]
|
|
|
|
[INTRA_MODES];
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
|
2013-06-06 22:14:04 +04:00
|
|
|
#ifdef MODE_STATS
|
|
|
|
extern void init_tx_count_stats();
|
|
|
|
extern void write_tx_count_stats();
|
2013-06-10 23:00:43 +04:00
|
|
|
extern void init_switchable_interp_stats();
|
|
|
|
extern void write_switchable_interp_stats();
|
2013-06-06 22:14:04 +04:00
|
|
|
#endif
|
2012-07-27 00:42:07 +04:00
|
|
|
|
2010-05-18 19:58:33 +04:00
|
|
|
#ifdef SPEEDSTATS
|
|
|
|
unsigned int frames_at_speed[16] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
|
|
|
|
#endif
|
|
|
|
|
2011-02-15 01:18:18 +03:00
|
|
|
#if defined(SECTIONBITS_OUTPUT)
|
|
|
|
extern unsigned __int64 Sectionbits[500];
|
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
extern void vp9_init_quantizer(VP9_COMP *cpi);
|
2011-12-12 22:27:25 +04:00
|
|
|
|
2010-09-29 15:03:19 +04:00
|
|
|
// Tables relating active max Q to active min Q
|
2011-11-23 15:32:20 +04:00
|
|
|
static int kf_low_motion_minq[QINDEX_RANGE];
|
|
|
|
static int kf_high_motion_minq[QINDEX_RANGE];
|
|
|
|
static int gf_low_motion_minq[QINDEX_RANGE];
|
|
|
|
static int gf_high_motion_minq[QINDEX_RANGE];
|
|
|
|
static int inter_minq[QINDEX_RANGE];
|
|
|
|
|
2013-07-19 01:05:06 +04:00
|
|
|
static INLINE void Scale2Ratio(int mode, int *hr, int *hs) {
|
|
|
|
switch (mode) {
|
|
|
|
case NORMAL:
|
|
|
|
*hr = 1;
|
|
|
|
*hs = 1;
|
|
|
|
break;
|
|
|
|
case FOURFIVE:
|
|
|
|
*hr = 4;
|
|
|
|
*hs = 5;
|
|
|
|
break;
|
|
|
|
case THREEFIVE:
|
|
|
|
*hr = 3;
|
|
|
|
*hs = 5;
|
|
|
|
break;
|
|
|
|
case ONETWO:
|
|
|
|
*hr = 1;
|
|
|
|
*hs = 2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*hr = 1;
|
|
|
|
*hs = 1;
|
|
|
|
assert(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-23 15:32:20 +04:00
|
|
|
// Functions to compute the active minq lookup table entries based on a
|
|
|
|
// formulaic approach to facilitate easier adjustment of the Q tables.
|
|
|
|
// The formulae were derived from computing a 3rd order polynomial best
|
|
|
|
// fit to the original data (after plotting real maxq vs minq (not q index))
|
2012-10-30 05:20:32 +04:00
|
|
|
static int calculate_minq_index(double maxq,
|
2013-03-28 01:22:30 +04:00
|
|
|
double x3, double x2, double x1, double c) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
2013-03-28 01:22:30 +04:00
|
|
|
const double minqtarget = MIN(((x3 * maxq + x2) * maxq + x1) * maxq + c,
|
|
|
|
maxq);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-04 20:40:39 +04:00
|
|
|
// Special case handling to deal with the step from q2.0
|
|
|
|
// down to lossless mode represented by q 1.0.
|
|
|
|
if (minqtarget <= 2.0)
|
|
|
|
return 0;
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < QINDEX_RANGE; i++) {
|
2012-10-30 23:58:42 +04:00
|
|
|
if (minqtarget <= vp9_convert_qindex_to_q(i))
|
2012-07-14 02:21:29 +04:00
|
|
|
return i;
|
|
|
|
}
|
2013-03-28 01:22:30 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return QINDEX_RANGE - 1;
|
2011-11-23 15:32:20 +04:00
|
|
|
}
|
2012-07-19 00:43:01 +04:00
|
|
|
|
2012-10-30 05:20:32 +04:00
|
|
|
static void init_minq_luts(void) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < QINDEX_RANGE; i++) {
|
2013-03-28 01:22:30 +04:00
|
|
|
const double maxq = vp9_convert_qindex_to_q(i);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
|
|
|
|
kf_low_motion_minq[i] = calculate_minq_index(maxq,
|
2013-04-16 21:59:39 +04:00
|
|
|
0.000001,
|
|
|
|
-0.0004,
|
|
|
|
0.15,
|
2012-07-14 02:21:29 +04:00
|
|
|
0.0);
|
|
|
|
kf_high_motion_minq[i] = calculate_minq_index(maxq,
|
2013-04-16 21:59:39 +04:00
|
|
|
0.000002,
|
|
|
|
-0.0012,
|
|
|
|
0.5,
|
2012-07-14 02:21:29 +04:00
|
|
|
0.0);
|
2013-04-16 21:59:39 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
gf_low_motion_minq[i] = calculate_minq_index(maxq,
|
|
|
|
0.0000015,
|
|
|
|
-0.0009,
|
|
|
|
0.33,
|
|
|
|
0.0);
|
|
|
|
gf_high_motion_minq[i] = calculate_minq_index(maxq,
|
|
|
|
0.0000021,
|
|
|
|
-0.00125,
|
|
|
|
0.45,
|
|
|
|
0.0);
|
|
|
|
inter_minq[i] = calculate_minq_index(maxq,
|
|
|
|
0.00000271,
|
|
|
|
-0.00113,
|
|
|
|
0.697,
|
|
|
|
0.0);
|
|
|
|
|
|
|
|
}
|
2011-11-23 15:32:20 +04:00
|
|
|
}
|
2012-11-07 18:50:25 +04:00
|
|
|
|
2012-11-09 03:44:39 +04:00
|
|
|
static void set_mvcost(MACROBLOCK *mb) {
|
|
|
|
if (mb->e_mbd.allow_high_precision_mv) {
|
|
|
|
mb->mvcost = mb->nmvcost_hp;
|
|
|
|
mb->mvsadcost = mb->nmvsadcost_hp;
|
|
|
|
} else {
|
|
|
|
mb->mvcost = mb->nmvcost;
|
|
|
|
mb->mvsadcost = mb->nmvsadcost;
|
|
|
|
}
|
|
|
|
}
|
2011-12-12 22:27:25 +04:00
|
|
|
|
2012-10-31 01:25:33 +04:00
|
|
|
void vp9_initialize_enc() {
|
2012-07-14 02:21:29 +04:00
|
|
|
static int init_done = 0;
|
|
|
|
|
|
|
|
if (!init_done) {
|
2012-10-31 03:25:53 +04:00
|
|
|
vp9_initialize_common();
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_tokenize_initialize();
|
2012-10-31 03:25:53 +04:00
|
|
|
vp9_init_quant_tables();
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_init_me_luts();
|
2012-07-14 02:21:29 +04:00
|
|
|
init_minq_luts();
|
2013-06-08 00:24:14 +04:00
|
|
|
// init_base_skip_probs();
|
2012-07-14 02:21:29 +04:00
|
|
|
init_done = 1;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-08-14 22:20:33 +04:00
|
|
|
static void setup_features(VP9_COMMON *cm) {
|
|
|
|
struct loopfilter *const lf = &cm->lf;
|
|
|
|
struct segmentation *const seg = &cm->seg;
|
2011-09-14 21:20:25 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Set up default state for MB feature flags
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->enabled = 0;
|
2011-11-08 19:40:32 +04:00
|
|
|
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->update_map = 0;
|
|
|
|
seg->update_data = 0;
|
|
|
|
vpx_memset(seg->tree_probs, 255, sizeof(seg->tree_probs));
|
2011-09-14 21:20:25 +04:00
|
|
|
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_clearall_segfeatures(seg);
|
2011-09-14 21:20:25 +04:00
|
|
|
|
2013-07-18 05:37:45 +04:00
|
|
|
lf->mode_ref_delta_enabled = 0;
|
|
|
|
lf->mode_ref_delta_update = 0;
|
|
|
|
vp9_zero(lf->ref_deltas);
|
|
|
|
vp9_zero(lf->mode_deltas);
|
|
|
|
vp9_zero(lf->last_ref_deltas);
|
|
|
|
vp9_zero(lf->last_mode_deltas);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-10 04:24:40 +04:00
|
|
|
set_default_lf_deltas(lf);
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void dealloc_compressor_data(VP9_COMP *cpi) {
|
2012-07-14 02:21:29 +04:00
|
|
|
// Delete sementation map
|
|
|
|
vpx_free(cpi->segmentation_map);
|
|
|
|
cpi->segmentation_map = 0;
|
|
|
|
vpx_free(cpi->common.last_frame_seg_map);
|
|
|
|
cpi->common.last_frame_seg_map = 0;
|
|
|
|
vpx_free(cpi->coding_context.last_frame_seg_map_copy);
|
|
|
|
cpi->coding_context.last_frame_seg_map_copy = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi->active_map);
|
|
|
|
cpi->active_map = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-02 05:23:04 +04:00
|
|
|
vp9_free_frame_buffers(&cpi->common);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-05-07 02:52:06 +04:00
|
|
|
vp9_free_frame_buffer(&cpi->last_frame_uf);
|
|
|
|
vp9_free_frame_buffer(&cpi->scaled_source);
|
|
|
|
vp9_free_frame_buffer(&cpi->alt_ref_buffer);
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_lookahead_destroy(cpi->lookahead);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi->tok);
|
|
|
|
cpi->tok = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Activity mask based per mb zbin adjustments
|
|
|
|
vpx_free(cpi->mb_activity_map);
|
|
|
|
cpi->mb_activity_map = 0;
|
|
|
|
vpx_free(cpi->mb_norm_activity_map);
|
|
|
|
cpi->mb_norm_activity_map = 0;
|
2011-05-12 20:01:55 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi->mb.pip);
|
|
|
|
cpi->mb.pip = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2011-12-06 18:48:52 +04:00
|
|
|
// Computes a q delta (in "q index" terms) to get from a starting q value
|
|
|
|
// to a target value
|
|
|
|
// target q value
|
2012-10-31 04:53:32 +04:00
|
|
|
static int compute_qdelta(VP9_COMP *cpi, double qstart, double qtarget) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
|
|
|
int start_index = cpi->worst_quality;
|
|
|
|
int target_index = cpi->worst_quality;
|
|
|
|
|
|
|
|
// Convert the average q value to an index.
|
|
|
|
for (i = cpi->best_quality; i < cpi->worst_quality; i++) {
|
|
|
|
start_index = i;
|
2012-10-30 23:58:42 +04:00
|
|
|
if (vp9_convert_qindex_to_q(i) >= qstart)
|
2012-07-14 02:21:29 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert the q target to an index
|
|
|
|
for (i = cpi->best_quality; i < cpi->worst_quality; i++) {
|
|
|
|
target_index = i;
|
2012-10-30 23:58:42 +04:00
|
|
|
if (vp9_convert_qindex_to_q(i) >= qtarget)
|
2012-07-14 02:21:29 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return target_index - start_index;
|
2011-12-06 18:48:52 +04:00
|
|
|
}
|
|
|
|
|
2013-02-05 14:13:25 +04:00
|
|
|
static void configure_static_seg_features(VP9_COMP *cpi) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2013-08-14 22:20:33 +04:00
|
|
|
struct segmentation *seg = &cm->seg;
|
2011-09-30 19:45:16 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
int high_q = (int)(cpi->avg_q > 48.0);
|
|
|
|
int qi_delta;
|
2011-10-28 18:27:23 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Disable and clear down for KF
|
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
|
|
|
// Clear down the global segmentation map
|
2013-04-26 22:57:17 +04:00
|
|
|
vpx_memset(cpi->segmentation_map, 0, cm->mi_rows * cm->mi_cols);
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->update_map = 0;
|
|
|
|
seg->update_data = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->static_mb_pct = 0;
|
2011-09-30 19:45:16 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Disable segmentation
|
2012-10-31 04:53:32 +04:00
|
|
|
vp9_disable_segmentation((VP9_PTR)cpi);
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Clear down the segment features.
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_clearall_segfeatures(seg);
|
2013-01-16 01:49:44 +04:00
|
|
|
} else if (cpi->refresh_alt_ref_frame) {
|
|
|
|
// If this is an alt ref frame
|
2012-07-14 02:21:29 +04:00
|
|
|
// Clear down the global segmentation map
|
2013-04-26 22:57:17 +04:00
|
|
|
vpx_memset(cpi->segmentation_map, 0, cm->mi_rows * cm->mi_cols);
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->update_map = 0;
|
|
|
|
seg->update_data = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->static_mb_pct = 0;
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Disable segmentation and individual segment features by default
|
2012-10-31 04:53:32 +04:00
|
|
|
vp9_disable_segmentation((VP9_PTR)cpi);
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_clearall_segfeatures(seg);
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Scan frames from current to arf frame.
|
|
|
|
// This function re-enables segmentation if appropriate.
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_update_mbgraph_stats(cpi);
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// If segmentation was enabled set those features needed for the
|
|
|
|
// arf itself.
|
2013-08-02 01:53:14 +04:00
|
|
|
if (seg->enabled) {
|
|
|
|
seg->update_map = 1;
|
|
|
|
seg->update_data = 1;
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
qi_delta = compute_qdelta(cpi, cpi->avg_q, (cpi->avg_q * 0.875));
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_set_segdata(seg, 1, SEG_LVL_ALT_Q, (qi_delta - 2));
|
|
|
|
vp9_set_segdata(seg, 1, SEG_LVL_ALT_LF, -2);
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_ALT_Q);
|
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_ALT_LF);
|
2011-10-05 14:26:00 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Where relevant assume segment data is delta data
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->abs_delta = SEGMENT_DELTADATA;
|
2011-11-09 18:05:28 +04:00
|
|
|
|
2011-09-30 19:45:16 +04:00
|
|
|
}
|
2013-08-02 01:53:14 +04:00
|
|
|
} else if (seg->enabled) {
|
2013-04-30 03:07:17 +04:00
|
|
|
// All other frames if segmentation has been enabled
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// First normal frame in a valid gf or alt ref group
|
2013-07-19 01:09:21 +04:00
|
|
|
if (cpi->frames_since_golden == 0) {
|
2013-01-28 19:22:53 +04:00
|
|
|
// Set up segment features for normal frames in an arf group
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->source_alt_ref_active) {
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->update_map = 0;
|
|
|
|
seg->update_data = 1;
|
|
|
|
seg->abs_delta = SEGMENT_DELTADATA;
|
2011-11-03 20:58:26 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
qi_delta = compute_qdelta(cpi, cpi->avg_q,
|
|
|
|
(cpi->avg_q * 1.125));
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_set_segdata(seg, 1, SEG_LVL_ALT_Q, (qi_delta + 2));
|
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_ALT_Q);
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_set_segdata(seg, 1, SEG_LVL_ALT_LF, -2);
|
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_ALT_LF);
|
2011-11-03 20:58:26 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Segment coding disabled for compred testing
|
|
|
|
if (high_q || (cpi->static_mb_pct == 100)) {
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_set_segdata(seg, 1, SEG_LVL_REF_FRAME, ALTREF_FRAME);
|
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_REF_FRAME);
|
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_SKIP);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2013-04-30 03:07:17 +04:00
|
|
|
} else {
|
|
|
|
// Disable segmentation and clear down features if alt ref
|
|
|
|
// is not active for this group
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
vp9_disable_segmentation((VP9_PTR)cpi);
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2013-04-26 22:57:17 +04:00
|
|
|
vpx_memset(cpi->segmentation_map, 0, cm->mi_rows * cm->mi_cols);
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->update_map = 0;
|
|
|
|
seg->update_data = 0;
|
2011-10-07 19:58:28 +04:00
|
|
|
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_clearall_segfeatures(seg);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2013-04-30 03:07:17 +04:00
|
|
|
} else if (cpi->is_src_frame_alt_ref) {
|
|
|
|
// Special case where we are coding over the top of a previous
|
|
|
|
// alt ref frame.
|
|
|
|
// Segment coding disabled for compred testing
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-01-28 19:22:53 +04:00
|
|
|
// Enable ref frame features for segment 0 as well
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_enable_segfeature(seg, 0, SEG_LVL_REF_FRAME);
|
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_REF_FRAME);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-01-28 19:22:53 +04:00
|
|
|
// All mbs should use ALTREF_FRAME
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_clear_segdata(seg, 0, SEG_LVL_REF_FRAME);
|
|
|
|
vp9_set_segdata(seg, 0, SEG_LVL_REF_FRAME, ALTREF_FRAME);
|
|
|
|
vp9_clear_segdata(seg, 1, SEG_LVL_REF_FRAME);
|
|
|
|
vp9_set_segdata(seg, 1, SEG_LVL_REF_FRAME, ALTREF_FRAME);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-01-28 19:22:53 +04:00
|
|
|
// Skip all MBs if high Q (0,0 mv and skip coeffs)
|
2012-07-14 02:21:29 +04:00
|
|
|
if (high_q) {
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_enable_segfeature(seg, 0, SEG_LVL_SKIP);
|
|
|
|
vp9_enable_segfeature(seg, 1, SEG_LVL_SKIP);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2013-07-24 18:58:26 +04:00
|
|
|
// Enable data update
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->update_data = 1;
|
2013-04-30 03:07:17 +04:00
|
|
|
} else {
|
|
|
|
// All other frames.
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// No updates.. leave things as they are.
|
2013-08-02 01:53:14 +04:00
|
|
|
seg->update_map = 0;
|
|
|
|
seg->update_data = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
}
|
2011-09-30 19:45:16 +04:00
|
|
|
}
|
|
|
|
|
2013-05-16 14:27:12 +04:00
|
|
|
#ifdef ENTROPY_STATS
|
|
|
|
void vp9_update_mode_context_stats(VP9_COMP *cpi) {
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
|
|
|
int i, j;
|
2013-08-23 05:40:34 +04:00
|
|
|
unsigned int (*inter_mode_counts)[INTER_MODES - 1][2] =
|
2013-06-05 02:25:16 +04:00
|
|
|
cm->fc.inter_mode_counts;
|
2013-08-23 05:40:34 +04:00
|
|
|
int64_t (*mv_ref_stats)[INTER_MODES - 1][2] = cpi->mv_ref_stats;
|
2013-05-16 14:27:12 +04:00
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
// Read the past stats counters
|
|
|
|
f = fopen("mode_context.bin", "rb");
|
|
|
|
if (!f) {
|
|
|
|
vpx_memset(cpi->mv_ref_stats, 0, sizeof(cpi->mv_ref_stats));
|
|
|
|
} else {
|
|
|
|
fread(cpi->mv_ref_stats, sizeof(cpi->mv_ref_stats), 1, f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add in the values for this frame
|
|
|
|
for (i = 0; i < INTER_MODE_CONTEXTS; i++) {
|
2013-08-23 05:40:34 +04:00
|
|
|
for (j = 0; j < INTER_MODES - 1; j++) {
|
2013-06-05 02:25:16 +04:00
|
|
|
mv_ref_stats[i][j][0] += (int64_t)inter_mode_counts[i][j][0];
|
|
|
|
mv_ref_stats[i][j][1] += (int64_t)inter_mode_counts[i][j][1];
|
2013-05-16 14:27:12 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write back the accumulated stats
|
|
|
|
f = fopen("mode_context.bin", "wb");
|
|
|
|
fwrite(cpi->mv_ref_stats, sizeof(cpi->mv_ref_stats), 1, f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void print_mode_context(VP9_COMP *cpi) {
|
|
|
|
FILE *f = fopen("vp9_modecont.c", "a");
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
fprintf(f, "#include \"vp9_entropy.h\"\n");
|
2013-06-05 22:21:44 +04:00
|
|
|
fprintf(
|
|
|
|
f,
|
2013-08-23 05:40:34 +04:00
|
|
|
"const int inter_mode_probs[INTER_MODE_CONTEXTS][INTER_MODES - 1] =");
|
2013-05-16 14:27:12 +04:00
|
|
|
fprintf(f, "{\n");
|
|
|
|
for (j = 0; j < INTER_MODE_CONTEXTS; j++) {
|
|
|
|
fprintf(f, " {/* %d */ ", j);
|
|
|
|
fprintf(f, " ");
|
2013-08-23 05:40:34 +04:00
|
|
|
for (i = 0; i < INTER_MODES - 1; i++) {
|
2013-05-16 14:27:12 +04:00
|
|
|
int this_prob;
|
|
|
|
int64_t count = cpi->mv_ref_stats[j][i][0] + cpi->mv_ref_stats[j][i][1];
|
|
|
|
if (count)
|
|
|
|
this_prob = ((cpi->mv_ref_stats[j][i][0] * 256) + (count >> 1)) / count;
|
|
|
|
else
|
|
|
|
this_prob = 128;
|
|
|
|
|
|
|
|
// context probs
|
|
|
|
fprintf(f, "%5d, ", this_prob);
|
|
|
|
}
|
|
|
|
fprintf(f, " },\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(f, "};\n");
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
#endif // ENTROPY_STATS
|
|
|
|
|
2011-09-30 19:45:16 +04:00
|
|
|
// DEBUG: Print out the segment id of each MB in the current frame.
|
2012-10-31 04:53:32 +04:00
|
|
|
static void print_seg_map(VP9_COMP *cpi) {
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 02:21:29 +04:00
|
|
|
int row, col;
|
|
|
|
int map_index = 0;
|
2013-03-28 01:22:30 +04:00
|
|
|
FILE *statsfile = fopen("segmap.stt", "a");
|
2011-09-30 19:45:16 +04:00
|
|
|
|
2013-03-28 01:22:30 +04:00
|
|
|
fprintf(statsfile, "%10d\n", cm->current_video_frame);
|
2011-09-30 19:45:16 +04:00
|
|
|
|
2013-04-26 22:57:17 +04:00
|
|
|
for (row = 0; row < cpi->common.mi_rows; row++) {
|
|
|
|
for (col = 0; col < cpi->common.mi_cols; col++) {
|
2013-03-28 01:22:30 +04:00
|
|
|
fprintf(statsfile, "%10d", cpi->segmentation_map[map_index]);
|
2012-07-14 02:21:29 +04:00
|
|
|
map_index++;
|
2011-09-30 19:45:16 +04:00
|
|
|
}
|
|
|
|
fprintf(statsfile, "\n");
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
fprintf(statsfile, "\n");
|
2011-09-30 19:45:16 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(statsfile);
|
2011-09-30 19:45:16 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void update_reference_segmentation_map(VP9_COMP *cpi) {
|
2013-01-06 06:20:25 +04:00
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
|
|
|
int row, col;
|
2013-09-11 21:45:44 +04:00
|
|
|
MODE_INFO **mi_8x8, **mi_8x8_ptr = cm->mi_grid_visible;
|
2013-01-06 06:20:25 +04:00
|
|
|
uint8_t *cache_ptr = cm->last_frame_seg_map, *cache;
|
|
|
|
|
2013-04-26 22:57:17 +04:00
|
|
|
for (row = 0; row < cm->mi_rows; row++) {
|
2013-09-11 21:45:44 +04:00
|
|
|
mi_8x8 = mi_8x8_ptr;
|
2013-01-06 06:20:25 +04:00
|
|
|
cache = cache_ptr;
|
2013-09-11 21:45:44 +04:00
|
|
|
for (col = 0; col < cm->mi_cols; col++, mi_8x8++, cache++)
|
|
|
|
cache[0] = mi_8x8[0]->mbmi.segment_id;
|
|
|
|
mi_8x8_ptr += cm->mode_info_stride;
|
2013-04-26 22:57:17 +04:00
|
|
|
cache_ptr += cm->mi_cols;
|
2012-08-21 01:43:34 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-10 04:24:40 +04:00
|
|
|
static void set_default_lf_deltas(struct loopfilter *lf) {
|
2013-07-18 05:37:45 +04:00
|
|
|
lf->mode_ref_delta_enabled = 1;
|
|
|
|
lf->mode_ref_delta_update = 1;
|
|
|
|
|
|
|
|
vp9_zero(lf->ref_deltas);
|
|
|
|
vp9_zero(lf->mode_deltas);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Test of ref frame deltas
|
2013-07-18 05:37:45 +04:00
|
|
|
lf->ref_deltas[INTRA_FRAME] = 2;
|
|
|
|
lf->ref_deltas[LAST_FRAME] = 0;
|
|
|
|
lf->ref_deltas[GOLDEN_FRAME] = -2;
|
|
|
|
lf->ref_deltas[ALTREF_FRAME] = -2;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-07-18 05:37:45 +04:00
|
|
|
lf->mode_deltas[0] = 0; // Zero
|
|
|
|
lf->mode_deltas[1] = 0; // New mv
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-09-11 03:13:15 +04:00
|
|
|
static void set_rd_speed_thresholds(VP9_COMP *cpi, int mode) {
|
2013-02-04 21:43:02 +04:00
|
|
|
SPEED_FEATURES *sf = &cpi->sf;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
// Set baseline threshold values
|
2013-04-02 21:24:56 +04:00
|
|
|
for (i = 0; i < MAX_MODES; ++i)
|
|
|
|
sf->thresh_mult[i] = mode == 0 ? -500 : 0;
|
2013-02-04 21:43:02 +04:00
|
|
|
|
|
|
|
sf->thresh_mult[THR_NEARESTMV] = 0;
|
2013-07-02 05:18:50 +04:00
|
|
|
sf->thresh_mult[THR_NEARESTG] = 0;
|
|
|
|
sf->thresh_mult[THR_NEARESTA] = 0;
|
|
|
|
|
2013-09-11 03:13:15 +04:00
|
|
|
sf->thresh_mult[THR_NEWMV] += 1000;
|
|
|
|
sf->thresh_mult[THR_COMP_NEARESTLA] += 1000;
|
|
|
|
sf->thresh_mult[THR_NEARMV] += 1000;
|
|
|
|
sf->thresh_mult[THR_COMP_NEARESTGA] += 1000;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_DC] += 1000;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_NEWG] += 1000;
|
|
|
|
sf->thresh_mult[THR_NEWA] += 1000;
|
|
|
|
sf->thresh_mult[THR_NEARA] += 1000;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_TM] += 1000;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_COMP_NEARLA] += 1500;
|
|
|
|
sf->thresh_mult[THR_COMP_NEWLA] += 2000;
|
|
|
|
sf->thresh_mult[THR_NEARG] += 1000;
|
|
|
|
sf->thresh_mult[THR_COMP_NEARGA] += 1500;
|
|
|
|
sf->thresh_mult[THR_COMP_NEWGA] += 2000;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_SPLITMV] += 2500;
|
|
|
|
sf->thresh_mult[THR_SPLITG] += 2500;
|
|
|
|
sf->thresh_mult[THR_SPLITA] += 2500;
|
|
|
|
sf->thresh_mult[THR_COMP_SPLITLA] += 4500;
|
|
|
|
sf->thresh_mult[THR_COMP_SPLITGA] += 4500;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_ZEROMV] += 2000;
|
|
|
|
sf->thresh_mult[THR_ZEROG] += 2000;
|
|
|
|
sf->thresh_mult[THR_ZEROA] += 2000;
|
|
|
|
sf->thresh_mult[THR_COMP_ZEROLA] += 2500;
|
|
|
|
sf->thresh_mult[THR_COMP_ZEROGA] += 2500;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_B_PRED] += 2500;
|
|
|
|
sf->thresh_mult[THR_H_PRED] += 2000;
|
|
|
|
sf->thresh_mult[THR_V_PRED] += 2000;
|
|
|
|
sf->thresh_mult[THR_D45_PRED ] += 2500;
|
|
|
|
sf->thresh_mult[THR_D135_PRED] += 2500;
|
|
|
|
sf->thresh_mult[THR_D117_PRED] += 2500;
|
|
|
|
sf->thresh_mult[THR_D153_PRED] += 2500;
|
|
|
|
sf->thresh_mult[THR_D207_PRED] += 2500;
|
|
|
|
sf->thresh_mult[THR_D63_PRED] += 2500;
|
2013-02-04 21:43:02 +04:00
|
|
|
|
2013-06-20 02:53:47 +04:00
|
|
|
if (cpi->sf.skip_lots_of_modes) {
|
2013-05-31 02:13:08 +04:00
|
|
|
for (i = 0; i < MAX_MODES; ++i)
|
|
|
|
sf->thresh_mult[i] = INT_MAX;
|
|
|
|
|
2013-07-19 17:04:53 +04:00
|
|
|
sf->thresh_mult[THR_DC] = 2000;
|
|
|
|
sf->thresh_mult[THR_TM] = 2000;
|
2013-06-19 23:16:45 +04:00
|
|
|
sf->thresh_mult[THR_NEWMV] = 4000;
|
|
|
|
sf->thresh_mult[THR_NEWG] = 4000;
|
|
|
|
sf->thresh_mult[THR_NEWA] = 4000;
|
2013-05-31 02:13:08 +04:00
|
|
|
sf->thresh_mult[THR_NEARESTMV] = 0;
|
2013-06-19 23:16:45 +04:00
|
|
|
sf->thresh_mult[THR_NEARESTG] = 0;
|
|
|
|
sf->thresh_mult[THR_NEARESTA] = 0;
|
|
|
|
sf->thresh_mult[THR_NEARMV] = 2000;
|
|
|
|
sf->thresh_mult[THR_NEARG] = 2000;
|
|
|
|
sf->thresh_mult[THR_NEARA] = 2000;
|
2013-05-31 02:13:08 +04:00
|
|
|
sf->thresh_mult[THR_COMP_NEARESTLA] = 2000;
|
2013-06-19 23:16:45 +04:00
|
|
|
sf->thresh_mult[THR_SPLITMV] = 2500;
|
|
|
|
sf->thresh_mult[THR_SPLITG] = 2500;
|
|
|
|
sf->thresh_mult[THR_SPLITA] = 2500;
|
2013-05-31 02:13:08 +04:00
|
|
|
sf->recode_loop = 0;
|
|
|
|
}
|
|
|
|
|
2013-02-05 14:13:25 +04:00
|
|
|
/* disable frame modes if flags not set */
|
|
|
|
if (!(cpi->ref_frame_flags & VP9_LAST_FLAG)) {
|
|
|
|
sf->thresh_mult[THR_NEWMV ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_NEARESTMV] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_ZEROMV ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_NEARMV ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_SPLITMV ] = INT_MAX;
|
|
|
|
}
|
|
|
|
if (!(cpi->ref_frame_flags & VP9_GOLD_FLAG)) {
|
|
|
|
sf->thresh_mult[THR_NEARESTG ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_ZEROG ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_NEARG ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_NEWG ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_SPLITG ] = INT_MAX;
|
|
|
|
}
|
|
|
|
if (!(cpi->ref_frame_flags & VP9_ALT_FLAG)) {
|
|
|
|
sf->thresh_mult[THR_NEARESTA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_ZEROA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_NEARA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_NEWA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_SPLITA ] = INT_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((cpi->ref_frame_flags & (VP9_LAST_FLAG | VP9_ALT_FLAG)) !=
|
|
|
|
(VP9_LAST_FLAG | VP9_ALT_FLAG)) {
|
|
|
|
sf->thresh_mult[THR_COMP_ZEROLA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_NEARESTLA] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_NEARLA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_NEWLA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_SPLITLA ] = INT_MAX;
|
|
|
|
}
|
|
|
|
if ((cpi->ref_frame_flags & (VP9_GOLD_FLAG | VP9_ALT_FLAG)) !=
|
|
|
|
(VP9_GOLD_FLAG | VP9_ALT_FLAG)) {
|
|
|
|
sf->thresh_mult[THR_COMP_ZEROGA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_NEARESTGA] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_NEARGA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_NEWGA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_SPLITGA ] = INT_MAX;
|
|
|
|
}
|
2013-06-30 04:34:51 +04:00
|
|
|
|
|
|
|
if (sf->disable_splitmv == 1) {
|
|
|
|
sf->thresh_mult[THR_SPLITMV ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_SPLITG ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_SPLITA ] = INT_MAX;
|
|
|
|
|
|
|
|
sf->thresh_mult[THR_COMP_SPLITLA ] = INT_MAX;
|
|
|
|
sf->thresh_mult[THR_COMP_SPLITGA ] = INT_MAX;
|
|
|
|
}
|
2013-02-04 21:43:02 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
void vp9_set_speed_features(VP9_COMP *cpi) {
|
2012-07-14 02:21:29 +04:00
|
|
|
SPEED_FEATURES *sf = &cpi->sf;
|
2013-02-04 21:43:02 +04:00
|
|
|
int mode = cpi->compressor_speed;
|
2013-05-09 01:35:42 +04:00
|
|
|
int speed = cpi->speed;
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
2013-07-02 05:18:50 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Only modes 0 and 1 supported for now in experimental code basae
|
2013-02-04 21:43:02 +04:00
|
|
|
if (mode > 1)
|
|
|
|
mode = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Initialise default mode frequency sampling variables
|
|
|
|
for (i = 0; i < MAX_MODES; i ++) {
|
|
|
|
cpi->mode_check_freq[i] = 0;
|
|
|
|
cpi->mode_test_hit_counts[i] = 0;
|
|
|
|
cpi->mode_chosen_counts[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// best quality defaults
|
|
|
|
sf->RD = 1;
|
|
|
|
sf->search_method = NSTEP;
|
|
|
|
sf->auto_filter = 1;
|
|
|
|
sf->recode_loop = 1;
|
2013-08-08 04:01:43 +04:00
|
|
|
sf->subpel_search_method = SUBPEL_TREE;
|
|
|
|
sf->subpel_iters_per_step = 2;
|
2013-04-02 21:24:56 +04:00
|
|
|
sf->optimize_coefficients = !cpi->oxcf.lossless;
|
2013-06-24 18:19:16 +04:00
|
|
|
sf->reduce_first_step_size = 0;
|
2013-06-26 20:06:25 +04:00
|
|
|
sf->auto_mv_step_size = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
sf->max_step_search_steps = MAX_MVSEARCH_STEPS;
|
2013-08-02 22:45:21 +04:00
|
|
|
sf->comp_inter_joint_search_thresh = BLOCK_4X4;
|
2013-07-02 15:34:41 +04:00
|
|
|
sf->adaptive_rd_thresh = 0;
|
2013-06-19 23:16:45 +04:00
|
|
|
sf->use_lastframe_partitioning = 0;
|
Tx size selection enhancements
(1) Refines the modeling function and uses that to add some speed
features. Specifically, intead of using a flag use_largest_txfm as
a speed feature, an enum tx_size_search_method is used, of which
two of the types are USE_FULL_RD and USE_LARGESTALL. Two other
new types are added:
USE_LARGESTINTRA (use largest only for intra)
USE_LARGESTINTRA_MODELINTER (use largest for intra, and model for
inter)
(2) Another change is that the framework for deciding transform type
is simplified to use a heuristic count based method rather than
an rd based method using txfm_cache. In practice the new method
is found to work just as well - with derf only -0.01 down.
The new method is more compatible with the new framework where
certain rd costs are based on full rd and certain others are
based on modeled rd or are not computed. In this patch the existing
rd based method is still kept for use in the USE_FULL_RD mode.
In the other modes, the count based method is used.
However the recommendation is to remove it eventually since the
benefit is limited, and will remove a lot of complications in
the code
(3) Finally a bug is fixed with the existing use_largest_txfm speed feature
that causes mismatches when the lossless mode and 4x4 WH transform is
forced.
Results on derf:
USE_FULL_RD: +0.03% (due to change in the tables), 0% encode time reduction
USE_LARGESTINTRA: -0.21%, 15% encode time reduction (this one is a
pretty good compromise)
USE_LARGESTINTRA_MODELINTER: -0.98%, 22% encode time reduction
(currently the benefit of modeling is limited for txfm size selection,
but keeping this enum as a placeholder) .
USE_LARGESTALL: -1.05%, 27% encode-time reduction (same as existing
use_largest_txfm speed feature).
Change-Id: I4d60a5f9ce78fbc90cddf2f97ed91d8bc0d4f936
2013-06-22 03:31:12 +04:00
|
|
|
sf->tx_size_search_method = USE_FULL_RD;
|
2013-08-08 02:22:51 +04:00
|
|
|
sf->use_lp32x32fdct = 0;
|
2013-09-12 21:06:47 +04:00
|
|
|
sf->adaptive_motion_search = 0;
|
2013-06-20 02:53:47 +04:00
|
|
|
sf->use_avoid_tested_higherror = 0;
|
2013-07-01 19:27:12 +04:00
|
|
|
sf->reference_masking = 0;
|
2013-06-20 02:53:47 +04:00
|
|
|
sf->skip_lots_of_modes = 0;
|
2013-06-20 18:17:01 +04:00
|
|
|
sf->partition_by_variance = 0;
|
2013-06-20 18:46:51 +04:00
|
|
|
sf->use_one_partition_size_always = 0;
|
2013-06-27 23:07:07 +04:00
|
|
|
sf->less_rectangular_check = 0;
|
2013-06-27 23:07:07 +04:00
|
|
|
sf->use_square_partition_only = 0;
|
2013-07-24 17:07:37 +04:00
|
|
|
sf->auto_min_max_partition_size = 0;
|
|
|
|
sf->auto_min_max_partition_interval = 0;
|
|
|
|
sf->auto_min_max_partition_count = 0;
|
|
|
|
sf->max_partition_size = BLOCK_64X64;
|
|
|
|
sf->min_partition_size = BLOCK_4X4;
|
2013-07-02 05:18:50 +04:00
|
|
|
sf->adjust_partitioning_from_last_frame = 0;
|
|
|
|
sf->last_partitioning_redo_frequency = 4;
|
2013-06-30 04:34:51 +04:00
|
|
|
sf->disable_splitmv = 0;
|
2013-07-04 01:47:54 +04:00
|
|
|
sf->mode_search_skip_flags = 0;
|
Speed feature to skip split partition based on var
Adds a speed feature to disable split partition search based on a
given threshold on the source variance. A tighter threshold derived
from the threshold provided is used to also disable horizontal and
vertical partitions.
Results on derfraw300:
threshold = 16, psnr = -0.057%, speedup ~1% (football)
threshold = 32, psnr = -0.150%, speedup ~4-5% (football)
threshold = 64, psnr = -0.570%, speedup ~10-12% (football)
Results on stdhdraw250:
threshold = 32, psnr = -0.18%, speedup is somewhat more than derf
because of a larger number of smoother blocks at higher resolution.
Based on these results, a threshold of 32 is chosen for speed 1,
and a threshold of 64 is chosen for speeds 2 and above.
Change-Id: If08912fb6c67fd4242d12a0d094783a99f52f6c6
2013-08-03 04:15:38 +04:00
|
|
|
sf->disable_split_var_thresh = 0;
|
Cleanup/enhancements of switchable filter search
Cleans up the switchable filter search logic. Also adds a
speed feature - a variance threshold - to disable filter search
if source variance is lower than this value.
Results: derfraw300
threshold = 16, psnr -0.238%, 4-5% speedup (tested on football)
threshold = 32, psnr -0.381%, 8-9% speedup (tested on football)
threshold = 64, psnr -0.611%, 12-13% speedup (tested on football)
threshold = 96, psnr -0.804%, 16-17% speedup (tested on football)
Based on these results, the threshold is chosen as 16 for speed 1,
32 for speed 2, 64 for speed 3 and 96 for speed 4.
Change-Id: Ib630d39192773b1983d3d349b97973768e170c04
2013-08-17 00:51:00 +04:00
|
|
|
sf->disable_filter_search_var_thresh = 0;
|
2013-08-22 20:23:02 +04:00
|
|
|
sf->intra_y_mode_mask = ALL_INTRA_MODES;
|
|
|
|
sf->intra_uv_mode_mask = ALL_INTRA_MODES;
|
2013-07-09 03:01:01 +04:00
|
|
|
sf->use_rd_breakout = 0;
|
2013-07-09 03:48:47 +04:00
|
|
|
sf->skip_encode_sb = 0;
|
2013-07-16 21:12:34 +04:00
|
|
|
sf->use_uv_intra_rd_estimate = 0;
|
2013-08-07 02:46:26 +04:00
|
|
|
sf->use_fast_lpf_pick = 0;
|
2013-08-28 02:07:50 +04:00
|
|
|
sf->use_fast_coef_updates = 0;
|
2013-07-04 01:43:23 +04:00
|
|
|
sf->using_small_partition_info = 0;
|
2013-09-05 04:15:05 +04:00
|
|
|
sf->mode_skip_start = MAX_MODES; // Mode index at which mode skip mask set
|
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
// Switch segmentation off.
|
|
|
|
sf->static_segmentation = 0;
|
|
|
|
#else
|
2013-04-28 02:49:49 +04:00
|
|
|
sf->static_segmentation = 0;
|
2013-04-03 02:08:50 +04:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-02-04 21:43:02 +04:00
|
|
|
switch (mode) {
|
2010-05-18 19:58:33 +04:00
|
|
|
case 0: // best quality mode
|
2012-07-14 02:21:29 +04:00
|
|
|
break;
|
|
|
|
|
2013-02-04 21:43:02 +04:00
|
|
|
case 1:
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
// Switch segmentation off.
|
|
|
|
sf->static_segmentation = 0;
|
2013-04-23 17:01:55 +04:00
|
|
|
#else
|
2013-06-07 16:27:08 +04:00
|
|
|
sf->static_segmentation = 0;
|
2013-04-03 02:08:50 +04:00
|
|
|
#endif
|
2013-07-02 15:34:41 +04:00
|
|
|
sf->use_avoid_tested_higherror = 1;
|
2013-08-23 17:00:54 +04:00
|
|
|
sf->adaptive_rd_thresh = MIN((speed + 1), 4);
|
2013-08-16 19:54:12 +04:00
|
|
|
|
2013-06-21 02:23:37 +04:00
|
|
|
if (speed == 1) {
|
2013-08-10 04:47:32 +04:00
|
|
|
sf->comp_inter_joint_search_thresh = BLOCK_SIZES;
|
2013-06-27 23:07:07 +04:00
|
|
|
sf->less_rectangular_check = 1;
|
Tx size selection enhancements
(1) Refines the modeling function and uses that to add some speed
features. Specifically, intead of using a flag use_largest_txfm as
a speed feature, an enum tx_size_search_method is used, of which
two of the types are USE_FULL_RD and USE_LARGESTALL. Two other
new types are added:
USE_LARGESTINTRA (use largest only for intra)
USE_LARGESTINTRA_MODELINTER (use largest for intra, and model for
inter)
(2) Another change is that the framework for deciding transform type
is simplified to use a heuristic count based method rather than
an rd based method using txfm_cache. In practice the new method
is found to work just as well - with derf only -0.01 down.
The new method is more compatible with the new framework where
certain rd costs are based on full rd and certain others are
based on modeled rd or are not computed. In this patch the existing
rd based method is still kept for use in the USE_FULL_RD mode.
In the other modes, the count based method is used.
However the recommendation is to remove it eventually since the
benefit is limited, and will remove a lot of complications in
the code
(3) Finally a bug is fixed with the existing use_largest_txfm speed feature
that causes mismatches when the lossless mode and 4x4 WH transform is
forced.
Results on derf:
USE_FULL_RD: +0.03% (due to change in the tables), 0% encode time reduction
USE_LARGESTINTRA: -0.21%, 15% encode time reduction (this one is a
pretty good compromise)
USE_LARGESTINTRA_MODELINTER: -0.98%, 22% encode time reduction
(currently the benefit of modeling is limited for txfm size selection,
but keeping this enum as a placeholder) .
USE_LARGESTALL: -1.05%, 27% encode-time reduction (same as existing
use_largest_txfm speed feature).
Change-Id: I4d60a5f9ce78fbc90cddf2f97ed91d8bc0d4f936
2013-06-22 03:31:12 +04:00
|
|
|
sf->tx_size_search_method = ((cpi->common.frame_type == KEY_FRAME ||
|
|
|
|
cpi->common.intra_only ||
|
|
|
|
cpi->common.show_frame == 0) ?
|
|
|
|
USE_FULL_RD :
|
2013-06-27 23:07:07 +04:00
|
|
|
USE_LARGESTALL);
|
|
|
|
sf->use_square_partition_only = !(cpi->common.frame_type == KEY_FRAME ||
|
|
|
|
cpi->common.intra_only ||
|
|
|
|
cpi->common.show_frame == 0);
|
2013-07-04 01:47:54 +04:00
|
|
|
sf->disable_splitmv =
|
|
|
|
(MIN(cpi->common.width, cpi->common.height) >= 720)? 1 : 0;
|
|
|
|
sf->mode_search_skip_flags = FLAG_SKIP_INTRA_DIRMISMATCH |
|
|
|
|
FLAG_SKIP_INTRA_BESTINTER |
|
2013-07-31 20:33:58 +04:00
|
|
|
FLAG_SKIP_COMP_BESTINTRA |
|
|
|
|
FLAG_SKIP_INTRA_LOWVAR;
|
2013-07-24 17:07:37 +04:00
|
|
|
sf->use_uv_intra_rd_estimate = 1;
|
2013-07-09 03:01:01 +04:00
|
|
|
sf->use_rd_breakout = 1;
|
2013-07-09 03:48:47 +04:00
|
|
|
sf->skip_encode_sb = 1;
|
2013-08-08 02:22:51 +04:00
|
|
|
sf->use_lp32x32fdct = 1;
|
2013-09-12 21:06:47 +04:00
|
|
|
sf->adaptive_motion_search = 1;
|
2013-07-12 20:52:24 +04:00
|
|
|
sf->auto_mv_step_size = 1;
|
2013-07-24 17:07:37 +04:00
|
|
|
|
|
|
|
sf->auto_min_max_partition_size = 1;
|
|
|
|
sf->auto_min_max_partition_interval = 1;
|
Refactor rd_pick_partition for parameter control
This commit changes the partition search order of superblocks from
{SPLIT, NONE, HORZ, VERT} to {NONE, SPLIT, HORZ, VERT} for
consistency with that of sub8x8 partition search. It enable the use
of early termination in partition search for all block sizes.
For ped_area_1080p 50 frames coded at 4000 kbps, it makes the runtime
goes down from 844305ms -> 818003ms (3% speed-up) at speed 0.
This will further move towards making the in-search partition types
configurable, hence unifying various speed-up approaches.
Some speed 1 and 2 features are turned off during the refactoring
process, including:
disable_split_var_thresh
using_small_partition_info
Stricter constraints are applied to use_square_partition_only for
right/bottom boundary blocks. Will bring back/refine these features
subsequently. At this point, it makes derf set at speed 1 about
0.45% higher in compression performance, and 9% down in run-time.
Change-Id: I3db9f9d1d1a0d6cbe2e50e49bd9eda1cf705f37c
2013-08-21 01:34:17 +04:00
|
|
|
// FIXME(jingning): temporarily turn off disable_split_var_thresh
|
|
|
|
// during refactoring process. will get this back after finishing
|
|
|
|
// the main framework of partition search type.
|
|
|
|
sf->disable_split_var_thresh = 0;
|
Cleanup/enhancements of switchable filter search
Cleans up the switchable filter search logic. Also adds a
speed feature - a variance threshold - to disable filter search
if source variance is lower than this value.
Results: derfraw300
threshold = 16, psnr -0.238%, 4-5% speedup (tested on football)
threshold = 32, psnr -0.381%, 8-9% speedup (tested on football)
threshold = 64, psnr -0.611%, 12-13% speedup (tested on football)
threshold = 96, psnr -0.804%, 16-17% speedup (tested on football)
Based on these results, the threshold is chosen as 16 for speed 1,
32 for speed 2, 64 for speed 3 and 96 for speed 4.
Change-Id: Ib630d39192773b1983d3d349b97973768e170c04
2013-08-17 00:51:00 +04:00
|
|
|
sf->disable_filter_search_var_thresh = 16;
|
2013-08-22 20:23:02 +04:00
|
|
|
|
|
|
|
sf->intra_y_mode_mask = INTRA_DC_TM_H_V;
|
|
|
|
sf->intra_uv_mode_mask = INTRA_DC_TM_H_V;
|
2013-08-28 02:07:50 +04:00
|
|
|
sf->use_fast_coef_updates = 1;
|
2013-09-05 04:15:05 +04:00
|
|
|
sf->mode_skip_start = 9;
|
2013-05-31 17:47:40 +04:00
|
|
|
}
|
2013-06-20 02:53:47 +04:00
|
|
|
if (speed == 2) {
|
2013-06-27 23:07:07 +04:00
|
|
|
sf->less_rectangular_check = 1;
|
2013-06-27 23:07:07 +04:00
|
|
|
sf->use_square_partition_only = 1;
|
2013-08-10 04:47:32 +04:00
|
|
|
sf->comp_inter_joint_search_thresh = BLOCK_SIZES;
|
2013-06-19 23:16:45 +04:00
|
|
|
sf->use_lastframe_partitioning = 1;
|
2013-07-02 05:18:50 +04:00
|
|
|
sf->adjust_partitioning_from_last_frame = 1;
|
|
|
|
sf->last_partitioning_redo_frequency = 3;
|
2013-07-04 01:47:54 +04:00
|
|
|
sf->tx_size_search_method = ((cpi->common.frame_type == KEY_FRAME ||
|
|
|
|
cpi->common.intra_only ||
|
|
|
|
cpi->common.show_frame == 0) ?
|
|
|
|
USE_FULL_RD :
|
|
|
|
USE_LARGESTALL);
|
|
|
|
sf->mode_search_skip_flags = FLAG_SKIP_INTRA_DIRMISMATCH |
|
|
|
|
FLAG_SKIP_INTRA_BESTINTER |
|
|
|
|
FLAG_SKIP_COMP_BESTINTRA |
|
2013-07-31 20:33:58 +04:00
|
|
|
FLAG_SKIP_COMP_REFMISMATCH |
|
|
|
|
FLAG_SKIP_INTRA_LOWVAR |
|
|
|
|
FLAG_EARLY_TERMINATE;
|
2013-08-22 20:23:02 +04:00
|
|
|
sf->intra_y_mode_mask = INTRA_DC_TM;
|
|
|
|
sf->intra_uv_mode_mask = INTRA_DC_TM;
|
2013-07-24 17:07:37 +04:00
|
|
|
sf->use_uv_intra_rd_estimate = 1;
|
2013-07-09 03:01:01 +04:00
|
|
|
sf->use_rd_breakout = 1;
|
2013-07-09 03:48:47 +04:00
|
|
|
sf->skip_encode_sb = 1;
|
2013-08-08 02:22:51 +04:00
|
|
|
sf->use_lp32x32fdct = 1;
|
2013-09-12 21:06:47 +04:00
|
|
|
sf->adaptive_motion_search = 1;
|
Refactor rd_pick_partition for parameter control
This commit changes the partition search order of superblocks from
{SPLIT, NONE, HORZ, VERT} to {NONE, SPLIT, HORZ, VERT} for
consistency with that of sub8x8 partition search. It enable the use
of early termination in partition search for all block sizes.
For ped_area_1080p 50 frames coded at 4000 kbps, it makes the runtime
goes down from 844305ms -> 818003ms (3% speed-up) at speed 0.
This will further move towards making the in-search partition types
configurable, hence unifying various speed-up approaches.
Some speed 1 and 2 features are turned off during the refactoring
process, including:
disable_split_var_thresh
using_small_partition_info
Stricter constraints are applied to use_square_partition_only for
right/bottom boundary blocks. Will bring back/refine these features
subsequently. At this point, it makes derf set at speed 1 about
0.45% higher in compression performance, and 9% down in run-time.
Change-Id: I3db9f9d1d1a0d6cbe2e50e49bd9eda1cf705f37c
2013-08-21 01:34:17 +04:00
|
|
|
sf->using_small_partition_info = 0;
|
2013-07-17 20:37:14 +04:00
|
|
|
sf->disable_splitmv =
|
|
|
|
(MIN(cpi->common.width, cpi->common.height) >= 720)? 1 : 0;
|
2013-07-12 20:52:24 +04:00
|
|
|
sf->auto_mv_step_size = 1;
|
2013-07-23 01:47:57 +04:00
|
|
|
sf->search_method = SQUARE;
|
2013-08-08 04:01:43 +04:00
|
|
|
sf->subpel_iters_per_step = 1;
|
2013-08-07 02:46:26 +04:00
|
|
|
sf->use_fast_lpf_pick = 1;
|
2013-08-16 19:54:12 +04:00
|
|
|
sf->auto_min_max_partition_size = 1;
|
|
|
|
sf->auto_min_max_partition_interval = 2;
|
Fixes on feature disabling split based on variance
Adds a couple of minor fixes, which may be absorbed in Jingning's
patch. Thanks to Guillaume for pointing these out.
Also adjusts the thresholds for speed 1 and 2 to 16 and 32
respectively, to keep quality drops small.
Results:
--------
derfraw300: threshold = 16, psnr -0.082%, speedup 2-3%
threshold = 32, psnr -0.218%, speedup 5-6%
stdhdraw250: threshold = 16, psnr -0.031%, speedup 2-3%
threshold = 32, psnr -0.273%, speedup 5-6%
Change-Id: I4b11ae8296cca6c2a9f644be7e40de7c423b8330
2013-08-22 03:19:35 +04:00
|
|
|
sf->disable_split_var_thresh = 32;
|
Cleanup/enhancements of switchable filter search
Cleans up the switchable filter search logic. Also adds a
speed feature - a variance threshold - to disable filter search
if source variance is lower than this value.
Results: derfraw300
threshold = 16, psnr -0.238%, 4-5% speedup (tested on football)
threshold = 32, psnr -0.381%, 8-9% speedup (tested on football)
threshold = 64, psnr -0.611%, 12-13% speedup (tested on football)
threshold = 96, psnr -0.804%, 16-17% speedup (tested on football)
Based on these results, the threshold is chosen as 16 for speed 1,
32 for speed 2, 64 for speed 3 and 96 for speed 4.
Change-Id: Ib630d39192773b1983d3d349b97973768e170c04
2013-08-17 00:51:00 +04:00
|
|
|
sf->disable_filter_search_var_thresh = 32;
|
2013-08-28 02:07:50 +04:00
|
|
|
sf->use_fast_coef_updates = 2;
|
2013-09-05 04:15:05 +04:00
|
|
|
sf->mode_skip_start = 9;
|
2013-06-20 02:53:47 +04:00
|
|
|
}
|
2013-06-20 18:17:01 +04:00
|
|
|
if (speed == 3) {
|
2013-08-10 04:47:32 +04:00
|
|
|
sf->comp_inter_joint_search_thresh = BLOCK_SIZES;
|
2013-06-20 18:17:01 +04:00
|
|
|
sf->partition_by_variance = 1;
|
Tx size selection enhancements
(1) Refines the modeling function and uses that to add some speed
features. Specifically, intead of using a flag use_largest_txfm as
a speed feature, an enum tx_size_search_method is used, of which
two of the types are USE_FULL_RD and USE_LARGESTALL. Two other
new types are added:
USE_LARGESTINTRA (use largest only for intra)
USE_LARGESTINTRA_MODELINTER (use largest for intra, and model for
inter)
(2) Another change is that the framework for deciding transform type
is simplified to use a heuristic count based method rather than
an rd based method using txfm_cache. In practice the new method
is found to work just as well - with derf only -0.01 down.
The new method is more compatible with the new framework where
certain rd costs are based on full rd and certain others are
based on modeled rd or are not computed. In this patch the existing
rd based method is still kept for use in the USE_FULL_RD mode.
In the other modes, the count based method is used.
However the recommendation is to remove it eventually since the
benefit is limited, and will remove a lot of complications in
the code
(3) Finally a bug is fixed with the existing use_largest_txfm speed feature
that causes mismatches when the lossless mode and 4x4 WH transform is
forced.
Results on derf:
USE_FULL_RD: +0.03% (due to change in the tables), 0% encode time reduction
USE_LARGESTINTRA: -0.21%, 15% encode time reduction (this one is a
pretty good compromise)
USE_LARGESTINTRA_MODELINTER: -0.98%, 22% encode time reduction
(currently the benefit of modeling is limited for txfm size selection,
but keeping this enum as a placeholder) .
USE_LARGESTALL: -1.05%, 27% encode-time reduction (same as existing
use_largest_txfm speed feature).
Change-Id: I4d60a5f9ce78fbc90cddf2f97ed91d8bc0d4f936
2013-06-22 03:31:12 +04:00
|
|
|
sf->tx_size_search_method = ((cpi->common.frame_type == KEY_FRAME ||
|
|
|
|
cpi->common.intra_only ||
|
|
|
|
cpi->common.show_frame == 0) ?
|
|
|
|
USE_FULL_RD :
|
|
|
|
USE_LARGESTALL);
|
2013-07-04 01:47:54 +04:00
|
|
|
sf->mode_search_skip_flags = FLAG_SKIP_INTRA_DIRMISMATCH |
|
|
|
|
FLAG_SKIP_INTRA_BESTINTER |
|
|
|
|
FLAG_SKIP_COMP_BESTINTRA |
|
2013-07-31 20:33:58 +04:00
|
|
|
FLAG_SKIP_COMP_REFMISMATCH |
|
|
|
|
FLAG_SKIP_INTRA_LOWVAR |
|
|
|
|
FLAG_EARLY_TERMINATE;
|
2013-07-09 03:01:01 +04:00
|
|
|
sf->use_rd_breakout = 1;
|
2013-07-09 03:48:47 +04:00
|
|
|
sf->skip_encode_sb = 1;
|
2013-08-08 02:22:51 +04:00
|
|
|
sf->use_lp32x32fdct = 1;
|
2013-07-17 20:37:14 +04:00
|
|
|
sf->disable_splitmv = 1;
|
2013-07-12 20:52:24 +04:00
|
|
|
sf->auto_mv_step_size = 1;
|
2013-07-23 01:47:57 +04:00
|
|
|
sf->search_method = BIGDIA;
|
2013-08-07 02:53:35 +04:00
|
|
|
sf->subpel_iters_per_step = 1;
|
Speed feature to skip split partition based on var
Adds a speed feature to disable split partition search based on a
given threshold on the source variance. A tighter threshold derived
from the threshold provided is used to also disable horizontal and
vertical partitions.
Results on derfraw300:
threshold = 16, psnr = -0.057%, speedup ~1% (football)
threshold = 32, psnr = -0.150%, speedup ~4-5% (football)
threshold = 64, psnr = -0.570%, speedup ~10-12% (football)
Results on stdhdraw250:
threshold = 32, psnr = -0.18%, speedup is somewhat more than derf
because of a larger number of smoother blocks at higher resolution.
Based on these results, a threshold of 32 is chosen for speed 1,
and a threshold of 64 is chosen for speeds 2 and above.
Change-Id: If08912fb6c67fd4242d12a0d094783a99f52f6c6
2013-08-03 04:15:38 +04:00
|
|
|
sf->disable_split_var_thresh = 64;
|
Cleanup/enhancements of switchable filter search
Cleans up the switchable filter search logic. Also adds a
speed feature - a variance threshold - to disable filter search
if source variance is lower than this value.
Results: derfraw300
threshold = 16, psnr -0.238%, 4-5% speedup (tested on football)
threshold = 32, psnr -0.381%, 8-9% speedup (tested on football)
threshold = 64, psnr -0.611%, 12-13% speedup (tested on football)
threshold = 96, psnr -0.804%, 16-17% speedup (tested on football)
Based on these results, the threshold is chosen as 16 for speed 1,
32 for speed 2, 64 for speed 3 and 96 for speed 4.
Change-Id: Ib630d39192773b1983d3d349b97973768e170c04
2013-08-17 00:51:00 +04:00
|
|
|
sf->disable_filter_search_var_thresh = 64;
|
2013-08-22 20:23:02 +04:00
|
|
|
sf->intra_y_mode_mask = INTRA_DC_ONLY;
|
|
|
|
sf->intra_uv_mode_mask = INTRA_DC_ONLY;
|
2013-08-28 02:07:50 +04:00
|
|
|
sf->use_fast_coef_updates = 2;
|
2013-09-05 04:15:05 +04:00
|
|
|
sf->mode_skip_start = 9;
|
2013-06-20 18:17:01 +04:00
|
|
|
}
|
2013-06-20 18:46:51 +04:00
|
|
|
if (speed == 4) {
|
2013-08-10 04:47:32 +04:00
|
|
|
sf->comp_inter_joint_search_thresh = BLOCK_SIZES;
|
2013-06-20 18:46:51 +04:00
|
|
|
sf->use_one_partition_size_always = 1;
|
2013-08-02 22:45:21 +04:00
|
|
|
sf->always_this_block_size = BLOCK_16X16;
|
Tx size selection enhancements
(1) Refines the modeling function and uses that to add some speed
features. Specifically, intead of using a flag use_largest_txfm as
a speed feature, an enum tx_size_search_method is used, of which
two of the types are USE_FULL_RD and USE_LARGESTALL. Two other
new types are added:
USE_LARGESTINTRA (use largest only for intra)
USE_LARGESTINTRA_MODELINTER (use largest for intra, and model for
inter)
(2) Another change is that the framework for deciding transform type
is simplified to use a heuristic count based method rather than
an rd based method using txfm_cache. In practice the new method
is found to work just as well - with derf only -0.01 down.
The new method is more compatible with the new framework where
certain rd costs are based on full rd and certain others are
based on modeled rd or are not computed. In this patch the existing
rd based method is still kept for use in the USE_FULL_RD mode.
In the other modes, the count based method is used.
However the recommendation is to remove it eventually since the
benefit is limited, and will remove a lot of complications in
the code
(3) Finally a bug is fixed with the existing use_largest_txfm speed feature
that causes mismatches when the lossless mode and 4x4 WH transform is
forced.
Results on derf:
USE_FULL_RD: +0.03% (due to change in the tables), 0% encode time reduction
USE_LARGESTINTRA: -0.21%, 15% encode time reduction (this one is a
pretty good compromise)
USE_LARGESTINTRA_MODELINTER: -0.98%, 22% encode time reduction
(currently the benefit of modeling is limited for txfm size selection,
but keeping this enum as a placeholder) .
USE_LARGESTALL: -1.05%, 27% encode-time reduction (same as existing
use_largest_txfm speed feature).
Change-Id: I4d60a5f9ce78fbc90cddf2f97ed91d8bc0d4f936
2013-06-22 03:31:12 +04:00
|
|
|
sf->tx_size_search_method = ((cpi->common.frame_type == KEY_FRAME ||
|
|
|
|
cpi->common.intra_only ||
|
|
|
|
cpi->common.show_frame == 0) ?
|
|
|
|
USE_FULL_RD :
|
|
|
|
USE_LARGESTALL);
|
2013-07-04 01:47:54 +04:00
|
|
|
sf->mode_search_skip_flags = FLAG_SKIP_INTRA_DIRMISMATCH |
|
|
|
|
FLAG_SKIP_INTRA_BESTINTER |
|
|
|
|
FLAG_SKIP_COMP_BESTINTRA |
|
2013-07-31 20:33:58 +04:00
|
|
|
FLAG_SKIP_COMP_REFMISMATCH |
|
|
|
|
FLAG_SKIP_INTRA_LOWVAR |
|
|
|
|
FLAG_EARLY_TERMINATE;
|
2013-07-09 03:01:01 +04:00
|
|
|
sf->use_rd_breakout = 1;
|
2013-08-08 02:22:51 +04:00
|
|
|
sf->use_lp32x32fdct = 1;
|
2013-07-03 20:54:06 +04:00
|
|
|
sf->optimize_coefficients = 0;
|
2013-07-12 20:52:24 +04:00
|
|
|
sf->auto_mv_step_size = 1;
|
2013-07-03 20:54:06 +04:00
|
|
|
// sf->reduce_first_step_size = 1;
|
|
|
|
// sf->reference_masking = 1;
|
2013-07-17 20:37:14 +04:00
|
|
|
|
|
|
|
sf->disable_splitmv = 1;
|
2013-07-23 01:47:57 +04:00
|
|
|
sf->search_method = HEX;
|
2013-08-07 02:53:35 +04:00
|
|
|
sf->subpel_iters_per_step = 1;
|
Speed feature to skip split partition based on var
Adds a speed feature to disable split partition search based on a
given threshold on the source variance. A tighter threshold derived
from the threshold provided is used to also disable horizontal and
vertical partitions.
Results on derfraw300:
threshold = 16, psnr = -0.057%, speedup ~1% (football)
threshold = 32, psnr = -0.150%, speedup ~4-5% (football)
threshold = 64, psnr = -0.570%, speedup ~10-12% (football)
Results on stdhdraw250:
threshold = 32, psnr = -0.18%, speedup is somewhat more than derf
because of a larger number of smoother blocks at higher resolution.
Based on these results, a threshold of 32 is chosen for speed 1,
and a threshold of 64 is chosen for speeds 2 and above.
Change-Id: If08912fb6c67fd4242d12a0d094783a99f52f6c6
2013-08-03 04:15:38 +04:00
|
|
|
sf->disable_split_var_thresh = 64;
|
Cleanup/enhancements of switchable filter search
Cleans up the switchable filter search logic. Also adds a
speed feature - a variance threshold - to disable filter search
if source variance is lower than this value.
Results: derfraw300
threshold = 16, psnr -0.238%, 4-5% speedup (tested on football)
threshold = 32, psnr -0.381%, 8-9% speedup (tested on football)
threshold = 64, psnr -0.611%, 12-13% speedup (tested on football)
threshold = 96, psnr -0.804%, 16-17% speedup (tested on football)
Based on these results, the threshold is chosen as 16 for speed 1,
32 for speed 2, 64 for speed 3 and 96 for speed 4.
Change-Id: Ib630d39192773b1983d3d349b97973768e170c04
2013-08-17 00:51:00 +04:00
|
|
|
sf->disable_filter_search_var_thresh = 96;
|
2013-08-30 03:29:53 +04:00
|
|
|
sf->intra_y_mode_mask = INTRA_DC_ONLY;
|
|
|
|
sf->intra_uv_mode_mask = INTRA_DC_ONLY;
|
2013-08-28 02:07:50 +04:00
|
|
|
sf->use_fast_coef_updates = 2;
|
2013-09-05 04:15:05 +04:00
|
|
|
sf->mode_skip_start = 9;
|
2013-06-20 18:46:51 +04:00
|
|
|
}
|
Tx size selection enhancements
(1) Refines the modeling function and uses that to add some speed
features. Specifically, intead of using a flag use_largest_txfm as
a speed feature, an enum tx_size_search_method is used, of which
two of the types are USE_FULL_RD and USE_LARGESTALL. Two other
new types are added:
USE_LARGESTINTRA (use largest only for intra)
USE_LARGESTINTRA_MODELINTER (use largest for intra, and model for
inter)
(2) Another change is that the framework for deciding transform type
is simplified to use a heuristic count based method rather than
an rd based method using txfm_cache. In practice the new method
is found to work just as well - with derf only -0.01 down.
The new method is more compatible with the new framework where
certain rd costs are based on full rd and certain others are
based on modeled rd or are not computed. In this patch the existing
rd based method is still kept for use in the USE_FULL_RD mode.
In the other modes, the count based method is used.
However the recommendation is to remove it eventually since the
benefit is limited, and will remove a lot of complications in
the code
(3) Finally a bug is fixed with the existing use_largest_txfm speed feature
that causes mismatches when the lossless mode and 4x4 WH transform is
forced.
Results on derf:
USE_FULL_RD: +0.03% (due to change in the tables), 0% encode time reduction
USE_LARGESTINTRA: -0.21%, 15% encode time reduction (this one is a
pretty good compromise)
USE_LARGESTINTRA_MODELINTER: -0.98%, 22% encode time reduction
(currently the benefit of modeling is limited for txfm size selection,
but keeping this enum as a placeholder) .
USE_LARGESTALL: -1.05%, 27% encode-time reduction (same as existing
use_largest_txfm speed feature).
Change-Id: I4d60a5f9ce78fbc90cddf2f97ed91d8bc0d4f936
2013-06-22 03:31:12 +04:00
|
|
|
break;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
}; /* switch */
|
2011-02-10 01:18:28 +03:00
|
|
|
|
2013-02-05 14:13:25 +04:00
|
|
|
// Set rd thresholds based on mode and speed setting
|
2013-09-11 03:13:15 +04:00
|
|
|
set_rd_speed_thresholds(cpi, mode);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Slow quant, dct and trellis not worthwhile for first pass
|
|
|
|
// so make sure they are always turned off.
|
|
|
|
if (cpi->pass == 1) {
|
|
|
|
sf->optimize_coefficients = 0;
|
|
|
|
}
|
2011-01-06 20:10:07 +03:00
|
|
|
|
2013-02-12 09:14:46 +04:00
|
|
|
cpi->mb.fwd_txm16x16 = vp9_short_fdct16x16;
|
|
|
|
cpi->mb.fwd_txm8x8 = vp9_short_fdct8x8;
|
|
|
|
cpi->mb.fwd_txm8x4 = vp9_short_fdct8x4;
|
|
|
|
cpi->mb.fwd_txm4x4 = vp9_short_fdct4x4;
|
2013-02-14 02:22:15 +04:00
|
|
|
if (cpi->oxcf.lossless || cpi->mb.e_mbd.lossless) {
|
2013-03-12 22:24:04 +04:00
|
|
|
cpi->mb.fwd_txm8x4 = vp9_short_walsh8x4;
|
|
|
|
cpi->mb.fwd_txm4x4 = vp9_short_walsh4x4;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
cpi->mb.quantize_b_4x4 = vp9_regular_quantize_b_4x4;
|
2012-03-21 22:22:33 +04:00
|
|
|
|
2013-08-07 02:53:35 +04:00
|
|
|
if (cpi->sf.subpel_search_method == SUBPEL_ITERATIVE) {
|
|
|
|
cpi->find_fractional_mv_step = vp9_find_best_sub_pixel_iterative;
|
2013-08-08 04:01:43 +04:00
|
|
|
cpi->find_fractional_mv_step_comp = vp9_find_best_sub_pixel_comp_iterative;
|
|
|
|
} else if (cpi->sf.subpel_search_method == SUBPEL_TREE) {
|
|
|
|
cpi->find_fractional_mv_step = vp9_find_best_sub_pixel_tree;
|
|
|
|
cpi->find_fractional_mv_step_comp = vp9_find_best_sub_pixel_comp_tree;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->mb.optimize = cpi->sf.optimize_coefficients == 1 && cpi->pass != 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#ifdef SPEEDSTATS
|
2013-05-09 01:35:42 +04:00
|
|
|
frames_at_speed[cpi->speed]++;
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
}
|
2013-02-04 21:43:02 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void alloc_raw_frame_buffers(VP9_COMP *cpi) {
|
2013-05-07 02:52:06 +04:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
cpi->lookahead = vp9_lookahead_init(cpi->oxcf.width, cpi->oxcf.height,
|
2013-05-07 02:52:06 +04:00
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->oxcf.lag_in_frames);
|
|
|
|
if (!cpi->lookahead)
|
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to allocate lag buffers");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-05-07 02:52:06 +04:00
|
|
|
if (vp9_realloc_frame_buffer(&cpi->alt_ref_buffer,
|
|
|
|
cpi->oxcf.width, cpi->oxcf.height,
|
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
|
|
|
VP9BORDERINPIXELS))
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to allocate altref buffer");
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
2010-09-03 00:17:52 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
static int alloc_partition_data(VP9_COMP *cpi) {
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi->mb.pip);
|
2010-12-07 01:21:37 +03:00
|
|
|
|
2013-07-03 21:54:50 +04:00
|
|
|
cpi->mb.pip = vpx_calloc(cpi->common.mode_info_stride *
|
|
|
|
(cpi->common.mi_rows + MI_BLOCK_SIZE),
|
2012-07-14 02:21:29 +04:00
|
|
|
sizeof(PARTITION_INFO));
|
|
|
|
if (!cpi->mb.pip)
|
|
|
|
return 1;
|
2010-09-03 00:17:52 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->mb.pi = cpi->mb.pip + cpi->common.mode_info_stride + 1;
|
2010-09-03 00:17:52 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
2010-09-03 00:17:52 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
void vp9_alloc_compressor_data(VP9_COMP *cpi) {
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
if (vp9_alloc_frame_buffers(cm, cm->width, cm->height))
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to allocate frame buffers");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
if (alloc_partition_data(cpi))
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to allocate partition data");
|
2010-09-03 00:17:52 +04:00
|
|
|
|
2013-05-07 02:52:06 +04:00
|
|
|
if (vp9_alloc_frame_buffer(&cpi->last_frame_uf,
|
|
|
|
cm->width, cm->height,
|
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
|
|
|
VP9BORDERINPIXELS))
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to allocate last frame buffer");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-05-07 02:52:06 +04:00
|
|
|
if (vp9_alloc_frame_buffer(&cpi->scaled_source,
|
|
|
|
cm->width, cm->height,
|
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
|
|
|
VP9BORDERINPIXELS))
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to allocate scaled source buffer");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi->tok);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
{
|
2013-03-28 21:42:23 +04:00
|
|
|
unsigned int tokens = get_token_alloc(cm->mb_rows, cm->mb_cols);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->tok, vpx_calloc(tokens, sizeof(*cpi->tok)));
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Data used for real time vc mode to see if gf needs refreshing
|
|
|
|
cpi->inter_zz_count = 0;
|
|
|
|
cpi->gf_bad_count = 0;
|
|
|
|
cpi->gf_update_recommended = 0;
|
2010-08-11 19:02:31 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi->mb_activity_map);
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->mb_activity_map,
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_calloc(sizeof(unsigned int),
|
|
|
|
cm->mb_rows * cm->mb_cols));
|
2010-08-11 19:02:31 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi->mb_norm_activity_map);
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->mb_norm_activity_map,
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_calloc(sizeof(unsigned int),
|
|
|
|
cm->mb_rows * cm->mb_cols));
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-07 02:22:17 +04:00
|
|
|
static void update_frame_size(VP9_COMP *cpi) {
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
|
|
|
|
2013-04-30 22:14:27 +04:00
|
|
|
vp9_update_frame_size(cm);
|
2013-02-07 02:22:17 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
// Update size of buffers local to this frame
|
2013-05-07 02:52:06 +04:00
|
|
|
if (vp9_realloc_frame_buffer(&cpi->last_frame_uf,
|
|
|
|
cm->width, cm->height,
|
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
|
|
|
VP9BORDERINPIXELS))
|
2013-02-07 02:22:17 +04:00
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to reallocate last frame buffer");
|
|
|
|
|
2013-05-07 02:52:06 +04:00
|
|
|
if (vp9_realloc_frame_buffer(&cpi->scaled_source,
|
|
|
|
cm->width, cm->height,
|
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
|
|
|
VP9BORDERINPIXELS))
|
2013-02-07 02:22:17 +04:00
|
|
|
vpx_internal_error(&cpi->common.error, VPX_CODEC_MEM_ERROR,
|
|
|
|
"Failed to reallocate scaled source buffer");
|
2013-03-05 03:21:45 +04:00
|
|
|
|
|
|
|
{
|
|
|
|
int y_stride = cpi->scaled_source.y_stride;
|
|
|
|
|
|
|
|
if (cpi->sf.search_method == NSTEP) {
|
|
|
|
vp9_init3smotion_compensation(&cpi->mb, y_stride);
|
|
|
|
} else if (cpi->sf.search_method == DIAMOND) {
|
|
|
|
vp9_init_dsmotion_compensation(&cpi->mb, y_stride);
|
|
|
|
}
|
|
|
|
}
|
2013-02-07 02:22:17 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-12 22:27:25 +04:00
|
|
|
// TODO perhaps change number of steps expose to outside world when setting
|
|
|
|
// max and min limits. Also this will likely want refining for the extended Q
|
|
|
|
// range.
|
|
|
|
//
|
|
|
|
// Table that converts 0-63 Q range values passed in outside to the Qindex
|
|
|
|
// range used internally.
|
2012-07-14 02:21:29 +04:00
|
|
|
static const int q_trans[] = {
|
|
|
|
0, 4, 8, 12, 16, 20, 24, 28,
|
|
|
|
32, 36, 40, 44, 48, 52, 56, 60,
|
|
|
|
64, 68, 72, 76, 80, 84, 88, 92,
|
|
|
|
96, 100, 104, 108, 112, 116, 120, 124,
|
|
|
|
128, 132, 136, 140, 144, 148, 152, 156,
|
|
|
|
160, 164, 168, 172, 176, 180, 184, 188,
|
|
|
|
192, 196, 200, 204, 208, 212, 216, 220,
|
|
|
|
224, 228, 232, 236, 240, 244, 249, 255,
|
2011-12-12 22:27:25 +04:00
|
|
|
};
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
int vp9_reverse_trans(int x) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < 64; i++)
|
|
|
|
if (q_trans[i] >= x)
|
|
|
|
return i;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return 63;
|
2010-05-18 19:58:33 +04:00
|
|
|
};
|
2013-07-13 04:12:46 +04:00
|
|
|
void vp9_new_framerate(VP9_COMP *cpi, double framerate) {
|
2013-04-02 21:24:56 +04:00
|
|
|
if (framerate < 0.1)
|
2012-07-14 02:21:29 +04:00
|
|
|
framerate = 30;
|
2010-10-15 00:19:06 +04:00
|
|
|
|
2013-07-13 04:12:46 +04:00
|
|
|
cpi->oxcf.framerate = framerate;
|
|
|
|
cpi->output_framerate = cpi->oxcf.framerate;
|
|
|
|
cpi->per_frame_bandwidth = (int)(cpi->oxcf.target_bandwidth / cpi->output_framerate);
|
|
|
|
cpi->av_per_frame_bandwidth = (int)(cpi->oxcf.target_bandwidth / cpi->output_framerate);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->min_frame_bandwidth = (int)(cpi->av_per_frame_bandwidth * cpi->oxcf.two_pass_vbrmin_section / 100);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
|
|
|
|
cpi->min_frame_bandwidth = MAX(cpi->min_frame_bandwidth, FRAME_OVERHEAD_BITS);
|
2012-03-02 05:24:30 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Set Maximum gf/arf interval
|
2013-03-06 15:33:43 +04:00
|
|
|
cpi->max_gf_interval = 16;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Extended interval for genuinely static scenes
|
|
|
|
cpi->twopass.static_scene_max_gf_interval = cpi->key_frame_frequency >> 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-06 15:33:43 +04:00
|
|
|
// Special conditions when alt ref frame enabled in lagged compress mode
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->oxcf.play_alternate && cpi->oxcf.lag_in_frames) {
|
|
|
|
if (cpi->max_gf_interval > cpi->oxcf.lag_in_frames - 1)
|
|
|
|
cpi->max_gf_interval = cpi->oxcf.lag_in_frames - 1;
|
2011-01-18 18:19:05 +03:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->twopass.static_scene_max_gf_interval > cpi->oxcf.lag_in_frames - 1)
|
|
|
|
cpi->twopass.static_scene_max_gf_interval = cpi->oxcf.lag_in_frames - 1;
|
|
|
|
}
|
2011-01-18 18:19:05 +03:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->max_gf_interval > cpi->twopass.static_scene_max_gf_interval)
|
|
|
|
cpi->max_gf_interval = cpi->twopass.static_scene_max_gf_interval;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-02-28 05:09:12 +04:00
|
|
|
static int64_t rescale(int val, int64_t num, int denom) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int64_t llnum = num;
|
|
|
|
int64_t llden = denom;
|
|
|
|
int64_t llval = val;
|
2010-08-20 19:04:10 +04:00
|
|
|
|
2013-02-28 05:09:12 +04:00
|
|
|
return (llval * llnum / llden);
|
2010-08-20 19:04:10 +04:00
|
|
|
}
|
|
|
|
|
2013-02-08 23:33:11 +04:00
|
|
|
static void set_tile_limits(VP9_COMP *cpi) {
|
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
2013-02-07 03:30:21 +04:00
|
|
|
|
2013-07-17 01:47:15 +04:00
|
|
|
int min_log2_tile_cols, max_log2_tile_cols;
|
|
|
|
vp9_get_tile_n_bits(cm->mi_cols, &min_log2_tile_cols, &max_log2_tile_cols);
|
2013-04-02 21:24:56 +04:00
|
|
|
|
2013-07-17 01:47:15 +04:00
|
|
|
cm->log2_tile_cols = clamp(cpi->oxcf.tile_columns,
|
|
|
|
min_log2_tile_cols, max_log2_tile_cols);
|
|
|
|
cm->log2_tile_rows = cpi->oxcf.tile_rows;
|
2013-02-07 03:30:21 +04:00
|
|
|
}
|
2010-08-20 19:04:10 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
static void init_config(VP9_PTR ptr, VP9_CONFIG *oxcf) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(ptr);
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 21:35:28 +04:00
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
2013-04-02 21:24:56 +04:00
|
|
|
int i;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->oxcf = *oxcf;
|
|
|
|
cpi->goldfreq = 7;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
cm->version = oxcf->version;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
cm->width = oxcf->width;
|
|
|
|
cm->height = oxcf->height;
|
2013-05-07 02:52:06 +04:00
|
|
|
cm->subsampling_x = 0;
|
|
|
|
cm->subsampling_y = 0;
|
|
|
|
vp9_alloc_compressor_data(cpi);
|
2013-03-15 01:36:08 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// change includes all joint functionality
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_change_config(ptr, oxcf);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Initialize active best and worst q and average q values.
|
|
|
|
cpi->active_worst_quality = cpi->oxcf.worst_allowed_q;
|
|
|
|
cpi->active_best_quality = cpi->oxcf.best_allowed_q;
|
|
|
|
cpi->avg_frame_qindex = cpi->oxcf.worst_allowed_q;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Initialise the starting buffer levels
|
|
|
|
cpi->buffer_level = cpi->oxcf.starting_buffer_level;
|
|
|
|
cpi->bits_off_target = cpi->oxcf.starting_buffer_level;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->rolling_target_bits = cpi->av_per_frame_bandwidth;
|
|
|
|
cpi->rolling_actual_bits = cpi->av_per_frame_bandwidth;
|
|
|
|
cpi->long_rolling_target_bits = cpi->av_per_frame_bandwidth;
|
|
|
|
cpi->long_rolling_actual_bits = cpi->av_per_frame_bandwidth;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->total_actual_bits = 0;
|
|
|
|
cpi->total_target_vs_actual = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->static_mb_pct = 0;
|
2012-01-11 18:05:57 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->lst_fb_idx = 0;
|
|
|
|
cpi->gld_fb_idx = 1;
|
|
|
|
cpi->alt_fb_idx = 2;
|
|
|
|
|
2013-09-05 19:55:47 +04:00
|
|
|
cpi->current_layer = 0;
|
|
|
|
cpi->use_svc = 0;
|
|
|
|
|
2013-02-08 23:33:11 +04:00
|
|
|
set_tile_limits(cpi);
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 21:35:28 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->fixed_divide[0] = 0;
|
|
|
|
for (i = 1; i < 512; i++)
|
|
|
|
cpi->fixed_divide[i] = 0x80000 / i;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
void vp9_change_config(VP9_PTR ptr, VP9_CONFIG *oxcf) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(ptr);
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 21:35:28 +04:00
|
|
|
VP9_COMMON *const cm = &cpi->common;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-28 01:22:30 +04:00
|
|
|
if (!cpi || !oxcf)
|
2012-07-14 02:21:29 +04:00
|
|
|
return;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
if (cm->version != oxcf->version) {
|
|
|
|
cm->version = oxcf->version;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->oxcf = *oxcf;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
switch (cpi->oxcf.Mode) {
|
|
|
|
// Real time and one pass deprecated in test code base
|
2010-05-18 19:58:33 +04:00
|
|
|
case MODE_FIRSTPASS:
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->pass = 1;
|
|
|
|
cpi->compressor_speed = 1;
|
|
|
|
break;
|
2012-02-16 21:02:17 +04:00
|
|
|
|
2010-05-18 19:58:33 +04:00
|
|
|
case MODE_SECONDPASS:
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->pass = 2;
|
|
|
|
cpi->compressor_speed = 1;
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->oxcf.cpu_used = clamp(cpi->oxcf.cpu_used, -5, 5);
|
2012-07-14 02:21:29 +04:00
|
|
|
break;
|
2012-02-16 21:02:17 +04:00
|
|
|
|
2010-05-18 19:58:33 +04:00
|
|
|
case MODE_SECONDPASS_BEST:
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->pass = 2;
|
|
|
|
cpi->compressor_speed = 0;
|
|
|
|
break;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->oxcf.worst_allowed_q = q_trans[oxcf->worst_allowed_q];
|
|
|
|
cpi->oxcf.best_allowed_q = q_trans[oxcf->best_allowed_q];
|
|
|
|
cpi->oxcf.cq_level = q_trans[cpi->oxcf.cq_level];
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->oxcf.lossless = oxcf->lossless;
|
|
|
|
if (cpi->oxcf.lossless) {
|
2013-05-20 21:03:17 +04:00
|
|
|
cpi->mb.e_mbd.inv_txm4x4_1_add = vp9_short_iwalsh4x4_1_add;
|
|
|
|
cpi->mb.e_mbd.inv_txm4x4_add = vp9_short_iwalsh4x4_add;
|
2013-02-20 19:27:35 +04:00
|
|
|
} else {
|
2013-05-20 21:03:17 +04:00
|
|
|
cpi->mb.e_mbd.inv_txm4x4_1_add = vp9_short_idct4x4_1_add;
|
|
|
|
cpi->mb.e_mbd.inv_txm4x4_add = vp9_short_idct4x4_add;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
Add lossless compression mode.
This commit adds lossless compression capability to the experimental
branch. The lossless experiment can be enabled using --enable-lossless
in configure. When the experiment is enabled, the encoder will use
lossless compression mode by command line option --lossless, and the
decoder automatically recognizes a losslessly encoded clip and decodes
accordingly.
To achieve the lossless coding, this commit has changed the following:
1. To encode at lossless mode, encoder forces the use of unit
quantizer, i.e, Q 0, where effective quantization is 1. Encoder also
disables the usage of 8x8 transform and allows only 4x4 transform;
2. At Q 0, the first order 4x4 DCT/IDCT have been switched over
to a pair of forward and inverse Walsh-Hadamard Transform
(http://goo.gl/EIsfy), with proper scaling applied to match the range
of the original 4x4 DCT/IDCT pair;
3. At Q 0, the second order remains to use the previous
walsh-hadamard transform pair. However, to maintain the reversibility
in second order transform at Q 0, scaling down is applied to first
order DC coefficients prior to forward transform, and scaling up is
applied to the second order output prior to quantization. Symmetric
upscaling and downscaling are added around inverse second order
transform;
4. At lossless mode, encoder also disables a number of minor
features to ensure no loss is introduced, these features includes:
a. Trellis quantization optimization
b. Loop filtering
c. Aggressive zero-binning, rounding and zero-bin boosting
d. Mode based zero-bin boosting
Lossless coding test was performed on all clips within the derf set,
to verify that the commit has achieved lossless compression for all
clips. The average compression ratio is around 2.57 to 1.
(http://goo.gl/dEShs)
Change-Id: Ia3aba7dd09df40dd590f93b9aba134defbc64e34
2012-06-14 06:03:31 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->baseline_gf_interval = DEFAULT_GF_INTERVAL;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
cpi->ref_frame_flags = VP9_ALT_FLAG | VP9_GOLD_FLAG | VP9_LAST_FLAG;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// cpi->use_golden_frame_only = 0;
|
|
|
|
// cpi->use_last_frame_only = 0;
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_golden_frame = 0;
|
|
|
|
cpi->refresh_last_frame = 1;
|
2013-04-27 01:39:58 +04:00
|
|
|
cm->refresh_frame_context = 1;
|
2013-05-30 04:16:00 +04:00
|
|
|
cm->reset_frame_context = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-14 22:20:33 +04:00
|
|
|
setup_features(cm);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->mb.e_mbd.allow_high_precision_mv = 0; // Default mv precision adaptation
|
2012-11-09 03:44:39 +04:00
|
|
|
set_mvcost(&cpi->mb);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
{
|
|
|
|
int i;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-07-23 15:09:04 +04:00
|
|
|
for (i = 0; i < MAX_SEGMENTS; i++)
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->segment_encode_breakout[i] = cpi->oxcf.encode_breakout;
|
|
|
|
}
|
|
|
|
|
|
|
|
// At the moment the first order values may not be > MAXQ
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->oxcf.fixed_q = MIN(cpi->oxcf.fixed_q, MAXQ);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// local file playback mode == really big buffer
|
|
|
|
if (cpi->oxcf.end_usage == USAGE_LOCAL_FILE_PLAYBACK) {
|
|
|
|
cpi->oxcf.starting_buffer_level = 60000;
|
|
|
|
cpi->oxcf.optimal_buffer_level = 60000;
|
|
|
|
cpi->oxcf.maximum_buffer_size = 240000;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert target bandwidth from Kbit/s to Bit/s
|
|
|
|
cpi->oxcf.target_bandwidth *= 1000;
|
|
|
|
|
2013-03-28 01:22:30 +04:00
|
|
|
cpi->oxcf.starting_buffer_level = rescale(cpi->oxcf.starting_buffer_level,
|
|
|
|
cpi->oxcf.target_bandwidth, 1000);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Set or reset optimal and maximum buffer levels.
|
|
|
|
if (cpi->oxcf.optimal_buffer_level == 0)
|
|
|
|
cpi->oxcf.optimal_buffer_level = cpi->oxcf.target_bandwidth / 8;
|
|
|
|
else
|
2013-03-28 01:22:30 +04:00
|
|
|
cpi->oxcf.optimal_buffer_level = rescale(cpi->oxcf.optimal_buffer_level,
|
|
|
|
cpi->oxcf.target_bandwidth, 1000);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
if (cpi->oxcf.maximum_buffer_size == 0)
|
|
|
|
cpi->oxcf.maximum_buffer_size = cpi->oxcf.target_bandwidth / 8;
|
|
|
|
else
|
2013-03-28 01:22:30 +04:00
|
|
|
cpi->oxcf.maximum_buffer_size = rescale(cpi->oxcf.maximum_buffer_size,
|
|
|
|
cpi->oxcf.target_bandwidth, 1000);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Set up frame rate and related parameters rate control values.
|
2013-07-13 04:12:46 +04:00
|
|
|
vp9_new_framerate(cpi, cpi->oxcf.framerate);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Set absolute upper and lower quality limits
|
2013-03-28 01:22:30 +04:00
|
|
|
cpi->worst_quality = cpi->oxcf.worst_allowed_q;
|
|
|
|
cpi->best_quality = cpi->oxcf.best_allowed_q;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// active values should only be modified if out of new range
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->active_worst_quality = clamp(cpi->active_worst_quality,
|
|
|
|
cpi->oxcf.best_allowed_q,
|
|
|
|
cpi->oxcf.worst_allowed_q);
|
|
|
|
|
|
|
|
cpi->active_best_quality = clamp(cpi->active_best_quality,
|
|
|
|
cpi->oxcf.best_allowed_q,
|
|
|
|
cpi->oxcf.worst_allowed_q);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->buffered_mode = cpi->oxcf.optimal_buffer_level > 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
cpi->cq_target_quality = cpi->oxcf.cq_level;
|
|
|
|
|
2013-06-11 02:55:03 +04:00
|
|
|
cm->mcomp_filter_type = DEFAULT_INTERP_FILTER;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->target_bandwidth = cpi->oxcf.target_bandwidth;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
cm->display_width = cpi->oxcf.width;
|
|
|
|
cm->display_height = cpi->oxcf.height;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// VP8 sharpness level mapping 0-7 (vs 0-10 in general VPx dialogs)
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->oxcf.Sharpness = MIN(7, cpi->oxcf.Sharpness);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-10 01:41:51 +04:00
|
|
|
cpi->common.lf.sharpness_level = cpi->oxcf.Sharpness;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-05-07 02:52:06 +04:00
|
|
|
if (cpi->initial_width) {
|
|
|
|
// Increasing the size of the frame beyond the first seen frame, or some
|
|
|
|
// otherwise signalled maximum size, is not supported.
|
|
|
|
// TODO(jkoleszar): exit gracefully.
|
|
|
|
assert(cm->width <= cpi->initial_width);
|
|
|
|
assert(cm->height <= cpi->initial_height);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2013-02-07 02:22:17 +04:00
|
|
|
update_frame_size(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->oxcf.fixed_q >= 0) {
|
|
|
|
cpi->last_q[0] = cpi->oxcf.fixed_q;
|
|
|
|
cpi->last_q[1] = cpi->oxcf.fixed_q;
|
|
|
|
cpi->last_boosted_qindex = cpi->oxcf.fixed_q;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-05-09 01:35:42 +04:00
|
|
|
cpi->speed = cpi->oxcf.cpu_used;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->oxcf.lag_in_frames == 0) {
|
2013-04-02 21:24:56 +04:00
|
|
|
// force to allowlag to 0 if lag_in_frames is 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->oxcf.allow_lag = 0;
|
2013-04-02 21:24:56 +04:00
|
|
|
} else if (cpi->oxcf.lag_in_frames > MAX_LAG_BUFFERS) {
|
|
|
|
// Limit on lag buffers as these are not currently dynamically allocated
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->oxcf.lag_in_frames = MAX_LAG_BUFFERS;
|
2013-04-02 21:24:56 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// YX Temp
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
vp9_zero(cpi->alt_ref_source);
|
|
|
|
#else
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->alt_ref_source = NULL;
|
2013-04-03 02:08:50 +04:00
|
|
|
#endif
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->is_src_frame_alt_ref = 0;
|
2011-05-11 06:57:51 +04:00
|
|
|
|
2010-05-18 19:58:33 +04:00
|
|
|
#if 0
|
2012-07-14 02:21:29 +04:00
|
|
|
// Experimental RD Code
|
|
|
|
cpi->frame_distortion = 0;
|
|
|
|
cpi->last_frame_distortion = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
|
2013-02-08 23:33:11 +04:00
|
|
|
set_tile_limits(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
#define M_LOG2_E 0.693147180559945309417
|
|
|
|
#define log2f(x) (log (x) / (float) M_LOG2_E)
|
2012-07-27 00:42:07 +04:00
|
|
|
|
|
|
|
static void cal_nmvjointsadcost(int *mvjointsadcost) {
|
|
|
|
mvjointsadcost[0] = 600;
|
|
|
|
mvjointsadcost[1] = 300;
|
|
|
|
mvjointsadcost[2] = 300;
|
|
|
|
mvjointsadcost[0] = 300;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cal_nmvsadcosts(int *mvsadcost[2]) {
|
|
|
|
int i = 1;
|
|
|
|
|
2013-03-28 01:22:30 +04:00
|
|
|
mvsadcost[0][0] = 0;
|
|
|
|
mvsadcost[1][0] = 0;
|
2012-07-27 00:42:07 +04:00
|
|
|
|
|
|
|
do {
|
|
|
|
double z = 256 * (2 * (log2f(8 * i) + .6));
|
2013-03-28 01:22:30 +04:00
|
|
|
mvsadcost[0][i] = (int)z;
|
|
|
|
mvsadcost[1][i] = (int)z;
|
|
|
|
mvsadcost[0][-i] = (int)z;
|
|
|
|
mvsadcost[1][-i] = (int)z;
|
2012-07-27 00:42:07 +04:00
|
|
|
} while (++i <= MV_MAX);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cal_nmvsadcosts_hp(int *mvsadcost[2]) {
|
|
|
|
int i = 1;
|
|
|
|
|
2013-03-28 01:22:30 +04:00
|
|
|
mvsadcost[0][0] = 0;
|
|
|
|
mvsadcost[1][0] = 0;
|
2012-07-27 00:42:07 +04:00
|
|
|
|
|
|
|
do {
|
|
|
|
double z = 256 * (2 * (log2f(8 * i) + .6));
|
2013-03-28 01:22:30 +04:00
|
|
|
mvsadcost[0][i] = (int)z;
|
|
|
|
mvsadcost[1][i] = (int)z;
|
|
|
|
mvsadcost[0][-i] = (int)z;
|
|
|
|
mvsadcost[1][-i] = (int)z;
|
2012-07-27 00:42:07 +04:00
|
|
|
} while (++i <= MV_MAX);
|
|
|
|
}
|
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
VP9_PTR vp9_create_compressor(VP9_CONFIG *oxcf) {
|
2013-08-23 17:00:54 +04:00
|
|
|
int i, j;
|
2012-07-14 02:21:29 +04:00
|
|
|
volatile union {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi;
|
|
|
|
VP9_PTR ptr;
|
2012-07-14 02:21:29 +04:00
|
|
|
} ctx;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi;
|
|
|
|
VP9_COMMON *cm;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
cpi = ctx.cpi = vpx_memalign(32, sizeof(VP9_COMP));
|
2012-07-14 02:21:29 +04:00
|
|
|
// Check that the CPI instance is valid
|
|
|
|
if (!cpi)
|
|
|
|
return 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cm = &cpi->common;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-07-26 01:13:44 +04:00
|
|
|
vp9_zero(*cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (setjmp(cm->error.jmp)) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_PTR ptr = ctx.ptr;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
ctx.cpi->common.error.setjmp = 0;
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_remove_compressor(&ptr);
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-06-28 21:36:20 +04:00
|
|
|
cm->error.setjmp = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->mb.ss, vpx_calloc(sizeof(search_site),
|
|
|
|
(MAX_MVSEARCH_STEPS * 8) + 1));
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-06-28 21:36:20 +04:00
|
|
|
vp9_create_common(cm);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
init_config((VP9_PTR)cpi, oxcf);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-06-28 21:36:20 +04:00
|
|
|
cm->current_video_frame = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->kf_overspend_bits = 0;
|
|
|
|
cpi->kf_bitrate_adjustment = 0;
|
2013-04-03 02:08:50 +04:00
|
|
|
cpi->frames_till_gf_update_due = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->gf_overspend_bits = 0;
|
2013-04-03 02:08:50 +04:00
|
|
|
cpi->non_gf_bitrate_adjustment = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Set reference frame sign bias for ALTREF frame to 1 (for now)
|
2013-06-28 21:36:20 +04:00
|
|
|
cm->ref_frame_sign_bias[ALTREF_FRAME] = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
cpi->baseline_gf_interval = DEFAULT_GF_INTERVAL;
|
|
|
|
|
|
|
|
cpi->gold_is_last = 0;
|
|
|
|
cpi->alt_is_last = 0;
|
|
|
|
cpi->gold_is_alt = 0;
|
|
|
|
|
2013-09-05 19:55:47 +04:00
|
|
|
// Spatial scalability
|
|
|
|
cpi->number_spatial_layers = oxcf->ss_number_layers;
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Create the encoder segmentation map and set all entries to 0
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->segmentation_map,
|
|
|
|
vpx_calloc(cm->mi_rows * cm->mi_cols, 1));
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// And a place holder structure is the coding context
|
|
|
|
// for use if we want to save and restore it
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->coding_context.last_frame_seg_map_copy,
|
|
|
|
vpx_calloc(cm->mi_rows * cm->mi_cols, 1));
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->active_map, vpx_calloc(cm->MBs, 1));
|
|
|
|
vpx_memset(cpi->active_map, 1, cm->MBs);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->active_map_enabled = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < (sizeof(cpi->mbgraph_stats) /
|
|
|
|
sizeof(cpi->mbgraph_stats[0])); i++) {
|
2013-06-28 21:36:20 +04:00
|
|
|
CHECK_MEM_ERROR(cm, cpi->mbgraph_stats[i].mb_stats,
|
|
|
|
vpx_calloc(cm->MBs *
|
|
|
|
sizeof(*cpi->mbgraph_stats[i].mb_stats), 1));
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2011-10-05 14:26:00 +04:00
|
|
|
|
2010-05-18 19:58:33 +04:00
|
|
|
#ifdef ENTROPY_STATS
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->pass != 1)
|
|
|
|
init_context_counters();
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
2013-05-16 16:40:32 +04:00
|
|
|
|
2013-06-06 22:14:04 +04:00
|
|
|
#ifdef MODE_STATS
|
|
|
|
init_tx_count_stats();
|
2013-06-10 23:00:43 +04:00
|
|
|
init_switchable_interp_stats();
|
2013-06-06 22:14:04 +04:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
/*Initialize the feed-forward activity masking.*/
|
|
|
|
cpi->activity_avg = 90 << 12;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->frames_since_key = 8; // Give a sensible default for the first frame.
|
|
|
|
cpi->key_frame_frequency = cpi->oxcf.key_freq;
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->this_key_frame_forced = 0;
|
|
|
|
cpi->next_key_frame_forced = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->source_alt_ref_pending = 0;
|
|
|
|
cpi->source_alt_ref_active = 0;
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_alt_ref_frame = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
// Turn multiple ARF usage on/off. This is a quick hack for the initial test
|
|
|
|
// version. It should eventually be set via the codec API.
|
|
|
|
cpi->multi_arf_enabled = 1;
|
|
|
|
|
|
|
|
if (cpi->multi_arf_enabled) {
|
|
|
|
cpi->sequence_number = 0;
|
|
|
|
cpi->frame_coding_order_period = 0;
|
|
|
|
vp9_zero(cpi->frame_coding_order);
|
|
|
|
vp9_zero(cpi->arf_buffer_idx);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->b_calculate_psnr = CONFIG_INTERNAL_STATS;
|
2011-04-29 20:37:59 +04:00
|
|
|
#if CONFIG_INTERNAL_STATS
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->b_calculate_ssimg = 0;
|
|
|
|
|
|
|
|
cpi->count = 0;
|
|
|
|
cpi->bytes = 0;
|
|
|
|
|
|
|
|
if (cpi->b_calculate_psnr) {
|
|
|
|
cpi->total_sq_error = 0.0;
|
|
|
|
cpi->total_sq_error2 = 0.0;
|
|
|
|
cpi->total_y = 0.0;
|
|
|
|
cpi->total_u = 0.0;
|
|
|
|
cpi->total_v = 0.0;
|
|
|
|
cpi->total = 0.0;
|
|
|
|
cpi->totalp_y = 0.0;
|
|
|
|
cpi->totalp_u = 0.0;
|
|
|
|
cpi->totalp_v = 0.0;
|
|
|
|
cpi->totalp = 0.0;
|
|
|
|
cpi->tot_recode_hits = 0;
|
|
|
|
cpi->summed_quality = 0;
|
|
|
|
cpi->summed_weights = 0;
|
2013-04-01 20:10:27 +04:00
|
|
|
cpi->summedp_quality = 0;
|
|
|
|
cpi->summedp_weights = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (cpi->b_calculate_ssimg) {
|
|
|
|
cpi->total_ssimg_y = 0;
|
|
|
|
cpi->total_ssimg_u = 0;
|
|
|
|
cpi->total_ssimg_v = 0;
|
|
|
|
cpi->total_ssimg_all = 0;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2011-05-31 20:37:45 +04:00
|
|
|
#endif
|
|
|
|
|
2013-01-14 23:49:30 +04:00
|
|
|
cpi->first_time_stamp_ever = INT64_MAX;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->frames_till_gf_update_due = 0;
|
|
|
|
cpi->key_frame_count = 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->ni_av_qi = cpi->oxcf.worst_allowed_q;
|
|
|
|
cpi->ni_tot_qi = 0;
|
|
|
|
cpi->ni_frames = 0;
|
|
|
|
cpi->tot_q = 0.0;
|
2012-10-30 23:58:42 +04:00
|
|
|
cpi->avg_q = vp9_convert_qindex_to_q(cpi->oxcf.worst_allowed_q);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->total_byte_count = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->rate_correction_factor = 1.0;
|
|
|
|
cpi->key_frame_rate_correction_factor = 1.0;
|
|
|
|
cpi->gf_rate_correction_factor = 1.0;
|
|
|
|
cpi->twopass.est_max_qcorrection_factor = 1.0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-27 00:42:07 +04:00
|
|
|
cal_nmvjointsadcost(cpi->mb.nmvjointsadcost);
|
|
|
|
cpi->mb.nmvcost[0] = &cpi->mb.nmvcosts[0][MV_MAX];
|
|
|
|
cpi->mb.nmvcost[1] = &cpi->mb.nmvcosts[1][MV_MAX];
|
|
|
|
cpi->mb.nmvsadcost[0] = &cpi->mb.nmvsadcosts[0][MV_MAX];
|
|
|
|
cpi->mb.nmvsadcost[1] = &cpi->mb.nmvsadcosts[1][MV_MAX];
|
|
|
|
cal_nmvsadcosts(cpi->mb.nmvsadcost);
|
|
|
|
|
|
|
|
cpi->mb.nmvcost_hp[0] = &cpi->mb.nmvcosts_hp[0][MV_MAX];
|
|
|
|
cpi->mb.nmvcost_hp[1] = &cpi->mb.nmvcosts_hp[1][MV_MAX];
|
|
|
|
cpi->mb.nmvsadcost_hp[0] = &cpi->mb.nmvsadcosts_hp[0][MV_MAX];
|
|
|
|
cpi->mb.nmvsadcost_hp[1] = &cpi->mb.nmvsadcosts_hp[1][MV_MAX];
|
|
|
|
cal_nmvsadcosts_hp(cpi->mb.nmvsadcost_hp);
|
2012-02-27 22:22:38 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
for (i = 0; i < KEY_FRAME_CONTEXT; i++)
|
2013-07-13 04:12:46 +04:00
|
|
|
cpi->prior_key_frame_distance[i] = (int)cpi->output_framerate;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#ifdef OUTPUT_YUV_SRC
|
2012-07-14 02:21:29 +04:00
|
|
|
yuv_file = fopen("bd.yuv", "ab");
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
2011-02-15 01:18:18 +03:00
|
|
|
#ifdef OUTPUT_YUV_REC
|
2012-07-14 02:21:29 +04:00
|
|
|
yuv_rec_file = fopen("rec.yuv", "wb");
|
2011-02-15 01:18:18 +03:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#if 0
|
2012-07-14 02:21:29 +04:00
|
|
|
framepsnr = fopen("framepsnr.stt", "a");
|
|
|
|
kf_list = fopen("kf_list.stt", "w");
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->output_pkt_list = oxcf->output_pkt_list;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-09-06 04:10:58 +04:00
|
|
|
cpi->enable_encode_breakout = 1;
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->pass == 1) {
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_init_first_pass(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
} else if (cpi->pass == 2) {
|
|
|
|
size_t packet_sz = sizeof(FIRSTPASS_STATS);
|
2012-11-06 02:22:59 +04:00
|
|
|
int packets = (int)(oxcf->two_pass_stats_in.sz / packet_sz);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->twopass.stats_in_start = oxcf->two_pass_stats_in.buf;
|
|
|
|
cpi->twopass.stats_in = cpi->twopass.stats_in_start;
|
|
|
|
cpi->twopass.stats_in_end = (void *)((char *)cpi->twopass.stats_in
|
|
|
|
+ (packets - 1) * packet_sz);
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_init_second_pass(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_set_speed_features(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-23 17:00:54 +04:00
|
|
|
// Default rd threshold factors for mode selection
|
|
|
|
for (i = 0; i < BLOCK_SIZES; ++i)
|
|
|
|
for (j = 0; j < MAX_MODES; ++j)
|
|
|
|
cpi->rd_thresh_freq_fact[i][j] = 32;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
#define BFP(BT, SDF, SDAF, VF, SVF, SVAF, SVFHH, SVFHV, SVFHHV, \
|
|
|
|
SDX3F, SDX8F, SDX4DF)\
|
2012-10-22 07:47:57 +04:00
|
|
|
cpi->fn_ptr[BT].sdf = SDF; \
|
2013-06-25 22:26:49 +04:00
|
|
|
cpi->fn_ptr[BT].sdaf = SDAF; \
|
2012-10-22 07:47:57 +04:00
|
|
|
cpi->fn_ptr[BT].vf = VF; \
|
|
|
|
cpi->fn_ptr[BT].svf = SVF; \
|
2013-05-07 20:45:28 +04:00
|
|
|
cpi->fn_ptr[BT].svaf = SVAF; \
|
2012-10-22 07:47:57 +04:00
|
|
|
cpi->fn_ptr[BT].svf_halfpix_h = SVFHH; \
|
|
|
|
cpi->fn_ptr[BT].svf_halfpix_v = SVFHV; \
|
|
|
|
cpi->fn_ptr[BT].svf_halfpix_hv = SVFHHV; \
|
|
|
|
cpi->fn_ptr[BT].sdx3f = SDX3F; \
|
|
|
|
cpi->fn_ptr[BT].sdx8f = SDX8F; \
|
|
|
|
cpi->fn_ptr[BT].sdx4df = SDX4DF;
|
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_32X16, vp9_sad32x16, vp9_sad32x16_avg,
|
|
|
|
vp9_variance32x16, vp9_sub_pixel_variance32x16,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance32x16, NULL, NULL,
|
2013-04-15 21:00:34 +04:00
|
|
|
NULL, NULL, NULL,
|
|
|
|
vp9_sad32x16x4d)
|
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_16X32, vp9_sad16x32, vp9_sad16x32_avg,
|
|
|
|
vp9_variance16x32, vp9_sub_pixel_variance16x32,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance16x32, NULL, NULL,
|
2013-04-15 21:00:34 +04:00
|
|
|
NULL, NULL, NULL,
|
|
|
|
vp9_sad16x32x4d)
|
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_64X32, vp9_sad64x32, vp9_sad64x32_avg,
|
|
|
|
vp9_variance64x32, vp9_sub_pixel_variance64x32,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance64x32, NULL, NULL,
|
2013-04-15 21:00:34 +04:00
|
|
|
NULL, NULL, NULL,
|
|
|
|
vp9_sad64x32x4d)
|
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_32X64, vp9_sad32x64, vp9_sad32x64_avg,
|
|
|
|
vp9_variance32x64, vp9_sub_pixel_variance32x64,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance32x64, NULL, NULL,
|
2013-04-15 21:00:34 +04:00
|
|
|
NULL, NULL, NULL,
|
|
|
|
vp9_sad32x64x4d)
|
2012-10-22 07:47:57 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_32X32, vp9_sad32x32, vp9_sad32x32_avg,
|
|
|
|
vp9_variance32x32, vp9_sub_pixel_variance32x32,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance32x32, vp9_variance_halfpixvar32x32_h,
|
|
|
|
vp9_variance_halfpixvar32x32_v,
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_variance_halfpixvar32x32_hv, vp9_sad32x32x3, vp9_sad32x32x8,
|
|
|
|
vp9_sad32x32x4d)
|
2013-01-06 06:20:25 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_64X64, vp9_sad64x64, vp9_sad64x64_avg,
|
|
|
|
vp9_variance64x64, vp9_sub_pixel_variance64x64,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance64x64, vp9_variance_halfpixvar64x64_h,
|
|
|
|
vp9_variance_halfpixvar64x64_v,
|
2013-01-06 06:20:25 +04:00
|
|
|
vp9_variance_halfpixvar64x64_hv, vp9_sad64x64x3, vp9_sad64x64x8,
|
|
|
|
vp9_sad64x64x4d)
|
2012-08-21 01:43:34 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_16X16, vp9_sad16x16, vp9_sad16x16_avg,
|
|
|
|
vp9_variance16x16, vp9_sub_pixel_variance16x16,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance16x16, vp9_variance_halfpixvar16x16_h,
|
|
|
|
vp9_variance_halfpixvar16x16_v,
|
|
|
|
vp9_variance_halfpixvar16x16_hv, vp9_sad16x16x3, vp9_sad16x16x8,
|
|
|
|
vp9_sad16x16x4d)
|
2012-10-22 07:47:57 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_16X8, vp9_sad16x8, vp9_sad16x8_avg,
|
|
|
|
vp9_variance16x8, vp9_sub_pixel_variance16x8,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance16x8, NULL, NULL, NULL,
|
|
|
|
vp9_sad16x8x3, vp9_sad16x8x8, vp9_sad16x8x4d)
|
2012-10-22 07:47:57 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_8X16, vp9_sad8x16, vp9_sad8x16_avg,
|
|
|
|
vp9_variance8x16, vp9_sub_pixel_variance8x16,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance8x16, NULL, NULL, NULL,
|
|
|
|
vp9_sad8x16x3, vp9_sad8x16x8, vp9_sad8x16x4d)
|
2012-10-22 07:47:57 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_8X8, vp9_sad8x8, vp9_sad8x8_avg,
|
|
|
|
vp9_variance8x8, vp9_sub_pixel_variance8x8,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance8x8, NULL, NULL, NULL,
|
|
|
|
vp9_sad8x8x3, vp9_sad8x8x8, vp9_sad8x8x4d)
|
2012-10-22 07:47:57 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_8X4, vp9_sad8x4, vp9_sad8x4_avg,
|
|
|
|
vp9_variance8x4, vp9_sub_pixel_variance8x4,
|
2013-05-15 23:19:59 +04:00
|
|
|
vp9_sub_pixel_avg_variance8x4, NULL, NULL,
|
2013-05-22 08:28:42 +04:00
|
|
|
NULL, NULL, vp9_sad8x4x8,
|
2013-05-15 23:19:59 +04:00
|
|
|
vp9_sad8x4x4d)
|
2013-05-05 09:09:43 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_4X8, vp9_sad4x8, vp9_sad4x8_avg,
|
|
|
|
vp9_variance4x8, vp9_sub_pixel_variance4x8,
|
2013-05-15 23:19:59 +04:00
|
|
|
vp9_sub_pixel_avg_variance4x8, NULL, NULL,
|
2013-05-22 08:28:42 +04:00
|
|
|
NULL, NULL, vp9_sad4x8x8,
|
2013-05-15 23:19:59 +04:00
|
|
|
vp9_sad4x8x4d)
|
2013-05-05 09:09:43 +04:00
|
|
|
|
2013-06-25 22:26:49 +04:00
|
|
|
BFP(BLOCK_4X4, vp9_sad4x4, vp9_sad4x4_avg,
|
|
|
|
vp9_variance4x4, vp9_sub_pixel_variance4x4,
|
2013-05-07 20:45:28 +04:00
|
|
|
vp9_sub_pixel_avg_variance4x4, NULL, NULL, NULL,
|
|
|
|
vp9_sad4x4x3, vp9_sad4x4x8, vp9_sad4x4x4d)
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-06 04:58:03 +04:00
|
|
|
cpi->full_search_sad = vp9_full_search_sad;
|
|
|
|
cpi->diamond_search_sad = vp9_diamond_search_sad;
|
|
|
|
cpi->refining_search_sad = vp9_refining_search_sad;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// make sure frame 1 is okay
|
|
|
|
cpi->error_bins[0] = cpi->common.MBs;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-31 01:25:33 +04:00
|
|
|
/* vp9_init_quantizer() is first called here. Add check in
|
|
|
|
* vp9_frame_init_quantizer() so that vp9_init_quantizer is only
|
|
|
|
* called later when needed. This will avoid unnecessary calls of
|
|
|
|
* vp9_init_quantizer() for every frame.
|
|
|
|
*/
|
|
|
|
vp9_init_quantizer(cpi);
|
2011-07-20 23:53:42 +04:00
|
|
|
|
2013-08-10 04:24:40 +04:00
|
|
|
vp9_loop_filter_init(cm);
|
2011-07-20 23:53:42 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->common.error.setjmp = 0;
|
2011-12-08 23:43:09 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_zero(cpi->y_uv_mode_count)
|
2011-12-08 23:43:09 +04:00
|
|
|
|
2013-08-21 15:34:14 +04:00
|
|
|
#ifdef MODE_TEST_HIT_STATS
|
|
|
|
vp9_zero(cpi->mode_test_hits)
|
|
|
|
#endif
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
return (VP9_PTR) cpi;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
void vp9_remove_compressor(VP9_PTR *ptr) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(*ptr);
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (!cpi)
|
|
|
|
return;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi && (cpi->common.current_video_frame > 0)) {
|
|
|
|
if (cpi->pass == 2) {
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_end_second_pass(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#ifdef ENTROPY_STATS
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->pass != 1) {
|
|
|
|
print_context_counters();
|
|
|
|
print_tree_update_probs();
|
2013-05-16 14:27:12 +04:00
|
|
|
print_mode_context(cpi);
|
2012-05-08 23:38:39 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
2013-08-03 04:10:15 +04:00
|
|
|
|
2013-06-06 22:14:04 +04:00
|
|
|
#ifdef MODE_STATS
|
2013-06-10 23:00:43 +04:00
|
|
|
if (cpi->pass != 1) {
|
2013-06-06 22:14:04 +04:00
|
|
|
write_tx_count_stats();
|
2013-06-10 23:00:43 +04:00
|
|
|
write_switchable_interp_stats();
|
|
|
|
}
|
2013-06-06 22:14:04 +04:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2011-04-29 20:37:59 +04:00
|
|
|
#if CONFIG_INTERNAL_STATS
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state();
|
2012-02-29 05:11:12 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// printf("\n8x8-4x4:%d-%d\n", cpi->t8x8_count, cpi->t4x4_count);
|
|
|
|
if (cpi->pass != 1) {
|
|
|
|
FILE *f = fopen("opsnr.stt", "a");
|
|
|
|
double time_encoded = (cpi->last_end_time_stamp_seen
|
|
|
|
- cpi->first_time_stamp_ever) / 10000000.000;
|
|
|
|
double total_encode_time = (cpi->time_receive_data + cpi->time_compress_data) / 1000.000;
|
|
|
|
double dr = (double)cpi->bytes * (double) 8 / (double)1000 / time_encoded;
|
2013-05-16 16:40:32 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->b_calculate_psnr) {
|
2013-01-16 01:49:44 +04:00
|
|
|
YV12_BUFFER_CONFIG *lst_yv12 =
|
2013-02-22 23:22:03 +04:00
|
|
|
&cpi->common.yv12_fb[cpi->common.ref_frame_map[cpi->lst_fb_idx]];
|
2013-04-01 20:10:27 +04:00
|
|
|
double samples = 3.0 / 2 * cpi->count *
|
|
|
|
lst_yv12->y_width * lst_yv12->y_height;
|
2012-10-30 23:58:42 +04:00
|
|
|
double total_psnr = vp9_mse2psnr(samples, 255.0, cpi->total_sq_error);
|
|
|
|
double total_psnr2 = vp9_mse2psnr(samples, 255.0, cpi->total_sq_error2);
|
2013-04-01 20:10:27 +04:00
|
|
|
double total_ssim = 100 * pow(cpi->summed_quality /
|
|
|
|
cpi->summed_weights, 8.0);
|
|
|
|
double total_ssimp = 100 * pow(cpi->summedp_quality /
|
|
|
|
cpi->summedp_weights, 8.0);
|
|
|
|
|
|
|
|
fprintf(f, "Bitrate\tAVGPsnr\tGLBPsnr\tAVPsnrP\tGLPsnrP\t"
|
|
|
|
"VPXSSIM\tVPSSIMP\t Time(ms)\n");
|
|
|
|
fprintf(f, "%7.2f\t%7.3f\t%7.3f\t%7.3f\t%7.3f\t%7.3f\t%7.3f\t%8.0f\n",
|
|
|
|
dr, cpi->total / cpi->count, total_psnr,
|
|
|
|
cpi->totalp / cpi->count, total_psnr2, total_ssim, total_ssimp,
|
2012-07-14 02:21:29 +04:00
|
|
|
total_encode_time);
|
2013-04-01 20:10:27 +04:00
|
|
|
// fprintf(f, "%7.3f\t%7.3f\t%7.3f\t%7.3f\t%7.3f\t%7.3f\t%8.0f %10ld\n",
|
|
|
|
// dr, cpi->total / cpi->count, total_psnr,
|
|
|
|
// cpi->totalp / cpi->count, total_psnr2, total_ssim,
|
|
|
|
// total_encode_time, cpi->tot_recode_hits);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->b_calculate_ssimg) {
|
2012-08-11 04:05:46 +04:00
|
|
|
fprintf(f, "BitRate\tSSIM_Y\tSSIM_U\tSSIM_V\tSSIM_A\t Time(ms)\n");
|
|
|
|
fprintf(f, "%7.2f\t%6.4f\t%6.4f\t%6.4f\t%6.4f\t%8.0f\n", dr,
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->total_ssimg_y / cpi->count, cpi->total_ssimg_u / cpi->count,
|
|
|
|
cpi->total_ssimg_v / cpi->count, cpi->total_ssimg_all / cpi->count, total_encode_time);
|
2012-01-11 18:05:57 +04:00
|
|
|
// fprintf(f, "%7.3f\t%6.4f\t%6.4f\t%6.4f\t%6.4f\t%8.0f %10ld\n", dr,
|
|
|
|
// cpi->total_ssimg_y / cpi->count, cpi->total_ssimg_u / cpi->count,
|
|
|
|
// cpi->total_ssimg_v / cpi->count, cpi->total_ssimg_all / cpi->count, total_encode_time, cpi->tot_recode_hits);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(f);
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2013-08-21 15:34:14 +04:00
|
|
|
#ifdef MODE_TEST_HIT_STATS
|
|
|
|
if (cpi->pass != 1) {
|
|
|
|
double norm_per_pixel_mode_tests = 0;
|
|
|
|
double norm_counts[BLOCK_SIZES];
|
|
|
|
int i;
|
|
|
|
int sb64_per_frame;
|
|
|
|
int norm_factors[BLOCK_SIZES] =
|
|
|
|
{256, 128, 128, 64, 32, 32, 16, 8, 8, 4, 2, 2, 1};
|
|
|
|
FILE *f = fopen("mode_hit_stats.stt", "a");
|
|
|
|
|
|
|
|
// On average, how many mode tests do we do
|
|
|
|
for (i = 0; i < BLOCK_SIZES; ++i) {
|
|
|
|
norm_counts[i] = (double)cpi->mode_test_hits[i] /
|
|
|
|
(double)norm_factors[i];
|
|
|
|
norm_per_pixel_mode_tests += norm_counts[i];
|
|
|
|
}
|
|
|
|
// Convert to a number per 64x64 and per frame
|
|
|
|
sb64_per_frame = ((cpi->common.height + 63) / 64) *
|
|
|
|
((cpi->common.width + 63) / 64);
|
|
|
|
norm_per_pixel_mode_tests =
|
|
|
|
norm_per_pixel_mode_tests /
|
|
|
|
(double)(cpi->common.current_video_frame * sb64_per_frame);
|
|
|
|
|
|
|
|
fprintf(f, "%6.4f\n", norm_per_pixel_mode_tests);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-05-18 19:58:33 +04:00
|
|
|
#ifdef ENTROPY_STATS
|
2012-07-14 02:21:29 +04:00
|
|
|
{
|
|
|
|
int i, j, k;
|
2012-11-28 01:59:17 +04:00
|
|
|
FILE *fmode = fopen("vp9_modecontext.c", "w");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-28 01:59:17 +04:00
|
|
|
fprintf(fmode, "\n#include \"vp9_entropymode.h\"\n\n");
|
2012-11-01 01:40:53 +04:00
|
|
|
fprintf(fmode, "const unsigned int vp9_kf_default_bmode_counts ");
|
2013-08-23 05:40:34 +04:00
|
|
|
fprintf(fmode, "[INTRA_MODES][INTRA_MODES]"
|
|
|
|
"[INTRA_MODES] =\n{\n");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-23 05:40:34 +04:00
|
|
|
for (i = 0; i < INTRA_MODES; i++) {
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fprintf(fmode, " { // Above Mode : %d\n", i);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-23 05:40:34 +04:00
|
|
|
for (j = 0; j < INTRA_MODES; j++) {
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fprintf(fmode, " {");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-23 05:40:34 +04:00
|
|
|
for (k = 0; k < INTRA_MODES; k++) {
|
2012-07-14 02:21:29 +04:00
|
|
|
if (!intra_mode_stats[i][j][k])
|
|
|
|
fprintf(fmode, " %5d, ", 1);
|
|
|
|
else
|
|
|
|
fprintf(fmode, " %5d, ", intra_mode_stats[i][j][k]);
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fprintf(fmode, "}, // left_mode %d\n", j);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fprintf(fmode, " },\n");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fprintf(fmode, "};\n");
|
|
|
|
fclose(fmode);
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if defined(SECTIONBITS_OUTPUT)
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (0) {
|
|
|
|
int i;
|
|
|
|
FILE *f = fopen("tokenbits.stt", "a");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < 28; i++)
|
|
|
|
fprintf(f, "%8d", (int)(Sectionbits[i] / 256));
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fprintf(f, "\n");
|
|
|
|
fclose(f);
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
2012-07-14 02:21:29 +04:00
|
|
|
{
|
|
|
|
printf("\n_pick_loop_filter_level:%d\n", cpi->time_pick_lpf / 1000);
|
|
|
|
printf("\n_frames recive_data encod_mb_row compress_frame Total\n");
|
2013-07-30 21:16:03 +04:00
|
|
|
printf("%6d %10ld %10ld %10ld %10ld\n", cpi->common.current_video_frame,
|
|
|
|
cpi->time_receive_data / 1000, cpi->time_encode_sb_row / 1000,
|
|
|
|
cpi->time_compress_data / 1000,
|
|
|
|
(cpi->time_receive_data + cpi->time_compress_data) / 1000);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
dealloc_compressor_data(cpi);
|
|
|
|
vpx_free(cpi->mb.ss);
|
|
|
|
vpx_free(cpi->tok);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < sizeof(cpi->mbgraph_stats) / sizeof(cpi->mbgraph_stats[0]); i++) {
|
|
|
|
vpx_free(cpi->mbgraph_stats[i].mb_stats);
|
|
|
|
}
|
2011-10-05 14:26:00 +04:00
|
|
|
|
2012-10-31 03:25:53 +04:00
|
|
|
vp9_remove_common(&cpi->common);
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_free(cpi);
|
|
|
|
*ptr = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#ifdef OUTPUT_YUV_SRC
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(yuv_file);
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
2011-02-15 01:18:18 +03:00
|
|
|
#ifdef OUTPUT_YUV_REC
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(yuv_rec_file);
|
2011-02-15 01:18:18 +03:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (keyfile)
|
|
|
|
fclose(keyfile);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (framepsnr)
|
|
|
|
fclose(framepsnr);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (kf_list)
|
|
|
|
fclose(kf_list);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-19 03:31:19 +04:00
|
|
|
static uint64_t calc_plane_error(uint8_t *orig, int orig_stride,
|
|
|
|
uint8_t *recon, int recon_stride,
|
2012-10-22 07:47:57 +04:00
|
|
|
unsigned int cols, unsigned int rows) {
|
2012-07-14 02:21:29 +04:00
|
|
|
unsigned int row, col;
|
|
|
|
uint64_t total_sse = 0;
|
|
|
|
int diff;
|
|
|
|
|
|
|
|
for (row = 0; row + 16 <= rows; row += 16) {
|
|
|
|
for (col = 0; col + 16 <= cols; col += 16) {
|
|
|
|
unsigned int sse;
|
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_mse16x16(orig + col, orig_stride, recon + col, recon_stride, &sse);
|
2012-07-14 02:21:29 +04:00
|
|
|
total_sse += sse;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle odd-sized width */
|
|
|
|
if (col < cols) {
|
2012-12-19 03:31:19 +04:00
|
|
|
unsigned int border_row, border_col;
|
|
|
|
uint8_t *border_orig = orig;
|
|
|
|
uint8_t *border_recon = recon;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
for (border_row = 0; border_row < 16; border_row++) {
|
|
|
|
for (border_col = col; border_col < cols; border_col++) {
|
|
|
|
diff = border_orig[border_col] - border_recon[border_col];
|
|
|
|
total_sse += diff * diff;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
border_orig += orig_stride;
|
|
|
|
border_recon += recon_stride;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
orig += orig_stride * 16;
|
|
|
|
recon += recon_stride * 16;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
/* Handle odd-sized height */
|
|
|
|
for (; row < rows; row++) {
|
|
|
|
for (col = 0; col < cols; col++) {
|
|
|
|
diff = orig[col] - recon[col];
|
|
|
|
total_sse += diff * diff;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
orig += orig_stride;
|
|
|
|
recon += recon_stride;
|
|
|
|
}
|
|
|
|
|
|
|
|
return total_sse;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void generate_psnr_packet(VP9_COMP *cpi) {
|
2012-07-14 02:21:29 +04:00
|
|
|
YV12_BUFFER_CONFIG *orig = cpi->Source;
|
|
|
|
YV12_BUFFER_CONFIG *recon = cpi->common.frame_to_show;
|
|
|
|
struct vpx_codec_cx_pkt pkt;
|
|
|
|
uint64_t sse;
|
|
|
|
int i;
|
2013-06-10 22:47:22 +04:00
|
|
|
unsigned int width = orig->y_crop_width;
|
|
|
|
unsigned int height = orig->y_crop_height;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
pkt.kind = VPX_CODEC_PSNR_PKT;
|
|
|
|
sse = calc_plane_error(orig->y_buffer, orig->y_stride,
|
|
|
|
recon->y_buffer, recon->y_stride,
|
2012-10-22 07:47:57 +04:00
|
|
|
width, height);
|
2012-07-14 02:21:29 +04:00
|
|
|
pkt.data.psnr.sse[0] = sse;
|
|
|
|
pkt.data.psnr.sse[1] = sse;
|
|
|
|
pkt.data.psnr.samples[0] = width * height;
|
|
|
|
pkt.data.psnr.samples[1] = width * height;
|
|
|
|
|
2013-06-10 22:47:22 +04:00
|
|
|
width = orig->uv_crop_width;
|
|
|
|
height = orig->uv_crop_height;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
sse = calc_plane_error(orig->u_buffer, orig->uv_stride,
|
|
|
|
recon->u_buffer, recon->uv_stride,
|
2012-10-22 07:47:57 +04:00
|
|
|
width, height);
|
2012-07-14 02:21:29 +04:00
|
|
|
pkt.data.psnr.sse[0] += sse;
|
|
|
|
pkt.data.psnr.sse[2] = sse;
|
|
|
|
pkt.data.psnr.samples[0] += width * height;
|
|
|
|
pkt.data.psnr.samples[2] = width * height;
|
|
|
|
|
|
|
|
sse = calc_plane_error(orig->v_buffer, orig->uv_stride,
|
|
|
|
recon->v_buffer, recon->uv_stride,
|
2012-10-22 07:47:57 +04:00
|
|
|
width, height);
|
2012-07-14 02:21:29 +04:00
|
|
|
pkt.data.psnr.sse[0] += sse;
|
|
|
|
pkt.data.psnr.sse[3] = sse;
|
|
|
|
pkt.data.psnr.samples[0] += width * height;
|
|
|
|
pkt.data.psnr.samples[3] = width * height;
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++)
|
2012-10-30 23:58:42 +04:00
|
|
|
pkt.data.psnr.psnr[i] = vp9_mse2psnr(pkt.data.psnr.samples[i], 255.0,
|
2012-11-06 02:22:59 +04:00
|
|
|
(double)pkt.data.psnr.sse[i]);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
vpx_codec_pkt_list_add(cpi->output_pkt_list, &pkt);
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_use_as_reference(VP9_PTR ptr, int ref_frame_flags) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(ptr);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (ref_frame_flags > 7)
|
|
|
|
return -1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->ref_frame_flags = ref_frame_flags;
|
|
|
|
return 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_update_reference(VP9_PTR ptr, int ref_frame_flags) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(ptr);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (ref_frame_flags > 7)
|
|
|
|
return -1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_golden_frame = 0;
|
|
|
|
cpi->refresh_alt_ref_frame = 0;
|
|
|
|
cpi->refresh_last_frame = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
if (ref_frame_flags & VP9_LAST_FLAG)
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_last_frame = 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
if (ref_frame_flags & VP9_GOLD_FLAG)
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_golden_frame = 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
if (ref_frame_flags & VP9_ALT_FLAG)
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_alt_ref_frame = 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-03-13 23:15:43 +04:00
|
|
|
int vp9_copy_reference_enc(VP9_PTR ptr, VP9_REFFRAME ref_frame_flag,
|
|
|
|
YV12_BUFFER_CONFIG *sd) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(ptr);
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 02:21:29 +04:00
|
|
|
int ref_fb_idx;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
if (ref_frame_flag == VP9_LAST_FLAG)
|
2013-02-22 23:22:03 +04:00
|
|
|
ref_fb_idx = cm->ref_frame_map[cpi->lst_fb_idx];
|
2012-11-01 01:40:53 +04:00
|
|
|
else if (ref_frame_flag == VP9_GOLD_FLAG)
|
2013-02-22 23:22:03 +04:00
|
|
|
ref_fb_idx = cm->ref_frame_map[cpi->gld_fb_idx];
|
2012-11-01 01:40:53 +04:00
|
|
|
else if (ref_frame_flag == VP9_ALT_FLAG)
|
2013-02-22 23:22:03 +04:00
|
|
|
ref_fb_idx = cm->ref_frame_map[cpi->alt_fb_idx];
|
2012-07-14 02:21:29 +04:00
|
|
|
else
|
|
|
|
return -1;
|
2010-07-22 16:07:32 +04:00
|
|
|
|
2012-11-02 04:53:44 +04:00
|
|
|
vp8_yv12_copy_frame(&cm->yv12_fb[ref_fb_idx], sd);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
2012-10-31 01:25:33 +04:00
|
|
|
|
2013-03-13 23:15:43 +04:00
|
|
|
int vp9_get_reference_enc(VP9_PTR ptr, int index, YV12_BUFFER_CONFIG **fb) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(ptr);
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
|
|
|
|
|
|
|
if (index < 0 || index >= NUM_REF_FRAMES)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
*fb = &cm->yv12_fb[cm->ref_frame_map[index]];
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
int vp9_set_reference_enc(VP9_PTR ptr, VP9_REFFRAME ref_frame_flag,
|
2012-10-31 01:25:33 +04:00
|
|
|
YV12_BUFFER_CONFIG *sd) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *)(ptr);
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
int ref_fb_idx;
|
2010-09-24 19:10:25 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
if (ref_frame_flag == VP9_LAST_FLAG)
|
2013-02-22 23:22:03 +04:00
|
|
|
ref_fb_idx = cm->ref_frame_map[cpi->lst_fb_idx];
|
2012-11-01 01:40:53 +04:00
|
|
|
else if (ref_frame_flag == VP9_GOLD_FLAG)
|
2013-02-22 23:22:03 +04:00
|
|
|
ref_fb_idx = cm->ref_frame_map[cpi->gld_fb_idx];
|
2012-11-01 01:40:53 +04:00
|
|
|
else if (ref_frame_flag == VP9_ALT_FLAG)
|
2013-02-22 23:22:03 +04:00
|
|
|
ref_fb_idx = cm->ref_frame_map[cpi->alt_fb_idx];
|
2012-07-14 02:21:29 +04:00
|
|
|
else
|
|
|
|
return -1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-02 04:53:44 +04:00
|
|
|
vp8_yv12_copy_frame(sd, &cm->yv12_fb[ref_fb_idx]);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
|
|
|
}
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_update_entropy(VP9_PTR comp, int update) {
|
2013-04-27 01:39:58 +04:00
|
|
|
((VP9_COMP *)comp)->common.refresh_frame_context = update;
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
#ifdef OUTPUT_YUV_SRC
|
2012-11-01 01:40:53 +04:00
|
|
|
void vp9_write_yuv_frame(YV12_BUFFER_CONFIG *s) {
|
2012-12-19 03:31:19 +04:00
|
|
|
uint8_t *src = s->y_buffer;
|
2012-07-14 02:21:29 +04:00
|
|
|
int h = s->y_height;
|
|
|
|
|
|
|
|
do {
|
|
|
|
fwrite(src, s->y_width, 1, yuv_file);
|
|
|
|
src += s->y_stride;
|
|
|
|
} while (--h);
|
|
|
|
|
|
|
|
src = s->u_buffer;
|
|
|
|
h = s->uv_height;
|
|
|
|
|
|
|
|
do {
|
|
|
|
fwrite(src, s->uv_width, 1, yuv_file);
|
|
|
|
src += s->uv_stride;
|
|
|
|
} while (--h);
|
|
|
|
|
|
|
|
src = s->v_buffer;
|
|
|
|
h = s->uv_height;
|
|
|
|
|
|
|
|
do {
|
|
|
|
fwrite(src, s->uv_width, 1, yuv_file);
|
|
|
|
src += s->uv_stride;
|
|
|
|
} while (--h);
|
2011-02-15 01:18:18 +03:00
|
|
|
}
|
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2011-02-15 01:18:18 +03:00
|
|
|
#ifdef OUTPUT_YUV_REC
|
2012-11-01 01:40:53 +04:00
|
|
|
void vp9_write_yuv_rec_frame(VP9_COMMON *cm) {
|
2012-07-14 02:21:29 +04:00
|
|
|
YV12_BUFFER_CONFIG *s = cm->frame_to_show;
|
2012-12-19 03:31:19 +04:00
|
|
|
uint8_t *src = s->y_buffer;
|
2013-03-21 03:41:30 +04:00
|
|
|
int h = cm->height;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
do {
|
|
|
|
fwrite(src, s->y_width, 1, yuv_rec_file);
|
|
|
|
src += s->y_stride;
|
|
|
|
} while (--h);
|
|
|
|
|
|
|
|
src = s->u_buffer;
|
2013-05-09 03:19:56 +04:00
|
|
|
h = s->uv_height;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
do {
|
|
|
|
fwrite(src, s->uv_width, 1, yuv_rec_file);
|
|
|
|
src += s->uv_stride;
|
|
|
|
} while (--h);
|
|
|
|
|
|
|
|
src = s->v_buffer;
|
2013-05-09 03:19:56 +04:00
|
|
|
h = s->uv_height;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
do {
|
|
|
|
fwrite(src, s->uv_width, 1, yuv_rec_file);
|
|
|
|
src += s->uv_stride;
|
|
|
|
} while (--h);
|
2013-05-16 04:55:08 +04:00
|
|
|
|
|
|
|
#if CONFIG_ALPHA
|
|
|
|
if (s->alpha_buffer) {
|
|
|
|
src = s->alpha_buffer;
|
|
|
|
h = s->alpha_height;
|
|
|
|
do {
|
|
|
|
fwrite(src, s->alpha_width, 1, yuv_rec_file);
|
|
|
|
src += s->alpha_stride;
|
|
|
|
} while (--h);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
[WIP] Add column-based tiling.
This patch adds column-based tiling. The idea is to make each tile
independently decodable (after reading the common frame header) and
also independendly encodable (minus within-frame cost adjustments in
the RD loop) to speed-up hardware & software en/decoders if they used
multi-threading. Column-based tiling has the added advantage (over
other tiling methods) that it minimizes realtime use-case latency,
since all threads can start encoding data as soon as the first SB-row
worth of data is available to the encoder.
There is some test code that does random tile ordering in the decoder,
to confirm that each tile is indeed independently decodable from other
tiles in the same frame. At tile edges, all contexts assume default
values (i.e. 0, 0 motion vector, no coefficients, DC intra4x4 mode),
and motion vector search and ordering do not cross tiles in the same
frame.
t log
Tile independence is not maintained between frames ATM, i.e. tile 0 of
frame 1 is free to use motion vectors that point into any tile of frame
0. We support 1 (i.e. no tiling), 2 or 4 column-tiles.
The loopfilter crosses tile boundaries. I discussed this briefly with Aki
and he says that's OK. An in-loop loopfilter would need to do some sync
between tile threads, but that shouldn't be a big issue.
Resuls: with tiling disabled, we go up slightly because of improved edge
use in the intra4x4 prediction. With 2 tiles, we lose about ~1% on derf,
~0.35% on HD and ~0.55% on STD/HD. With 4 tiles, we lose another ~1.5%
on derf ~0.77% on HD and ~0.85% on STD/HD. Most of this loss is
concentrated in the low-bitrate end of clips, and most of it is because
of the loss of edges at tile boundaries and the resulting loss of intra
predictors.
TODO:
- more tiles (perhaps allow row-based tiling also, and max. 8 tiles)?
- maybe optionally (for EC purposes), motion vectors themselves
should not cross tile edges, or we should emulate such borders as
if they were off-frame, to limit error propagation to within one
tile only. This doesn't have to be the default behaviour but could
be an optional bitstream flag.
Change-Id: I5951c3a0742a767b20bc9fb5af685d9892c2c96f
2013-02-01 21:35:28 +04:00
|
|
|
fflush(yuv_rec_file);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
#endif
|
2011-02-15 01:18:18 +03:00
|
|
|
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 08:55:14 +04:00
|
|
|
static void scale_and_extend_frame(YV12_BUFFER_CONFIG *src_fb,
|
|
|
|
YV12_BUFFER_CONFIG *dst_fb) {
|
2013-03-14 04:09:05 +04:00
|
|
|
const int in_w = src_fb->y_crop_width;
|
|
|
|
const int in_h = src_fb->y_crop_height;
|
|
|
|
const int out_w = dst_fb->y_crop_width;
|
|
|
|
const int out_h = dst_fb->y_crop_height;
|
2013-05-14 03:02:29 +04:00
|
|
|
int x, y, i;
|
|
|
|
|
2013-05-16 04:55:08 +04:00
|
|
|
uint8_t *srcs[4] = {src_fb->y_buffer, src_fb->u_buffer, src_fb->v_buffer,
|
|
|
|
src_fb->alpha_buffer};
|
|
|
|
int src_strides[4] = {src_fb->y_stride, src_fb->uv_stride, src_fb->uv_stride,
|
|
|
|
src_fb->alpha_stride};
|
2013-05-14 03:02:29 +04:00
|
|
|
|
2013-05-16 04:55:08 +04:00
|
|
|
uint8_t *dsts[4] = {dst_fb->y_buffer, dst_fb->u_buffer, dst_fb->v_buffer,
|
|
|
|
dst_fb->alpha_buffer};
|
|
|
|
int dst_strides[4] = {dst_fb->y_stride, dst_fb->uv_stride, dst_fb->uv_stride,
|
|
|
|
dst_fb->alpha_stride};
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 08:55:14 +04:00
|
|
|
|
|
|
|
for (y = 0; y < out_h; y += 16) {
|
|
|
|
for (x = 0; x < out_w; x += 16) {
|
2013-05-14 03:02:29 +04:00
|
|
|
for (i = 0; i < MAX_MB_PLANE; ++i) {
|
|
|
|
const int factor = i == 0 ? 1 : 2;
|
|
|
|
const int x_q4 = x * (16 / factor) * in_w / out_w;
|
|
|
|
const int y_q4 = y * (16 / factor) * in_h / out_h;
|
|
|
|
const int src_stride = src_strides[i];
|
|
|
|
const int dst_stride = dst_strides[i];
|
|
|
|
uint8_t *src = srcs[i] + y / factor * in_h / out_h * src_stride +
|
|
|
|
x / factor * in_w / out_w;
|
2013-06-12 21:09:56 +04:00
|
|
|
uint8_t *dst = dsts[i] + y / factor * dst_stride + x / factor;
|
2013-05-14 03:02:29 +04:00
|
|
|
|
|
|
|
vp9_convolve8(src, src_stride, dst, dst_stride,
|
|
|
|
vp9_sub_pel_filters_8[x_q4 & 0xf], 16 * in_w / out_w,
|
|
|
|
vp9_sub_pel_filters_8[y_q4 & 0xf], 16 * in_h / out_h,
|
|
|
|
16 / factor, 16 / factor);
|
|
|
|
}
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 08:55:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vp8_yv12_extend_frame_borders(dst_fb);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void update_alt_ref_frame_stats(VP9_COMP *cpi) {
|
2012-07-14 02:21:29 +04:00
|
|
|
// this frame refreshes means next frames don't unless specified by user
|
2013-07-19 01:09:21 +04:00
|
|
|
cpi->frames_since_golden = 0;
|
2011-02-15 01:18:18 +03:00
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
if (!cpi->multi_arf_enabled)
|
|
|
|
#endif
|
|
|
|
// Clear the alternate reference update pending flag.
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->source_alt_ref_pending = 0;
|
2011-02-15 01:18:18 +03:00
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
// Set the alternate reference frame active flag
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->source_alt_ref_active = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2012-10-31 04:53:32 +04:00
|
|
|
static void update_golden_frame_stats(VP9_COMP *cpi) {
|
2012-07-14 02:21:29 +04:00
|
|
|
// Update the Golden frame usage counts.
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_golden_frame) {
|
2011-01-10 12:14:10 +03:00
|
|
|
// this frame refreshes means next frames don't unless specified by user
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_golden_frame = 0;
|
2013-07-19 01:09:21 +04:00
|
|
|
cpi->frames_since_golden = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// ******** Fixed Q test code only ************
|
|
|
|
// If we are going to use the ALT reference for the next group of frames set a flag to say so.
|
|
|
|
if (cpi->oxcf.fixed_q >= 0 &&
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->oxcf.play_alternate && !cpi->refresh_alt_ref_frame) {
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->source_alt_ref_pending = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->frames_till_gf_update_due = cpi->baseline_gf_interval;
|
2013-09-05 19:55:47 +04:00
|
|
|
|
|
|
|
// TODO(ivan): for SVC encoder, GF automatic update is disabled by using a
|
|
|
|
// large GF_interval
|
|
|
|
if (cpi->use_svc) {
|
|
|
|
cpi->frames_till_gf_update_due = INT_MAX;
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!cpi->source_alt_ref_pending)
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->source_alt_ref_active = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Decrement count down till next gf
|
|
|
|
if (cpi->frames_till_gf_update_due > 0)
|
|
|
|
cpi->frames_till_gf_update_due--;
|
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
} else if (!cpi->refresh_alt_ref_frame) {
|
2012-07-14 02:21:29 +04:00
|
|
|
// Decrement count down till next gf
|
|
|
|
if (cpi->frames_till_gf_update_due > 0)
|
|
|
|
cpi->frames_till_gf_update_due--;
|
|
|
|
|
2013-07-19 01:09:21 +04:00
|
|
|
if (cpi->frames_till_alt_ref_frame)
|
|
|
|
cpi->frames_till_alt_ref_frame--;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-07-19 01:09:21 +04:00
|
|
|
cpi->frames_since_golden++;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-10-30 05:20:32 +04:00
|
|
|
static int find_fp_qindex() {
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
2011-12-02 18:57:21 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < QINDEX_RANGE; i++) {
|
2012-10-30 23:58:42 +04:00
|
|
|
if (vp9_convert_qindex_to_q(i) >= 30.0) {
|
2012-07-14 02:21:29 +04:00
|
|
|
break;
|
2011-12-02 18:57:21 +04:00
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-12-02 02:50:14 +03:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (i == QINDEX_RANGE)
|
|
|
|
i--;
|
2011-12-02 18:57:21 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return i;
|
2011-12-02 18:57:21 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void Pass1Encode(VP9_COMP *cpi, unsigned long *size, unsigned char *dest, unsigned int *frame_flags) {
|
2012-07-14 02:21:29 +04:00
|
|
|
(void) size;
|
|
|
|
(void) dest;
|
|
|
|
(void) frame_flags;
|
2011-12-02 18:57:21 +04:00
|
|
|
|
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_set_quantizer(cpi, find_fp_qindex());
|
|
|
|
vp9_first_pass(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
2012-04-07 03:38:34 +04:00
|
|
|
|
2012-08-14 03:50:03 +04:00
|
|
|
#define WRITE_RECON_BUFFER 0
|
2011-11-16 04:16:30 +04:00
|
|
|
#if WRITE_RECON_BUFFER
|
2012-07-14 02:21:29 +04:00
|
|
|
void write_cx_frame_to_file(YV12_BUFFER_CONFIG *frame, int this_frame) {
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// write the frame
|
|
|
|
FILE *yframe;
|
|
|
|
int i;
|
|
|
|
char filename[255];
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
sprintf(filename, "cx\\y%04d.raw", this_frame);
|
|
|
|
yframe = fopen(filename, "wb");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < frame->y_height; i++)
|
|
|
|
fwrite(frame->y_buffer + i * frame->y_stride,
|
|
|
|
frame->y_width, 1, yframe);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(yframe);
|
|
|
|
sprintf(filename, "cx\\u%04d.raw", this_frame);
|
|
|
|
yframe = fopen(filename, "wb");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < frame->uv_height; i++)
|
|
|
|
fwrite(frame->u_buffer + i * frame->uv_stride,
|
|
|
|
frame->uv_width, 1, yframe);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(yframe);
|
|
|
|
sprintf(filename, "cx\\v%04d.raw", this_frame);
|
|
|
|
yframe = fopen(filename, "wb");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < frame->uv_height; i++)
|
|
|
|
fwrite(frame->v_buffer + i * frame->uv_stride,
|
|
|
|
frame->uv_width, 1, yframe);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(yframe);
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
static double compute_edge_pixel_proportion(YV12_BUFFER_CONFIG *frame) {
|
2012-03-19 18:53:05 +04:00
|
|
|
#define EDGE_THRESH 128
|
2012-07-14 02:21:29 +04:00
|
|
|
int i, j;
|
|
|
|
int num_edge_pels = 0;
|
|
|
|
int num_pels = (frame->y_height - 2) * (frame->y_width - 2);
|
2012-12-19 03:31:19 +04:00
|
|
|
uint8_t *prev = frame->y_buffer + 1;
|
|
|
|
uint8_t *curr = frame->y_buffer + 1 + frame->y_stride;
|
|
|
|
uint8_t *next = frame->y_buffer + 1 + 2 * frame->y_stride;
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 1; i < frame->y_height - 1; i++) {
|
|
|
|
for (j = 1; j < frame->y_width - 1; j++) {
|
|
|
|
/* Sobel hor and ver gradients */
|
|
|
|
int v = 2 * (curr[1] - curr[-1]) + (prev[1] - prev[-1]) + (next[1] - next[-1]);
|
|
|
|
int h = 2 * (prev[0] - next[0]) + (prev[1] - next[1]) + (prev[-1] - next[-1]);
|
|
|
|
h = (h < 0 ? -h : h);
|
|
|
|
v = (v < 0 ? -v : v);
|
2013-04-02 21:24:56 +04:00
|
|
|
if (h > EDGE_THRESH || v > EDGE_THRESH)
|
|
|
|
num_edge_pels++;
|
2012-07-14 02:21:29 +04:00
|
|
|
curr++;
|
|
|
|
prev++;
|
|
|
|
next++;
|
|
|
|
}
|
|
|
|
curr += frame->y_stride - frame->y_width + 2;
|
|
|
|
prev += frame->y_stride - frame->y_width + 2;
|
|
|
|
next += frame->y_stride - frame->y_width + 2;
|
|
|
|
}
|
2013-04-02 21:24:56 +04:00
|
|
|
return (double)num_edge_pels / num_pels;
|
2012-03-19 18:53:05 +04:00
|
|
|
}
|
|
|
|
|
2012-04-07 03:38:34 +04:00
|
|
|
// Function to test for conditions that indicate we should loop
|
2010-11-17 18:12:04 +03:00
|
|
|
// back and recode a frame.
|
2012-12-15 00:35:33 +04:00
|
|
|
static int recode_loop_test(VP9_COMP *cpi,
|
|
|
|
int high_limit, int low_limit,
|
|
|
|
int q, int maxq, int minq) {
|
2013-04-16 02:24:39 +04:00
|
|
|
int force_recode = 0;
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Is frame recode allowed at all
|
2013-04-03 02:08:50 +04:00
|
|
|
// Yes if either recode mode 1 is selected or mode two is selected
|
2012-07-14 02:21:29 +04:00
|
|
|
// and the frame is a key frame. golden frame or alt_ref_frame
|
|
|
|
if ((cpi->sf.recode_loop == 1) ||
|
|
|
|
((cpi->sf.recode_loop == 2) &&
|
|
|
|
((cm->frame_type == KEY_FRAME) ||
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_golden_frame ||
|
|
|
|
cpi->refresh_alt_ref_frame))) {
|
2012-07-14 02:21:29 +04:00
|
|
|
// General over and under shoot tests
|
|
|
|
if (((cpi->projected_frame_size > high_limit) && (q < maxq)) ||
|
|
|
|
((cpi->projected_frame_size < low_limit) && (q > minq))) {
|
2013-04-16 02:24:39 +04:00
|
|
|
force_recode = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
// Special Constrained quality tests
|
|
|
|
else if (cpi->oxcf.end_usage == USAGE_CONSTRAINED_QUALITY) {
|
|
|
|
// Undershoot and below auto cq level
|
2013-04-02 21:24:56 +04:00
|
|
|
if (q > cpi->cq_target_quality &&
|
|
|
|
cpi->projected_frame_size < ((cpi->this_frame_target * 7) >> 3)) {
|
2013-04-16 02:24:39 +04:00
|
|
|
force_recode = 1;
|
2013-04-02 21:24:56 +04:00
|
|
|
} else if (q > cpi->oxcf.cq_level &&
|
|
|
|
cpi->projected_frame_size < cpi->min_frame_bandwidth &&
|
|
|
|
cpi->active_best_quality > cpi->oxcf.cq_level) {
|
|
|
|
// Severe undershoot and between auto and user cq level
|
2013-04-16 02:24:39 +04:00
|
|
|
force_recode = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->active_best_quality = cpi->oxcf.cq_level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return force_recode;
|
2010-11-17 18:12:04 +03:00
|
|
|
}
|
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
static void update_reference_frames(VP9_COMP * const cpi) {
|
|
|
|
VP9_COMMON * const cm = &cpi->common;
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// At this point the new frame has been encoded.
|
|
|
|
// If any buffer copy / swapping is signaled it should be done here.
|
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
2013-01-16 01:49:44 +04:00
|
|
|
ref_cnt_fb(cm->fb_idx_ref_cnt,
|
2013-02-22 23:22:03 +04:00
|
|
|
&cm->ref_frame_map[cpi->gld_fb_idx], cm->new_fb_idx);
|
2013-01-16 01:49:44 +04:00
|
|
|
ref_cnt_fb(cm->fb_idx_ref_cnt,
|
2013-02-22 23:22:03 +04:00
|
|
|
&cm->ref_frame_map[cpi->alt_fb_idx], cm->new_fb_idx);
|
2013-04-03 02:08:50 +04:00
|
|
|
}
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
else if (!cpi->multi_arf_enabled && cpi->refresh_golden_frame &&
|
|
|
|
!cpi->refresh_alt_ref_frame) {
|
|
|
|
#else
|
2013-09-05 19:55:47 +04:00
|
|
|
else if (cpi->refresh_golden_frame && !cpi->refresh_alt_ref_frame &&
|
|
|
|
!cpi->use_svc) {
|
2013-04-03 02:08:50 +04:00
|
|
|
#endif
|
2013-01-17 00:19:42 +04:00
|
|
|
/* Preserve the previously existing golden frame and update the frame in
|
|
|
|
* the alt ref slot instead. This is highly specific to the current use of
|
|
|
|
* alt-ref as a forward reference, and this needs to be generalized as
|
|
|
|
* other uses are implemented (like RTC/temporal scaling)
|
|
|
|
*
|
2013-04-03 02:08:50 +04:00
|
|
|
* The update to the buffer in the alt ref slot was signaled in
|
2013-01-17 00:19:42 +04:00
|
|
|
* vp9_pack_bitstream(), now swap the buffer pointers so that it's treated
|
|
|
|
* as the golden frame next time.
|
|
|
|
*/
|
|
|
|
int tmp;
|
|
|
|
|
|
|
|
ref_cnt_fb(cm->fb_idx_ref_cnt,
|
2013-02-22 23:22:03 +04:00
|
|
|
&cm->ref_frame_map[cpi->alt_fb_idx], cm->new_fb_idx);
|
2013-01-17 00:19:42 +04:00
|
|
|
|
|
|
|
tmp = cpi->alt_fb_idx;
|
|
|
|
cpi->alt_fb_idx = cpi->gld_fb_idx;
|
|
|
|
cpi->gld_fb_idx = tmp;
|
2013-04-03 02:08:50 +04:00
|
|
|
} else { /* For non key/golden frames */
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_alt_ref_frame) {
|
2013-04-03 02:08:50 +04:00
|
|
|
int arf_idx = cpi->alt_fb_idx;
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
if (cpi->multi_arf_enabled) {
|
|
|
|
arf_idx = cpi->arf_buffer_idx[cpi->sequence_number + 1];
|
|
|
|
}
|
|
|
|
#endif
|
2013-01-16 01:49:44 +04:00
|
|
|
ref_cnt_fb(cm->fb_idx_ref_cnt,
|
2013-04-03 02:08:50 +04:00
|
|
|
&cm->ref_frame_map[arf_idx], cm->new_fb_idx);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2011-04-20 16:12:23 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_golden_frame) {
|
|
|
|
ref_cnt_fb(cm->fb_idx_ref_cnt,
|
2013-02-22 23:22:03 +04:00
|
|
|
&cm->ref_frame_map[cpi->gld_fb_idx], cm->new_fb_idx);
|
2011-04-20 16:12:23 +04:00
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2011-04-20 16:12:23 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_last_frame) {
|
|
|
|
ref_cnt_fb(cm->fb_idx_ref_cnt,
|
2013-02-22 23:22:03 +04:00
|
|
|
&cm->ref_frame_map[cpi->lst_fb_idx], cm->new_fb_idx);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2011-04-20 16:12:23 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void loopfilter_frame(VP9_COMP *cpi, VP9_COMMON *cm) {
|
2013-07-18 05:37:45 +04:00
|
|
|
MACROBLOCKD *xd = &cpi->mb.e_mbd;
|
2013-08-10 01:41:51 +04:00
|
|
|
struct loopfilter *lf = &cm->lf;
|
2013-07-18 05:37:45 +04:00
|
|
|
if (xd->lossless) {
|
2013-08-02 01:53:14 +04:00
|
|
|
lf->filter_level = 0;
|
2013-02-20 19:27:35 +04:00
|
|
|
} else {
|
2012-07-14 02:21:29 +04:00
|
|
|
struct vpx_usec_timer timer;
|
2011-02-25 14:42:05 +03:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state();
|
2011-02-25 14:42:05 +03:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_usec_timer_start(&timer);
|
2013-04-11 00:45:22 +04:00
|
|
|
|
2013-08-07 23:29:45 +04:00
|
|
|
vp9_pick_filter_level(cpi->Source, cpi, cpi->sf.use_fast_lpf_pick);
|
2011-02-25 14:42:05 +03:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_usec_timer_mark(&timer);
|
|
|
|
cpi->time_pick_lpf += vpx_usec_timer_elapsed(&timer);
|
|
|
|
}
|
2011-02-25 14:42:05 +03:00
|
|
|
|
2013-08-02 01:53:14 +04:00
|
|
|
if (lf->filter_level > 0) {
|
|
|
|
vp9_set_alt_lf_level(cpi, lf->filter_level);
|
2013-08-07 23:29:45 +04:00
|
|
|
vp9_loop_filter_frame(cm, xd, lf->filter_level, 0, 0);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2011-02-25 14:42:05 +03:00
|
|
|
|
2013-07-16 01:59:59 +04:00
|
|
|
vp9_extend_frame_inner_borders(cm->frame_to_show,
|
2013-07-18 05:37:45 +04:00
|
|
|
cm->subsampling_x, cm->subsampling_y);
|
2011-02-25 14:42:05 +03:00
|
|
|
}
|
|
|
|
|
2013-02-21 00:34:31 +04:00
|
|
|
static void scale_references(VP9_COMP *cpi) {
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
|
|
|
int i;
|
2013-09-05 19:55:47 +04:00
|
|
|
int refs[ALLOWED_REFS_PER_FRAME] = {cpi->lst_fb_idx, cpi->gld_fb_idx,
|
|
|
|
cpi->alt_fb_idx};
|
2013-02-21 00:34:31 +04:00
|
|
|
|
|
|
|
for (i = 0; i < 3; i++) {
|
2013-09-05 19:55:47 +04:00
|
|
|
YV12_BUFFER_CONFIG *ref = &cm->yv12_fb[cm->ref_frame_map[refs[i]]];
|
2013-02-21 00:34:31 +04:00
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
if (ref->y_crop_width != cm->width ||
|
|
|
|
ref->y_crop_height != cm->height) {
|
2013-02-21 00:34:31 +04:00
|
|
|
int new_fb = get_free_fb(cm);
|
|
|
|
|
2013-05-07 02:52:06 +04:00
|
|
|
vp9_realloc_frame_buffer(&cm->yv12_fb[new_fb],
|
|
|
|
cm->width, cm->height,
|
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
|
|
|
VP9BORDERINPIXELS);
|
2013-02-21 00:34:31 +04:00
|
|
|
scale_and_extend_frame(ref, &cm->yv12_fb[new_fb]);
|
|
|
|
cpi->scaled_ref_idx[i] = new_fb;
|
|
|
|
} else {
|
2013-09-05 19:55:47 +04:00
|
|
|
cpi->scaled_ref_idx[i] = cm->ref_frame_map[refs[i]];
|
|
|
|
cm->fb_idx_ref_cnt[cm->ref_frame_map[refs[i]]]++;
|
2013-02-21 00:34:31 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void release_scaled_references(VP9_COMP *cpi) {
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
|
|
|
int i;
|
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
for (i = 0; i < 3; i++)
|
2013-02-21 00:34:31 +04:00
|
|
|
cm->fb_idx_ref_cnt[cpi->scaled_ref_idx[i]]--;
|
|
|
|
}
|
|
|
|
|
2013-06-25 02:46:15 +04:00
|
|
|
static void full_to_model_count(unsigned int *model_count,
|
|
|
|
unsigned int *full_count) {
|
|
|
|
int n;
|
|
|
|
model_count[ZERO_TOKEN] = full_count[ZERO_TOKEN];
|
|
|
|
model_count[ONE_TOKEN] = full_count[ONE_TOKEN];
|
|
|
|
model_count[TWO_TOKEN] = full_count[TWO_TOKEN];
|
|
|
|
for (n = THREE_TOKEN; n < DCT_EOB_TOKEN; ++n)
|
|
|
|
model_count[TWO_TOKEN] += full_count[n];
|
|
|
|
model_count[DCT_EOB_MODEL_TOKEN] = full_count[DCT_EOB_TOKEN];
|
|
|
|
}
|
|
|
|
|
|
|
|
static void full_to_model_counts(
|
|
|
|
vp9_coeff_count_model *model_count, vp9_coeff_count *full_count) {
|
|
|
|
int i, j, k, l;
|
|
|
|
for (i = 0; i < BLOCK_TYPES; ++i)
|
|
|
|
for (j = 0; j < REF_TYPES; ++j)
|
|
|
|
for (k = 0; k < COEF_BANDS; ++k)
|
|
|
|
for (l = 0; l < PREV_COEF_CONTEXTS; ++l) {
|
|
|
|
if (l >= 3 && k == 0)
|
|
|
|
continue;
|
2013-06-28 21:36:20 +04:00
|
|
|
full_to_model_count(model_count[i][j][k][l], full_count[i][j][k][l]);
|
2013-06-25 02:46:15 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-19 03:31:19 +04:00
|
|
|
static void encode_frame_to_data_rate(VP9_COMP *cpi,
|
|
|
|
unsigned long *size,
|
|
|
|
unsigned char *dest,
|
|
|
|
unsigned int *frame_flags) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 02:21:29 +04:00
|
|
|
MACROBLOCKD *xd = &cpi->mb.e_mbd;
|
2013-05-31 20:18:59 +04:00
|
|
|
TX_SIZE t;
|
2013-04-02 21:24:56 +04:00
|
|
|
int q;
|
2012-07-14 02:21:29 +04:00
|
|
|
int frame_over_shoot_limit;
|
|
|
|
int frame_under_shoot_limit;
|
|
|
|
|
2013-04-16 02:24:39 +04:00
|
|
|
int loop = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
int loop_count;
|
|
|
|
|
|
|
|
int q_low;
|
|
|
|
int q_high;
|
|
|
|
|
|
|
|
int top_index;
|
|
|
|
int bottom_index;
|
2013-04-16 02:24:39 +04:00
|
|
|
int active_worst_qchanged = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-16 02:24:39 +04:00
|
|
|
int overshoot_seen = 0;
|
|
|
|
int undershoot_seen = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
SPEED_FEATURES *sf = &cpi->sf;
|
2013-07-12 20:52:24 +04:00
|
|
|
unsigned int max_mv_def = MIN(cpi->common.width, cpi->common.height);
|
2013-08-14 22:20:33 +04:00
|
|
|
struct segmentation *seg = &cm->seg;
|
2012-04-17 01:53:37 +04:00
|
|
|
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 08:55:14 +04:00
|
|
|
/* Scale the source buffer, if required */
|
2013-06-08 04:20:50 +04:00
|
|
|
if (cm->mi_cols * 8 != cpi->un_scaled_source->y_width ||
|
|
|
|
cm->mi_rows * 8 != cpi->un_scaled_source->y_height) {
|
Spatial resamping of ZEROMV predictors
This patch allows coding frames using references of different
resolution, in ZEROMV mode. For compound prediction, either
reference may be scaled.
To test, I use the resize_test and enable WRITE_RECON_BUFFER
in vp9_onyxd_if.c. It's also useful to apply this patch to
test/i420_video_source.h:
--- a/test/i420_video_source.h
+++ b/test/i420_video_source.h
@@ -93,6 +93,7 @@ class I420VideoSource : public VideoSource {
virtual void FillFrame() {
// Read a frame from input_file.
+ if (frame_ != 3)
if (fread(img_->img_data, raw_sz_, 1, input_file_) == 0) {
limit_ = frame_;
}
This forces the frame that the resolution changes on to be coded
with no motion, only scaling, and improves the quality of the
result.
Change-Id: I1ee75d19a437ff801192f767fd02a36bcbd1d496
2013-02-25 08:55:14 +04:00
|
|
|
scale_and_extend_frame(cpi->un_scaled_source, &cpi->scaled_source);
|
|
|
|
cpi->Source = &cpi->scaled_source;
|
|
|
|
} else {
|
|
|
|
cpi->Source = cpi->un_scaled_source;
|
|
|
|
}
|
|
|
|
|
2013-02-21 00:34:31 +04:00
|
|
|
scale_references(cpi);
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Clear down mmx registers to allow floating point in what follows
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state();
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-09-10 09:42:35 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// For an alt ref frame in 2 pass we skip the call to the second
|
|
|
|
// pass function that sets the target bandwidth so must set it here
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_alt_ref_frame) {
|
2013-04-03 02:08:50 +04:00
|
|
|
// Per frame bit target for the alt ref frame
|
|
|
|
cpi->per_frame_bandwidth = cpi->twopass.gf_bits;
|
2012-11-06 02:22:59 +04:00
|
|
|
// per second target bitrate
|
|
|
|
cpi->target_bandwidth = (int)(cpi->twopass.gf_bits *
|
2013-07-13 04:12:46 +04:00
|
|
|
cpi->output_framerate);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Clear zbin over-quant value and mode boost values.
|
|
|
|
cpi->zbin_mode_boost = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Enable or disable mode based tweaking of the zbin
|
|
|
|
// For 2 Pass Only used where GF/ARF prediction quality
|
|
|
|
// is above a threshold
|
|
|
|
cpi->zbin_mode_boost = 0;
|
2013-02-20 19:27:35 +04:00
|
|
|
|
2013-03-22 19:47:17 +04:00
|
|
|
// if (cpi->oxcf.lossless)
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->zbin_mode_boost_enabled = 0;
|
2013-03-22 19:47:17 +04:00
|
|
|
// else
|
2013-04-16 02:24:39 +04:00
|
|
|
// cpi->zbin_mode_boost_enabled = 1;
|
2013-02-20 19:27:35 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Current default encoder behaviour for the altref sign bias
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->common.ref_frame_sign_bias[ALTREF_FRAME] = cpi->source_alt_ref_active;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
// Check to see if a key frame is signaled
|
2012-07-14 02:21:29 +04:00
|
|
|
// For two pass with auto key frame enabled cm->frame_type may already be set, but not for one pass.
|
|
|
|
if ((cm->current_video_frame == 0) ||
|
|
|
|
(cm->frame_flags & FRAMEFLAGS_KEY) ||
|
|
|
|
(cpi->oxcf.auto_key && (cpi->frames_since_key % cpi->key_frame_frequency == 0))) {
|
|
|
|
// Key frame from VFW/auto-keyframe/first frame
|
|
|
|
cm->frame_type = KEY_FRAME;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set default state for segment based loop filter update flags
|
2013-08-10 01:41:51 +04:00
|
|
|
cm->lf.mode_ref_delta_update = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-07-12 20:52:24 +04:00
|
|
|
// Initialize cpi->mv_step_param to default based on max resolution
|
|
|
|
cpi->mv_step_param = vp9_init_search_range(cpi, max_mv_def);
|
|
|
|
// Initialize cpi->max_mv_magnitude and cpi->mv_step_param if appropriate.
|
|
|
|
if (sf->auto_mv_step_size) {
|
|
|
|
if ((cpi->common.frame_type == KEY_FRAME) || cpi->common.intra_only) {
|
|
|
|
// initialize max_mv_magnitude for use in the first INTER frame
|
|
|
|
// after a key/intra-only frame
|
|
|
|
cpi->max_mv_magnitude = max_mv_def;
|
|
|
|
} else {
|
|
|
|
if (cm->show_frame)
|
|
|
|
// allow mv_steps to correspond to twice the max mv magnitude found
|
|
|
|
// in the previous frame, capped by the default max_mv_magnitude based
|
|
|
|
// on resolution
|
|
|
|
cpi->mv_step_param = vp9_init_search_range(
|
|
|
|
cpi, MIN(max_mv_def, 2 * cpi->max_mv_magnitude));
|
|
|
|
cpi->max_mv_magnitude = 0;
|
|
|
|
}
|
|
|
|
}
|
2012-11-09 22:52:08 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Set various flags etc to special state if it is a key frame
|
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
|
|
|
// Reset the loop filter deltas and segmentation map
|
2013-08-14 22:20:33 +04:00
|
|
|
setup_features(cm);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// If segmentation is enabled force a map update for key frames
|
2013-08-02 01:53:14 +04:00
|
|
|
if (seg->enabled) {
|
|
|
|
seg->update_map = 1;
|
|
|
|
seg->update_data = 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// The alternate reference frame cannot be active for a key frame
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->source_alt_ref_active = 0;
|
2011-10-05 14:26:00 +04:00
|
|
|
|
2013-01-15 18:43:35 +04:00
|
|
|
cm->error_resilient_mode = (cpi->oxcf.error_resilient_mode != 0);
|
2013-01-25 23:30:28 +04:00
|
|
|
cm->frame_parallel_decoding_mode =
|
|
|
|
(cpi->oxcf.frame_parallel_decoding_mode != 0);
|
|
|
|
if (cm->error_resilient_mode) {
|
|
|
|
cm->frame_parallel_decoding_mode = 1;
|
2013-05-30 04:16:00 +04:00
|
|
|
cm->reset_frame_context = 0;
|
2013-04-27 01:39:58 +04:00
|
|
|
cm->refresh_frame_context = 0;
|
2013-01-25 23:30:28 +04:00
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-23 17:01:55 +04:00
|
|
|
// Configure experimental use of segmentation for enhanced coding of
|
|
|
|
// static regions if indicated.
|
2013-02-05 14:13:25 +04:00
|
|
|
// Only allowed for now in second pass of two pass (as requires lagged coding)
|
2013-04-03 02:08:50 +04:00
|
|
|
// and if the relevant speed feature flag is set.
|
2013-02-05 14:13:25 +04:00
|
|
|
if ((cpi->pass == 2) && (cpi->sf.static_segmentation)) {
|
|
|
|
configure_static_seg_features(cpi);
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Decide how big to make the frame
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_pick_frame_size(cpi);
|
2012-06-07 16:39:20 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state();
|
2012-06-07 16:39:20 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Set an active best quality and if necessary active worst quality
|
2013-04-02 21:24:56 +04:00
|
|
|
q = cpi->active_worst_quality;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
2013-04-16 21:59:39 +04:00
|
|
|
#if !CONFIG_MULTIPLE_ARF
|
2013-08-30 03:21:44 +04:00
|
|
|
// Special case for key frames forced because we have reached
|
|
|
|
// the maximum key frame interval. Here force the Q to a range
|
|
|
|
// based on the ambient Q to reduce the risk of popping
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->this_key_frame_forced) {
|
2013-03-28 23:46:35 +04:00
|
|
|
int delta_qindex;
|
|
|
|
int qindex = cpi->last_boosted_qindex;
|
|
|
|
double last_boosted_q = vp9_convert_qindex_to_q(qindex);
|
|
|
|
|
|
|
|
delta_qindex = compute_qdelta(cpi, last_boosted_q,
|
|
|
|
(last_boosted_q * 0.75));
|
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
cpi->active_best_quality = MAX(qindex + delta_qindex,
|
|
|
|
cpi->best_quality);
|
2013-04-16 21:59:39 +04:00
|
|
|
} else {
|
|
|
|
int high = 5000;
|
|
|
|
int low = 400;
|
|
|
|
double q_adj_factor = 1.0;
|
|
|
|
double q_val;
|
|
|
|
|
|
|
|
// Baseline value derived from cpi->active_worst_quality and kf boost
|
|
|
|
if (cpi->kf_boost > high) {
|
|
|
|
cpi->active_best_quality = kf_low_motion_minq[q];
|
|
|
|
} else if (cpi->kf_boost < low) {
|
|
|
|
cpi->active_best_quality = kf_high_motion_minq[q];
|
|
|
|
} else {
|
|
|
|
const int gap = high - low;
|
|
|
|
const int offset = high - cpi->kf_boost;
|
|
|
|
const int qdiff = kf_high_motion_minq[q] - kf_low_motion_minq[q];
|
|
|
|
const int adjustment = ((offset * qdiff) + (gap >> 1)) / gap;
|
|
|
|
|
|
|
|
cpi->active_best_quality = kf_low_motion_minq[q] + adjustment;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allow somewhat lower kf minq with small image formats.
|
|
|
|
if ((cm->width * cm->height) <= (352 * 288)) {
|
|
|
|
q_adj_factor -= 0.25;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make a further adjustment based on the kf zero motion measure.
|
|
|
|
q_adj_factor += 0.05 - (0.001 * (double)cpi->kf_zeromotion_pct);
|
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
// Convert the adjustment factor to a qindex delta
|
|
|
|
// on active_best_quality.
|
2013-04-16 21:59:39 +04:00
|
|
|
q_val = vp9_convert_qindex_to_q(cpi->active_best_quality);
|
|
|
|
cpi->active_best_quality +=
|
2013-08-30 03:21:44 +04:00
|
|
|
compute_qdelta(cpi, q_val, (q_val * q_adj_factor));
|
2012-01-11 18:05:57 +04:00
|
|
|
}
|
2013-04-16 21:59:39 +04:00
|
|
|
#else
|
|
|
|
double current_q;
|
2013-04-03 02:08:50 +04:00
|
|
|
// Force the KF quantizer to be 30% of the active_worst_quality.
|
|
|
|
current_q = vp9_convert_qindex_to_q(cpi->active_worst_quality);
|
|
|
|
cpi->active_best_quality = cpi->active_worst_quality
|
|
|
|
+ compute_qdelta(cpi, current_q, current_q * 0.3);
|
|
|
|
#endif
|
2013-01-16 01:49:44 +04:00
|
|
|
} else if (cpi->refresh_golden_frame || cpi->refresh_alt_ref_frame) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int high = 2000;
|
|
|
|
int low = 400;
|
|
|
|
|
|
|
|
// Use the lower of cpi->active_worst_quality and recent
|
|
|
|
// average Q as basis for GF/ARF Q limit unless last frame was
|
|
|
|
// a key frame.
|
2013-04-02 21:24:56 +04:00
|
|
|
if (cpi->frames_since_key > 1 &&
|
|
|
|
cpi->avg_frame_qindex < cpi->active_worst_quality) {
|
|
|
|
q = cpi->avg_frame_qindex;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
// For constrained quality dont allow Q less than the cq level
|
2013-04-02 21:24:56 +04:00
|
|
|
if (cpi->oxcf.end_usage == USAGE_CONSTRAINED_QUALITY &&
|
|
|
|
q < cpi->cq_target_quality) {
|
|
|
|
q = cpi->cq_target_quality;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2013-04-02 21:24:56 +04:00
|
|
|
if (cpi->gfu_boost > high) {
|
|
|
|
cpi->active_best_quality = gf_low_motion_minq[q];
|
|
|
|
} else if (cpi->gfu_boost < low) {
|
|
|
|
cpi->active_best_quality = gf_high_motion_minq[q];
|
|
|
|
} else {
|
|
|
|
const int gap = high - low;
|
|
|
|
const int offset = high - cpi->gfu_boost;
|
|
|
|
const int qdiff = gf_high_motion_minq[q] - gf_low_motion_minq[q];
|
|
|
|
const int adjustment = ((offset * qdiff) + (gap >> 1)) / gap;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->active_best_quality = gf_low_motion_minq[q] + adjustment;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Constrained quality use slightly lower active best.
|
2013-04-02 21:24:56 +04:00
|
|
|
if (cpi->oxcf.end_usage == USAGE_CONSTRAINED_QUALITY)
|
|
|
|
cpi->active_best_quality = cpi->active_best_quality * 15 / 16;
|
2013-08-30 03:21:44 +04:00
|
|
|
|
|
|
|
// TODO(debargha): Refine the logic below
|
|
|
|
if (cpi->oxcf.end_usage == USAGE_CONSTANT_QUALITY) {
|
|
|
|
if (!cpi->refresh_alt_ref_frame) {
|
2013-09-10 04:31:58 +04:00
|
|
|
cpi->active_best_quality = cpi->cq_target_quality;
|
2013-08-30 03:21:44 +04:00
|
|
|
} else {
|
|
|
|
if (cpi->frames_since_key > 1) {
|
|
|
|
if (cpi->gfu_boost > high) {
|
|
|
|
cpi->active_best_quality = cpi->cq_target_quality * 6 / 16;
|
|
|
|
} else if (cpi->gfu_boost < low) {
|
2013-09-10 04:31:58 +04:00
|
|
|
cpi->active_best_quality = cpi->cq_target_quality * 11 / 16;
|
2013-08-30 03:21:44 +04:00
|
|
|
} else {
|
|
|
|
const int gap = high - low;
|
|
|
|
const int offset = high - cpi->gfu_boost;
|
2013-09-10 04:31:58 +04:00
|
|
|
const int qdiff = cpi->cq_target_quality * 5 / 16;
|
2013-08-30 03:21:44 +04:00
|
|
|
const int adjustment = ((offset * qdiff) + (gap >> 1)) / gap;
|
|
|
|
cpi->active_best_quality = cpi->cq_target_quality * 6 / 16
|
|
|
|
+ adjustment;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
} else {
|
2013-08-30 03:21:44 +04:00
|
|
|
if (cpi->oxcf.end_usage == USAGE_CONSTANT_QUALITY) {
|
|
|
|
cpi->active_best_quality = cpi->cq_target_quality;
|
|
|
|
} else {
|
2013-02-26 20:53:03 +04:00
|
|
|
#ifdef ONE_SHOT_Q_ESTIMATE
|
|
|
|
#ifdef STRICT_ONE_SHOT_Q
|
2013-08-30 03:21:44 +04:00
|
|
|
cpi->active_best_quality = q;
|
2013-02-26 20:53:03 +04:00
|
|
|
#else
|
2013-08-30 03:21:44 +04:00
|
|
|
cpi->active_best_quality = inter_minq[q];
|
2013-02-26 20:53:03 +04:00
|
|
|
#endif
|
|
|
|
#else
|
2013-08-30 03:21:44 +04:00
|
|
|
cpi->active_best_quality = inter_minq[q];
|
2013-02-26 20:53:03 +04:00
|
|
|
#endif
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
// For the constant/constrained quality mode we don't want
|
|
|
|
// q to fall below the cq level.
|
|
|
|
if ((cpi->oxcf.end_usage == USAGE_CONSTRAINED_QUALITY) &&
|
|
|
|
(cpi->active_best_quality < cpi->cq_target_quality)) {
|
|
|
|
// If we are strongly undershooting the target rate in the last
|
|
|
|
// frames then use the user passed in cq value not the auto
|
|
|
|
// cq value.
|
|
|
|
if (cpi->rolling_actual_bits < cpi->min_frame_bandwidth)
|
|
|
|
cpi->active_best_quality = cpi->oxcf.cq_level;
|
|
|
|
else
|
|
|
|
cpi->active_best_quality = cpi->cq_target_quality;
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clip the active best and worst quality values to limits
|
|
|
|
if (cpi->active_worst_quality > cpi->worst_quality)
|
|
|
|
cpi->active_worst_quality = cpi->worst_quality;
|
|
|
|
|
|
|
|
if (cpi->active_best_quality < cpi->best_quality)
|
|
|
|
cpi->active_best_quality = cpi->best_quality;
|
|
|
|
|
|
|
|
if (cpi->active_best_quality > cpi->worst_quality)
|
|
|
|
cpi->active_best_quality = cpi->worst_quality;
|
|
|
|
|
|
|
|
if (cpi->active_worst_quality < cpi->active_best_quality)
|
|
|
|
cpi->active_worst_quality = cpi->active_best_quality;
|
|
|
|
|
2012-12-21 02:56:19 +04:00
|
|
|
// Special case code to try and match quality with forced key frames
|
2013-08-30 03:21:44 +04:00
|
|
|
if (cpi->oxcf.end_usage == USAGE_CONSTANT_QUALITY) {
|
|
|
|
q = cpi->active_best_quality;
|
|
|
|
} else if ((cm->frame_type == KEY_FRAME) && cpi->this_key_frame_forced) {
|
2013-04-02 21:24:56 +04:00
|
|
|
q = cpi->last_boosted_qindex;
|
2012-07-14 02:21:29 +04:00
|
|
|
} else {
|
|
|
|
// Determine initial Q to try
|
2013-04-02 21:24:56 +04:00
|
|
|
q = vp9_regulate_q(cpi, cpi->this_frame_target);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 01:25:33 +04:00
|
|
|
vp9_compute_frame_size_bounds(cpi, &frame_under_shoot_limit,
|
|
|
|
&frame_over_shoot_limit);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
// Force the quantizer determined by the coding order pattern.
|
2013-08-30 03:21:44 +04:00
|
|
|
if (cpi->multi_arf_enabled && (cm->frame_type != KEY_FRAME) &&
|
|
|
|
cpi->oxcf.end_usage != USAGE_CONSTANT_QUALITY) {
|
2013-04-03 02:08:50 +04:00
|
|
|
double new_q;
|
|
|
|
double current_q = vp9_convert_qindex_to_q(cpi->active_worst_quality);
|
|
|
|
int level = cpi->this_frame_weight;
|
|
|
|
assert(level >= 0);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
// Set quantizer steps at 10% increments.
|
|
|
|
new_q = current_q * (1.0 - (0.2 * (cpi->max_arf_level - level)));
|
|
|
|
q = cpi->active_worst_quality + compute_qdelta(cpi, current_q, new_q);
|
|
|
|
|
|
|
|
bottom_index = q;
|
|
|
|
top_index = q;
|
|
|
|
q_low = q;
|
|
|
|
q_high = q;
|
|
|
|
|
|
|
|
printf("frame:%d q:%d\n", cm->current_video_frame, q);
|
|
|
|
} else {
|
|
|
|
#endif
|
|
|
|
// Limit Q range for the adaptive loop.
|
|
|
|
bottom_index = cpi->active_best_quality;
|
|
|
|
top_index = cpi->active_worst_quality;
|
|
|
|
q_low = cpi->active_best_quality;
|
|
|
|
q_high = cpi->active_worst_quality;
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
}
|
|
|
|
#endif
|
2012-07-14 02:21:29 +04:00
|
|
|
loop_count = 0;
|
2013-07-26 01:13:44 +04:00
|
|
|
vp9_zero(cpi->rd_tx_select_threshes);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cm->frame_type != KEY_FRAME) {
|
2013-09-12 02:16:36 +04:00
|
|
|
cm->mcomp_filter_type = DEFAULT_INTERP_FILTER;
|
2012-07-14 02:21:29 +04:00
|
|
|
/* TODO: Decide this more intelligently */
|
2013-04-02 21:24:56 +04:00
|
|
|
xd->allow_high_precision_mv = q < HIGH_PRECISION_MV_QTHRESH;
|
2012-11-09 03:44:39 +04:00
|
|
|
set_mvcost(&cpi->mb);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2012-02-27 22:22:38 +04:00
|
|
|
|
2013-09-04 21:02:08 +04:00
|
|
|
#if CONFIG_VP9_POSTPROC
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->oxcf.noise_sensitivity > 0) {
|
|
|
|
int l = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
switch (cpi->oxcf.noise_sensitivity) {
|
|
|
|
case 1:
|
|
|
|
l = 20;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
l = 40;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
l = 60;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
case 5:
|
|
|
|
l = 100;
|
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
l = 150;
|
|
|
|
break;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-05-14 22:01:57 +04:00
|
|
|
vp9_denoise(cpi->Source, cpi->Source, l);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef OUTPUT_YUV_SRC
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_write_yuv_frame(cpi->Source);
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
do {
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state(); // __asm emms;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
vp9_set_quantizer(cpi, q);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (loop_count == 0) {
|
2011-12-08 23:43:09 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Set up entropy depending on frame type.
|
2013-01-15 18:43:35 +04:00
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
|
|
|
/* Choose which entropy context to use. When using a forward reference
|
2013-03-26 19:34:56 +04:00
|
|
|
* frame, it immediately follows the keyframe, and thus benefits from
|
2013-04-03 02:08:50 +04:00
|
|
|
* using the same entropy context established by the keyframe.
|
|
|
|
* Otherwise, use the default context 0.
|
2013-03-26 19:34:56 +04:00
|
|
|
*/
|
2013-01-15 18:43:35 +04:00
|
|
|
cm->frame_context_idx = cpi->oxcf.play_alternate;
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_setup_key_frame(cpi);
|
2013-01-15 18:43:35 +04:00
|
|
|
} else {
|
2013-03-26 19:34:56 +04:00
|
|
|
/* Choose which entropy context to use. Currently there are only two
|
|
|
|
* contexts used, one for normal frames and one for alt ref frames.
|
|
|
|
*/
|
2013-01-15 18:43:35 +04:00
|
|
|
cpi->common.frame_context_idx = cpi->refresh_alt_ref_frame;
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_setup_inter_frame(cpi);
|
2013-01-15 18:43:35 +04:00
|
|
|
}
|
2013-04-28 02:49:49 +04:00
|
|
|
}
|
2013-04-23 17:01:55 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// transform / motion compensation build reconstruction frame
|
2012-04-17 01:53:37 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_encode_frame(cpi);
|
2011-02-25 14:42:05 +03:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Update the skip mb flag probabilities based on the distribution
|
|
|
|
// seen in the last encoder iteration.
|
2013-06-08 00:24:14 +04:00
|
|
|
// update_base_skip_probs(cpi);
|
2012-04-11 17:37:48 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state(); // __asm emms;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Dummy pack of the bitstream using up to date stats to get an
|
|
|
|
// accurate estimate of output frame size to determine if we need
|
|
|
|
// to recode.
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_save_coding_context(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->dummy_packing = 1;
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_pack_bitstream(cpi, dest, size);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->projected_frame_size = (*size) << 3;
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_restore_coding_context(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
if (frame_over_shoot_limit == 0)
|
|
|
|
frame_over_shoot_limit = 1;
|
2013-04-16 02:24:39 +04:00
|
|
|
active_worst_qchanged = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Special case handling for forced key frames
|
2013-08-30 03:21:44 +04:00
|
|
|
if (cpi->oxcf.end_usage == USAGE_CONSTANT_QUALITY) {
|
|
|
|
loop = 0;
|
|
|
|
} else {
|
|
|
|
if ((cm->frame_type == KEY_FRAME) && cpi->this_key_frame_forced) {
|
|
|
|
int last_q = q;
|
|
|
|
int kf_err = vp9_calc_ss_err(cpi->Source,
|
|
|
|
&cm->yv12_fb[cm->new_fb_idx]);
|
|
|
|
|
|
|
|
int high_err_target = cpi->ambient_err;
|
|
|
|
int low_err_target = cpi->ambient_err >> 1;
|
|
|
|
|
|
|
|
// Prevent possible divide by zero error below for perfect KF
|
|
|
|
kf_err += !kf_err;
|
|
|
|
|
|
|
|
// The key frame is not good enough or we can afford
|
|
|
|
// to make it better without undue risk of popping.
|
|
|
|
if ((kf_err > high_err_target &&
|
|
|
|
cpi->projected_frame_size <= frame_over_shoot_limit) ||
|
|
|
|
(kf_err > low_err_target &&
|
|
|
|
cpi->projected_frame_size <= frame_under_shoot_limit)) {
|
|
|
|
// Lower q_high
|
|
|
|
q_high = q > q_low ? q - 1 : q_low;
|
|
|
|
|
|
|
|
// Adjust Q
|
|
|
|
q = (q * high_err_target) / kf_err;
|
|
|
|
q = MIN(q, (q_high + q_low) >> 1);
|
|
|
|
} else if (kf_err < low_err_target &&
|
|
|
|
cpi->projected_frame_size >= frame_under_shoot_limit) {
|
|
|
|
// The key frame is much better than the previous frame
|
|
|
|
// Raise q_low
|
|
|
|
q_low = q < q_high ? q + 1 : q_high;
|
|
|
|
|
|
|
|
// Adjust Q
|
|
|
|
q = (q * low_err_target) / kf_err;
|
|
|
|
q = MIN(q, (q_high + q_low + 1) >> 1);
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
// Clamp Q to upper and lower limits:
|
|
|
|
q = clamp(q, q_low, q_high);
|
|
|
|
|
|
|
|
loop = q != last_q;
|
|
|
|
} else if (recode_loop_test(
|
|
|
|
cpi, frame_over_shoot_limit, frame_under_shoot_limit,
|
|
|
|
q, top_index, bottom_index)) {
|
|
|
|
// Is the projected frame size out of range and are we allowed
|
|
|
|
// to attempt to recode.
|
|
|
|
int last_q = q;
|
|
|
|
int retries = 0;
|
|
|
|
|
|
|
|
// Frame size out of permitted range:
|
|
|
|
// Update correction factor & compute new Q to try...
|
|
|
|
|
|
|
|
// Frame is too large
|
|
|
|
if (cpi->projected_frame_size > cpi->this_frame_target) {
|
|
|
|
// Raise Qlow as to at least the current value
|
|
|
|
q_low = q < q_high ? q + 1 : q_high;
|
|
|
|
|
|
|
|
if (undershoot_seen || loop_count > 1) {
|
|
|
|
// Update rate_correction_factor unless
|
|
|
|
// cpi->active_worst_quality has changed.
|
|
|
|
if (!active_worst_qchanged)
|
|
|
|
vp9_update_rate_correction_factors(cpi, 1);
|
|
|
|
|
|
|
|
q = (q_high + q_low + 1) / 2;
|
|
|
|
} else {
|
|
|
|
// Update rate_correction_factor unless
|
|
|
|
// cpi->active_worst_quality has changed.
|
|
|
|
if (!active_worst_qchanged)
|
|
|
|
vp9_update_rate_correction_factors(cpi, 0);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
q = vp9_regulate_q(cpi, cpi->this_frame_target);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
while (q < q_low && retries < 10) {
|
|
|
|
vp9_update_rate_correction_factors(cpi, 0);
|
|
|
|
q = vp9_regulate_q(cpi, cpi->this_frame_target);
|
|
|
|
retries++;
|
|
|
|
}
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
overshoot_seen = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
} else {
|
2013-08-30 03:21:44 +04:00
|
|
|
// Frame is too small
|
|
|
|
q_high = q > q_low ? q - 1 : q_low;
|
|
|
|
|
|
|
|
if (overshoot_seen || loop_count > 1) {
|
|
|
|
// Update rate_correction_factor unless
|
|
|
|
// cpi->active_worst_quality has changed.
|
|
|
|
if (!active_worst_qchanged)
|
|
|
|
vp9_update_rate_correction_factors(cpi, 1);
|
|
|
|
|
|
|
|
q = (q_high + q_low) / 2;
|
|
|
|
} else {
|
|
|
|
// Update rate_correction_factor unless
|
|
|
|
// cpi->active_worst_quality has changed.
|
|
|
|
if (!active_worst_qchanged)
|
|
|
|
vp9_update_rate_correction_factors(cpi, 0);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
q = vp9_regulate_q(cpi, cpi->this_frame_target);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
// Special case reset for qlow for constrained quality.
|
|
|
|
// This should only trigger where there is very substantial
|
|
|
|
// undershoot on a frame and the auto cq level is above
|
|
|
|
// the user passsed in value.
|
|
|
|
if (cpi->oxcf.end_usage == USAGE_CONSTRAINED_QUALITY && q < q_low) {
|
|
|
|
q_low = q;
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
while (q > q_high && retries < 10) {
|
|
|
|
vp9_update_rate_correction_factors(cpi, 0);
|
|
|
|
q = vp9_regulate_q(cpi, cpi->this_frame_target);
|
|
|
|
retries++;
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
undershoot_seen = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-11-17 18:12:04 +03:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
// Clamp Q to upper and lower limits:
|
|
|
|
q = clamp(q, q_low, q_high);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-08-30 03:21:44 +04:00
|
|
|
loop = q != last_q;
|
|
|
|
} else {
|
|
|
|
loop = 0;
|
|
|
|
}
|
2013-04-02 21:24:56 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->is_src_frame_alt_ref)
|
2013-04-16 02:24:39 +04:00
|
|
|
loop = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
if (loop) {
|
2012-07-14 02:21:29 +04:00
|
|
|
loop_count++;
|
2012-12-21 02:56:19 +04:00
|
|
|
|
2011-04-29 20:37:59 +04:00
|
|
|
#if CONFIG_INTERNAL_STATS
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->tot_recode_hits++;
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
2011-01-17 20:23:11 +03:00
|
|
|
}
|
2013-04-02 21:24:56 +04:00
|
|
|
} while (loop);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Special case code to reduce pulsing when key frames are forced at a
|
|
|
|
// fixed interval. Note the reconstruction error if it is the frame before
|
|
|
|
// the force key frame
|
|
|
|
if (cpi->next_key_frame_forced && (cpi->twopass.frames_to_key == 0)) {
|
2012-10-30 23:58:42 +04:00
|
|
|
cpi->ambient_err = vp9_calc_ss_err(cpi->Source,
|
2012-10-22 07:47:57 +04:00
|
|
|
&cm->yv12_fb[cm->new_fb_idx]);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (cm->frame_type == KEY_FRAME)
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_last_frame = 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->frame_to_show = &cm->yv12_fb[cm->new_fb_idx];
|
2011-02-25 14:42:05 +03:00
|
|
|
|
2011-11-16 04:16:30 +04:00
|
|
|
#if WRITE_RECON_BUFFER
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cm->show_frame)
|
|
|
|
write_cx_frame_to_file(cm->frame_to_show,
|
|
|
|
cm->current_video_frame);
|
|
|
|
else
|
|
|
|
write_cx_frame_to_file(cm->frame_to_show,
|
|
|
|
cm->current_video_frame + 1000);
|
2011-11-16 04:16:30 +04:00
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Pick the loop filter level for the frame.
|
|
|
|
loopfilter_frame(cpi, cm);
|
2012-01-28 16:20:14 +04:00
|
|
|
|
2013-05-17 23:50:40 +04:00
|
|
|
#if WRITE_RECON_BUFFER
|
|
|
|
if (cm->show_frame)
|
|
|
|
write_cx_frame_to_file(cm->frame_to_show,
|
|
|
|
cm->current_video_frame + 2000);
|
|
|
|
else
|
|
|
|
write_cx_frame_to_file(cm->frame_to_show,
|
|
|
|
cm->current_video_frame + 3000);
|
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// build the bitstream
|
|
|
|
cpi->dummy_packing = 0;
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_pack_bitstream(cpi, dest, size);
|
2011-02-18 16:46:07 +03:00
|
|
|
|
2013-08-14 22:20:33 +04:00
|
|
|
if (cm->seg.update_map)
|
2012-08-21 01:43:34 +04:00
|
|
|
update_reference_segmentation_map(cpi);
|
|
|
|
|
2013-02-21 00:34:31 +04:00
|
|
|
release_scaled_references(cpi);
|
2013-01-16 01:49:44 +04:00
|
|
|
update_reference_frames(cpi);
|
2013-05-17 17:40:25 +04:00
|
|
|
|
2013-05-31 20:18:59 +04:00
|
|
|
for (t = TX_4X4; t <= TX_32X32; t++)
|
2013-07-24 04:02:08 +04:00
|
|
|
full_to_model_counts(cpi->common.counts.coef[t],
|
2013-06-25 02:46:15 +04:00
|
|
|
cpi->coef_counts[t]);
|
2013-01-25 23:30:28 +04:00
|
|
|
if (!cpi->common.error_resilient_mode &&
|
2013-02-20 22:16:24 +04:00
|
|
|
!cpi->common.frame_parallel_decoding_mode) {
|
2013-01-15 18:43:35 +04:00
|
|
|
vp9_adapt_coef_probs(&cpi->common);
|
2013-02-20 22:16:24 +04:00
|
|
|
}
|
2013-04-26 22:57:17 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->common.frame_type != KEY_FRAME) {
|
2013-07-24 04:02:08 +04:00
|
|
|
FRAME_COUNTS *counts = &cpi->common.counts;
|
|
|
|
|
|
|
|
vp9_copy(counts->y_mode, cpi->y_mode_count);
|
|
|
|
vp9_copy(counts->uv_mode, cpi->y_uv_mode_count);
|
|
|
|
vp9_copy(counts->partition, cpi->partition_count);
|
|
|
|
vp9_copy(counts->intra_inter, cpi->intra_inter_count);
|
|
|
|
vp9_copy(counts->comp_inter, cpi->comp_inter_count);
|
|
|
|
vp9_copy(counts->single_ref, cpi->single_ref_count);
|
|
|
|
vp9_copy(counts->comp_ref, cpi->comp_ref_count);
|
|
|
|
counts->mv = cpi->NMVcount;
|
2013-01-25 23:30:28 +04:00
|
|
|
if (!cpi->common.error_resilient_mode &&
|
|
|
|
!cpi->common.frame_parallel_decoding_mode) {
|
|
|
|
vp9_adapt_mode_probs(&cpi->common);
|
2013-06-19 05:28:10 +04:00
|
|
|
vp9_adapt_mv_probs(&cpi->common, cpi->mb.e_mbd.allow_high_precision_mv);
|
2013-01-25 23:30:28 +04:00
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2012-02-27 22:22:38 +04:00
|
|
|
|
2013-05-16 14:27:12 +04:00
|
|
|
#ifdef ENTROPY_STATS
|
|
|
|
vp9_update_mode_context_stats(cpi);
|
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
/* Move storing frame_type out of the above loop since it is also
|
|
|
|
* needed in motion search besides loopfilter */
|
|
|
|
cm->last_frame_type = cm->frame_type;
|
|
|
|
|
|
|
|
// Update rate control heuristics
|
|
|
|
cpi->total_byte_count += (*size);
|
|
|
|
cpi->projected_frame_size = (*size) << 3;
|
|
|
|
|
|
|
|
if (!active_worst_qchanged)
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_update_rate_correction_factors(cpi, 2);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
cpi->last_q[cm->frame_type] = cm->base_qindex;
|
|
|
|
|
|
|
|
// Keep record of last boosted (KF/KF/ARF) Q value.
|
|
|
|
// If the current frame is coded at a lower Q then we also update it.
|
|
|
|
// If all mbs in this group are skipped only update if the Q value is
|
|
|
|
// better than that already stored.
|
|
|
|
// This is used to help set quality in forced key frames to reduce popping
|
|
|
|
if ((cm->base_qindex < cpi->last_boosted_qindex) ||
|
|
|
|
((cpi->static_mb_pct < 100) &&
|
|
|
|
((cm->frame_type == KEY_FRAME) ||
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_alt_ref_frame ||
|
|
|
|
(cpi->refresh_golden_frame && !cpi->is_src_frame_alt_ref)))) {
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->last_boosted_qindex = cm->base_qindex;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_adjust_key_frame_context(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Keep a record of ambient average Q.
|
|
|
|
if (cm->frame_type != KEY_FRAME)
|
|
|
|
cpi->avg_frame_qindex = (2 + 3 * cpi->avg_frame_qindex + cm->base_qindex) >> 2;
|
|
|
|
|
|
|
|
// Keep a record from which we can calculate the average Q excluding GF updates and key frames
|
2013-04-02 21:24:56 +04:00
|
|
|
if (cm->frame_type != KEY_FRAME &&
|
|
|
|
!cpi->refresh_golden_frame &&
|
|
|
|
!cpi->refresh_alt_ref_frame) {
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->ni_frames++;
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->tot_q += vp9_convert_qindex_to_q(q);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->avg_q = cpi->tot_q / (double)cpi->ni_frames;
|
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
// Calculate the average Q for normal inter frames (not key or GFU frames).
|
|
|
|
cpi->ni_tot_qi += q;
|
|
|
|
cpi->ni_av_qi = cpi->ni_tot_qi / cpi->ni_frames;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Update the buffer level variable.
|
|
|
|
// Non-viewable frames are a special case and are treated as pure overhead.
|
|
|
|
if (!cm->show_frame)
|
|
|
|
cpi->bits_off_target -= cpi->projected_frame_size;
|
|
|
|
else
|
|
|
|
cpi->bits_off_target += cpi->av_per_frame_bandwidth - cpi->projected_frame_size;
|
|
|
|
|
|
|
|
// Clip the buffer level at the maximum buffer size
|
|
|
|
if (cpi->bits_off_target > cpi->oxcf.maximum_buffer_size)
|
|
|
|
cpi->bits_off_target = cpi->oxcf.maximum_buffer_size;
|
|
|
|
|
2013-02-25 16:36:38 +04:00
|
|
|
// Rolling monitors of whether we are over or underspending used to help
|
|
|
|
// regulate min and Max Q in two pass.
|
|
|
|
if (cm->frame_type != KEY_FRAME) {
|
|
|
|
cpi->rolling_target_bits =
|
|
|
|
((cpi->rolling_target_bits * 3) + cpi->this_frame_target + 2) / 4;
|
|
|
|
cpi->rolling_actual_bits =
|
|
|
|
((cpi->rolling_actual_bits * 3) + cpi->projected_frame_size + 2) / 4;
|
|
|
|
cpi->long_rolling_target_bits =
|
|
|
|
((cpi->long_rolling_target_bits * 31) + cpi->this_frame_target + 16) / 32;
|
|
|
|
cpi->long_rolling_actual_bits =
|
|
|
|
((cpi->long_rolling_actual_bits * 31) +
|
|
|
|
cpi->projected_frame_size + 16) / 32;
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Actual bits spent
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->total_actual_bits += cpi->projected_frame_size;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Debug stats
|
|
|
|
cpi->total_target_vs_actual += (cpi->this_frame_target - cpi->projected_frame_size);
|
|
|
|
|
|
|
|
cpi->buffer_level = cpi->bits_off_target;
|
|
|
|
|
|
|
|
// Update bits left to the kf and gf groups to account for overshoot or undershoot on these frames
|
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
|
|
|
cpi->twopass.kf_group_bits += cpi->this_frame_target - cpi->projected_frame_size;
|
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->twopass.kf_group_bits = MAX(cpi->twopass.kf_group_bits, 0);
|
2013-01-16 01:49:44 +04:00
|
|
|
} else if (cpi->refresh_golden_frame || cpi->refresh_alt_ref_frame) {
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->twopass.gf_group_bits += cpi->this_frame_target - cpi->projected_frame_size;
|
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->twopass.gf_group_bits = MAX(cpi->twopass.gf_group_bits, 0);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Update the skip mb flag probabilities based on the distribution seen
|
|
|
|
// in this frame.
|
2013-06-08 00:24:14 +04:00
|
|
|
// update_base_skip_probs(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-09-10 04:31:58 +04:00
|
|
|
#if CONFIG_INTERNAL_STATS
|
2012-07-14 02:21:29 +04:00
|
|
|
{
|
2013-08-30 03:21:44 +04:00
|
|
|
FILE *f = fopen("tmp.stt", cm->current_video_frame ? "a" : "w");
|
2012-07-14 02:21:29 +04:00
|
|
|
int recon_err;
|
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state(); // __asm emms;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
recon_err = vp9_calc_ss_err(cpi->Source,
|
2012-10-22 07:47:57 +04:00
|
|
|
&cm->yv12_fb[cm->new_fb_idx]);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-05-25 22:41:25 +04:00
|
|
|
if (cpi->twopass.total_left_stats.coded_error != 0.0)
|
2013-08-30 03:21:44 +04:00
|
|
|
fprintf(f, "%10d %10d %10d %10d %10d %10d %10d %10d %10d"
|
2012-07-14 02:21:29 +04:00
|
|
|
"%7.2f %7.2f %7.2f %7.2f %7.2f %7.2f %7.2f"
|
2013-03-12 19:33:40 +04:00
|
|
|
"%6d %6d %5d %5d %5d %8.2f %10d %10.3f"
|
2012-07-14 02:21:29 +04:00
|
|
|
"%10.3f %8d %10d %10d %10d\n",
|
|
|
|
cpi->common.current_video_frame, cpi->this_frame_target,
|
2012-12-23 19:20:10 +04:00
|
|
|
cpi->projected_frame_size, 0, //loop_size_estimate,
|
2012-07-14 02:21:29 +04:00
|
|
|
(cpi->projected_frame_size - cpi->this_frame_target),
|
|
|
|
(int)cpi->total_target_vs_actual,
|
2013-03-12 19:33:40 +04:00
|
|
|
(int)(cpi->oxcf.starting_buffer_level - cpi->bits_off_target),
|
2012-07-14 02:21:29 +04:00
|
|
|
(int)cpi->total_actual_bits,
|
2013-08-30 03:21:44 +04:00
|
|
|
cm->base_qindex,
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_convert_qindex_to_q(cm->base_qindex),
|
2012-10-31 03:25:53 +04:00
|
|
|
(double)vp9_dc_quant(cm->base_qindex, 0) / 4.0,
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_convert_qindex_to_q(cpi->active_best_quality),
|
|
|
|
vp9_convert_qindex_to_q(cpi->active_worst_quality),
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->avg_q,
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_convert_qindex_to_q(cpi->ni_av_qi),
|
|
|
|
vp9_convert_qindex_to_q(cpi->cq_target_quality),
|
2013-03-12 19:33:40 +04:00
|
|
|
cpi->refresh_last_frame,
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_golden_frame, cpi->refresh_alt_ref_frame,
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->frame_type, cpi->gfu_boost,
|
|
|
|
cpi->twopass.est_max_qcorrection_factor,
|
|
|
|
(int)cpi->twopass.bits_left,
|
2013-05-25 22:41:25 +04:00
|
|
|
cpi->twopass.total_left_stats.coded_error,
|
2012-07-14 02:21:29 +04:00
|
|
|
(double)cpi->twopass.bits_left /
|
2013-05-25 22:41:25 +04:00
|
|
|
cpi->twopass.total_left_stats.coded_error,
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->tot_recode_hits, recon_err, cpi->kf_boost,
|
|
|
|
cpi->kf_zeromotion_pct);
|
2011-08-12 22:51:36 +04:00
|
|
|
else
|
2013-08-30 03:21:44 +04:00
|
|
|
fprintf(f, "%10d %10d %10d %10d %10d %10d %10d %10d %10d"
|
2012-07-14 02:21:29 +04:00
|
|
|
"%7.2f %7.2f %7.2f %7.2f %7.2f %7.2f %7.2f"
|
2013-03-12 19:33:40 +04:00
|
|
|
"%5d %5d %5d %8d %8d %8.2f %10d %10.3f"
|
2012-07-14 02:21:29 +04:00
|
|
|
"%8d %10d %10d %10d\n",
|
|
|
|
cpi->common.current_video_frame,
|
|
|
|
cpi->this_frame_target, cpi->projected_frame_size,
|
2012-12-23 19:20:10 +04:00
|
|
|
0, //loop_size_estimate,
|
2012-07-14 02:21:29 +04:00
|
|
|
(cpi->projected_frame_size - cpi->this_frame_target),
|
|
|
|
(int)cpi->total_target_vs_actual,
|
2013-03-12 19:33:40 +04:00
|
|
|
(int)(cpi->oxcf.starting_buffer_level - cpi->bits_off_target),
|
2012-07-14 02:21:29 +04:00
|
|
|
(int)cpi->total_actual_bits,
|
2013-08-30 03:21:44 +04:00
|
|
|
cm->base_qindex,
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_convert_qindex_to_q(cm->base_qindex),
|
2012-10-31 03:25:53 +04:00
|
|
|
(double)vp9_dc_quant(cm->base_qindex, 0) / 4.0,
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_convert_qindex_to_q(cpi->active_best_quality),
|
|
|
|
vp9_convert_qindex_to_q(cpi->active_worst_quality),
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->avg_q,
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_convert_qindex_to_q(cpi->ni_av_qi),
|
|
|
|
vp9_convert_qindex_to_q(cpi->cq_target_quality),
|
2013-03-12 19:33:40 +04:00
|
|
|
cpi->refresh_last_frame,
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_golden_frame, cpi->refresh_alt_ref_frame,
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->frame_type, cpi->gfu_boost,
|
|
|
|
cpi->twopass.est_max_qcorrection_factor,
|
|
|
|
(int)cpi->twopass.bits_left,
|
2013-05-25 22:41:25 +04:00
|
|
|
cpi->twopass.total_left_stats.coded_error,
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->tot_recode_hits, recon_err, cpi->kf_boost,
|
|
|
|
cpi->kf_zeromotion_pct);
|
|
|
|
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
if (0) {
|
|
|
|
FILE *fmodes = fopen("Modes.stt", "a");
|
|
|
|
int i;
|
|
|
|
|
|
|
|
fprintf(fmodes, "%6d:%1d:%1d:%1d ",
|
|
|
|
cpi->common.current_video_frame,
|
2013-01-16 01:49:44 +04:00
|
|
|
cm->frame_type, cpi->refresh_golden_frame,
|
|
|
|
cpi->refresh_alt_ref_frame);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
for (i = 0; i < MAX_MODES; i++)
|
|
|
|
fprintf(fmodes, "%5d ", cpi->mode_chosen_counts[i]);
|
|
|
|
|
|
|
|
fprintf(fmodes, "\n");
|
|
|
|
|
|
|
|
fclose(fmodes);
|
|
|
|
}
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2011-09-30 19:45:16 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
2012-07-14 02:21:29 +04:00
|
|
|
// Debug stats for segment feature experiments.
|
|
|
|
print_seg_map(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// If this was a kf or Gf note the Q
|
2013-01-16 01:49:44 +04:00
|
|
|
if ((cm->frame_type == KEY_FRAME)
|
|
|
|
|| cpi->refresh_golden_frame || cpi->refresh_alt_ref_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->last_kf_gf_q = cm->base_qindex;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_golden_frame == 1)
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->frame_flags = cm->frame_flags | FRAMEFLAGS_GOLDEN;
|
|
|
|
else
|
|
|
|
cm->frame_flags = cm->frame_flags&~FRAMEFLAGS_GOLDEN;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_alt_ref_frame == 1)
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->frame_flags = cm->frame_flags | FRAMEFLAGS_ALTREF;
|
|
|
|
else
|
|
|
|
cm->frame_flags = cm->frame_flags&~FRAMEFLAGS_ALTREF;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_last_frame & cpi->refresh_golden_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->gold_is_last = 1;
|
2013-01-16 01:49:44 +04:00
|
|
|
else if (cpi->refresh_last_frame ^ cpi->refresh_golden_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->gold_is_last = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_last_frame & cpi->refresh_alt_ref_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->alt_is_last = 1;
|
2013-01-16 01:49:44 +04:00
|
|
|
else if (cpi->refresh_last_frame ^ cpi->refresh_alt_ref_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->alt_is_last = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->refresh_alt_ref_frame & cpi->refresh_golden_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->gold_is_alt = 1;
|
2013-01-16 01:49:44 +04:00
|
|
|
else if (cpi->refresh_alt_ref_frame ^ cpi->refresh_golden_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->gold_is_alt = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-11-01 01:40:53 +04:00
|
|
|
cpi->ref_frame_flags = VP9_ALT_FLAG | VP9_GOLD_FLAG | VP9_LAST_FLAG;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->gold_is_last)
|
2012-11-01 01:40:53 +04:00
|
|
|
cpi->ref_frame_flags &= ~VP9_GOLD_FLAG;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->alt_is_last)
|
2012-11-01 01:40:53 +04:00
|
|
|
cpi->ref_frame_flags &= ~VP9_ALT_FLAG;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->gold_is_alt)
|
2012-11-01 01:40:53 +04:00
|
|
|
cpi->ref_frame_flags &= ~VP9_ALT_FLAG;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (cpi->oxcf.play_alternate && cpi->refresh_alt_ref_frame
|
|
|
|
&& (cm->frame_type != KEY_FRAME))
|
2012-07-14 02:21:29 +04:00
|
|
|
// Update the alternate reference frame stats as appropriate.
|
|
|
|
update_alt_ref_frame_stats(cpi);
|
|
|
|
else
|
|
|
|
// Update the Golden frame stats as appropriate.
|
|
|
|
update_golden_frame_stats(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cm->frame_type == KEY_FRAME) {
|
|
|
|
// Tell the caller that the frame was coded as a key frame
|
|
|
|
*frame_flags = cm->frame_flags | FRAMEFLAGS_KEY;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
// Reset the sequence number.
|
|
|
|
if (cpi->multi_arf_enabled) {
|
|
|
|
cpi->sequence_number = 0;
|
|
|
|
cpi->frame_coding_order_period = cpi->new_frame_coding_order_period;
|
|
|
|
cpi->new_frame_coding_order_period = -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// As this frame is a key frame the next defaults to an inter frame.
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->frame_type = INTER_FRAME;
|
|
|
|
} else {
|
|
|
|
*frame_flags = cm->frame_flags&~FRAMEFLAGS_KEY;
|
2013-04-03 02:08:50 +04:00
|
|
|
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
/* Increment position in the coded frame sequence. */
|
|
|
|
if (cpi->multi_arf_enabled) {
|
|
|
|
++cpi->sequence_number;
|
|
|
|
if (cpi->sequence_number >= cpi->frame_coding_order_period) {
|
|
|
|
cpi->sequence_number = 0;
|
|
|
|
cpi->frame_coding_order_period = cpi->new_frame_coding_order_period;
|
|
|
|
cpi->new_frame_coding_order_period = -1;
|
|
|
|
}
|
|
|
|
cpi->this_frame_weight = cpi->arf_weight[cpi->sequence_number];
|
|
|
|
assert(cpi->this_frame_weight >= 0);
|
|
|
|
}
|
|
|
|
#endif
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Clear the one shot update flags for segmentation map and mode/ref loop filter deltas.
|
2013-08-14 22:20:33 +04:00
|
|
|
cm->seg.update_map = 0;
|
|
|
|
cm->seg.update_data = 0;
|
2013-08-10 01:41:51 +04:00
|
|
|
cm->lf.mode_ref_delta_update = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-02-21 00:34:31 +04:00
|
|
|
// keep track of the last coded dimensions
|
2013-03-21 03:41:30 +04:00
|
|
|
cm->last_width = cm->width;
|
|
|
|
cm->last_height = cm->height;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-07-11 18:47:28 +04:00
|
|
|
// reset to normal state now that we are done.
|
2013-05-27 01:40:49 +04:00
|
|
|
cm->last_show_frame = cm->show_frame;
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cm->show_frame) {
|
2013-07-11 18:47:28 +04:00
|
|
|
// current mip will be the prev_mip for the next frame
|
|
|
|
MODE_INFO *temp = cm->prev_mip;
|
2013-09-11 21:45:44 +04:00
|
|
|
MODE_INFO **temp2 = cm->prev_mi_grid_base;
|
2013-07-11 18:47:28 +04:00
|
|
|
cm->prev_mip = cm->mip;
|
|
|
|
cm->mip = temp;
|
2013-09-11 21:45:44 +04:00
|
|
|
cm->prev_mi_grid_base = cm->mi_grid_base;
|
|
|
|
cm->mi_grid_base = temp2;
|
2013-07-11 18:47:28 +04:00
|
|
|
|
|
|
|
// update the upper left visible macroblock ptrs
|
|
|
|
cm->mi = cm->mip + cm->mode_info_stride + 1;
|
2013-09-11 21:45:44 +04:00
|
|
|
cm->mi_grid_visible = cm->mi_grid_base + cm->mode_info_stride + 1;
|
2013-07-11 18:47:28 +04:00
|
|
|
|
|
|
|
// Don't increment frame counters if this was an altref buffer
|
|
|
|
// update not a real frame
|
2013-04-03 02:08:50 +04:00
|
|
|
++cm->current_video_frame;
|
|
|
|
++cpi->frames_since_key;
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2013-07-11 18:47:28 +04:00
|
|
|
// restore prev_mi
|
|
|
|
cm->prev_mi = cm->prev_mip + cm->mode_info_stride + 1;
|
2013-09-11 21:45:44 +04:00
|
|
|
cm->prev_mi_grid_visible = cm->prev_mi_grid_base + cm->mode_info_stride + 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-09-11 21:45:44 +04:00
|
|
|
#if 0
|
2012-07-14 02:21:29 +04:00
|
|
|
{
|
|
|
|
char filename[512];
|
|
|
|
FILE *recon_file;
|
|
|
|
sprintf(filename, "enc%04d.yuv", (int) cm->current_video_frame);
|
|
|
|
recon_file = fopen(filename, "wb");
|
2013-02-22 23:22:03 +04:00
|
|
|
fwrite(cm->yv12_fb[cm->ref_frame_map[cpi->lst_fb_idx]].buffer_alloc,
|
|
|
|
cm->yv12_fb[cm->ref_frame_map[cpi->lst_fb_idx]].frame_size,
|
2013-01-16 01:49:44 +04:00
|
|
|
1, recon_file);
|
2012-07-14 02:21:29 +04:00
|
|
|
fclose(recon_file);
|
|
|
|
}
|
2010-09-24 19:10:25 +04:00
|
|
|
#endif
|
2012-02-27 22:22:38 +04:00
|
|
|
#ifdef OUTPUT_YUV_REC
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_write_yuv_rec_frame(cm);
|
2010-12-02 02:50:14 +03:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static void Pass2Encode(VP9_COMP *cpi, unsigned long *size,
|
2012-10-31 01:25:33 +04:00
|
|
|
unsigned char *dest, unsigned int *frame_flags) {
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-09-06 04:10:58 +04:00
|
|
|
cpi->enable_encode_breakout = 1;
|
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (!cpi->refresh_alt_ref_frame)
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_second_pass(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
encode_frame_to_data_rate(cpi, size, dest, frame_flags);
|
2013-07-02 05:18:50 +04:00
|
|
|
// vp9_print_modes_and_motion_vectors(&cpi->common, "encode.stt");
|
2013-02-25 16:36:38 +04:00
|
|
|
#ifdef DISABLE_RC_LONG_TERM_MEM
|
|
|
|
cpi->twopass.bits_left -= cpi->this_frame_target;
|
|
|
|
#else
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->twopass.bits_left -= 8 * *size;
|
2013-02-25 16:36:38 +04:00
|
|
|
#endif
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-01-16 01:49:44 +04:00
|
|
|
if (!cpi->refresh_alt_ref_frame) {
|
2013-07-13 04:12:46 +04:00
|
|
|
double lower_bounds_min_rate = FRAME_OVERHEAD_BITS * cpi->oxcf.framerate;
|
2012-07-14 02:21:29 +04:00
|
|
|
double two_pass_min_rate = (double)(cpi->oxcf.target_bandwidth
|
|
|
|
* cpi->oxcf.two_pass_vbrmin_section / 100);
|
2012-03-02 05:24:30 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (two_pass_min_rate < lower_bounds_min_rate)
|
|
|
|
two_pass_min_rate = lower_bounds_min_rate;
|
2012-03-02 05:24:30 +04:00
|
|
|
|
2013-07-13 04:12:46 +04:00
|
|
|
cpi->twopass.bits_left += (int64_t)(two_pass_min_rate / cpi->oxcf.framerate);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-09-05 19:55:47 +04:00
|
|
|
static void check_initial_width(VP9_COMP *cpi, YV12_BUFFER_CONFIG *sd) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2013-05-07 02:52:06 +04:00
|
|
|
if (!cpi->initial_width) {
|
|
|
|
// TODO(jkoleszar): Support 1/4 subsampling?
|
2013-09-05 19:55:47 +04:00
|
|
|
cm->subsampling_x = (sd != NULL) && sd->uv_width < sd->y_width;
|
|
|
|
cm->subsampling_y = (sd != NULL) && sd->uv_height < sd->y_height;
|
2013-05-07 02:52:06 +04:00
|
|
|
alloc_raw_frame_buffers(cpi);
|
|
|
|
|
|
|
|
cpi->initial_width = cm->width;
|
|
|
|
cpi->initial_height = cm->height;
|
|
|
|
}
|
2013-09-05 19:55:47 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int vp9_receive_raw_frame(VP9_PTR ptr, unsigned int frame_flags,
|
|
|
|
YV12_BUFFER_CONFIG *sd, int64_t time_stamp,
|
|
|
|
int64_t end_time) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *) ptr;
|
|
|
|
struct vpx_usec_timer timer;
|
|
|
|
int res = 0;
|
|
|
|
|
|
|
|
check_initial_width(cpi, sd);
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_usec_timer_start(&timer);
|
2012-10-31 01:25:33 +04:00
|
|
|
if (vp9_lookahead_push(cpi->lookahead, sd, time_stamp, end_time, frame_flags,
|
|
|
|
cpi->active_map_enabled ? cpi->active_map : NULL))
|
2012-07-14 02:21:29 +04:00
|
|
|
res = -1;
|
|
|
|
vpx_usec_timer_mark(&timer);
|
|
|
|
cpi->time_receive_data += vpx_usec_timer_elapsed(&timer);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return res;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
2011-04-13 22:00:18 +04:00
|
|
|
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
static int frame_is_reference(const VP9_COMP *cpi) {
|
|
|
|
const VP9_COMMON *cm = &cpi->common;
|
2011-07-07 18:38:23 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
return cm->frame_type == KEY_FRAME ||
|
|
|
|
cpi->refresh_last_frame ||
|
|
|
|
cpi->refresh_golden_frame ||
|
|
|
|
cpi->refresh_alt_ref_frame ||
|
2013-04-27 01:39:58 +04:00
|
|
|
cm->refresh_frame_context ||
|
2013-08-10 01:41:51 +04:00
|
|
|
cm->lf.mode_ref_delta_update ||
|
2013-08-14 22:20:33 +04:00
|
|
|
cm->seg.update_map ||
|
|
|
|
cm->seg.update_data;
|
2011-07-07 18:38:23 +04:00
|
|
|
}
|
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
int is_next_frame_arf(VP9_COMP *cpi) {
|
|
|
|
// Negative entry in frame_coding_order indicates an ARF at this position.
|
|
|
|
return cpi->frame_coding_order[cpi->sequence_number + 1] < 0 ? 1 : 0;
|
|
|
|
}
|
|
|
|
#endif
|
2011-07-07 18:38:23 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_get_compressed_data(VP9_PTR ptr, unsigned int *frame_flags,
|
2012-10-31 01:25:33 +04:00
|
|
|
unsigned long *size, unsigned char *dest,
|
|
|
|
int64_t *time_stamp, int64_t *time_end, int flush) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *) ptr;
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2012-07-14 02:21:29 +04:00
|
|
|
struct vpx_usec_timer cmptimer;
|
|
|
|
YV12_BUFFER_CONFIG *force_src_buffer = NULL;
|
2013-04-03 23:22:50 +04:00
|
|
|
int i;
|
2013-04-03 02:08:50 +04:00
|
|
|
// FILE *fp_out = fopen("enc_frame_type.txt", "a");
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (!cpi)
|
|
|
|
return -1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_usec_timer_start(&cmptimer);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->source = NULL;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->mb.e_mbd.allow_high_precision_mv = ALTREF_HIGH_PRECISION_MV;
|
2012-11-09 03:44:39 +04:00
|
|
|
set_mvcost(&cpi->mb);
|
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
// Should we code an alternate reference frame.
|
|
|
|
if (cpi->oxcf.play_alternate && cpi->source_alt_ref_pending) {
|
|
|
|
int frames_to_arf;
|
|
|
|
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
assert(!cpi->multi_arf_enabled ||
|
|
|
|
cpi->frame_coding_order[cpi->sequence_number] < 0);
|
|
|
|
|
|
|
|
if (cpi->multi_arf_enabled && (cpi->pass == 2))
|
|
|
|
frames_to_arf = (-cpi->frame_coding_order[cpi->sequence_number])
|
|
|
|
- cpi->next_frame_in_order;
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
frames_to_arf = cpi->frames_till_gf_update_due;
|
|
|
|
|
|
|
|
assert(frames_to_arf < cpi->twopass.frames_to_key);
|
|
|
|
|
|
|
|
if ((cpi->source = vp9_lookahead_peek(cpi->lookahead, frames_to_arf))) {
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
cpi->alt_ref_source[cpi->arf_buffered] = cpi->source;
|
|
|
|
#else
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->alt_ref_source = cpi->source;
|
2013-04-03 02:08:50 +04:00
|
|
|
#endif
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->oxcf.arnr_max_frames > 0) {
|
2013-04-03 02:08:50 +04:00
|
|
|
// Produce the filtered ARF frame.
|
|
|
|
// TODO(agrange) merge these two functions.
|
|
|
|
configure_arnr_filter(cpi, cm->current_video_frame + frames_to_arf,
|
|
|
|
cpi->gfu_boost);
|
|
|
|
vp9_temporal_filter_prepare(cpi, frames_to_arf);
|
2013-08-22 21:37:28 +04:00
|
|
|
vp9_extend_frame_borders(&cpi->alt_ref_buffer,
|
|
|
|
cm->subsampling_x, cm->subsampling_y);
|
2012-07-14 02:21:29 +04:00
|
|
|
force_src_buffer = &cpi->alt_ref_buffer;
|
|
|
|
}
|
2013-04-03 02:08:50 +04:00
|
|
|
|
|
|
|
cm->show_frame = 0;
|
2013-05-30 04:16:00 +04:00
|
|
|
cm->intra_only = 0;
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_alt_ref_frame = 1;
|
|
|
|
cpi->refresh_golden_frame = 0;
|
|
|
|
cpi->refresh_last_frame = 0;
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->is_src_frame_alt_ref = 0;
|
2013-04-03 02:08:50 +04:00
|
|
|
|
|
|
|
// TODO(agrange) This needs to vary depending on where the next ARF is.
|
2013-07-19 01:09:21 +04:00
|
|
|
cpi->frames_till_alt_ref_frame = frames_to_arf;
|
2013-04-03 02:08:50 +04:00
|
|
|
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
if (!cpi->multi_arf_enabled)
|
|
|
|
#endif
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->source_alt_ref_pending = 0; // Clear Pending altf Ref flag.
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cpi->source) {
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
int i;
|
|
|
|
#endif
|
2012-10-30 23:58:42 +04:00
|
|
|
if ((cpi->source = vp9_lookahead_pop(cpi->lookahead, flush))) {
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->show_frame = 1;
|
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
// Is this frame the ARF overlay.
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->is_src_frame_alt_ref = 0;
|
2013-04-03 02:08:50 +04:00
|
|
|
for (i = 0; i < cpi->arf_buffered; ++i) {
|
|
|
|
if (cpi->source == cpi->alt_ref_source[i]) {
|
2013-04-16 02:24:39 +04:00
|
|
|
cpi->is_src_frame_alt_ref = 1;
|
|
|
|
cpi->refresh_golden_frame = 1;
|
2013-04-03 02:08:50 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->is_src_frame_alt_ref = cpi->alt_ref_source
|
|
|
|
&& (cpi->source == cpi->alt_ref_source);
|
2013-04-03 02:08:50 +04:00
|
|
|
#endif
|
2013-03-12 19:33:40 +04:00
|
|
|
if (cpi->is_src_frame_alt_ref) {
|
2013-04-03 02:08:50 +04:00
|
|
|
// Current frame is an ARF overlay frame.
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
cpi->alt_ref_source[i] = NULL;
|
|
|
|
#else
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->alt_ref_source = NULL;
|
2013-04-03 02:08:50 +04:00
|
|
|
#endif
|
|
|
|
// Don't refresh the last buffer for an ARF overlay frame. It will
|
|
|
|
// become the GF so preserve last as an alternative prediction option.
|
|
|
|
cpi->refresh_last_frame = 0;
|
2013-03-12 19:33:40 +04:00
|
|
|
}
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
++cpi->next_frame_in_order;
|
|
|
|
#endif
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cpi->source) {
|
2013-04-02 21:24:56 +04:00
|
|
|
cpi->un_scaled_source = cpi->Source = force_src_buffer ? force_src_buffer
|
|
|
|
: &cpi->source->img;
|
2012-07-14 02:21:29 +04:00
|
|
|
*time_stamp = cpi->source->ts_start;
|
|
|
|
*time_end = cpi->source->ts_end;
|
|
|
|
*frame_flags = cpi->source->flags;
|
2013-04-03 02:08:50 +04:00
|
|
|
|
|
|
|
// fprintf(fp_out, " Frame:%d", cm->current_video_frame);
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
if (cpi->multi_arf_enabled) {
|
|
|
|
// fprintf(fp_out, " seq_no:%d this_frame_weight:%d",
|
|
|
|
// cpi->sequence_number, cpi->this_frame_weight);
|
|
|
|
} else {
|
|
|
|
// fprintf(fp_out, "\n");
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// fprintf(fp_out, "\n");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
if ((cm->frame_type != KEY_FRAME) && (cpi->pass == 2))
|
|
|
|
cpi->source_alt_ref_pending = is_next_frame_arf(cpi);
|
|
|
|
#endif
|
2012-07-14 02:21:29 +04:00
|
|
|
} else {
|
|
|
|
*size = 0;
|
|
|
|
if (flush && cpi->pass == 1 && !cpi->twopass.first_pass_done) {
|
2012-10-30 23:58:42 +04:00
|
|
|
vp9_end_first_pass(cpi); /* get last stats packet */
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->twopass.first_pass_done = 1;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
// fclose(fp_out);
|
2012-07-14 02:21:29 +04:00
|
|
|
return -1;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->source->ts_start < cpi->first_time_stamp_ever) {
|
|
|
|
cpi->first_time_stamp_ever = cpi->source->ts_start;
|
|
|
|
cpi->last_end_time_stamp_seen = cpi->source->ts_start;
|
|
|
|
}
|
2011-05-12 23:04:06 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// adjust frame rates based on timestamps given
|
2013-01-16 01:49:44 +04:00
|
|
|
if (!cpi->refresh_alt_ref_frame) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int64_t this_duration;
|
|
|
|
int step = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->source->ts_start == cpi->first_time_stamp_ever) {
|
|
|
|
this_duration = cpi->source->ts_end - cpi->source->ts_start;
|
|
|
|
step = 1;
|
|
|
|
} else {
|
2013-04-02 21:24:56 +04:00
|
|
|
int64_t last_duration = cpi->last_end_time_stamp_seen
|
|
|
|
- cpi->last_time_stamp_seen;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
this_duration = cpi->source->ts_end - cpi->last_end_time_stamp_seen;
|
2013-04-02 21:24:56 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// do a step update if the duration changes by 10%
|
|
|
|
if (last_duration)
|
2012-11-06 02:22:59 +04:00
|
|
|
step = (int)((this_duration - last_duration) * 10 / last_duration);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (this_duration) {
|
2013-04-02 21:24:56 +04:00
|
|
|
if (step) {
|
2013-07-13 04:12:46 +04:00
|
|
|
vp9_new_framerate(cpi, 10000000.0 / this_duration);
|
2013-04-02 21:24:56 +04:00
|
|
|
} else {
|
|
|
|
// Average this frame's rate into the last second's average
|
|
|
|
// frame rate. If we haven't seen 1 second yet, then average
|
|
|
|
// over the whole interval seen.
|
|
|
|
const double interval = MIN((double)(cpi->source->ts_end
|
|
|
|
- cpi->first_time_stamp_ever), 10000000.0);
|
2013-07-13 04:12:46 +04:00
|
|
|
double avg_duration = 10000000.0 / cpi->oxcf.framerate;
|
2012-07-14 02:21:29 +04:00
|
|
|
avg_duration *= (interval - avg_duration + this_duration);
|
|
|
|
avg_duration /= interval;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-07-13 04:12:46 +04:00
|
|
|
vp9_new_framerate(cpi, 10000000.0 / avg_duration);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->last_time_stamp_seen = cpi->source->ts_start;
|
|
|
|
cpi->last_end_time_stamp_seen = cpi->source->ts_end;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// start with a 0 size frame
|
|
|
|
*size = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// Clear down mmx registers
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state(); // __asm emms;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-01-15 03:52:20 +04:00
|
|
|
/* find a free buffer for the new frame, releasing the reference previously
|
|
|
|
* held.
|
|
|
|
*/
|
|
|
|
cm->fb_idx_ref_cnt[cm->new_fb_idx]--;
|
|
|
|
cm->new_fb_idx = get_free_fb(cm);
|
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if CONFIG_MULTIPLE_ARF
|
|
|
|
/* Set up the correct ARF frame. */
|
|
|
|
if (cpi->refresh_alt_ref_frame) {
|
|
|
|
++cpi->arf_buffered;
|
|
|
|
}
|
|
|
|
if (cpi->multi_arf_enabled && (cm->frame_type != KEY_FRAME) &&
|
|
|
|
(cpi->pass == 2)) {
|
|
|
|
cpi->alt_fb_idx = cpi->arf_buffer_idx[cpi->sequence_number];
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-02-22 23:22:03 +04:00
|
|
|
/* Get the mapping of L/G/A to the reference buffer pool */
|
|
|
|
cm->active_ref_idx[0] = cm->ref_frame_map[cpi->lst_fb_idx];
|
|
|
|
cm->active_ref_idx[1] = cm->ref_frame_map[cpi->gld_fb_idx];
|
|
|
|
cm->active_ref_idx[2] = cm->ref_frame_map[cpi->alt_fb_idx];
|
|
|
|
|
2013-04-03 02:08:50 +04:00
|
|
|
#if 0 // CONFIG_MULTIPLE_ARF
|
|
|
|
if (cpi->multi_arf_enabled) {
|
|
|
|
fprintf(fp_out, " idx(%d, %d, %d, %d) active(%d, %d, %d)",
|
|
|
|
cpi->lst_fb_idx, cpi->gld_fb_idx, cpi->alt_fb_idx, cm->new_fb_idx,
|
|
|
|
cm->active_ref_idx[0], cm->active_ref_idx[1], cm->active_ref_idx[2]);
|
|
|
|
if (cpi->refresh_alt_ref_frame)
|
|
|
|
fprintf(fp_out, " type:ARF");
|
|
|
|
if (cpi->is_src_frame_alt_ref)
|
|
|
|
fprintf(fp_out, " type:OVERLAY[%d]", cpi->alt_fb_idx);
|
|
|
|
fprintf(fp_out, "\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
cm->frame_type = INTER_FRAME;
|
|
|
|
cm->frame_flags = *frame_flags;
|
|
|
|
|
2013-04-17 22:45:35 +04:00
|
|
|
// Reset the frame pointers to the current frame size
|
2013-05-07 02:52:06 +04:00
|
|
|
vp9_realloc_frame_buffer(&cm->yv12_fb[cm->new_fb_idx],
|
|
|
|
cm->width, cm->height,
|
|
|
|
cm->subsampling_x, cm->subsampling_y,
|
|
|
|
VP9BORDERINPIXELS);
|
2013-02-07 02:22:17 +04:00
|
|
|
|
2013-04-17 22:45:35 +04:00
|
|
|
// Calculate scaling factors for each of the 3 available references
|
2013-05-15 04:10:17 +04:00
|
|
|
for (i = 0; i < ALLOWED_REFS_PER_FRAME; ++i)
|
|
|
|
vp9_setup_scale_factors(cm, i);
|
2013-04-03 23:22:50 +04:00
|
|
|
|
Convert subpixel filters to use convolve framework
Update the code to call the new convolution functions to do subpixel
prediction rather than the existing functions. Remove the old C and
assembly code, since it is unused. This causes a 50% performance
reduction on the decoder, but that will be resolved when the asm for
the new functions is available.
There is no consensus for whether 6-tap or 2-tap predictors will be
supported in the final codec, so these filters are implemented in
terms of the 8-tap code, so that quality testing of these modes
can continue. Implementing the lower complexity algorithms is a
simple exercise, should it be necessary.
This code produces slightly better results in the EIGHTTAP_SMOOTH
case, since the filter is now applied in only one direction when
the subpel motion is only in one direction. Like the previous code,
the filtering is skipped entirely on full-pel MVs. This combination
seems to give the best quality gains, but this may be indicative of a
bug in the encoder's filter selection, since the encoder could
achieve the result of skipping the filtering on full-pel by selecting
one of the other filters. This should be revisited.
Quality gains on derf positive on almost all clips. The only clip
that seemed to be hurt at all datarates was football
(-0.115% PSNR average, -0.587% min). Overall averages 0.375% PSNR,
0.347% SSIM.
Change-Id: I7d469716091b1d89b4b08adde5863999319d69ff
2013-01-29 04:59:03 +04:00
|
|
|
vp9_setup_interp_filters(&cpi->mb.e_mbd, DEFAULT_INTERP_FILTER, cm);
|
2013-04-03 23:22:50 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->pass == 1) {
|
|
|
|
Pass1Encode(cpi, size, dest, frame_flags);
|
|
|
|
} else if (cpi->pass == 2) {
|
|
|
|
Pass2Encode(cpi, size, dest, frame_flags);
|
2012-09-10 09:42:35 +04:00
|
|
|
} else {
|
2012-07-14 02:21:29 +04:00
|
|
|
encode_frame_to_data_rate(cpi, size, dest, frame_flags);
|
2012-09-10 09:42:35 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-27 01:39:58 +04:00
|
|
|
if (cm->refresh_frame_context)
|
|
|
|
cm->frame_contexts[cm->frame_context_idx] = cm->fc;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (*size > 0) {
|
2013-03-12 19:33:40 +04:00
|
|
|
// if its a dropped frame honor the requests on subsequent frames
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->droppable = !frame_is_reference(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
// return to normal state
|
2013-05-30 04:16:00 +04:00
|
|
|
cm->reset_frame_context = 0;
|
2013-04-27 01:39:58 +04:00
|
|
|
cm->refresh_frame_context = 1;
|
2013-01-16 01:49:44 +04:00
|
|
|
cpi->refresh_alt_ref_frame = 0;
|
|
|
|
cpi->refresh_golden_frame = 0;
|
|
|
|
cpi->refresh_last_frame = 1;
|
2012-07-14 02:21:29 +04:00
|
|
|
cm->frame_type = INTER_FRAME;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
vpx_usec_timer_mark(&cmptimer);
|
|
|
|
cpi->time_compress_data += vpx_usec_timer_elapsed(&cmptimer);
|
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
if (cpi->b_calculate_psnr && cpi->pass != 1 && cm->show_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
generate_psnr_packet(cpi);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2011-04-29 20:37:59 +04:00
|
|
|
#if CONFIG_INTERNAL_STATS
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->pass != 1) {
|
|
|
|
cpi->bytes += *size;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cm->show_frame) {
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->count++;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->b_calculate_psnr) {
|
|
|
|
double ye, ue, ve;
|
|
|
|
double frame_psnr;
|
|
|
|
YV12_BUFFER_CONFIG *orig = cpi->Source;
|
|
|
|
YV12_BUFFER_CONFIG *recon = cpi->common.frame_to_show;
|
|
|
|
YV12_BUFFER_CONFIG *pp = &cm->post_proc_buffer;
|
|
|
|
int y_samples = orig->y_height * orig->y_width;
|
|
|
|
int uv_samples = orig->uv_height * orig->uv_width;
|
|
|
|
int t_samples = y_samples + 2 * uv_samples;
|
2012-11-06 02:22:59 +04:00
|
|
|
double sq_error;
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2012-11-06 02:22:59 +04:00
|
|
|
ye = (double)calc_plane_error(orig->y_buffer, orig->y_stride,
|
2013-06-10 22:47:22 +04:00
|
|
|
recon->y_buffer, recon->y_stride,
|
|
|
|
orig->y_crop_width, orig->y_crop_height);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2012-11-06 02:22:59 +04:00
|
|
|
ue = (double)calc_plane_error(orig->u_buffer, orig->uv_stride,
|
2013-06-10 22:47:22 +04:00
|
|
|
recon->u_buffer, recon->uv_stride,
|
|
|
|
orig->uv_crop_width, orig->uv_crop_height);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2012-11-06 02:22:59 +04:00
|
|
|
ve = (double)calc_plane_error(orig->v_buffer, orig->uv_stride,
|
2013-06-10 22:47:22 +04:00
|
|
|
recon->v_buffer, recon->uv_stride,
|
|
|
|
orig->uv_crop_width, orig->uv_crop_height);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
sq_error = ye + ue + ve;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
frame_psnr = vp9_mse2psnr(t_samples, 255.0, sq_error);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
cpi->total_y += vp9_mse2psnr(y_samples, 255.0, ye);
|
|
|
|
cpi->total_u += vp9_mse2psnr(uv_samples, 255.0, ue);
|
|
|
|
cpi->total_v += vp9_mse2psnr(uv_samples, 255.0, ve);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->total_sq_error += sq_error;
|
|
|
|
cpi->total += frame_psnr;
|
|
|
|
{
|
|
|
|
double frame_psnr2, frame_ssim2 = 0;
|
|
|
|
double weight = 0;
|
2013-09-04 21:02:08 +04:00
|
|
|
#if CONFIG_VP9_POSTPROC
|
2012-11-28 22:00:25 +04:00
|
|
|
vp9_deblock(cm->frame_to_show, &cm->post_proc_buffer,
|
2013-08-12 22:41:09 +04:00
|
|
|
cm->lf.filter_level * 10 / 6);
|
2012-10-14 05:49:44 +04:00
|
|
|
#endif
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state();
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2012-11-06 02:22:59 +04:00
|
|
|
ye = (double)calc_plane_error(orig->y_buffer, orig->y_stride,
|
2013-06-10 22:47:22 +04:00
|
|
|
pp->y_buffer, pp->y_stride,
|
|
|
|
orig->y_crop_width, orig->y_crop_height);
|
2011-03-08 23:23:40 +03:00
|
|
|
|
2012-11-06 02:22:59 +04:00
|
|
|
ue = (double)calc_plane_error(orig->u_buffer, orig->uv_stride,
|
2013-06-10 22:47:22 +04:00
|
|
|
pp->u_buffer, pp->uv_stride,
|
|
|
|
orig->uv_crop_width, orig->uv_crop_height);
|
2011-03-08 23:23:40 +03:00
|
|
|
|
2012-11-06 02:22:59 +04:00
|
|
|
ve = (double)calc_plane_error(orig->v_buffer, orig->uv_stride,
|
2013-06-10 22:47:22 +04:00
|
|
|
pp->v_buffer, pp->uv_stride,
|
|
|
|
orig->uv_crop_width, orig->uv_crop_height);
|
2011-03-08 23:23:40 +03:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
sq_error = ye + ue + ve;
|
2011-03-08 23:23:40 +03:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
frame_psnr2 = vp9_mse2psnr(t_samples, 255.0, sq_error);
|
2011-03-08 17:05:18 +03:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
cpi->totalp_y += vp9_mse2psnr(y_samples, 255.0, ye);
|
|
|
|
cpi->totalp_u += vp9_mse2psnr(uv_samples, 255.0, ue);
|
|
|
|
cpi->totalp_v += vp9_mse2psnr(uv_samples, 255.0, ve);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->total_sq_error2 += sq_error;
|
|
|
|
cpi->totalp += frame_psnr2;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
frame_ssim2 = vp9_calc_ssim(cpi->Source,
|
2013-04-01 20:10:27 +04:00
|
|
|
recon, 1, &weight);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
cpi->summed_quality += frame_ssim2 * weight;
|
|
|
|
cpi->summed_weights += weight;
|
2013-04-01 20:10:27 +04:00
|
|
|
|
|
|
|
frame_ssim2 = vp9_calc_ssim(cpi->Source,
|
|
|
|
&cm->post_proc_buffer, 1, &weight);
|
|
|
|
|
|
|
|
cpi->summedp_quality += frame_ssim2 * weight;
|
|
|
|
cpi->summedp_weights += weight;
|
2010-12-02 02:50:14 +03:00
|
|
|
#if 0
|
2012-07-14 02:21:29 +04:00
|
|
|
{
|
|
|
|
FILE *f = fopen("q_used.stt", "a");
|
|
|
|
fprintf(f, "%5d : Y%f7.3:U%f7.3:V%f7.3:F%f7.3:S%7.3f\n",
|
|
|
|
cpi->common.current_video_frame, y2, u2, v2,
|
|
|
|
frame_psnr2, frame_ssim2);
|
|
|
|
fclose(f);
|
|
|
|
}
|
2010-12-02 02:50:14 +03:00
|
|
|
#endif
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->b_calculate_ssimg) {
|
|
|
|
double y, u, v, frame_all;
|
2012-10-30 23:58:42 +04:00
|
|
|
frame_all = vp9_calc_ssimg(cpi->Source, cm->frame_to_show,
|
2012-10-22 07:47:57 +04:00
|
|
|
&y, &u, &v);
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->total_ssimg_y += y;
|
|
|
|
cpi->total_ssimg_u += u;
|
|
|
|
cpi->total_ssimg_v += v;
|
|
|
|
cpi->total_ssimg_all += frame_all;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
|
|
|
#endif
|
2013-04-03 02:08:50 +04:00
|
|
|
// fclose(fp_out);
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_get_preview_raw_frame(VP9_PTR comp, YV12_BUFFER_CONFIG *dest,
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_ppflags_t *flags) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *) comp;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-06 00:23:34 +04:00
|
|
|
if (!cpi->common.show_frame)
|
2012-07-14 02:21:29 +04:00
|
|
|
return -1;
|
|
|
|
else {
|
|
|
|
int ret;
|
2013-09-04 21:02:08 +04:00
|
|
|
#if CONFIG_VP9_POSTPROC
|
2013-08-10 04:24:40 +04:00
|
|
|
ret = vp9_post_proc_frame(&cpi->common, dest, flags);
|
2010-05-18 19:58:33 +04:00
|
|
|
#else
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->common.frame_to_show) {
|
|
|
|
*dest = *cpi->common.frame_to_show;
|
2013-03-21 03:41:30 +04:00
|
|
|
dest->y_width = cpi->common.width;
|
|
|
|
dest->y_height = cpi->common.height;
|
|
|
|
dest->uv_height = cpi->common.height / 2;
|
2012-07-14 02:21:29 +04:00
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
ret = -1;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-09-04 21:02:08 +04:00
|
|
|
#endif // !CONFIG_VP9_POSTPROC
|
2012-11-01 01:40:53 +04:00
|
|
|
vp9_clear_system_state();
|
2012-07-14 02:21:29 +04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_set_roimap(VP9_PTR comp, unsigned char *map, unsigned int rows,
|
2013-07-23 15:09:04 +04:00
|
|
|
unsigned int cols, int delta_q[MAX_SEGMENTS],
|
|
|
|
int delta_lf[MAX_SEGMENTS],
|
|
|
|
unsigned int threshold[MAX_SEGMENTS]) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *) comp;
|
2013-07-23 15:09:04 +04:00
|
|
|
signed char feature_data[SEG_LVL_MAX][MAX_SEGMENTS];
|
2013-08-14 22:20:33 +04:00
|
|
|
struct segmentation *seg = &cpi->common.seg;
|
2012-07-14 02:21:29 +04:00
|
|
|
int i;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (cpi->common.mb_rows != rows || cpi->common.mb_cols != cols)
|
|
|
|
return -1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (!map) {
|
2012-10-31 04:53:32 +04:00
|
|
|
vp9_disable_segmentation((VP9_PTR)cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the segmentation Map
|
2012-10-31 04:53:32 +04:00
|
|
|
vp9_set_segmentation_map((VP9_PTR)cpi, map);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Activate segmentation.
|
2012-10-31 04:53:32 +04:00
|
|
|
vp9_enable_segmentation((VP9_PTR)cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
|
2013-08-10 03:25:01 +04:00
|
|
|
// Set up the quant, LF and breakout threshold segment data
|
2013-07-23 15:09:04 +04:00
|
|
|
for (i = 0; i < MAX_SEGMENTS; i++) {
|
2013-04-24 16:04:45 +04:00
|
|
|
feature_data[SEG_LVL_ALT_Q][i] = delta_q[i];
|
|
|
|
feature_data[SEG_LVL_ALT_LF][i] = delta_lf[i];
|
|
|
|
cpi->segment_encode_breakout[i] = threshold[i];
|
|
|
|
}
|
2012-07-14 02:21:29 +04:00
|
|
|
|
|
|
|
// Enable the loop and quant changes in the feature mask
|
2013-07-23 15:09:04 +04:00
|
|
|
for (i = 0; i < MAX_SEGMENTS; i++) {
|
2012-07-14 02:21:29 +04:00
|
|
|
if (delta_q[i])
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_enable_segfeature(seg, i, SEG_LVL_ALT_Q);
|
2012-07-14 02:21:29 +04:00
|
|
|
else
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_disable_segfeature(seg, i, SEG_LVL_ALT_Q);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (delta_lf[i])
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_enable_segfeature(seg, i, SEG_LVL_ALT_LF);
|
2012-07-14 02:21:29 +04:00
|
|
|
else
|
2013-08-02 01:53:14 +04:00
|
|
|
vp9_disable_segfeature(seg, i, SEG_LVL_ALT_LF);
|
2012-07-14 02:21:29 +04:00
|
|
|
}
|
2011-10-05 14:26:00 +04:00
|
|
|
|
2013-08-10 03:25:01 +04:00
|
|
|
// Initialize the feature data structure
|
2012-07-14 02:21:29 +04:00
|
|
|
// SEGMENT_DELTADATA 0, SEGMENT_ABSDATA 1
|
2012-10-31 04:53:32 +04:00
|
|
|
vp9_set_segment_data((VP9_PTR)cpi, &feature_data[0][0], SEGMENT_DELTADATA);
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_set_active_map(VP9_PTR comp, unsigned char *map,
|
2012-10-31 01:25:33 +04:00
|
|
|
unsigned int rows, unsigned int cols) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *) comp;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
if (rows == cpi->common.mb_rows && cols == cpi->common.mb_cols) {
|
|
|
|
if (map) {
|
|
|
|
vpx_memcpy(cpi->active_map, map, rows * cols);
|
|
|
|
cpi->active_map_enabled = 1;
|
2013-04-02 21:24:56 +04:00
|
|
|
} else {
|
2012-07-14 02:21:29 +04:00
|
|
|
cpi->active_map_enabled = 0;
|
2013-04-02 21:24:56 +04:00
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
// cpi->active_map_enabled = 0;
|
|
|
|
return -1;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_set_internal_size(VP9_PTR comp,
|
2012-10-31 01:25:33 +04:00
|
|
|
VPX_SCALING horiz_mode, VPX_SCALING vert_mode) {
|
2012-10-31 04:53:32 +04:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *) comp;
|
2013-02-21 00:34:31 +04:00
|
|
|
VP9_COMMON *cm = &cpi->common;
|
2013-03-15 01:36:08 +04:00
|
|
|
int hr = 0, hs = 0, vr = 0, vs = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-04-02 21:24:56 +04:00
|
|
|
if (horiz_mode > ONETWO || vert_mode > ONETWO)
|
2012-07-14 02:21:29 +04:00
|
|
|
return -1;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-15 01:36:08 +04:00
|
|
|
Scale2Ratio(horiz_mode, &hr, &hs);
|
|
|
|
Scale2Ratio(vert_mode, &vr, &vs);
|
2013-02-21 22:38:27 +04:00
|
|
|
|
2013-03-15 01:36:08 +04:00
|
|
|
// always go to the next whole number
|
2013-03-21 03:41:30 +04:00
|
|
|
cm->width = (hs - 1 + cpi->oxcf.width * hr) / hs;
|
|
|
|
cm->height = (vs - 1 + cpi->oxcf.height * vr) / vs;
|
2013-02-21 00:34:31 +04:00
|
|
|
|
2013-03-21 03:41:30 +04:00
|
|
|
assert(cm->width <= cpi->initial_width);
|
|
|
|
assert(cm->height <= cpi->initial_height);
|
2013-02-21 00:34:31 +04:00
|
|
|
update_frame_size(cpi);
|
2012-07-14 02:21:29 +04:00
|
|
|
return 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-09-05 19:55:47 +04:00
|
|
|
int vp9_set_size_literal(VP9_PTR comp, unsigned int width,
|
|
|
|
unsigned int height) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *)comp;
|
|
|
|
VP9_COMMON *cm = &cpi->common;
|
|
|
|
|
|
|
|
check_initial_width(cpi, NULL);
|
|
|
|
|
|
|
|
if (width) {
|
|
|
|
cm->width = width;
|
|
|
|
if (cm->width * 5 < cpi->initial_width) {
|
|
|
|
cm->width = cpi->initial_width / 5 + 1;
|
|
|
|
printf("Warning: Desired width too small, changed to %d \n", cm->width);
|
|
|
|
}
|
|
|
|
if (cm->width > cpi->initial_width) {
|
|
|
|
cm->width = cpi->initial_width;
|
|
|
|
printf("Warning: Desired width too large, changed to %d \n", cm->width);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (height) {
|
|
|
|
cm->height = height;
|
|
|
|
if (cm->height * 5 < cpi->initial_height) {
|
|
|
|
cm->height = cpi->initial_height / 5 + 1;
|
|
|
|
printf("Warning: Desired height too small, changed to %d \n", cm->height);
|
|
|
|
}
|
|
|
|
if (cm->height > cpi->initial_height) {
|
|
|
|
cm->height = cpi->initial_height;
|
|
|
|
printf("Warning: Desired height too large, changed to %d \n", cm->height);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(cm->width <= cpi->initial_width);
|
|
|
|
assert(cm->height <= cpi->initial_height);
|
|
|
|
update_frame_size(cpi);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int vp9_switch_layer(VP9_PTR comp, int layer) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *)comp;
|
|
|
|
|
|
|
|
if (cpi->use_svc) {
|
|
|
|
cpi->current_layer = layer;
|
|
|
|
|
|
|
|
// Use buffer i for layer i LST
|
|
|
|
cpi->lst_fb_idx = layer;
|
|
|
|
|
|
|
|
// Use buffer i-1 for layer i Alt (Inter-layer prediction)
|
|
|
|
if (layer != 0) cpi->alt_fb_idx = layer - 1;
|
|
|
|
|
|
|
|
// Use the rest for Golden
|
|
|
|
if (layer < 2 * cpi->number_spatial_layers - NUM_REF_FRAMES)
|
|
|
|
cpi->gld_fb_idx = cpi->lst_fb_idx;
|
|
|
|
else
|
|
|
|
cpi->gld_fb_idx = 2 * cpi->number_spatial_layers - 1 - layer;
|
|
|
|
|
|
|
|
printf("Switching to layer %d:\n", layer);
|
|
|
|
printf("Using references: LST/GLD/ALT [%d|%d|%d]\n", cpi->lst_fb_idx,
|
|
|
|
cpi->gld_fb_idx, cpi->alt_fb_idx);
|
|
|
|
} else {
|
|
|
|
printf("Switching layer not supported. Enable SVC first \n");
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-09-05 19:55:47 +04:00
|
|
|
void vp9_set_svc(VP9_PTR comp, int use_svc) {
|
|
|
|
VP9_COMP *cpi = (VP9_COMP *)comp;
|
|
|
|
cpi->use_svc = use_svc;
|
|
|
|
if (cpi->use_svc) printf("Enabled SVC encoder \n");
|
|
|
|
return;
|
|
|
|
}
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-10-30 23:58:42 +04:00
|
|
|
int vp9_calc_ss_err(YV12_BUFFER_CONFIG *source, YV12_BUFFER_CONFIG *dest) {
|
2012-07-14 02:21:29 +04:00
|
|
|
int i, j;
|
2013-03-28 01:22:30 +04:00
|
|
|
int total = 0;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2012-12-19 03:31:19 +04:00
|
|
|
uint8_t *src = source->y_buffer;
|
|
|
|
uint8_t *dst = dest->y_buffer;
|
2010-05-18 19:58:33 +04:00
|
|
|
|
2013-03-28 01:22:30 +04:00
|
|
|
// Loop through the Y plane raw and reconstruction data summing
|
|
|
|
// (square differences)
|
2012-07-14 02:21:29 +04:00
|
|
|
for (i = 0; i < source->y_height; i += 16) {
|
|
|
|
for (j = 0; j < source->y_width; j += 16) {
|
|
|
|
unsigned int sse;
|
2013-03-28 01:22:30 +04:00
|
|
|
total += vp9_mse16x16(src + j, source->y_stride, dst + j, dest->y_stride,
|
2012-10-22 07:47:57 +04:00
|
|
|
&sse);
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-07-14 02:21:29 +04:00
|
|
|
src += 16 * source->y_stride;
|
|
|
|
dst += 16 * dest->y_stride;
|
|
|
|
}
|
|
|
|
|
2013-03-28 01:22:30 +04:00
|
|
|
return total;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|
2011-03-18 00:07:59 +03:00
|
|
|
|
|
|
|
|
2012-10-31 04:53:32 +04:00
|
|
|
int vp9_get_quantizer(VP9_PTR c) {
|
2013-04-02 21:24:56 +04:00
|
|
|
return ((VP9_COMP *)c)->common.base_qindex;
|
2010-05-18 19:58:33 +04:00
|
|
|
}
|