summaryrefslogtreecommitdiff
path: root/libavfilter
diff options
context:
space:
mode:
authorClément Bœsch <clement@stupeflix.com>2015-08-31 15:18:34 +0200
committerClément Bœsch <u@pkh.me>2015-12-05 12:24:15 +0100
commit560d1e7b4992657fad2461420663861cc5524603 (patch)
treef9b06e7993e8fb084a768a0a6d7ced195c8ea435 /libavfilter
parent3b0f63e110d47fbd856297bd2e322f5e3b6881ee (diff)
avfilter/codecview: add QP support
Diffstat (limited to 'libavfilter')
-rw-r--r--libavfilter/vf_codecview.c43
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 }