avfilter.c 11.6 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
int avfilter_insert_filter(AVFilterLink *link, AVFilterContext *filt,
                           unsigned in, unsigned out)
{
102 103
    AVFilterFormats *formats;

104 105 106 107 108 109 110 111 112 113 114 115 116 117 118
    av_log(NULL, AV_LOG_INFO, "auto-inserting filter '%s'\n",
            filt->filter->name);

    link->dst->inputs[link->dstpad] = NULL;
    if(avfilter_link(filt, out, link->dst, link->dstpad)) {
        /* failed to link output filter to new filter */
        link->dst->inputs[link->dstpad] = link;
        return -1;
    }

    /* re-hookup the link to the new destination filter we inserted */
    link->dst = filt;
    link->dstpad = in;
    filt->inputs[in] = link;

119 120
    /* if any information on supported colorspaces already exists on the
     * link, we need to preserve that */
121 122 123
    if((formats = link->out_formats))
        avfilter_formats_changeref(&link->out_formats,
                                   &filt->outputs[out]->out_formats);
124

125 126 127
    return 0;
}

128
int avfilter_config_links(AVFilterContext *filter)
129 130
{
    int (*config_link)(AVFilterLink *);
131 132 133 134
    unsigned i;

    for(i = 0; i < filter->input_count; i ++) {
        AVFilterLink *link;
135

136 137 138 139 140 141 142 143 144 145 146 147 148
        if(!(link = filter->inputs[i])) continue;

        switch(link->init_state) {
        case AVLINK_INIT:
            continue;
        case AVLINK_STARTINIT:
            av_log(filter, AV_LOG_ERROR, "circular filter chain detected\n");
            return -1;
        case AVLINK_UNINIT:
            link->init_state = AVLINK_STARTINIT;

            if(avfilter_config_links(link->src))
                return -1;
V
Vitor Sessak 已提交
149

V
Vitor Sessak 已提交
150 151 152 153
            if(!(config_link = link_spad(link).config_props))
                config_link  = avfilter_default_config_output_link;
            if(config_link(link))
                return -1;
154

V
Vitor Sessak 已提交
155 156 157
            if((config_link = link_dpad(link).config_props))
            if(config_link(link))
                return -1;
158

159 160 161 162
            link->init_state = AVLINK_INIT;
        }
    }

V
Vitor Sessak 已提交
163 164 165 166 167 168 169
    return 0;
}

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

V
Vitor Sessak 已提交
170 171
    if(link_dpad(link).get_video_buffer)
        ret = link_dpad(link).get_video_buffer(link, perms);
V
Vitor Sessak 已提交
172 173 174 175 176 177 178

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

    return ret;
}

179
int avfilter_request_frame(AVFilterLink *link)
V
Vitor Sessak 已提交
180
{
V
Vitor Sessak 已提交
181 182
    if(link_spad(link).request_frame)
        return link_spad(link).request_frame(link);
183
    else if(link->src->inputs[0])
184 185
        return avfilter_request_frame(link->src->inputs[0]);
    else return -1;
V
Vitor Sessak 已提交
186 187 188 189 190 191 192 193
}

/* 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 *);

194
    if(!(start_frame = link_dpad(link).start_frame))
V
Vitor Sessak 已提交
195 196
        start_frame = avfilter_default_start_frame;

197
    /* prepare to copy the picture if it has insufficient permissions */
