X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=libavfilter%2Favfiltergraph.c;h=cad601be1ef49f98379d7e54197e49f0033b5ab1;hb=8bca34933e7f459dd4cdfd392840e0044a3b6c6e;hp=269d566d5f50121934ec06b089412951e1b35fcf;hpb=4ec42240cad351bbd8ca619dd0f38064eeec29dd;p=ffmpeg diff --git a/libavfilter/avfiltergraph.c b/libavfilter/avfiltergraph.c index 269d566d5f5..cad601be1ef 100644 --- a/libavfilter/avfiltergraph.c +++ b/libavfilter/avfiltergraph.c @@ -1,7 +1,7 @@ /* * filter graphs - * copyright (c) 2008 Vitor Sessak - * copyright (c) 2007 Bobby Bingham + * Copyright (c) 2008 Vitor Sessak + * Copyright (c) 2007 Bobby Bingham * * This file is part of FFmpeg. * @@ -25,34 +25,61 @@ #include "avfilter.h" #include "avfiltergraph.h" +#include "internal.h" -void avfilter_graph_destroy(AVFilterGraph *graph) +AVFilterGraph *avfilter_graph_alloc(void) { - for(; graph->filter_count > 0; graph->filter_count --) - avfilter_destroy(graph->filters[graph->filter_count - 1]); + return av_mallocz(sizeof(AVFilterGraph)); +} + +void avfilter_graph_free(AVFilterGraph *graph) +{ + for (; graph->filter_count > 0; graph->filter_count --) + avfilter_free(graph->filters[graph->filter_count - 1]); av_freep(&graph->scale_sws_opts); av_freep(&graph->filters); } int avfilter_graph_add_filter(AVFilterGraph *graph, AVFilterContext *filter) { - graph->filters = av_realloc(graph->filters, - sizeof(AVFilterContext*) * ++graph->filter_count); + AVFilterContext **filters = av_realloc(graph->filters, + sizeof(AVFilterContext*) * (graph->filter_count+1)); + if (!filters) + return AVERROR(ENOMEM); - if (!graph->filters) - return -1; + graph->filters = filters; + graph->filters[graph->filter_count++] = filter; - graph->filters[graph->filter_count - 1] = filter; + return 0; +} +int avfilter_graph_create_filter(AVFilterContext **filt_ctx, AVFilter *filt, + const char *name, const char *args, void *opaque, + AVFilterGraph *graph_ctx) +{ + int ret; + + if ((ret = avfilter_open(filt_ctx, filt, name)) < 0) + goto fail; + if ((ret = avfilter_init_filter(*filt_ctx, args, opaque)) < 0) + goto fail; + if ((ret = avfilter_graph_add_filter(graph_ctx, *filt_ctx)) < 0) + goto fail; return 0; + +fail: + if (*filt_ctx) + avfilter_free(*filt_ctx); + *filt_ctx = NULL; + return ret; } -int avfilter_graph_check_validity(AVFilterGraph *graph, AVClass *log_ctx) +int ff_avfilter_graph_check_validity(AVFilterGraph *graph, AVClass *log_ctx) { AVFilterContext *filt; int i, j; - for (i=0; i < graph->filter_count; i++) { + for (i = 0; i < graph->filter_count; i++) { filt = graph->filters[i]; for (j = 0; j < filt->input_count; j++) { @@ -77,7 +104,7 @@ int avfilter_graph_check_validity(AVFilterGraph *graph, AVClass *log_ctx) return 0; } -int avfilter_graph_config_links(AVFilterGraph *graph, AVClass *log_ctx) +int ff_avfilter_graph_config_links(AVFilterGraph *graph, AVClass *log_ctx) { AVFilterContext *filt; int i, ret; @@ -98,60 +125,58 @@ AVFilterContext *avfilter_graph_get_filter(AVFilterGraph *graph, char *name) { int i; - for(i = 0; i < graph->filter_count; i ++) - if(graph->filters[i]->name && !strcmp(name, graph->filters[i]->name)) + for (i = 0; i < graph->filter_count; i++) + if (graph->filters[i]->name && !strcmp(name, graph->filters[i]->name)) return graph->filters[i]; return NULL; } -static int query_formats(AVFilterGraph *graph) +static int query_formats(AVFilterGraph *graph, AVClass *log_ctx) { - int i, j; + int i, j, ret; int scaler_count = 0; char inst_name[30]; - /* ask all the sub-filters for their supported colorspaces */ - for(i = 0; i < graph->filter_count; i ++) { - if(graph->filters[i]->filter->query_formats) + /* ask all the sub-filters for their supported media formats */ + for (i = 0; i < graph->filter_count; i++) { + if (graph->filters[i]->filter->query_formats) graph->filters[i]->filter->query_formats(graph->filters[i]); else avfilter_default_query_formats(graph->filters[i]); } /* go through and merge as many format lists as possible */ - for(i = 0; i < graph->filter_count; i ++) { + for (i = 0; i < graph->filter_count; i++) { AVFilterContext *filter = graph->filters[i]; - for(j = 0; j < filter->input_count; j ++) { + for (j = 0; j < filter->input_count; j++) { AVFilterLink *link = filter->inputs[j]; - if(link && link->in_formats != link->out_formats) { - if(!avfilter_merge_formats(link->in_formats, - link->out_formats)) { + if (link && link->in_formats != link->out_formats) { + if (!avfilter_merge_formats(link->in_formats, + link->out_formats)) { AVFilterContext *scale; char scale_args[256]; /* couldn't merge format lists. auto-insert scale filter */ snprintf(inst_name, sizeof(inst_name), "auto-inserted scaler %d", - scaler_count); - scale = - avfilter_open(avfilter_get_by_name("scale"),inst_name); - + scaler_count++); snprintf(scale_args, sizeof(scale_args), "0:0:%s", graph->scale_sws_opts); - if(!scale || scale->filter->init(scale, scale_args, NULL) || - avfilter_insert_filter(link, scale, 0, 0)) { - avfilter_destroy(scale); - return -1; - } - - if (avfilter_graph_add_filter(graph, scale) < 0) - return -1; + if ((ret = avfilter_graph_create_filter(&scale, avfilter_get_by_name("scale"), + inst_name, scale_args, NULL, graph)) < 0) + return ret; + if ((ret = avfilter_insert_filter(link, scale, 0, 0)) < 0) + return ret; scale->filter->query_formats(scale); - if(!avfilter_merge_formats(scale-> inputs[0]->in_formats, - scale-> inputs[0]->out_formats)|| - !avfilter_merge_formats(scale->outputs[0]->in_formats, - scale->outputs[0]->out_formats)) + if (((link = scale-> inputs[0]) && + !avfilter_merge_formats(link->in_formats, link->out_formats)) || + ((link = scale->outputs[0]) && + !avfilter_merge_formats(link->in_formats, link->out_formats))) { + av_log(log_ctx, AV_LOG_ERROR, + "Impossible to convert between the formats supported by the filter " + "'%s' and the filter '%s'\n", link->src->name, link->dst->name); return -1; + } } } } @@ -162,7 +187,7 @@ static int query_formats(AVFilterGraph *graph) static void pick_format(AVFilterLink *link) { - if(!link || !link->in_formats) + if (!link || !link->in_formats) return; link->in_formats->format_count = 1; @@ -176,26 +201,39 @@ static void pick_formats(AVFilterGraph *graph) { int i, j; - for(i = 0; i < graph->filter_count; i ++) { + for (i = 0; i < graph->filter_count; i++) { AVFilterContext *filter = graph->filters[i]; - for(j = 0; j < filter->input_count; j ++) + for (j = 0; j < filter->input_count; j++) pick_format(filter->inputs[j]); - for(j = 0; j < filter->output_count; j ++) + for (j = 0; j < filter->output_count; j++) pick_format(filter->outputs[j]); } } -int avfilter_graph_config_formats(AVFilterGraph *graph) +int ff_avfilter_graph_config_formats(AVFilterGraph *graph, AVClass *log_ctx) { /* find supported formats from sub-filters, and merge along links */ - if(query_formats(graph)) + if (query_formats(graph, log_ctx)) return -1; /* Once everything is merged, it's possible that we'll still have - * multiple valid colorspace choices. We pick the first one. */ + * multiple valid media format choices. We pick the first one. */ pick_formats(graph); return 0; } +int avfilter_graph_config(AVFilterGraph *graphctx, AVClass *log_ctx) +{ + int ret; + + if ((ret = ff_avfilter_graph_check_validity(graphctx, log_ctx))) + return ret; + if ((ret = ff_avfilter_graph_config_formats(graphctx, log_ctx))) + return ret; + if ((ret = ff_avfilter_graph_config_links(graphctx, log_ctx))) + return ret; + + return 0; +}