avfilter.c 12.8 KB
Newer Older
V
Vitor Sessak 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/*
 * Filter layer
 * copyright (c) 2007 Bobby Bingham
 *
 * This file is part of FFmpeg.
 *
 * FFmpeg is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * FFmpeg is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with FFmpeg; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */

22
#include <stdarg.h>
V
Vitor Sessak 已提交
23 24 25 26 27
#include <stdlib.h>
#include <string.h>
#include <stdio.h>

#include "avfilter.h"
V
Vitor Sessak 已提交
28
#include "allfilters.h"
V
Vitor Sessak 已提交
29

30 31 32 33 34 35
/** list of registered filters */
struct FilterList
{
    AVFilter *filter;
    struct FilterList *next;
} *filters = NULL;
V
Vitor Sessak 已提交
36

V
Vitor Sessak 已提交
37 38 39 40
/** helper macros to get the in/out pad on the dst/src filter */
#define link_dpad(link)     link->dst-> input_pads[link->dstpad]
#define link_spad(link)     link->src->output_pads[link->srcpad]

41
AVFilterPicRef *avfilter_ref_pic(AVFilterPicRef *ref, int pmask)
V
Vitor Sessak 已提交
42 43
{
    AVFilterPicRef *ret = av_malloc(sizeof(AVFilterPicRef));
V
Vitor Sessak 已提交
44
    *ret = *ref;
45
    ret->perms &= pmask;
V
Vitor Sessak 已提交
46 47 48 49 50 51
    ret->pic->refcount ++;
    return ret;
}

void avfilter_unref_pic(AVFilterPicRef *ref)
{
52
    if(-- ref->pic->refcount == 0)
V
Vitor Sessak 已提交
53 54 55 56
        ref->pic->free(ref->pic);
    av_free(ref);
}

57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77
void avfilter_insert_pad(unsigned idx, unsigned *count, size_t padidx_off,
                         AVFilterPad **pads, AVFilterLink ***links,
                         AVFilterPad *newpad)
{
    unsigned i;

    idx = FFMIN(idx, *count);

    *pads  = av_realloc(*pads,  sizeof(AVFilterPad)   * (*count + 1));
    *links = av_realloc(*links, sizeof(AVFilterLink*) * (*count + 1));
    memmove(*pads +idx+1, *pads +idx, sizeof(AVFilterPad)   * (*count-idx));
    memmove(*links+idx+1, *links+idx, sizeof(AVFilterLink*) * (*count-idx));
    memcpy(*pads+idx, newpad, sizeof(AVFilterPad));
    (*links)[idx] = NULL;

    (*count) ++;
    for(i = idx+1; i < *count; i ++)
        if(*links[i])
            (*(unsigned *)((uint8_t *)(*links[i]) + padidx_off)) ++;
}

V
Vitor Sessak 已提交
78 79 80 81 82
int avfilter_link(AVFilterContext *src, unsigned srcpad,
                  AVFilterContext *dst, unsigned dstpad)
{
    AVFilterLink *link;

V
Vitor Sessak 已提交
83 84
    if(src->output_count <= srcpad || dst->input_count <= dstpad ||
       src->outputs[srcpad]        || dst->inputs[dstpad])
V
Vitor Sessak 已提交
85 86 87
        return -1;

    src->outputs[srcpad] =
88
    dst->inputs[dstpad]  = link = av_mallocz(sizeof(AVFilterLink));
V
Vitor Sessak 已提交
89

V
Vitor Sessak 已提交
90 91 92 93
    link->src     = src;
    link->dst     = dst;
    link->srcpad  = srcpad;
    link->dstpad  = dstpad;
94 95 96 97 98
    link->format  = -1;

    return 0;
}

99 100 101 102 103 104 105 106 107 108 109 110
static int common_format(int *fmts0, int *fmts1)
{
    int i, j;

    for(i = 0; fmts0[i] != -1; i ++)
        for(j = 0; fmts1[j] != -1; j ++)
            if(fmts0[i] == fmts1[j])
                return fmts0[i];

    return -1;
}

