Imported Debian version 2.5.3~trusty1
[deb_ffmpeg.git] / ffmpeg / libavfilter / split.c
CommitLineData
2ba45a60
DM
1/*
2 * Copyright (c) 2007 Bobby Bingham
3 *
4 * This file is part of FFmpeg.
5 *
6 * FFmpeg is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * FFmpeg is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with FFmpeg; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
21/**
22 * @file
23 * audio and video splitter
24 */
25
26#include <stdio.h>
27
28#include "libavutil/attributes.h"
29#include "libavutil/internal.h"
30#include "libavutil/mem.h"
31#include "libavutil/opt.h"
32
33#include "avfilter.h"
34#include "audio.h"
35#include "internal.h"
36#include "video.h"
37
38typedef struct SplitContext {
39 const AVClass *class;
40 int nb_outputs;
41} SplitContext;
42
43static av_cold int split_init(AVFilterContext *ctx)
44{
45 SplitContext *s = ctx->priv;
46 int i;
47
48 for (i = 0; i < s->nb_outputs; i++) {
49 char name[32];
50 AVFilterPad pad = { 0 };
51
52 snprintf(name, sizeof(name), "output%d", i);
53 pad.type = ctx->filter->inputs[0].type;
54 pad.name = av_strdup(name);
0e279ba6
DM
55 if (!pad.name)
56 return AVERROR(ENOMEM);
2ba45a60
DM
57
58 ff_insert_outpad(ctx, i, &pad);
59 }
60
61 return 0;
62}
63
64static av_cold void split_uninit(AVFilterContext *ctx)
65{
66 int i;
67
68 for (i = 0; i < ctx->nb_outputs; i++)
69 av_freep(&ctx->output_pads[i].name);
70}
71
72static int filter_frame(AVFilterLink *inlink, AVFrame *frame)
73{
74 AVFilterContext *ctx = inlink->dst;
75 int i, ret = AVERROR_EOF;
76
77 for (i = 0; i < ctx->nb_outputs; i++) {
78 AVFrame *buf_out;
79
80 if (ctx->outputs[i]->closed)
81 continue;
82 buf_out = av_frame_clone(frame);
83 if (!buf_out) {
84 ret = AVERROR(ENOMEM);
85 break;
86 }
87
88 ret = ff_filter_frame(ctx->outputs[i], buf_out);
89 if (ret < 0)
90 break;
91 }
92 av_frame_free(&frame);
93 return ret;
94}
95
96#define OFFSET(x) offsetof(SplitContext, x)
97#define FLAGS AV_OPT_FLAG_AUDIO_PARAM | AV_OPT_FLAG_VIDEO_PARAM
98static const AVOption options[] = {
99 { "outputs", "set number of outputs", OFFSET(nb_outputs), AV_OPT_TYPE_INT, { .i64 = 2 }, 1, INT_MAX, FLAGS },
100 { NULL }
101};
102
103#define split_options options
104AVFILTER_DEFINE_CLASS(split);
105
106#define asplit_options options
107AVFILTER_DEFINE_CLASS(asplit);
108
109static const AVFilterPad avfilter_vf_split_inputs[] = {
110 {
111 .name = "default",
112 .type = AVMEDIA_TYPE_VIDEO,
113 .filter_frame = filter_frame,
114 },
115 { NULL }
116};
117
118AVFilter ff_vf_split = {
119 .name = "split",
120 .description = NULL_IF_CONFIG_SMALL("Pass on the input to N video outputs."),
121 .priv_size = sizeof(SplitContext),
122 .priv_class = &split_class,
123 .init = split_init,
124 .uninit = split_uninit,
125 .inputs = avfilter_vf_split_inputs,
126 .outputs = NULL,
127 .flags = AVFILTER_FLAG_DYNAMIC_OUTPUTS,
128};
129
130static const AVFilterPad avfilter_af_asplit_inputs[] = {
131 {
132 .name = "default",
133 .type = AVMEDIA_TYPE_AUDIO,
134 .filter_frame = filter_frame,
135 },
136 { NULL }
137};
138
139AVFilter ff_af_asplit = {
140 .name = "asplit",
141 .description = NULL_IF_CONFIG_SMALL("Pass on the audio input to N audio outputs."),
142 .priv_size = sizeof(SplitContext),
143 .priv_class = &asplit_class,
144 .init = split_init,
145 .uninit = split_uninit,
146 .inputs = avfilter_af_asplit_inputs,
147 .outputs = NULL,
148 .flags = AVFILTER_FLAG_DYNAMIC_OUTPUTS,
149};