2010-05-18 17:58:33 +02:00
|
|
|
/*
|
2010-09-09 14:16:39 +02:00
|
|
|
* Copyright (c) 2010 The WebM project authors. All Rights Reserved.
|
2010-05-18 17:58:33 +02:00
|
|
|
*
|
2010-06-18 18:39:21 +02:00
|
|
|
* Use of this source code is governed by a BSD-style license
|
2010-06-04 22:19:40 +02: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 18:39:21 +02:00
|
|
|
* in the file PATENTS. All contributing project authors may
|
2010-06-04 22:19:40 +02:00
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
2010-05-18 17:58:33 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
2010-05-24 17:39:59 +02:00
|
|
|
#include "vpx/vpx_codec.h"
|
|
|
|
#include "vpx/internal/vpx_codec_internal.h"
|
2010-05-18 17:58:33 +02:00
|
|
|
#include "vpx_version.h"
|
2012-11-27 22:59:17 +01:00
|
|
|
#include "vp9/encoder/vp9_onyx_int.h"
|
2012-11-05 21:37:14 +01:00
|
|
|
#include "vpx/vp8cx.h"
|
2012-11-27 22:59:17 +01:00
|
|
|
#include "vp9/encoder/vp9_firstpass.h"
|
|
|
|
#include "vp9/common/vp9_onyx.h"
|
2013-03-13 20:15:43 +01:00
|
|
|
#include "vp9/vp9_iface_common.h"
|
2010-05-18 17:58:33 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
struct vp9_extracfg {
|
2012-07-14 00:21:29 +02:00
|
|
|
struct vpx_codec_pkt_list *pkt_list;
|
|
|
|
int cpu_used; /** available cpu percentage in 1/16*/
|
|
|
|
unsigned int enable_auto_alt_ref; /** if encoder decides to uses alternate reference frame */
|
|
|
|
unsigned int noise_sensitivity;
|
|
|
|
unsigned int Sharpness;
|
|
|
|
unsigned int static_thresh;
|
[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 18:35:28 +01:00
|
|
|
unsigned int tile_columns;
|
2013-02-08 20:33:11 +01:00
|
|
|
unsigned int tile_rows;
|
2012-07-14 00:21:29 +02:00
|
|
|
unsigned int arnr_max_frames; /* alt_ref Noise Reduction Max Frame Count */
|
|
|
|
unsigned int arnr_strength; /* alt_ref Noise Reduction Strength */
|
|
|
|
unsigned int arnr_type; /* alt_ref filter type */
|
|
|
|
unsigned int experimental;
|
|
|
|
vp8e_tuning tuning;
|
|
|
|
unsigned int cq_level; /* constrained quality level */
|
|
|
|
unsigned int rc_max_intra_bitrate_pct;
|
2012-11-05 21:37:14 +01:00
|
|
|
unsigned int lossless;
|
2013-03-27 19:07:26 +01:00
|
|
|
unsigned int frame_parallel_decoding_mode;
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
struct extraconfig_map {
|
|
|
|
int usage;
|
2013-05-23 03:04:08 +02:00
|
|
|
struct vp9_extracfg cfg;
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
static const struct extraconfig_map extracfg_map[] = {
|
|
|
|
{
|
|
|
|
0,
|
2010-05-18 17:58:33 +02:00
|
|
|
{
|
2012-07-14 00:21:29 +02:00
|
|
|
NULL,
|
|
|
|
0, /* cpu_used */
|
2013-08-19 23:16:26 +02:00
|
|
|
1, /* enable_auto_alt_ref */
|
2012-07-14 00:21:29 +02:00
|
|
|
0, /* noise_sensitivity */
|
|
|
|
0, /* Sharpness */
|
|
|
|
0, /* static_thresh */
|
2013-02-07 00:30:21 +01:00
|
|
|
0, /* tile_columns */
|
2013-02-08 20:33:11 +01:00
|
|
|
0, /* tile_rows */
|
2013-08-19 23:16:26 +02:00
|
|
|
7, /* arnr_max_frames */
|
|
|
|
5, /* arnr_strength */
|
2012-07-14 00:21:29 +02:00
|
|
|
3, /* arnr_type*/
|
|
|
|
0, /* experimental mode */
|
|
|
|
0, /* tuning*/
|
|
|
|
10, /* cq_level */
|
|
|
|
0, /* rc_max_intra_bitrate_pct */
|
2012-11-05 21:37:14 +01:00
|
|
|
0, /* lossless */
|
2013-03-27 19:07:26 +01:00
|
|
|
0, /* frame_parallel_decoding_mode */
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
struct vpx_codec_alg_priv {
|
|
|
|
vpx_codec_priv_t base;
|
|
|
|
vpx_codec_enc_cfg_t cfg;
|
2013-05-23 03:04:08 +02:00
|
|
|
struct vp9_extracfg vp8_cfg;
|
2012-10-31 22:40:53 +01:00
|
|
|
VP9_CONFIG oxcf;
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_PTR cpi;
|
2012-07-14 00:21:29 +02:00
|
|
|
unsigned char *cx_data;
|
|
|
|
unsigned int cx_data_sz;
|
2012-11-15 21:19:07 +01:00
|
|
|
unsigned char *pending_cx_data;
|
|
|
|
unsigned int pending_cx_data_sz;
|
2013-03-05 21:23:34 +01:00
|
|
|
int pending_frame_count;
|
|
|
|
uint32_t pending_frame_sizes[8];
|
|
|
|
uint32_t pending_frame_magnitude;
|
2012-07-14 00:21:29 +02:00
|
|
|
vpx_image_t preview_img;
|
|
|
|
vp8_postproc_cfg_t preview_ppcfg;
|
|
|
|
vpx_codec_pkt_list_decl(64) pkt_list; // changed to accomendate the maximum number of lagged frames allowed
|
|
|
|
unsigned int fixed_kf_cntr;
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static vpx_codec_err_t
|
|
|
|
update_error_state(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
const struct vpx_internal_error_info *error) {
|
|
|
|
vpx_codec_err_t res;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if ((res = error->error_code))
|
|
|
|
ctx->base.err_detail = error->has_detail
|
|
|
|
? error->detail
|
|
|
|
: NULL;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return res;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-17 15:08:47 +01:00
|
|
|
#undef ERROR
|
2010-05-18 17:58:33 +02:00
|
|
|
#define ERROR(str) do {\
|
2012-07-14 00:21:29 +02:00
|
|
|
ctx->base.err_detail = str;\
|
|
|
|
return VPX_CODEC_INVALID_PARAM;\
|
|
|
|
} while(0)
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
#define RANGE_CHECK(p,memb,lo,hi) do {\
|
2012-07-14 00:21:29 +02:00
|
|
|
if(!(((p)->memb == lo || (p)->memb > (lo)) && (p)->memb <= hi)) \
|
|
|
|
ERROR(#memb " out of range ["#lo".."#hi"]");\
|
|
|
|
} while(0)
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2010-06-11 20:33:49 +02:00
|
|
|
#define RANGE_CHECK_HI(p,memb,hi) do {\
|
2012-07-14 00:21:29 +02:00
|
|
|
if(!((p)->memb <= (hi))) \
|
|
|
|
ERROR(#memb " out of range [.."#hi"]");\
|
|
|
|
} while(0)
|
2010-06-11 20:33:49 +02:00
|
|
|
|
2010-05-18 17:58:33 +02:00
|
|
|
#define RANGE_CHECK_LO(p,memb,lo) do {\
|
2012-07-14 00:21:29 +02:00
|
|
|
if(!((p)->memb >= (lo))) \
|
|
|
|
ERROR(#memb " out of range ["#lo"..]");\
|
|
|
|
} while(0)
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
#define RANGE_CHECK_BOOL(p,memb) do {\
|
2012-07-14 00:21:29 +02:00
|
|
|
if(!!((p)->memb) != (p)->memb) ERROR(#memb " expected boolean");\
|
|
|
|
} while(0)
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
static vpx_codec_err_t validate_config(vpx_codec_alg_priv_t *ctx,
|
|
|
|
const vpx_codec_enc_cfg_t *cfg,
|
2013-05-23 03:04:08 +02:00
|
|
|
const struct vp9_extracfg *vp8_cfg) {
|
2013-02-28 02:09:12 +01:00
|
|
|
RANGE_CHECK(cfg, g_w, 1, 65535); /* 16 bits available */
|
|
|
|
RANGE_CHECK(cfg, g_h, 1, 65535); /* 16 bits available */
|
2012-07-14 00:21:29 +02:00
|
|
|
RANGE_CHECK(cfg, g_timebase.den, 1, 1000000000);
|
|
|
|
RANGE_CHECK(cfg, g_timebase.num, 1, cfg->g_timebase.den);
|
|
|
|
RANGE_CHECK_HI(cfg, g_profile, 3);
|
2012-11-05 21:37:14 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
RANGE_CHECK_HI(cfg, rc_max_quantizer, 63);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_min_quantizer, cfg->rc_max_quantizer);
|
2012-11-05 21:37:14 +01:00
|
|
|
RANGE_CHECK_BOOL(vp8_cfg, lossless);
|
|
|
|
if (vp8_cfg->lossless) {
|
|
|
|
RANGE_CHECK_HI(cfg, rc_max_quantizer, 0);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_min_quantizer, 0);
|
|
|
|
}
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
RANGE_CHECK_HI(cfg, g_threads, 64);
|
|
|
|
RANGE_CHECK_HI(cfg, g_lag_in_frames, MAX_LAG_BUFFERS);
|
|
|
|
RANGE_CHECK(cfg, rc_end_usage, VPX_VBR, VPX_CQ);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_undershoot_pct, 1000);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_overshoot_pct, 1000);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_2pass_vbr_bias_pct, 100);
|
|
|
|
RANGE_CHECK(cfg, kf_mode, VPX_KF_DISABLED, VPX_KF_AUTO);
|
|
|
|
// RANGE_CHECK_BOOL(cfg, g_delete_firstpassfile);
|
|
|
|
RANGE_CHECK_BOOL(cfg, rc_resize_allowed);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_dropframe_thresh, 100);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_resize_up_thresh, 100);
|
|
|
|
RANGE_CHECK_HI(cfg, rc_resize_down_thresh, 100);
|
|
|
|
RANGE_CHECK(cfg, g_pass, VPX_RC_ONE_PASS, VPX_RC_LAST_PASS);
|
|
|
|
|
|
|
|
/* VP8 does not support a lower bound on the keyframe interval in
|
|
|
|
* automatic keyframe placement mode.
|
|
|
|
*/
|
|
|
|
if (cfg->kf_mode != VPX_KF_DISABLED && cfg->kf_min_dist != cfg->kf_max_dist
|
|
|
|
&& cfg->kf_min_dist > 0)
|
|
|
|
ERROR("kf_min_dist not supported in auto mode, use 0 "
|
|
|
|
"or kf_max_dist instead.");
|
|
|
|
|
|
|
|
RANGE_CHECK_BOOL(vp8_cfg, enable_auto_alt_ref);
|
|
|
|
RANGE_CHECK(vp8_cfg, cpu_used, -16, 16);
|
|
|
|
|
|
|
|
RANGE_CHECK_HI(vp8_cfg, noise_sensitivity, 6);
|
|
|
|
|
2013-02-07 00:30:21 +01:00
|
|
|
RANGE_CHECK(vp8_cfg, tile_columns, 0, 6);
|
2013-02-08 20:33:11 +01:00
|
|
|
RANGE_CHECK(vp8_cfg, tile_rows, 0, 2);
|
2012-07-14 00:21:29 +02:00
|
|
|
RANGE_CHECK_HI(vp8_cfg, Sharpness, 7);
|
|
|
|
RANGE_CHECK(vp8_cfg, arnr_max_frames, 0, 15);
|
|
|
|
RANGE_CHECK_HI(vp8_cfg, arnr_strength, 6);
|
|
|
|
RANGE_CHECK(vp8_cfg, arnr_type, 1, 3);
|
|
|
|
RANGE_CHECK(vp8_cfg, cq_level, 0, 63);
|
|
|
|
|
|
|
|
if (cfg->g_pass == VPX_RC_LAST_PASS) {
|
|
|
|
size_t packet_sz = sizeof(FIRSTPASS_STATS);
|
2012-11-05 23:22:59 +01:00
|
|
|
int n_packets = (int)(cfg->rc_twopass_stats_in.sz / packet_sz);
|
2012-07-14 00:21:29 +02:00
|
|
|
FIRSTPASS_STATS *stats;
|
|
|
|
|
|
|
|
if (!cfg->rc_twopass_stats_in.buf)
|
|
|
|
ERROR("rc_twopass_stats_in.buf not set.");
|
|
|
|
|
|
|
|
if (cfg->rc_twopass_stats_in.sz % packet_sz)
|
|
|
|
ERROR("rc_twopass_stats_in.sz indicates truncated packet.");
|
|
|
|
|
|
|
|
if (cfg->rc_twopass_stats_in.sz < 2 * packet_sz)
|
|
|
|
ERROR("rc_twopass_stats_in requires at least two packets.");
|
|
|
|
|
|
|
|
stats = (void *)((char *)cfg->rc_twopass_stats_in.buf
|
|
|
|
+ (n_packets - 1) * packet_sz);
|
|
|
|
|
|
|
|
if ((int)(stats->count + 0.5) != n_packets - 1)
|
|
|
|
ERROR("rc_twopass_stats_in missing EOS stats packet");
|
|
|
|
}
|
|
|
|
|
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static vpx_codec_err_t validate_img(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
const vpx_image_t *img) {
|
|
|
|
switch (img->fmt) {
|
2010-05-21 05:22:39 +02:00
|
|
|
case VPX_IMG_FMT_YV12:
|
|
|
|
case VPX_IMG_FMT_I420:
|
2013-05-07 00:52:06 +02:00
|
|
|
case VPX_IMG_FMT_I422:
|
|
|
|
case VPX_IMG_FMT_I444:
|
2012-07-14 00:21:29 +02:00
|
|
|
break;
|
2010-05-18 17:58:33 +02:00
|
|
|
default:
|
2013-05-07 00:52:06 +02:00
|
|
|
ERROR("Invalid image format. Only YV12, I420, I422, I444 images are "
|
|
|
|
"supported.");
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if ((img->d_w != ctx->cfg.g_w) || (img->d_h != ctx->cfg.g_h))
|
|
|
|
ERROR("Image size must match encoder init configuration size");
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t set_vp9e_config(VP9_CONFIG *oxcf,
|
2010-05-18 17:58:33 +02:00
|
|
|
vpx_codec_enc_cfg_t cfg,
|
2013-05-23 03:04:08 +02:00
|
|
|
struct vp9_extracfg vp8_cfg) {
|
2013-03-21 00:41:30 +01:00
|
|
|
oxcf->version = cfg.g_profile | (vp8_cfg.experimental ? 0x4 : 0);
|
|
|
|
oxcf->width = cfg.g_w;
|
|
|
|
oxcf->height = cfg.g_h;
|
2012-07-14 00:21:29 +02:00
|
|
|
/* guess a frame rate if out of whack, use 30 */
|
2013-07-13 02:12:46 +02:00
|
|
|
oxcf->framerate = (double)(cfg.g_timebase.den) / (double)(cfg.g_timebase.num);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-07-13 02:12:46 +02:00
|
|
|
if (oxcf->framerate > 180) {
|
|
|
|
oxcf->framerate = 30;
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
switch (cfg.g_pass) {
|
2010-05-18 17:58:33 +02:00
|
|
|
case VPX_RC_ONE_PASS:
|
2012-07-14 00:21:29 +02:00
|
|
|
oxcf->Mode = MODE_BESTQUALITY;
|
|
|
|
break;
|
2010-05-18 17:58:33 +02:00
|
|
|
case VPX_RC_FIRST_PASS:
|
2012-07-14 00:21:29 +02:00
|
|
|
oxcf->Mode = MODE_FIRSTPASS;
|
|
|
|
break;
|
2010-05-18 17:58:33 +02:00
|
|
|
case VPX_RC_LAST_PASS:
|
2012-07-14 00:21:29 +02:00
|
|
|
oxcf->Mode = MODE_SECONDPASS_BEST;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cfg.g_pass == VPX_RC_FIRST_PASS) {
|
|
|
|
oxcf->allow_lag = 0;
|
|
|
|
oxcf->lag_in_frames = 0;
|
|
|
|
} else {
|
|
|
|
oxcf->allow_lag = (cfg.g_lag_in_frames) > 0;
|
|
|
|
oxcf->lag_in_frames = cfg.g_lag_in_frames;
|
|
|
|
}
|
|
|
|
|
|
|
|
// VBR only supported for now.
|
|
|
|
// CBR code has been deprectated for experimental phase.
|
|
|
|
// CQ mode not yet tested
|
|
|
|
oxcf->end_usage = USAGE_LOCAL_FILE_PLAYBACK;
|
|
|
|
/*if (cfg.rc_end_usage == VPX_CQ)
|
|
|
|
oxcf->end_usage = USAGE_CONSTRAINED_QUALITY;
|
|
|
|
else
|
|
|
|
oxcf->end_usage = USAGE_LOCAL_FILE_PLAYBACK;*/
|
|
|
|
|
|
|
|
oxcf->target_bandwidth = cfg.rc_target_bitrate;
|
|
|
|
oxcf->rc_max_intra_bitrate_pct = vp8_cfg.rc_max_intra_bitrate_pct;
|
|
|
|
|
|
|
|
oxcf->best_allowed_q = cfg.rc_min_quantizer;
|
|
|
|
oxcf->worst_allowed_q = cfg.rc_max_quantizer;
|
|
|
|
oxcf->cq_level = vp8_cfg.cq_level;
|
|
|
|
oxcf->fixed_q = -1;
|
|
|
|
|
|
|
|
oxcf->under_shoot_pct = cfg.rc_undershoot_pct;
|
|
|
|
oxcf->over_shoot_pct = cfg.rc_overshoot_pct;
|
|
|
|
|
|
|
|
oxcf->maximum_buffer_size = cfg.rc_buf_sz;
|
|
|
|
oxcf->starting_buffer_level = cfg.rc_buf_initial_sz;
|
|
|
|
oxcf->optimal_buffer_level = cfg.rc_buf_optimal_sz;
|
|
|
|
|
|
|
|
oxcf->two_pass_vbrbias = cfg.rc_2pass_vbr_bias_pct;
|
|
|
|
oxcf->two_pass_vbrmin_section = cfg.rc_2pass_vbr_minsection_pct;
|
|
|
|
oxcf->two_pass_vbrmax_section = cfg.rc_2pass_vbr_maxsection_pct;
|
|
|
|
|
|
|
|
oxcf->auto_key = cfg.kf_mode == VPX_KF_AUTO
|
|
|
|
&& cfg.kf_min_dist != cfg.kf_max_dist;
|
|
|
|
// oxcf->kf_min_dist = cfg.kf_min_dis;
|
|
|
|
oxcf->key_freq = cfg.kf_max_dist;
|
|
|
|
|
|
|
|
// oxcf->delete_first_pass_file = cfg.g_delete_firstpassfile;
|
|
|
|
// strcpy(oxcf->first_pass_file, cfg.g_firstpass_file);
|
|
|
|
|
|
|
|
oxcf->cpu_used = vp8_cfg.cpu_used;
|
|
|
|
oxcf->encode_breakout = vp8_cfg.static_thresh;
|
|
|
|
oxcf->play_alternate = vp8_cfg.enable_auto_alt_ref;
|
|
|
|
oxcf->noise_sensitivity = vp8_cfg.noise_sensitivity;
|
|
|
|
oxcf->Sharpness = vp8_cfg.Sharpness;
|
|
|
|
|
|
|
|
oxcf->two_pass_stats_in = cfg.rc_twopass_stats_in;
|
|
|
|
oxcf->output_pkt_list = vp8_cfg.pkt_list;
|
|
|
|
|
|
|
|
oxcf->arnr_max_frames = vp8_cfg.arnr_max_frames;
|
|
|
|
oxcf->arnr_strength = vp8_cfg.arnr_strength;
|
|
|
|
oxcf->arnr_type = vp8_cfg.arnr_type;
|
|
|
|
|
|
|
|
oxcf->tuning = vp8_cfg.tuning;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
[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 18:35:28 +01:00
|
|
|
oxcf->tile_columns = vp8_cfg.tile_columns;
|
2013-02-08 20:33:11 +01:00
|
|
|
oxcf->tile_rows = vp8_cfg.tile_rows;
|
[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 18:35:28 +01:00
|
|
|
|
2012-11-05 21:37:14 +01:00
|
|
|
oxcf->lossless = vp8_cfg.lossless;
|
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 04:03:31 +02:00
|
|
|
|
2013-01-15 15:43:35 +01:00
|
|
|
oxcf->error_resilient_mode = cfg.g_error_resilient;
|
2013-03-27 19:07:26 +01:00
|
|
|
oxcf->frame_parallel_decoding_mode = vp8_cfg.frame_parallel_decoding_mode;
|
2012-07-14 00:21:29 +02:00
|
|
|
/*
|
2013-01-15 15:43:35 +01:00
|
|
|
printf("Current VP9 Settings: \n");
|
|
|
|
printf("target_bandwidth: %d\n", oxcf->target_bandwidth);
|
|
|
|
printf("noise_sensitivity: %d\n", oxcf->noise_sensitivity);
|
|
|
|
printf("Sharpness: %d\n", oxcf->Sharpness);
|
|
|
|
printf("cpu_used: %d\n", oxcf->cpu_used);
|
|
|
|
printf("Mode: %d\n", oxcf->Mode);
|
|
|
|
// printf("delete_first_pass_file: %d\n", oxcf->delete_first_pass_file);
|
|
|
|
printf("auto_key: %d\n", oxcf->auto_key);
|
|
|
|
printf("key_freq: %d\n", oxcf->key_freq);
|
|
|
|
printf("end_usage: %d\n", oxcf->end_usage);
|
|
|
|
printf("under_shoot_pct: %d\n", oxcf->under_shoot_pct);
|
|
|
|
printf("over_shoot_pct: %d\n", oxcf->over_shoot_pct);
|
|
|
|
printf("starting_buffer_level: %d\n", oxcf->starting_buffer_level);
|
|
|
|
printf("optimal_buffer_level: %d\n", oxcf->optimal_buffer_level);
|
|
|
|
printf("maximum_buffer_size: %d\n", oxcf->maximum_buffer_size);
|
|
|
|
printf("fixed_q: %d\n", oxcf->fixed_q);
|
|
|
|
printf("worst_allowed_q: %d\n", oxcf->worst_allowed_q);
|
|
|
|
printf("best_allowed_q: %d\n", oxcf->best_allowed_q);
|
|
|
|
printf("two_pass_vbrbias: %d\n", oxcf->two_pass_vbrbias);
|
|
|
|
printf("two_pass_vbrmin_section: %d\n", oxcf->two_pass_vbrmin_section);
|
|
|
|
printf("two_pass_vbrmax_section: %d\n", oxcf->two_pass_vbrmax_section);
|
|
|
|
printf("allow_lag: %d\n", oxcf->allow_lag);
|
|
|
|
printf("lag_in_frames: %d\n", oxcf->lag_in_frames);
|
|
|
|
printf("play_alternate: %d\n", oxcf->play_alternate);
|
|
|
|
printf("Version: %d\n", oxcf->Version);
|
|
|
|
printf("encode_breakout: %d\n", oxcf->encode_breakout);
|
|
|
|
printf("error resilient: %d\n", oxcf->error_resilient_mode);
|
2013-01-25 20:30:28 +01:00
|
|
|
printf("frame parallel detokenization: %d\n",
|
|
|
|
oxcf->frame_parallel_decoding_mode);
|
2012-07-14 00:21:29 +02:00
|
|
|
*/
|
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_set_config(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
const vpx_codec_enc_cfg_t *cfg) {
|
|
|
|
vpx_codec_err_t res;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if ((cfg->g_w != ctx->cfg.g_w) || (cfg->g_h != ctx->cfg.g_h))
|
|
|
|
ERROR("Cannot change width or height after initialization");
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Prevent increasing lag_in_frames. This check is stricter than it needs
|
|
|
|
* to be -- the limit is not increasing past the first lag_in_frames
|
|
|
|
* value, but we don't track the initial config, only the last successful
|
|
|
|
* config.
|
|
|
|
*/
|
|
|
|
if ((cfg->g_lag_in_frames > ctx->cfg.g_lag_in_frames))
|
|
|
|
ERROR("Cannot increase lag_in_frames");
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
res = validate_config(ctx, cfg, &ctx->vp8_cfg);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (!res) {
|
|
|
|
ctx->cfg = *cfg;
|
2013-05-23 03:04:08 +02:00
|
|
|
set_vp9e_config(&ctx->oxcf, ctx->cfg, ctx->vp8_cfg);
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_change_config(ctx->cpi, &ctx->oxcf);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return res;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-30 22:25:33 +01:00
|
|
|
int vp9_reverse_trans(int q);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
|
|
|
|
static vpx_codec_err_t get_param(vpx_codec_alg_priv_t *ctx,
|
|
|
|
int ctrl_id,
|
2012-07-14 00:21:29 +02:00
|
|
|
va_list args) {
|
|
|
|
void *arg = va_arg(args, void *);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
#define MAP(id, var) case id: *(RECAST(id, arg)) = var; break
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (!arg)
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
switch (ctrl_id) {
|
2012-10-30 20:58:42 +01:00
|
|
|
MAP(VP8E_GET_LAST_QUANTIZER, vp9_get_quantizer(ctx->cpi));
|
2012-10-30 22:25:33 +01:00
|
|
|
MAP(VP8E_GET_LAST_QUANTIZER_64,
|
|
|
|
vp9_reverse_trans(vp9_get_quantizer(ctx->cpi)));
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
#undef MAP
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static vpx_codec_err_t set_param(vpx_codec_alg_priv_t *ctx,
|
|
|
|
int ctrl_id,
|
2012-07-14 00:21:29 +02:00
|
|
|
va_list args) {
|
|
|
|
vpx_codec_err_t res = VPX_CODEC_OK;
|
2013-05-23 03:04:08 +02:00
|
|
|
struct vp9_extracfg xcfg = ctx->vp8_cfg;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
#define MAP(id, var) case id: var = CAST(id, args); break;
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
switch (ctrl_id) {
|
2013-08-19 23:16:26 +02:00
|
|
|
MAP(VP8E_SET_CPUUSED, xcfg.cpu_used);
|
|
|
|
MAP(VP8E_SET_ENABLEAUTOALTREF, xcfg.enable_auto_alt_ref);
|
|
|
|
MAP(VP8E_SET_NOISE_SENSITIVITY, xcfg.noise_sensitivity);
|
|
|
|
MAP(VP8E_SET_SHARPNESS, xcfg.Sharpness);
|
|
|
|
MAP(VP8E_SET_STATIC_THRESHOLD, xcfg.static_thresh);
|
|
|
|
MAP(VP9E_SET_TILE_COLUMNS, xcfg.tile_columns);
|
|
|
|
MAP(VP9E_SET_TILE_ROWS, xcfg.tile_rows);
|
|
|
|
MAP(VP8E_SET_ARNR_MAXFRAMES, xcfg.arnr_max_frames);
|
|
|
|
MAP(VP8E_SET_ARNR_STRENGTH, xcfg.arnr_strength);
|
|
|
|
MAP(VP8E_SET_ARNR_TYPE, xcfg.arnr_type);
|
|
|
|
MAP(VP8E_SET_TUNING, xcfg.tuning);
|
|
|
|
MAP(VP8E_SET_CQ_LEVEL, xcfg.cq_level);
|
|
|
|
MAP(VP8E_SET_MAX_INTRA_BITRATE_PCT, xcfg.rc_max_intra_bitrate_pct);
|
|
|
|
MAP(VP9E_SET_LOSSLESS, xcfg.lossless);
|
2013-03-27 19:07:26 +01:00
|
|
|
MAP(VP9E_SET_FRAME_PARALLEL_DECODING, xcfg.frame_parallel_decoding_mode);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
res = validate_config(ctx, &ctx->cfg, &xcfg);
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
ctx->vp8_cfg = xcfg;
|
2013-05-23 03:04:08 +02:00
|
|
|
set_vp9e_config(&ctx->oxcf, ctx->cfg, ctx->vp8_cfg);
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_change_config(ctx->cpi, &ctx->oxcf);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2010-05-18 17:58:33 +02:00
|
|
|
#undef MAP
|
|
|
|
}
|
2010-06-01 17:14:25 +02:00
|
|
|
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_common_init(vpx_codec_ctx_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int experimental) {
|
2012-11-05 21:37:14 +01:00
|
|
|
vpx_codec_err_t res = VPX_CODEC_OK;
|
2012-07-14 00:21:29 +02:00
|
|
|
struct vpx_codec_alg_priv *priv;
|
|
|
|
vpx_codec_enc_cfg_t *cfg;
|
|
|
|
unsigned int i;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_PTR optr;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (!ctx->priv) {
|
|
|
|
priv = calloc(1, sizeof(struct vpx_codec_alg_priv));
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (!priv) {
|
|
|
|
return VPX_CODEC_MEM_ERROR;
|
|
|
|
}
|
2011-02-22 14:02:05 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
ctx->priv = &priv->base;
|
|
|
|
ctx->priv->sz = sizeof(*ctx->priv);
|
|
|
|
ctx->priv->iface = ctx->iface;
|
|
|
|
ctx->priv->alg_priv = priv;
|
|
|
|
ctx->priv->init_flags = ctx->init_flags;
|
2012-11-05 21:37:14 +01:00
|
|
|
ctx->priv->enc.total_encoders = 1;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
if (ctx->config.enc) {
|
|
|
|
/* Update the reference to the config structure to an
|
|
|
|
* internal copy.
|
|
|
|
*/
|
|
|
|
ctx->priv->alg_priv->cfg = *ctx->config.enc;
|
|
|
|
ctx->config.enc = &ctx->priv->alg_priv->cfg;
|
|
|
|
}
|
2011-02-22 14:02:05 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
cfg = &ctx->priv->alg_priv->cfg;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Select the extra vp6 configuration table based on the current
|
|
|
|
* usage value. If the current usage value isn't found, use the
|
|
|
|
* values for usage case 0.
|
|
|
|
*/
|
|
|
|
for (i = 0;
|
|
|
|
extracfg_map[i].usage && extracfg_map[i].usage != cfg->g_usage;
|
|
|
|
i++);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
priv->vp8_cfg = extracfg_map[i].cfg;
|
|
|
|
priv->vp8_cfg.pkt_list = &priv->pkt_list.head;
|
|
|
|
priv->vp8_cfg.experimental = experimental;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-04-03 00:08:50 +02:00
|
|
|
// TODO(agrange) Check the limits set on this buffer, or the check that is
|
2013-05-23 03:04:08 +02:00
|
|
|
// applied in vp9e_encode.
|
2013-04-03 00:08:50 +02:00
|
|
|
priv->cx_data_sz = priv->cfg.g_w * priv->cfg.g_h * 3 / 2 * 8;
|
|
|
|
// priv->cx_data_sz = priv->cfg.g_w * priv->cfg.g_h * 3 / 2 * 2;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (priv->cx_data_sz < 4096) priv->cx_data_sz = 4096;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
priv->cx_data = malloc(priv->cx_data_sz);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (!priv->cx_data) {
|
|
|
|
return VPX_CODEC_MEM_ERROR;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-30 22:25:33 +01:00
|
|
|
vp9_initialize_enc();
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
res = validate_config(priv, &priv->cfg, &priv->vp8_cfg);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (!res) {
|
2013-05-23 03:04:08 +02:00
|
|
|
set_vp9e_config(&ctx->priv->alg_priv->oxcf,
|
2012-07-14 00:21:29 +02:00
|
|
|
ctx->priv->alg_priv->cfg,
|
|
|
|
ctx->priv->alg_priv->vp8_cfg);
|
2012-10-30 20:58:42 +01:00
|
|
|
optr = vp9_create_compressor(&ctx->priv->alg_priv->oxcf);
|
2011-02-22 14:02:05 +01:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (!optr)
|
|
|
|
res = VPX_CODEC_MEM_ERROR;
|
|
|
|
else
|
|
|
|
ctx->priv->alg_priv->cpi = optr;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return res;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2010-06-01 17:14:25 +02:00
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_init(vpx_codec_ctx_t *ctx,
|
2012-11-05 21:37:14 +01:00
|
|
|
vpx_codec_priv_enc_mr_cfg_t *data) {
|
2013-05-23 03:04:08 +02:00
|
|
|
return vp9e_common_init(ctx, 0);
|
2010-06-01 17:14:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if CONFIG_EXPERIMENTAL
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_exp_init(vpx_codec_ctx_t *ctx,
|
2012-11-05 21:37:14 +01:00
|
|
|
vpx_codec_priv_enc_mr_cfg_t *data) {
|
2013-05-23 03:04:08 +02:00
|
|
|
return vp9e_common_init(ctx, 1);
|
2010-06-01 17:14:25 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_destroy(vpx_codec_alg_priv_t *ctx) {
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
free(ctx->cx_data);
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_remove_compressor(&ctx->cpi);
|
2012-07-14 00:21:29 +02:00
|
|
|
free(ctx);
|
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void pick_quickcompress_mode(vpx_codec_alg_priv_t *ctx,
|
|
|
|
unsigned long duration,
|
2012-07-14 00:21:29 +02:00
|
|
|
unsigned long deadline) {
|
|
|
|
unsigned int new_qc;
|
|
|
|
|
|
|
|
/* Use best quality mode if no deadline is given. */
|
|
|
|
if (deadline)
|
|
|
|
new_qc = MODE_GOODQUALITY;
|
|
|
|
else
|
|
|
|
new_qc = MODE_BESTQUALITY;
|
|
|
|
|
|
|
|
if (ctx->cfg.g_pass == VPX_RC_FIRST_PASS)
|
|
|
|
new_qc = MODE_FIRSTPASS;
|
|
|
|
else if (ctx->cfg.g_pass == VPX_RC_LAST_PASS)
|
|
|
|
new_qc = (new_qc == MODE_BESTQUALITY)
|
|
|
|
? MODE_SECONDPASS_BEST
|
|
|
|
: MODE_SECONDPASS;
|
|
|
|
|
|
|
|
if (ctx->oxcf.Mode != new_qc) {
|
|
|
|
ctx->oxcf.Mode = new_qc;
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_change_config(ctx->cpi, &ctx->oxcf);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-13 00:33:38 +01:00
|
|
|
static int write_superframe_index(vpx_codec_alg_priv_t *ctx) {
|
2013-03-05 21:23:34 +01:00
|
|
|
uint8_t marker = 0xc0;
|
|
|
|
int mag, mask, index_sz;
|
|
|
|
|
|
|
|
assert(ctx->pending_frame_count);
|
|
|
|
assert(ctx->pending_frame_count <= 8);
|
|
|
|
|
|
|
|
/* Add the number of frames to the marker byte */
|
|
|
|
marker |= ctx->pending_frame_count - 1;
|
|
|
|
|
|
|
|
/* Choose the magnitude */
|
|
|
|
for (mag = 0, mask = 0xff; mag < 4; mag++) {
|
|
|
|
if (ctx->pending_frame_magnitude < mask)
|
|
|
|
break;
|
|
|
|
mask <<= 8;
|
|
|
|
mask |= 0xff;
|
|
|
|
}
|
|
|
|
marker |= mag << 3;
|
|
|
|
|
|
|
|
/* Write the index */
|
|
|
|
index_sz = 2 + (mag + 1) * ctx->pending_frame_count;
|
|
|
|
if (ctx->pending_cx_data_sz + index_sz < ctx->cx_data_sz) {
|
|
|
|
uint8_t *x = ctx->pending_cx_data + ctx->pending_cx_data_sz;
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
*x++ = marker;
|
|
|
|
for (i = 0; i < ctx->pending_frame_count; i++) {
|
|
|
|
int this_sz = ctx->pending_frame_sizes[i];
|
|
|
|
|
|
|
|
for (j = 0; j <= mag; j++) {
|
|
|
|
*x++ = this_sz & 0xff;
|
|
|
|
this_sz >>= 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*x++ = marker;
|
|
|
|
ctx->pending_cx_data_sz += index_sz;
|
|
|
|
}
|
2013-03-13 00:33:38 +01:00
|
|
|
return index_sz;
|
2013-03-05 21:23:34 +01:00
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_encode(vpx_codec_alg_priv_t *ctx,
|
2010-05-18 17:58:33 +02:00
|
|
|
const vpx_image_t *img,
|
|
|
|
vpx_codec_pts_t pts,
|
|
|
|
unsigned long duration,
|
|
|
|
vpx_enc_frame_flags_t flags,
|
2012-07-14 00:21:29 +02:00
|
|
|
unsigned long deadline) {
|
|
|
|
vpx_codec_err_t res = VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (img)
|
|
|
|
res = validate_img(ctx, img);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
pick_quickcompress_mode(ctx, duration, deadline);
|
|
|
|
vpx_codec_pkt_list_init(&ctx->pkt_list);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Handle Flags */
|
|
|
|
if (((flags & VP8_EFLAG_NO_UPD_GF) && (flags & VP8_EFLAG_FORCE_GF))
|
|
|
|
|| ((flags & VP8_EFLAG_NO_UPD_ARF) && (flags & VP8_EFLAG_FORCE_ARF))) {
|
|
|
|
ctx->base.err_detail = "Conflicting flags.";
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & (VP8_EFLAG_NO_REF_LAST | VP8_EFLAG_NO_REF_GF
|
|
|
|
| VP8_EFLAG_NO_REF_ARF)) {
|
|
|
|
int ref = 7;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & VP8_EFLAG_NO_REF_LAST)
|
2012-10-31 22:40:53 +01:00
|
|
|
ref ^= VP9_LAST_FLAG;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & VP8_EFLAG_NO_REF_GF)
|
2012-10-31 22:40:53 +01:00
|
|
|
ref ^= VP9_GOLD_FLAG;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & VP8_EFLAG_NO_REF_ARF)
|
2012-10-31 22:40:53 +01:00
|
|
|
ref ^= VP9_ALT_FLAG;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_use_as_reference(ctx->cpi, ref);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & (VP8_EFLAG_NO_UPD_LAST | VP8_EFLAG_NO_UPD_GF
|
|
|
|
| VP8_EFLAG_NO_UPD_ARF | VP8_EFLAG_FORCE_GF
|
|
|
|
| VP8_EFLAG_FORCE_ARF)) {
|
|
|
|
int upd = 7;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & VP8_EFLAG_NO_UPD_LAST)
|
2012-10-31 22:40:53 +01:00
|
|
|
upd ^= VP9_LAST_FLAG;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & VP8_EFLAG_NO_UPD_GF)
|
2012-10-31 22:40:53 +01:00
|
|
|
upd ^= VP9_GOLD_FLAG;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & VP8_EFLAG_NO_UPD_ARF)
|
2012-10-31 22:40:53 +01:00
|
|
|
upd ^= VP9_ALT_FLAG;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_update_reference(ctx->cpi, upd);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (flags & VP8_EFLAG_NO_UPD_ENTROPY) {
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_update_entropy(ctx->cpi, 0);
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Handle fixed keyframe intervals */
|
|
|
|
if (ctx->cfg.kf_mode == VPX_KF_AUTO
|
|
|
|
&& ctx->cfg.kf_min_dist == ctx->cfg.kf_max_dist) {
|
|
|
|
if (++ctx->fixed_kf_cntr > ctx->cfg.kf_min_dist) {
|
|
|
|
flags |= VPX_EFLAG_FORCE_KF;
|
|
|
|
ctx->fixed_kf_cntr = 1;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Initialize the encoder instance on the first frame*/
|
|
|
|
if (!res && ctx->cpi) {
|
|
|
|
unsigned int lib_flags;
|
|
|
|
YV12_BUFFER_CONFIG sd;
|
|
|
|
int64_t dst_time_stamp, dst_end_time_stamp;
|
|
|
|
unsigned long size, cx_data_sz;
|
|
|
|
unsigned char *cx_data;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Set up internal flags */
|
|
|
|
if (ctx->base.init_flags & VPX_CODEC_USE_PSNR)
|
2012-10-31 01:53:32 +01:00
|
|
|
((VP9_COMP *)ctx->cpi)->b_calculate_psnr = 1;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
// if (ctx->base.init_flags & VPX_CODEC_USE_OUTPUT_PARTITION)
|
2012-10-31 01:53:32 +01:00
|
|
|
// ((VP9_COMP *)ctx->cpi)->output_partition = 1;
|
2011-06-13 16:42:27 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Convert API flags to internal codec lib flags */
|
|
|
|
lib_flags = (flags & VPX_EFLAG_FORCE_KF) ? FRAMEFLAGS_KEY : 0;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* vp8 use 10,000,000 ticks/second as time stamp */
|
|
|
|
dst_time_stamp = pts * 10000000 * ctx->cfg.g_timebase.num / ctx->cfg.g_timebase.den;
|
|
|
|
dst_end_time_stamp = (pts + duration) * 10000000 * ctx->cfg.g_timebase.num / ctx->cfg.g_timebase.den;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (img != NULL) {
|
|
|
|
res = image2yuvconfig(img, &sd);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-02-07 00:54:52 +01:00
|
|
|
if (vp9_receive_raw_frame(ctx->cpi, lib_flags,
|
2012-07-14 00:21:29 +02:00
|
|
|
&sd, dst_time_stamp, dst_end_time_stamp)) {
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *)ctx->cpi;
|
2012-07-14 00:21:29 +02:00
|
|
|
res = update_error_state(ctx, &cpi->common.error);
|
|
|
|
}
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-11-15 21:19:07 +01:00
|
|
|
cx_data = ctx->cx_data;
|
|
|
|
cx_data_sz = ctx->cx_data_sz;
|
2012-07-14 00:21:29 +02:00
|
|
|
lib_flags = 0;
|
|
|
|
|
2012-11-15 21:19:07 +01:00
|
|
|
/* Any pending invisible frames? */
|
|
|
|
if (ctx->pending_cx_data) {
|
|
|
|
memmove(cx_data, ctx->pending_cx_data, ctx->pending_cx_data_sz);
|
|
|
|
ctx->pending_cx_data = cx_data;
|
|
|
|
cx_data += ctx->pending_cx_data_sz;
|
|
|
|
cx_data_sz -= ctx->pending_cx_data_sz;
|
|
|
|
|
|
|
|
/* TODO: this is a minimal check, the underlying codec doesn't respect
|
|
|
|
* the buffer size anyway.
|
|
|
|
*/
|
|
|
|
if (cx_data_sz < ctx->cx_data_sz / 2) {
|
|
|
|
ctx->base.err_detail = "Compressed data buffer too small";
|
|
|
|
return VPX_CODEC_ERROR;
|
|
|
|
}
|
2012-10-29 23:57:44 +01:00
|
|
|
}
|
|
|
|
|
2012-10-30 22:25:33 +01:00
|
|
|
while (cx_data_sz >= ctx->cx_data_sz / 2 &&
|
|
|
|
-1 != vp9_get_compressed_data(ctx->cpi, &lib_flags, &size,
|
|
|
|
cx_data, &dst_time_stamp,
|
|
|
|
&dst_end_time_stamp, !img)) {
|
2012-07-14 00:21:29 +02:00
|
|
|
if (size) {
|
|
|
|
vpx_codec_pts_t round, delta;
|
|
|
|
vpx_codec_cx_pkt_t pkt;
|
2012-10-31 01:53:32 +01:00
|
|
|
VP9_COMP *cpi = (VP9_COMP *)ctx->cpi;
|
2012-07-14 00:21:29 +02:00
|
|
|
|
2013-04-03 00:08:50 +02:00
|
|
|
/* Pack invisible frames with the next visible frame */
|
2012-10-29 23:57:44 +01:00
|
|
|
if (!cpi->common.show_frame) {
|
2012-11-15 21:19:07 +01:00
|
|
|
if (!ctx->pending_cx_data)
|
|
|
|
ctx->pending_cx_data = cx_data;
|
|
|
|
ctx->pending_cx_data_sz += size;
|
2013-03-05 21:23:34 +01:00
|
|
|
ctx->pending_frame_sizes[ctx->pending_frame_count++] = size;
|
|
|
|
ctx->pending_frame_magnitude |= size;
|
2012-10-29 23:57:44 +01:00
|
|
|
cx_data += size;
|
|
|
|
cx_data_sz -= size;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* Add the frame packet to the list of returned packets. */
|
|
|
|
round = 1000000 * ctx->cfg.g_timebase.num / 2 - 1;
|
|
|
|
delta = (dst_end_time_stamp - dst_time_stamp);
|
|
|
|
pkt.kind = VPX_CODEC_CX_FRAME_PKT;
|
|
|
|
pkt.data.frame.pts =
|
|
|
|
(dst_time_stamp * ctx->cfg.g_timebase.den + round)
|
|
|
|
/ ctx->cfg.g_timebase.num / 10000000;
|
2012-11-05 23:22:59 +01:00
|
|
|
pkt.data.frame.duration = (unsigned long)
|
|
|
|
((delta * ctx->cfg.g_timebase.den + round)
|
|
|
|
/ ctx->cfg.g_timebase.num / 10000000);
|
2012-07-14 00:21:29 +02:00
|
|
|
pkt.data.frame.flags = lib_flags << 16;
|
|
|
|
|
|
|
|
if (lib_flags & FRAMEFLAGS_KEY)
|
|
|
|
pkt.data.frame.flags |= VPX_FRAME_IS_KEY;
|
|
|
|
|
|
|
|
if (!cpi->common.show_frame) {
|
|
|
|
pkt.data.frame.flags |= VPX_FRAME_IS_INVISIBLE;
|
|
|
|
|
|
|
|
// This timestamp should be as close as possible to the
|
|
|
|
// prior PTS so that if a decoder uses pts to schedule when
|
|
|
|
// to do this, we start right after last frame was decoded.
|
|
|
|
// Invisible frames have no duration.
|
|
|
|
pkt.data.frame.pts = ((cpi->last_time_stamp_seen
|
|
|
|
* ctx->cfg.g_timebase.den + round)
|
|
|
|
/ ctx->cfg.g_timebase.num / 10000000) + 1;
|
|
|
|
pkt.data.frame.duration = 0;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (cpi->droppable)
|
|
|
|
pkt.data.frame.flags |= VPX_FRAME_IS_DROPPABLE;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/*if (cpi->output_partition)
|
2010-05-18 17:58:33 +02:00
|
|
|
{
|
2012-07-14 00:21:29 +02:00
|
|
|
int i;
|
|
|
|
const int num_partitions = 1;
|
|
|
|
|
|
|
|
pkt.data.frame.flags |= VPX_FRAME_IS_FRAGMENT;
|
|
|
|
|
|
|
|
for (i = 0; i < num_partitions; ++i)
|
2010-05-18 17:58:33 +02:00
|
|
|
{
|
2012-07-14 00:21:29 +02:00
|
|
|
pkt.data.frame.buf = cx_data;
|
|
|
|
pkt.data.frame.sz = cpi->partition_sz[i];
|
|
|
|
pkt.data.frame.partition_id = i;
|
|
|
|
// don't set the fragment bit for the last partition
|
|
|
|
if (i == (num_partitions - 1))
|
|
|
|
pkt.data.frame.flags &= ~VPX_FRAME_IS_FRAGMENT;
|
|
|
|
vpx_codec_pkt_list_add(&ctx->pkt_list.head, &pkt);
|
|
|
|
cx_data += cpi->partition_sz[i];
|
|
|
|
cx_data_sz -= cpi->partition_sz[i];
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
else*/
|
|
|
|
{
|
2012-11-15 21:19:07 +01:00
|
|
|
if (ctx->pending_cx_data) {
|
2013-03-05 21:23:34 +01:00
|
|
|
ctx->pending_frame_sizes[ctx->pending_frame_count++] = size;
|
|
|
|
ctx->pending_frame_magnitude |= size;
|
|
|
|
ctx->pending_cx_data_sz += size;
|
2013-03-13 00:33:38 +01:00
|
|
|
size += write_superframe_index(ctx);
|
2012-11-15 21:19:07 +01:00
|
|
|
pkt.data.frame.buf = ctx->pending_cx_data;
|
2013-03-05 21:23:34 +01:00
|
|
|
pkt.data.frame.sz = ctx->pending_cx_data_sz;
|
2012-11-15 21:19:07 +01:00
|
|
|
ctx->pending_cx_data = NULL;
|
|
|
|
ctx->pending_cx_data_sz = 0;
|
2013-03-05 21:23:34 +01:00
|
|
|
ctx->pending_frame_count = 0;
|
|
|
|
ctx->pending_frame_magnitude = 0;
|
2012-10-29 23:57:44 +01:00
|
|
|
} else {
|
|
|
|
pkt.data.frame.buf = cx_data;
|
2012-11-15 21:19:07 +01:00
|
|
|
pkt.data.frame.sz = size;
|
2012-10-29 23:57:44 +01:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
pkt.data.frame.partition_id = -1;
|
|
|
|
vpx_codec_pkt_list_add(&ctx->pkt_list.head, &pkt);
|
|
|
|
cx_data += size;
|
|
|
|
cx_data_sz -= size;
|
|
|
|
}
|
|
|
|
|
|
|
|
// printf("timestamp: %lld, duration: %d\n", pkt->data.frame.pts, pkt->data.frame.duration);
|
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
return res;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static const vpx_codec_cx_pkt_t *vp9e_get_cxdata(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
vpx_codec_iter_t *iter) {
|
|
|
|
return vpx_codec_pkt_list_get(&ctx->pkt_list.head, iter);
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_set_reference(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
|
|
|
vpx_ref_frame_t *data = va_arg(args, vpx_ref_frame_t *);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (data) {
|
|
|
|
vpx_ref_frame_t *frame = (vpx_ref_frame_t *)data;
|
|
|
|
YV12_BUFFER_CONFIG sd;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
image2yuvconfig(&frame->img, &sd);
|
2012-10-30 22:25:33 +01:00
|
|
|
vp9_set_reference_enc(ctx->cpi, frame->frame_type, &sd);
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
|
|
|
} else
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_copy_reference(vpx_codec_alg_priv_t *ctx,
|
2013-03-13 20:15:43 +01:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
vpx_ref_frame_t *data = va_arg(args, vpx_ref_frame_t *);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (data) {
|
|
|
|
vpx_ref_frame_t *frame = (vpx_ref_frame_t *)data;
|
|
|
|
YV12_BUFFER_CONFIG sd;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
image2yuvconfig(&frame->img, &sd);
|
2013-03-13 20:15:43 +01:00
|
|
|
vp9_copy_reference_enc(ctx->cpi, frame->frame_type, &sd);
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
|
|
|
} else
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-03-13 20:15:43 +01:00
|
|
|
static vpx_codec_err_t get_reference(vpx_codec_alg_priv_t *ctx,
|
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
|
|
|
vp9_ref_frame_t *data = va_arg(args, vp9_ref_frame_t *);
|
|
|
|
|
|
|
|
if (data) {
|
|
|
|
YV12_BUFFER_CONFIG* fb;
|
|
|
|
|
|
|
|
vp9_get_reference_enc(ctx->cpi, data->idx, &fb);
|
|
|
|
yuvconfig2image(&data->img, fb, NULL);
|
|
|
|
return VPX_CODEC_OK;
|
|
|
|
} else {
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_set_previewpp(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
2013-09-04 19:02:08 +02:00
|
|
|
#if CONFIG_VP9_POSTPROC
|
2012-07-14 00:21:29 +02:00
|
|
|
vp8_postproc_cfg_t *data = va_arg(args, vp8_postproc_cfg_t *);
|
|
|
|
(void)ctr_id;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (data) {
|
|
|
|
ctx->preview_ppcfg = *((vp8_postproc_cfg_t *)data);
|
|
|
|
return VPX_CODEC_OK;
|
|
|
|
} else
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
2010-08-20 22:06:56 +02:00
|
|
|
#else
|
2012-07-14 00:21:29 +02:00
|
|
|
(void)ctx;
|
|
|
|
(void)ctr_id;
|
|
|
|
(void)args;
|
|
|
|
return VPX_CODEC_INCAPABLE;
|
2010-08-20 22:06:56 +02:00
|
|
|
#endif
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_image_t *vp9e_get_preview(vpx_codec_alg_priv_t *ctx) {
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
YV12_BUFFER_CONFIG sd;
|
2012-10-31 22:40:53 +01:00
|
|
|
vp9_ppflags_t flags = {0};
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (ctx->preview_ppcfg.post_proc_flag) {
|
|
|
|
flags.post_proc_flag = ctx->preview_ppcfg.post_proc_flag;
|
|
|
|
flags.deblocking_level = ctx->preview_ppcfg.deblocking_level;
|
|
|
|
flags.noise_level = ctx->preview_ppcfg.noise_level;
|
|
|
|
}
|
2010-11-05 00:03:36 +01:00
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
if (0 == vp9_get_preview_raw_frame(ctx->cpi, &sd, &flags)) {
|
2013-05-07 00:52:06 +02:00
|
|
|
yuvconfig2image(&ctx->preview_img, &sd, NULL);
|
2012-07-14 00:21:29 +02:00
|
|
|
return &ctx->preview_img;
|
|
|
|
} else
|
|
|
|
return NULL;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_update_entropy(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
|
|
|
int update = va_arg(args, int);
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_update_entropy(ctx->cpi, update);
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_update_reference(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
|
|
|
int update = va_arg(args, int);
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_update_reference(ctx->cpi, update);
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_use_reference(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
|
|
|
int reference_flag = va_arg(args, int);
|
2012-10-30 20:58:42 +01:00
|
|
|
vp9_use_as_reference(ctx->cpi, reference_flag);
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_set_roi_map(vpx_codec_alg_priv_t *ctx,
|
2010-05-18 17:58:33 +02:00
|
|
|
int ctr_id,
|
2012-07-14 00:21:29 +02:00
|
|
|
va_list args) {
|
|
|
|
vpx_roi_map_t *data = va_arg(args, vpx_roi_map_t *);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (data) {
|
|
|
|
vpx_roi_map_t *roi = (vpx_roi_map_t *)data;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-30 22:25:33 +01:00
|
|
|
if (!vp9_set_roimap(ctx->cpi, roi->roi_map, roi->rows, roi->cols,
|
|
|
|
roi->delta_q, roi->delta_lf, roi->static_threshold))
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
else
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
|
|
|
} else
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_set_activemap(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
|
|
|
vpx_active_map_t *data = va_arg(args, vpx_active_map_t *);
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
if (data) {
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
vpx_active_map_t *map = (vpx_active_map_t *)data;
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-10-30 20:58:42 +01:00
|
|
|
if (!vp9_set_active_map(ctx->cpi, map->active_map, map->rows, map->cols))
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_OK;
|
2010-05-18 17:58:33 +02:00
|
|
|
else
|
2012-07-14 00:21:29 +02:00
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
|
|
|
} else
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_err_t vp9e_set_scalemode(vpx_codec_alg_priv_t *ctx,
|
2012-07-14 00:21:29 +02:00
|
|
|
int ctr_id,
|
|
|
|
va_list args) {
|
|
|
|
|
|
|
|
vpx_scaling_mode_t *data = va_arg(args, vpx_scaling_mode_t *);
|
|
|
|
|
|
|
|
if (data) {
|
|
|
|
int res;
|
|
|
|
vpx_scaling_mode_t scalemode = *(vpx_scaling_mode_t *)data;
|
2012-10-30 22:25:33 +01:00
|
|
|
res = vp9_set_internal_size(ctx->cpi, scalemode.h_scaling_mode,
|
|
|
|
scalemode.v_scaling_mode);
|
2012-07-14 00:21:29 +02:00
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
return VPX_CODEC_OK;
|
|
|
|
} else
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
|
|
|
} else
|
|
|
|
return VPX_CODEC_INVALID_PARAM;
|
2010-05-18 17:58:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_ctrl_fn_map_t vp9e_ctf_maps[] = {
|
|
|
|
{VP8_SET_REFERENCE, vp9e_set_reference},
|
|
|
|
{VP8_COPY_REFERENCE, vp9e_copy_reference},
|
|
|
|
{VP8_SET_POSTPROC, vp9e_set_previewpp},
|
|
|
|
{VP8E_UPD_ENTROPY, vp9e_update_entropy},
|
|
|
|
{VP8E_UPD_REFERENCE, vp9e_update_reference},
|
|
|
|
{VP8E_USE_REFERENCE, vp9e_use_reference},
|
|
|
|
{VP8E_SET_ROI_MAP, vp9e_set_roi_map},
|
|
|
|
{VP8E_SET_ACTIVEMAP, vp9e_set_activemap},
|
|
|
|
{VP8E_SET_SCALEMODE, vp9e_set_scalemode},
|
2012-07-14 00:21:29 +02:00
|
|
|
{VP8E_SET_CPUUSED, set_param},
|
|
|
|
{VP8E_SET_NOISE_SENSITIVITY, set_param},
|
|
|
|
{VP8E_SET_ENABLEAUTOALTREF, set_param},
|
|
|
|
{VP8E_SET_SHARPNESS, set_param},
|
|
|
|
{VP8E_SET_STATIC_THRESHOLD, set_param},
|
[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 18:35:28 +01:00
|
|
|
{VP9E_SET_TILE_COLUMNS, set_param},
|
2013-02-08 20:33:11 +01:00
|
|
|
{VP9E_SET_TILE_ROWS, set_param},
|
2012-07-14 00:21:29 +02:00
|
|
|
{VP8E_GET_LAST_QUANTIZER, get_param},
|
|
|
|
{VP8E_GET_LAST_QUANTIZER_64, get_param},
|
|
|
|
{VP8E_SET_ARNR_MAXFRAMES, set_param},
|
2013-08-19 23:16:26 +02:00
|
|
|
{VP8E_SET_ARNR_STRENGTH, set_param},
|
|
|
|
{VP8E_SET_ARNR_TYPE, set_param},
|
2012-07-14 00:21:29 +02:00
|
|
|
{VP8E_SET_TUNING, set_param},
|
|
|
|
{VP8E_SET_CQ_LEVEL, set_param},
|
|
|
|
{VP8E_SET_MAX_INTRA_BITRATE_PCT, set_param},
|
2012-11-05 21:37:14 +01:00
|
|
|
{VP9E_SET_LOSSLESS, set_param},
|
2013-07-03 16:25:29 +02:00
|
|
|
{VP9E_SET_FRAME_PARALLEL_DECODING, set_param},
|
2013-03-13 20:15:43 +01:00
|
|
|
{VP9_GET_REFERENCE, get_reference},
|
2012-07-14 00:21:29 +02:00
|
|
|
{ -1, NULL},
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
2013-05-23 03:04:08 +02:00
|
|
|
static vpx_codec_enc_cfg_map_t vp9e_usage_cfg_map[] = {
|
2012-07-14 00:21:29 +02:00
|
|
|
{
|
2010-05-18 17:58:33 +02:00
|
|
|
0,
|
|
|
|
{
|
2012-07-14 00:21:29 +02:00
|
|
|
0, /* g_usage */
|
|
|
|
0, /* g_threads */
|
|
|
|
0, /* g_profile */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
320, /* g_width */
|
|
|
|
240, /* g_height */
|
|
|
|
{1, 30}, /* g_timebase */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
0, /* g_error_resilient */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
VPX_RC_ONE_PASS, /* g_pass */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2013-08-19 23:16:26 +02:00
|
|
|
25, /* g_lag_in_frames */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
0, /* rc_dropframe_thresh */
|
|
|
|
0, /* rc_resize_allowed */
|
|
|
|
60, /* rc_resize_down_thresold */
|
|
|
|
30, /* rc_resize_up_thresold */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
VPX_VBR, /* rc_end_usage */
|
2010-05-18 17:58:33 +02:00
|
|
|
#if VPX_ENCODER_ABI_VERSION > (1 + VPX_CODEC_ABI_VERSION)
|
2012-07-14 00:21:29 +02:00
|
|
|
{0}, /* rc_twopass_stats_in */
|
2010-05-18 17:58:33 +02:00
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
256, /* rc_target_bandwidth */
|
2013-08-19 23:16:26 +02:00
|
|
|
0, /* rc_min_quantizer */
|
2012-07-14 00:21:29 +02:00
|
|
|
63, /* rc_max_quantizer */
|
|
|
|
100, /* rc_undershoot_pct */
|
|
|
|
100, /* rc_overshoot_pct */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
6000, /* rc_max_buffer_size */
|
|
|
|
4000, /* rc_buffer_initial_size; */
|
|
|
|
5000, /* rc_buffer_optimal_size; */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
50, /* rc_two_pass_vbrbias */
|
|
|
|
0, /* rc_two_pass_vbrmin_section */
|
2013-08-19 23:16:26 +02:00
|
|
|
2000, /* rc_two_pass_vbrmax_section */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
2012-07-14 00:21:29 +02:00
|
|
|
/* keyframing settings (kf) */
|
|
|
|
VPX_KF_AUTO, /* g_kfmode*/
|
|
|
|
0, /* kf_min_dist */
|
|
|
|
9999, /* kf_max_dist */
|
2010-05-18 17:58:33 +02:00
|
|
|
|
|
|
|
#if VPX_ENCODER_ABI_VERSION == (1 + VPX_CODEC_ABI_VERSION)
|
2012-07-14 00:21:29 +02:00
|
|
|
1, /* g_delete_first_pass_file */
|
|
|
|
"vp8.fpf" /* first pass filename */
|
2010-05-18 17:58:33 +02:00
|
|
|
#endif
|
2012-07-14 00:21:29 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
{ -1, {NOT_IMPLEMENTED}}
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef VERSION_STRING
|
|
|
|
#define VERSION_STRING
|
|
|
|
#endif
|
2012-11-09 02:09:30 +01:00
|
|
|
CODEC_INTERFACE(vpx_codec_vp9_cx) = {
|
|
|
|
"WebM Project VP9 Encoder" VERSION_STRING,
|
2012-07-14 00:21:29 +02:00
|
|
|
VPX_CODEC_INTERNAL_ABI_VERSION,
|
|
|
|
VPX_CODEC_CAP_ENCODER | VPX_CODEC_CAP_PSNR |
|
|
|
|
VPX_CODEC_CAP_OUTPUT_PARTITION,
|
|
|
|
/* vpx_codec_caps_t caps; */
|
2013-05-23 03:04:08 +02:00
|
|
|
vp9e_init, /* vpx_codec_init_fn_t init; */
|
|
|
|
vp9e_destroy, /* vpx_codec_destroy_fn_t destroy; */
|
|
|
|
vp9e_ctf_maps, /* vpx_codec_ctrl_fn_map_t *ctrl_maps; */
|
2012-07-14 00:21:29 +02:00
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_get_mmap_fn_t get_mmap; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_set_mmap_fn_t set_mmap; */
|
|
|
|
{
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_peek_si_fn_t peek_si; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_get_si_fn_t get_si; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_decode_fn_t decode; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_frame_get_fn_t frame_get; */
|
|
|
|
},
|
|
|
|
{
|
2013-05-23 03:04:08 +02:00
|
|
|
vp9e_usage_cfg_map, /* vpx_codec_enc_cfg_map_t peek_si; */
|
|
|
|
vp9e_encode, /* vpx_codec_encode_fn_t encode; */
|
|
|
|
vp9e_get_cxdata, /* vpx_codec_get_cx_data_fn_t frame_get; */
|
|
|
|
vp9e_set_config,
|
2012-07-14 00:21:29 +02:00
|
|
|
NOT_IMPLEMENTED,
|
2013-05-23 03:04:08 +02:00
|
|
|
vp9e_get_preview,
|
2012-07-14 00:21:29 +02:00
|
|
|
} /* encoder functions */
|
2010-05-18 17:58:33 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-06-01 17:14:25 +02:00
|
|
|
#if CONFIG_EXPERIMENTAL
|
2012-07-23 21:32:59 +02:00
|
|
|
|
2012-11-09 02:09:30 +01:00
|
|
|
CODEC_INTERFACE(vpx_codec_vp9x_cx) = {
|
2012-07-14 00:21:29 +02:00
|
|
|
"VP8 Experimental Encoder" VERSION_STRING,
|
|
|
|
VPX_CODEC_INTERNAL_ABI_VERSION,
|
|
|
|
VPX_CODEC_CAP_ENCODER | VPX_CODEC_CAP_PSNR,
|
|
|
|
/* vpx_codec_caps_t caps; */
|
2013-05-23 03:04:08 +02:00
|
|
|
vp9e_exp_init, /* vpx_codec_init_fn_t init; */
|
|
|
|
vp9e_destroy, /* vpx_codec_destroy_fn_t destroy; */
|
|
|
|
vp9e_ctf_maps, /* vpx_codec_ctrl_fn_map_t *ctrl_maps; */
|
2012-07-14 00:21:29 +02:00
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_get_mmap_fn_t get_mmap; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_set_mmap_fn_t set_mmap; */
|
|
|
|
{
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_peek_si_fn_t peek_si; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_get_si_fn_t get_si; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_decode_fn_t decode; */
|
|
|
|
NOT_IMPLEMENTED, /* vpx_codec_frame_get_fn_t frame_get; */
|
|
|
|
},
|
|
|
|
{
|
2013-05-23 03:04:08 +02:00
|
|
|
vp9e_usage_cfg_map, /* vpx_codec_enc_cfg_map_t peek_si; */
|
|
|
|
vp9e_encode, /* vpx_codec_encode_fn_t encode; */
|
|
|
|
vp9e_get_cxdata, /* vpx_codec_get_cx_data_fn_t frame_get; */
|
|
|
|
vp9e_set_config,
|
2012-07-14 00:21:29 +02:00
|
|
|
NOT_IMPLEMENTED,
|
2013-05-23 03:04:08 +02:00
|
|
|
vp9e_get_preview,
|
2012-07-14 00:21:29 +02:00
|
|
|
} /* encoder functions */
|
2010-06-01 17:14:25 +02:00
|
|
|
};
|
|
|
|
#endif
|