111 112
int avfilter_config_link(AVFilterLink *link)
{
113
    int *fmts[3] = {NULL,NULL,NULL};
114
    int (*config_link)(AVFilterLink *);
115
    int *(*query_formats)(AVFilterLink *link);
116

117 118 119
    AVFilterContext *scale;
    AVFilterLink *link2 = NULL;

120 121
    if(!link)
        return 0;
V
Vitor Sessak 已提交
122

123
    /* find a format both filters support */
124 125 126
    if(!(query_formats = link_spad(link).query_formats))
        query_formats = avfilter_default_query_output_formats;
    fmts[0] = query_formats(link);
V
Vitor Sessak 已提交
127
    fmts[1] = link_dpad(link).query_formats(link);
128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153
    if((link->format = common_format(fmts[0], fmts[1])) == -1) {
        /* no common format found.  insert scale filter to convert */
        if(!(scale = avfilter_open(&avfilter_vf_scale, NULL)))
            goto format_done;
        if(scale->filter->init(scale, NULL, NULL)) {
            avfilter_destroy(scale);
            goto format_done;
        }

        link->dst->inputs[link->dstpad] = NULL;
        if(avfilter_link(scale, 0, link->dst, link->dstpad)) {
            link->dst->inputs[link->dstpad] = link;
            goto format_done;
        }
        link2 = scale->outputs[0];
        link->dst    = scale;
        link->dstpad = 0;
        scale->inputs[0] = link;

        /* now try again to find working colorspaces.
         * XXX: is it safe to assume that the scale filter always supports the
         * same input and output colorspaces? */
        fmts[2] = scale->input_pads[0].query_formats(link);
        link->format  = common_format(fmts[0], fmts[2]);
        link2->format = common_format(fmts[1], fmts[2]);
    }
154 155 156 157

format_done:
    av_free(fmts[0]);
    av_free(fmts[1]);
158 159 160 161 162 163 164 165 166 167
    av_free(fmts[2]);
    if(link->format == -1 || (link2 && link2->format == -1)) {
        if(link2) {
            link->dst    = link2->dst;
            link->dstpad = link2->dstpad;
            link->dst->inputs[link->dstpad] = link;
            link->format = -1;
            avfilter_destroy(scale);
            av_free(link2);
        }
168
        return -1;
169
    }
170

V
Vitor Sessak 已提交
171
    if(!(config_link = link_spad(link).config_props))
172 173 174
        config_link  = avfilter_default_config_output_link;
    if(config_link(link))
            return -1;
175

V
Vitor Sessak 已提交
176
    if(!(config_link = link_dpad(link).config_props))
177 178 179
        config_link  = avfilter_default_config_input_link;
    if(config_link(link))
            return -1;
180

181 182 183 184 185 186 187 188 189 190 191 192
    if(link2) {
        if(!(config_link = link_spad(link2).config_props))
            config_link  = avfilter_default_config_output_link;
        if(config_link(link2))
                return -1;

        if(!(config_link = link_dpad(link2).config_props))
            config_link  = avfilter_default_config_input_link;
        if(config_link(link2))
                return -1;
    }

V
Vitor Sessak 已提交
193 194 195 196 197 198 199
    return 0;
}

AVFilterPicRef *avfilter_get_video_buffer(AVFilterLink *link, int perms)
{
    AVFilterPicRef *ret = NULL;

V
Vitor Sessak 已提交
200 201
    if(link_dpad(link).get_video_buffer)
        ret = link_dpad(link).get_video_buffer(link, perms);
V
Vitor Sessak 已提交
202 203 204 205 206 207 208

    if(!ret)
        ret = avfilter_default_get_video_buffer(link, perms);

    return ret;
}

209
int avfilter_request_frame(AVFilterLink *link)
V
Vitor Sessak 已提交
210
{
V
Vitor Sessak 已提交
211 212
    if(link_spad(link).request_frame)
        return link_spad(link).request_frame(link);
213
    else if(link->src->inputs[0])
214 215
        return avfilter_request_frame(link->src->inputs[0]);
    else return -1;
V
Vitor Sessak 已提交
216 217 218 219 220 221 222 223
}

/* XXX: should we do the duplicating of the picture ref here, instead of
 * forcing the source filter to do it? */
void avfilter_start_frame(AVFilterLink *link, AVFilterPicRef *picref)
{
    void (*start_frame)(AVFilterLink *, AVFilterPicRef *);

224
    if(!(start_frame = link_dpad(link).start_frame))
V
Vitor Sessak 已提交
225 226
        start_frame = avfilter_default_start_frame;

227
    /* prepare to copy the picture if it has insufficient permissions */
228 229
    if((link_dpad(link).min_perms & picref->perms) != link_dpad(link).min_perms ||
        link_dpad(link).rej_perms & picref->perms) {
230 231
        av_log(link->dst, AV_LOG_INFO,
                "frame copy needed (have perms %x, need %x, reject %x)\n",
232
                picref->perms,
233 234 235 236 237 238 239 240 241
                link_dpad(link).min_perms, link_dpad(link).rej_perms);

        link->cur_pic = avfilter_default_get_video_buffer(link, link_dpad(link).min_perms);
        link->srcpic = picref;
    }
    else
        link->cur_pic = picref;

    start_frame(link, link->cur_pic);
V
Vitor Sessak 已提交
242 243 244 245 246 247
}

