diff options
author | Clément Bœsch <clement@stupeflix.com> | 2015-08-31 15:18:34 +0200 |
---|---|---|
committer | Clément Bœsch <u@pkh.me> | 2015-12-05 12:24:15 +0100 |
commit | 560d1e7b4992657fad2461420663861cc5524603 (patch) | |
tree | f9b06e7993e8fb084a768a0a6d7ced195c8ea435 /libavfilter | |
parent | 3b0f63e110d47fbd856297bd2e322f5e3b6881ee (diff) | |
download | ffmpeg-560d1e7b4992657fad2461420663861cc5524603.tar.gz |
avfilter/codecview: add QP support
Diffstat (limited to 'libavfilter')
-rw-r--r-- | libavfilter/vf_codecview.c | 43 |
1 files changed, 42 insertions, 1 deletions
diff --git a/libavfilter/vf_codecview.c b/libavfilter/vf_codecview.c index 7a59b104d8..ecf000c38e 100644 --- a/libavfilter/vf_codecview.c +++ b/libavfilter/vf_codecview.c @@ -27,7 +27,6 @@ * libavcodec/mpegvideo.c. * * TODO: segmentation - * TODO: quantization */ #include "libavutil/imgutils.h" @@ -43,6 +42,8 @@ typedef struct { const AVClass *class; unsigned mv; + int hsub, vsub; + int qp; } CodecViewContext; #define OFFSET(x) offsetof(CodecViewContext, x) @@ -52,6 +53,7 @@ static const AVOption codecview_options[] = { {"pf", "forward predicted MVs of P-frames", 0, AV_OPT_TYPE_CONST, {.i64 = MV_P_FOR }, INT_MIN, INT_MAX, FLAGS, "mv"}, {"bf", "forward predicted MVs of B-frames", 0, AV_OPT_TYPE_CONST, {.i64 = MV_B_FOR }, INT_MIN, INT_MAX, FLAGS, "mv"}, {"bb", "backward predicted MVs of B-frames", 0, AV_OPT_TYPE_CONST, {.i64 = MV_B_BACK }, INT_MIN, INT_MAX, FLAGS, "mv"}, + { "qp", NULL, OFFSET(qp), AV_OPT_TYPE_BOOL, {.i64=0}, 0, 1, .flags = FLAGS }, { NULL } }; @@ -198,6 +200,31 @@ static int filter_frame(AVFilterLink *inlink, AVFrame *frame) CodecViewContext *s = ctx->priv; AVFilterLink *outlink = ctx->outputs[0]; + if (s->qp) { + int qstride, qp_type; + int8_t *qp_table = av_frame_get_qp_table(frame, &qstride, &qp_type); + + if (qp_table) { + int x, y; + const int w = FF_CEIL_RSHIFT(frame->width, s->hsub); + const int h = FF_CEIL_RSHIFT(frame->height, s->vsub); + uint8_t *pu = frame->data[1]; + uint8_t *pv = frame->data[2]; + const int lzu = frame->linesize[1]; + const int lzv = frame->linesize[2]; + + for (y = 0; y < h; y++) { + for (x = 0; x < w; x++) { + const int qp = ff_norm_qscale(qp_table[(y >> 3) * qstride + (x >> 3)], qp_type) * 128/31; + pu[x] = pv[x] = qp; + } + pu += lzu; + pv += lzv; + } + } + } + + if (s->mv) { AVFrameSideData *sd = av_frame_get_side_data(frame, AV_FRAME_DATA_MOTION_VECTORS); if (sd) { int i; @@ -213,14 +240,28 @@ static int filter_frame(AVFilterLink *inlink, AVFrame *frame) 100, 0, mv->source > 0); } } + } + return ff_filter_frame(outlink, frame); } +static int config_input(AVFilterLink *inlink) +{ + AVFilterContext *ctx = inlink->dst; + CodecViewContext *s = ctx->priv; + const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(inlink->format); + + s->hsub = desc->log2_chroma_w; + s->vsub = desc->log2_chroma_h; + return 0; +} + static const AVFilterPad codecview_inputs[] = { { .name = "default", .type = AVMEDIA_TYPE_VIDEO, .filter_frame = filter_frame, + .config_props = config_input, .needs_writable = 1, }, { NULL } |