annotate ffmpeg/libavfilter/vf_geq.c @ 13:844d341cf643 tip

Back up before ISMIR
author Yading Song <yading.song@eecs.qmul.ac.uk>
date Thu, 31 Oct 2013 13:17:06 +0000
parents 6840f77b83aa
children
rev   line source
yading@10 1 /*
yading@10 2 * Copyright (C) 2006 Michael Niedermayer <michaelni@gmx.at>
yading@10 3 * Copyright (C) 2012 Clément Bœsch <ubitux@gmail.com>
yading@10 4 *
yading@10 5 * This file is part of FFmpeg.
yading@10 6 *
yading@10 7 * FFmpeg is free software; you can redistribute it and/or modify
yading@10 8 * it under the terms of the GNU General Public License as published by
yading@10 9 * the Free Software Foundation; either version 2 of the License, or
yading@10 10 * (at your option) any later version.
yading@10 11 *
yading@10 12 * FFmpeg is distributed in the hope that it will be useful,
yading@10 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
yading@10 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
yading@10 15 * GNU General Public License for more details.
yading@10 16 *
yading@10 17 * You should have received a copy of the GNU General Public License along
yading@10 18 * with FFmpeg; if not, write to the Free Software Foundation, Inc.,
yading@10 19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
yading@10 20 */
yading@10 21
yading@10 22 /**
yading@10 23 * @file
yading@10 24 * Generic equation change filter
yading@10 25 * Originally written by Michael Niedermayer for the MPlayer project, and
yading@10 26 * ported by Clément Bœsch for FFmpeg.
yading@10 27 */
yading@10 28
yading@10 29 #include "libavutil/avstring.h"
yading@10 30 #include "libavutil/eval.h"
yading@10 31 #include "libavutil/opt.h"
yading@10 32 #include "libavutil/pixdesc.h"
yading@10 33 #include "internal.h"
yading@10 34
yading@10 35 typedef struct {
yading@10 36 const AVClass *class;
yading@10 37 AVExpr *e[4]; ///< expressions for each plane
yading@10 38 char *expr_str[4]; ///< expression strings for each plane
yading@10 39 int framenum; ///< frame counter
yading@10 40 AVFrame *picref; ///< current input buffer
yading@10 41 int hsub, vsub; ///< chroma subsampling
yading@10 42 int planes; ///< number of planes
yading@10 43 } GEQContext;
yading@10 44
yading@10 45 #define OFFSET(x) offsetof(GEQContext, x)
yading@10 46 #define FLAGS AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_FILTERING_PARAM
yading@10 47
yading@10 48 static const AVOption geq_options[] = {
yading@10 49 { "lum_expr", "set luminance expression", OFFSET(expr_str[0]), AV_OPT_TYPE_STRING, {.str=NULL}, CHAR_MIN, CHAR_MAX, FLAGS },
yading@10 50 { "cb_expr", "set chroma blue expression", OFFSET(expr_str[1]), AV_OPT_TYPE_STRING, {.str=NULL}, CHAR_MIN, CHAR_MAX, FLAGS },
yading@10 51 { "cr_expr", "set chroma red expression", OFFSET(expr_str[2]), AV_OPT_TYPE_STRING, {.str=NULL}, CHAR_MIN, CHAR_MAX, FLAGS },
yading@10 52 { "alpha_expr", "set alpha expression", OFFSET(expr_str[3]), AV_OPT_TYPE_STRING, {.str=NULL}, CHAR_MIN, CHAR_MAX, FLAGS },
yading@10 53 {NULL},
yading@10 54 };
yading@10 55
yading@10 56 AVFILTER_DEFINE_CLASS(geq);
yading@10 57
yading@10 58 static inline double getpix(void *priv, double x, double y, int plane)
yading@10 59 {
yading@10 60 int xi, yi;
yading@10 61 GEQContext *geq = priv;
yading@10 62 AVFrame *picref = geq->picref;
yading@10 63 const uint8_t *src = picref->data[plane];
yading@10 64 const int linesize = picref->linesize[plane];
yading@10 65 const int w = picref->width >> ((plane == 1 || plane == 2) ? geq->hsub : 0);
yading@10 66 const int h = picref->height >> ((plane == 1 || plane == 2) ? geq->vsub : 0);
yading@10 67
yading@10 68 if (!src)
yading@10 69 return 0;
yading@10 70
yading@10 71 xi = x = av_clipf(x, 0, w - 2);
yading@10 72 yi = y = av_clipf(y, 0, h - 2);
yading@10 73
yading@10 74 x -= xi;
yading@10 75 y -= yi;
yading@10 76
yading@10 77 return (1-y)*((1-x)*src[xi + yi * linesize] + x*src[xi + 1 + yi * linesize])
yading@10 78 + y *((1-x)*src[xi + (yi+1) * linesize] + x*src[xi + 1 + (yi+1) * linesize]);
yading@10 79 }
yading@10 80
yading@10 81 //TODO: cubic interpolate
yading@10 82 //TODO: keep the last few frames
yading@10 83 static double lum(void *priv, double x, double y) { return getpix(priv, x, y, 0); }
yading@10 84 static double cb(void *priv, double x, double y) { return getpix(priv, x, y, 1); }
yading@10 85 static double cr(void *priv, double x, double y) { return getpix(priv, x, y, 2); }
yading@10 86 static double alpha(void *priv, double x, double y) { return getpix(priv, x, y, 3); }
yading@10 87
yading@10 88 static const char *const var_names[] = { "X", "Y", "W", "H", "N", "SW", "SH", "T", NULL };
yading@10 89 enum { VAR_X, VAR_Y, VAR_W, VAR_H, VAR_N, VAR_SW, VAR_SH, VAR_T, VAR_VARS_NB };
yading@10 90
yading@10 91 static av_cold int geq_init(AVFilterContext *ctx)
yading@10 92 {
yading@10 93 GEQContext *geq = ctx->priv;
yading@10 94 int plane, ret = 0;
yading@10 95
yading@10 96 if (!geq->expr_str[0]) {
yading@10 97 av_log(ctx, AV_LOG_ERROR, "Luminance expression is mandatory\n");
yading@10 98 ret = AVERROR(EINVAL);
yading@10 99 goto end;
yading@10 100 }
yading@10 101
yading@10 102 if (!geq->expr_str[1] && !geq->expr_str[2]) {
yading@10 103 /* No chroma at all: fallback on luma */
yading@10 104 geq->expr_str[1] = av_strdup(geq->expr_str[0]);
yading@10 105 geq->expr_str[2] = av_strdup(geq->expr_str[0]);
yading@10 106 } else {
yading@10 107 /* One chroma unspecified, fallback on the other */
yading@10 108 if (!geq->expr_str[1]) geq->expr_str[1] = av_strdup(geq->expr_str[2]);
yading@10 109 if (!geq->expr_str[2]) geq->expr_str[2] = av_strdup(geq->expr_str[1]);
yading@10 110 }
yading@10 111
yading@10 112 if (!geq->expr_str[3])
yading@10 113 geq->expr_str[3] = av_strdup("255");
yading@10 114
yading@10 115 if (!geq->expr_str[1] || !geq->expr_str[2] || !geq->expr_str[3]) {
yading@10 116 ret = AVERROR(ENOMEM);
yading@10 117 goto end;
yading@10 118 }
yading@10 119
yading@10 120 for (plane = 0; plane < 4; plane++) {
yading@10 121 static double (*p[])(void *, double, double) = { lum, cb, cr, alpha };
yading@10 122 static const char *const func2_names[] = { "lum", "cb", "cr", "alpha", "p", NULL };
yading@10 123 double (*func2[])(void *, double, double) = { lum, cb, cr, alpha, p[plane], NULL };
yading@10 124
yading@10 125 ret = av_expr_parse(&geq->e[plane], geq->expr_str[plane], var_names,
yading@10 126 NULL, NULL, func2_names, func2, 0, ctx);
yading@10 127 if (ret < 0)
yading@10 128 break;
yading@10 129 }
yading@10 130
yading@10 131 end:
yading@10 132 return ret;
yading@10 133 }
yading@10 134
yading@10 135 static int geq_query_formats(AVFilterContext *ctx)
yading@10 136 {
yading@10 137 static const enum PixelFormat pix_fmts[] = {
yading@10 138 AV_PIX_FMT_YUV444P, AV_PIX_FMT_YUV422P, AV_PIX_FMT_YUV420P,
yading@10 139 AV_PIX_FMT_YUV411P, AV_PIX_FMT_YUV410P, AV_PIX_FMT_YUV440P,
yading@10 140 AV_PIX_FMT_YUVA444P, AV_PIX_FMT_YUVA422P, AV_PIX_FMT_YUVA420P,
yading@10 141 AV_PIX_FMT_GRAY8,
yading@10 142 AV_PIX_FMT_NONE
yading@10 143 };
yading@10 144 ff_set_common_formats(ctx, ff_make_format_list(pix_fmts));
yading@10 145 return 0;
yading@10 146 }
yading@10 147
yading@10 148 static int geq_config_props(AVFilterLink *inlink)
yading@10 149 {
yading@10 150 GEQContext *geq = inlink->dst->priv;
yading@10 151 const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(inlink->format);
yading@10 152
yading@10 153 geq->hsub = desc->log2_chroma_w;
yading@10 154 geq->vsub = desc->log2_chroma_h;
yading@10 155 geq->planes = desc->nb_components;
yading@10 156 return 0;
yading@10 157 }
yading@10 158
yading@10 159 static int geq_filter_frame(AVFilterLink *inlink, AVFrame *in)
yading@10 160 {
yading@10 161 int plane;
yading@10 162 GEQContext *geq = inlink->dst->priv;
yading@10 163 AVFilterLink *outlink = inlink->dst->outputs[0];
yading@10 164 AVFrame *out;
yading@10 165 double values[VAR_VARS_NB] = {
yading@10 166 [VAR_N] = geq->framenum++,
yading@10 167 [VAR_T] = in->pts == AV_NOPTS_VALUE ? NAN : in->pts * av_q2d(inlink->time_base),
yading@10 168 };
yading@10 169
yading@10 170 geq->picref = in;
yading@10 171 out = ff_get_video_buffer(outlink, outlink->w, outlink->h);
yading@10 172 if (!out) {
yading@10 173 av_frame_free(&in);
yading@10 174 return AVERROR(ENOMEM);
yading@10 175 }
yading@10 176 av_frame_copy_props(out, in);
yading@10 177
yading@10 178 for (plane = 0; plane < geq->planes && out->data[plane]; plane++) {
yading@10 179 int x, y;
yading@10 180 uint8_t *dst = out->data[plane];
yading@10 181 const int linesize = out->linesize[plane];
yading@10 182 const int w = inlink->w >> ((plane == 1 || plane == 2) ? geq->hsub : 0);
yading@10 183 const int h = inlink->h >> ((plane == 1 || plane == 2) ? geq->vsub : 0);
yading@10 184
yading@10 185 values[VAR_W] = w;
yading@10 186 values[VAR_H] = h;
yading@10 187 values[VAR_SW] = w / (double)inlink->w;
yading@10 188 values[VAR_SH] = h / (double)inlink->h;
yading@10 189
yading@10 190 for (y = 0; y < h; y++) {
yading@10 191 values[VAR_Y] = y;
yading@10 192 for (x = 0; x < w; x++) {
yading@10 193 values[VAR_X] = x;
yading@10 194 dst[x] = av_expr_eval(geq->e[plane], values, geq);
yading@10 195 }
yading@10 196 dst += linesize;
yading@10 197 }
yading@10 198 }
yading@10 199
yading@10 200 av_frame_free(&geq->picref);
yading@10 201 return ff_filter_frame(outlink, out);
yading@10 202 }
yading@10 203
yading@10 204 static av_cold void geq_uninit(AVFilterContext *ctx)
yading@10 205 {
yading@10 206 int i;
yading@10 207 GEQContext *geq = ctx->priv;
yading@10 208
yading@10 209 for (i = 0; i < FF_ARRAY_ELEMS(geq->e); i++)
yading@10 210 av_expr_free(geq->e[i]);
yading@10 211 }
yading@10 212
yading@10 213 static const AVFilterPad geq_inputs[] = {
yading@10 214 {
yading@10 215 .name = "default",
yading@10 216 .type = AVMEDIA_TYPE_VIDEO,
yading@10 217 .config_props = geq_config_props,
yading@10 218 .filter_frame = geq_filter_frame,
yading@10 219 },
yading@10 220 { NULL }
yading@10 221 };
yading@10 222
yading@10 223 static const AVFilterPad geq_outputs[] = {
yading@10 224 {
yading@10 225 .name = "default",
yading@10 226 .type = AVMEDIA_TYPE_VIDEO,
yading@10 227 },
yading@10 228 { NULL }
yading@10 229 };
yading@10 230
yading@10 231 AVFilter avfilter_vf_geq = {
yading@10 232 .name = "geq",
yading@10 233 .description = NULL_IF_CONFIG_SMALL("Apply generic equation to each pixel."),
yading@10 234 .priv_size = sizeof(GEQContext),
yading@10 235 .init = geq_init,
yading@10 236 .uninit = geq_uninit,
yading@10 237 .query_formats = geq_query_formats,
yading@10 238 .inputs = geq_inputs,
yading@10 239 .outputs = geq_outputs,
yading@10 240 .priv_class = &geq_class,
yading@10 241 };