void avfilter_end_frame(AVFilterLink *link)
{
    void (*end_frame)(AVFilterLink *);

248 249 250 251 252 253 254
    /* unreference the source picture if we're feeding the destination filter
     * a copied version dues to permission issues */
    if(link->srcpic) {
        avfilter_unref_pic(link->srcpic);
        link->srcpic = NULL;
    }

255
    if(!(end_frame = link_dpad(link).end_frame))
V
Vitor Sessak 已提交
256 257 258 259 260
        end_frame = avfilter_default_end_frame;

    end_frame(link);
}

261
void avfilter_draw_slice(AVFilterLink *link, int y, int h)
V
Vitor Sessak 已提交
262
{
263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294
    uint8_t *src[4], *dst[4];
    int i, j, hsub, vsub;

    /* copy the slice if needed for permission reasons */
    if(link->srcpic) {
        avcodec_get_chroma_sub_sample(link->format, &hsub, &vsub);

        src[0] = link->srcpic-> data[0] + y * link->srcpic-> linesize[0];
        dst[0] = link->cur_pic->data[0] + y * link->cur_pic->linesize[0];
        for(i = 1; i < 4; i ++) {
            if(link->srcpic->data[i]) {
                src[i] = link->srcpic-> data[i] + (y >> vsub) * link->srcpic-> linesize[i];
                dst[i] = link->cur_pic->data[i] + (y >> vsub) * link->cur_pic->linesize[i];
            } else
                src[i] = dst[i] = NULL;
        }
        for(j = 0; j < h; j ++) {
            memcpy(dst[0], src[0], link->cur_pic->linesize[0]);
            src[0] += link->srcpic ->linesize[0];
            dst[0] += link->cur_pic->linesize[0];
        }
        for(i = 1; i < 4; i ++) {
            if(!src[i]) continue;

            for(j = 0; j < h >> vsub; j ++) {
                memcpy(dst[i], src[i], link->cur_pic->linesize[i]);
                src[i] += link->srcpic ->linesize[i];
                dst[i] += link->cur_pic->linesize[i];
            }
        }
    }

V
Vitor Sessak 已提交
295
    if(!link_dpad(link).draw_slice)
V
Vitor Sessak 已提交
296 297
        return;

V
Vitor Sessak 已提交
298
    link_dpad(link).draw_slice(link, y, h);
V
Vitor Sessak 已提交
299 300 301 302
}

AVFilter *avfilter_get_by_name(char *name)
{
303 304 305 306 307
    struct FilterList *filt;

    for(filt = filters; filt; filt = filt->next)
        if(!strcmp(filt->filter->name, name))
            return filt->filter;
V
Vitor Sessak 已提交
308 309 310 311 312 313 314

    return NULL;
}

/* FIXME: insert in order, rather than insert at end + resort */
void avfilter_register(AVFilter *filter)
{
315 316 317 318 319
    struct FilterList *newfilt = av_malloc(sizeof(struct FilterList));

    newfilt->filter = filter;
    newfilt->next   = filters;
    filters         = newfilt;
V
Vitor Sessak 已提交
320 321 322 323
}

void avfilter_init(void)
{
V
Vitor Sessak 已提交
324 325 326 327 328 329 330 331 332 333 334
    avfilter_register(&avfilter_vsrc_dummy);
    avfilter_register(&avfilter_vsrc_ppm);
    avfilter_register(&avfilter_vf_crop);
    avfilter_register(&avfilter_vf_fifo);
    avfilter_register(&avfilter_vf_fps);
    avfilter_register(&avfilter_vf_graph);
    avfilter_register(&avfilter_vf_graphdesc);
    avfilter_register(&avfilter_vf_graphfile);
    avfilter_register(&avfilter_vf_overlay);
    avfilter_register(&avfilter_vf_passthrough);
    avfilter_register(&avfilter_vf_rgb2bgr);
335
    avfilter_register(&avfilter_vf_scale);
V
Vitor Sessak 已提交
336 337 338
    avfilter_register(&avfilter_vf_slicify);
    avfilter_register(&avfilter_vf_split);
    avfilter_register(&avfilter_vf_vflip);
V
Vitor Sessak 已提交
339 340 341 342
}