198 199
    if((link_dpad(link).min_perms & picref->perms) != link_dpad(link).min_perms ||
        link_dpad(link).rej_perms & picref->perms) {
200 201
        av_log(link->dst, AV_LOG_INFO,
                "frame copy needed (have perms %x, need %x, reject %x)\n",
202
                picref->perms,
203 204 205 206 207 208 209 210 211
                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 已提交
212 213 214 215 216 217
}

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

218 219 220 221 222 223 224
    /* 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;
    }

225
    if(!(end_frame = link_dpad(link).end_frame))
V
Vitor Sessak 已提交
226 227 228 229 230
        end_frame = avfilter_default_end_frame;

    end_frame(link);
}

231
void avfilter_draw_slice(AVFilterLink *link, int y, int h)
V
Vitor Sessak 已提交
232
{
233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264
    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 已提交
265
    if(!link_dpad(link).draw_slice)
V
Vitor Sessak 已提交
266 267
        return;

V
Vitor Sessak 已提交
268
    link_dpad(link).draw_slice(link, y, h);
V
Vitor Sessak 已提交
269 270 271 272
}

AVFilter *avfilter_get_by_name(char *name)
{
273 274 275 276 277
    struct FilterList *filt;

    for(filt = filters; filt; filt = filt->next)
        if(!strcmp(filt->filter->name, name))
            return filt->filter;
V
Vitor Sessak 已提交
278 279 280 281 282 283

    return NULL;
}

void avfilter_register(AVFilter *filter)
{
284 285 286 287 288
    struct FilterList *newfilt = av_malloc(sizeof(struct FilterList));

    newfilt->filter = filter;
    newfilt->next   = filters;
    filters         = newfilt;
V
Vitor Sessak 已提交
289 290 291 292
}

void avfilter_init(void)
{
V
Vitor Sessak 已提交
293 294
    avfilter_register(&avfilter_vf_crop);
    avfilter_register(&avfilter_vf_fifo);
295
    avfilter_register(&avfilter_vf_format);
V
Vitor Sessak 已提交
296 297 298 299
    avfilter_register(&avfilter_vf_fps);
    avfilter_register(&avfilter_vf_graph);
    avfilter_register(&avfilter_vf_graphdesc);
    avfilter_register(&avfilter_vf_graphfile);
V
Vitor Sessak 已提交
300
    avfilter_register(&avfilter_vf_hflip);
V
Vitor Sessak 已提交
301
    avfilter_register(&avfilter_vf_negate);
302
    avfilter_register(&avfilter_vf_noformat);
V
Vitor Sessak 已提交
303
    avfilter_register(&avfilter_vf_overlay);
304
    avfilter_register(&avfilter_vf_rotate);
305
    avfilter_register(&avfilter_vf_scale);
306
    avfilter_register(&avfilter_vf_setpts);
V
Vitor Sessak 已提交
307 308
    avfilter_register(&avfilter_vf_slicify);
    avfilter_register(&avfilter_vf_split);
V
Vitor Sessak 已提交
309
    avfilter_register(&avfilter_vf_transpose);
V
Vitor Sessak 已提交
310
    avfilter_register(&avfilter_vf_vflip);
V
Vitor Sessak 已提交
311 312 313 314
}

void avfilter_uninit(void)
{
315 316 317 318 319 320
    struct FilterList *tmp;

    for(; filters; filters = tmp) {
        tmp = filters->next;
        av_free(filters);
    }
V
Vitor Sessak 已提交
321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337
}

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;
}

338
AVFilterContext *avfilter_open(AVFilter *filter, char *inst_name)
V
Vitor Sessak 已提交
339
{
V
Vitor Sessak 已提交
340 341 342 343 344 345
    AVFilterContext *ret;

    if (!filter)
        return 0;

    ret = av_malloc(sizeof(AVFilterContext));
V
Vitor Sessak 已提交
346 347 348 349

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

V
Vitor Sessak 已提交
353
    ret->input_count  = pad_count(filter->inputs);
354 355
    ret->input_pads   = av_malloc(sizeof(AVFilterPad) * ret->input_count);
    memcpy(ret->input_pads, filter->inputs, sizeof(AVFilterPad)*ret->input_count);
V
Vitor Sessak 已提交
356
    ret->inputs       = av_mallocz(sizeof(AVFilterLink*) * ret->input_count);
357

V
Vitor Sessak 已提交
358
    ret->output_count = pad_count(filter->outputs);
359 360
    ret->output_pads  = av_malloc(sizeof(AVFilterPad) * ret->output_count);
    memcpy(ret->output_pads, filter->outputs, sizeof(AVFilterPad)*ret->output_count);
V
Vitor Sessak 已提交
361 362
    ret->outputs      = av_mallocz(sizeof(AVFilterLink*) * ret->output_count);

V
Vitor Sessak 已提交
363 364 365 366 367 368 369 370 371 372
    return ret;
}

void avfilter_destroy(AVFilterContext *filter)
{
    int i;

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

373
    for(i = 0; i < filter->input_count; i ++) {
V
Vitor Sessak 已提交
374 375
        if(filter->inputs[i])
            filter->inputs[i]->src->outputs[filter->inputs[i]->srcpad] = NULL;
V
Vitor Sessak 已提交
376
        av_freep(&filter->inputs[i]);
V
Vitor Sessak 已提交
377
    }
378
    for(i = 0; i < filter->output_count; i ++) {
V
Vitor Sessak 已提交
379 380
        if(filter->outputs[i])
            filter->outputs[i]->dst->inputs[filter->outputs[i]->dstpad] = NULL;
V
Vitor Sessak 已提交
381
        av_freep(&filter->outputs[i]);
V
Vitor Sessak 已提交
382 383
    }

V
Vitor Sessak 已提交
384 385 386 387 388 389 390
    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 已提交
391 392 393
    av_free(filter);
}

394
int avfilter_init_filter(AVFilterContext *filter, const char *args, void *opaque)
V
Vitor Sessak 已提交
395
{
V
Vitor Sessak 已提交
396
    int ret;
V
Vitor Sessak 已提交
397 398

    if(filter->filter->init)
399
        if((ret = filter->filter->init(filter, args, opaque))) return ret;
V
Vitor Sessak 已提交
400 401 402
    return 0;
}