libavfilter/split.c
a38a00ed
 /*
7d5297b3
  * Copyright (c) 2007 Bobby Bingham
a38a00ed
  *
  * 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
  */
 
7d5297b3
 /**
  * @file
0e82d317
  * audio and video splitter
7d5297b3
  */
 
1d9c2dc8
 #include <stdio.h>
 
093804a9
 #include "libavutil/attributes.h"
1d9c2dc8
 #include "libavutil/internal.h"
 #include "libavutil/mem.h"
73d5d405
 #include "libavutil/opt.h"
 
02aa0701
 #define FF_INTERNAL_FIELDS 1
 #include "framequeue.h"
 
a38a00ed
 #include "avfilter.h"
150227e8
 #include "audio.h"
b1fe7940
 #include "formats.h"
fa417fcd
 #include "internal.h"
c04c533f
 #include "video.h"
a38a00ed
 
73d5d405
 typedef struct SplitContext {
     const AVClass *class;
     int nb_outputs;
 } SplitContext;
 
093804a9
 static av_cold int split_init(AVFilterContext *ctx)
fd18ee0f
 {
73d5d405
     SplitContext *s = ctx->priv;
5e706a2a
     int i, ret;
fd18ee0f
 
73d5d405
     for (i = 0; i < s->nb_outputs; i++) {
fd18ee0f
         char name[32];
         AVFilterPad pad = { 0 };
 
         snprintf(name, sizeof(name), "output%d", i);
3d2515a8
         pad.type = ctx->filter->inputs[0].type;
fd18ee0f
         pad.name = av_strdup(name);
145a8471
         if (!pad.name)
             return AVERROR(ENOMEM);
fd18ee0f
 
5e706a2a
         if ((ret = ff_insert_outpad(ctx, i, &pad)) < 0) {
             av_freep(&pad.name);
             return ret;
         }
fd18ee0f
     }
 
     return 0;
 }
 
093804a9
 static av_cold void split_uninit(AVFilterContext *ctx)
fd18ee0f
 {
     int i;
 
9baeff95
     for (i = 0; i < ctx->nb_outputs; i++)
fd18ee0f
         av_freep(&ctx->output_pads[i].name);
 }
 
7e350379
 static int filter_frame(AVFilterLink *inlink, AVFrame *frame)
a38a00ed
 {
fd18ee0f
     AVFilterContext *ctx = inlink->dst;
a9cfe656
     int i, ret = AVERROR_EOF;
fd18ee0f
 
ebc8d974
     for (i = 0; i < ctx->nb_outputs; i++) {
a05a44e2
         AVFrame *buf_out;
a9cfe656
 
02aa0701
         if (ctx->outputs[i]->status_in)
a9cfe656
             continue;
a05a44e2
         buf_out = av_frame_clone(frame);
69d4420a
         if (!buf_out) {
             ret = AVERROR(ENOMEM);
e9b992d0
             break;
69d4420a
         }
fd18ee0f
 
69d4420a
         ret = ff_filter_frame(ctx->outputs[i], buf_out);
d4f89906
         if (ret < 0)
             break;
     }
7e350379
     av_frame_free(&frame);
d4f89906
     return ret;
a38a00ed
 }
 
73d5d405
 #define OFFSET(x) offsetof(SplitContext, x)
7b7c338e
 #define FLAGS (AV_OPT_FLAG_AUDIO_PARAM | AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_FILTERING_PARAM)
73d5d405
 static const AVOption options[] = {
3453b019
     { "outputs", "set number of outputs", OFFSET(nb_outputs), AV_OPT_TYPE_INT, { .i64 = 2 }, 1, INT_MAX, FLAGS },
b211607b
     { NULL }
73d5d405
 };
 
ba32afd7
 #define split_options options
 AVFILTER_DEFINE_CLASS(split);
73d5d405
 
ba32afd7
 #define asplit_options options
 AVFILTER_DEFINE_CLASS(asplit);
73d5d405
 
568c70e7
 static const AVFilterPad avfilter_vf_split_inputs[] = {
     {
b211607b
         .name         = "default",
         .type         = AVMEDIA_TYPE_VIDEO,
         .filter_frame = filter_frame,
568c70e7
     },
     { NULL }
 };
 
cd43ca04
 AVFilter ff_vf_split = {
b211607b
     .name        = "split",
10121557
     .description = NULL_IF_CONFIG_SMALL("Pass on the input to N video outputs."),
b211607b
     .priv_size   = sizeof(SplitContext),
     .priv_class  = &split_class,
     .init        = split_init,
     .uninit      = split_uninit,
     .inputs      = avfilter_vf_split_inputs,
     .outputs     = NULL,
     .flags       = AVFILTER_FLAG_DYNAMIC_OUTPUTS,
a38a00ed
 };
150227e8
 
568c70e7
 static const AVFilterPad avfilter_af_asplit_inputs[] = {
     {
b211607b
         .name         = "default",
         .type         = AVMEDIA_TYPE_AUDIO,
         .filter_frame = filter_frame,
568c70e7
     },
     { NULL }
 };
 
cd43ca04
 AVFilter ff_af_asplit = {
afeb3590
     .name        = "asplit",
150227e8
     .description = NULL_IF_CONFIG_SMALL("Pass on the audio input to N audio outputs."),
b211607b
     .priv_size   = sizeof(SplitContext),
     .priv_class  = &asplit_class,
     .init        = split_init,
     .uninit      = split_uninit,
     .inputs      = avfilter_af_asplit_inputs,
     .outputs     = NULL,
     .flags       = AVFILTER_FLAG_DYNAMIC_OUTPUTS,
150227e8
 };