void avfilter_uninit(void)
{
343 344 345 346 347 348
    struct FilterList *tmp;

    for(; filters; filters = tmp) {
        tmp = filters->next;
        av_free(filters);
    }
V
Vitor Sessak 已提交
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365
}

static int pad_count(const AVFilterPad *pads)
{
    AVFilterPad *p = (AVFilterPad *) pads;
    int count;

    for(count = 0; p->name; count ++) p ++;
    return count;
}

static const char *filter_name(void *p)
{
    AVFilterContext *filter = p;
    return filter->filter->name;
}

366
AVFilterContext *avfilter_open(AVFilter *filter, char *inst_name)
V
Vitor Sessak 已提交
367 368 369 370 371 372
{
    AVFilterContext *ret = av_malloc(sizeof(AVFilterContext));

    ret->av_class = av_mallocz(sizeof(AVClass));
    ret->av_class->item_name = filter_name;
    ret->filter   = filter;
373
    ret->name     = inst_name ? av_strdup(inst_name) : NULL;
V
Vitor Sessak 已提交
374 375
    ret->priv     = av_mallocz(filter->priv_size);

V
Vitor Sessak 已提交
376
    ret->input_count  = pad_count(filter->inputs);
377 378
    ret->input_pads   = av_malloc(sizeof(AVFilterPad) * ret->input_count);
    memcpy(ret->input_pads, filter->inputs, sizeof(AVFilterPad)*ret->input_count);
V
Vitor Sessak 已提交
379
    ret->inputs       = av_mallocz(sizeof(AVFilterLink*) * ret->input_count);
380

V
Vitor Sessak 已提交
381
    ret->output_count = pad_count(filter->outputs);
382 383
    ret->output_pads  = av_malloc(sizeof(AVFilterPad) * ret->output_count);
    memcpy(ret->output_pads, filter->outputs, sizeof(AVFilterPad)*ret->output_count);
V
Vitor Sessak 已提交
384 385
    ret->outputs      = av_mallocz(sizeof(AVFilterLink*) * ret->output_count);

V
Vitor Sessak 已提交
386 387 388 389 390 391 392 393 394 395
    return ret;
}

void avfilter_destroy(AVFilterContext *filter)
{
    int i;

    if(filter->filter->uninit)
        filter->filter->uninit(filter);

396
    for(i = 0; i < filter->input_count; i ++) {
V
Vitor Sessak 已提交
397 398
        if(filter->inputs[i])
            filter->inputs[i]->src->outputs[filter->inputs[i]->srcpad] = NULL;
V
Vitor Sessak 已提交
399
        av_freep(&filter->inputs[i]);
V
Vitor Sessak 已提交
400
    }
401
    for(i = 0; i < filter->output_count; i ++) {
V
Vitor Sessak 已提交
402 403
        if(filter->outputs[i])
            filter->outputs[i]->dst->inputs[filter->outputs[i]->dstpad] = NULL;
V
Vitor Sessak 已提交
404
        av_freep(&filter->outputs[i]);
V
Vitor Sessak 已提交
405 406
    }

V
Vitor Sessak 已提交
407 408 409 410 411 412 413
    av_freep(&filter->name);
    av_freep(&filter->input_pads);
    av_freep(&filter->output_pads);
    av_freep(&filter->inputs);
    av_freep(&filter->outputs);
    av_freep(&filter->priv);
    av_freep(&filter->av_class);
V
Vitor Sessak 已提交
414 415 416
    av_free(filter);
}

417
int avfilter_init_filter(AVFilterContext *filter, const char *args, void *opaque)
V
Vitor Sessak 已提交
418
{
V
Vitor Sessak 已提交
419
    int ret;
V
Vitor Sessak 已提交
420 421

    if(filter->filter->init)
422
        if((ret = filter->filter->init(filter, args, opaque))) return ret;
V
Vitor Sessak 已提交
423 424 425
    return 0;
}

426 427 428 429 430 431 432 433 434 435 436 437 438 439 440
int *avfilter_make_format_list(int len, ...)
{
    int *ret, i;
    va_list vl;

    ret = av_malloc(sizeof(int) * (len + 1));
    va_start(vl, len);
    for(i = 0; i < len; i ++)
        ret[i] = va_arg(vl, int);
    va_end(vl);
    ret[len] = -1;

    return ret;
}