yading@10
|
1 /*
|
yading@10
|
2 * This file is part of Libav.
|
yading@10
|
3 *
|
yading@10
|
4 * Libav is free software; you can redistribute it and/or
|
yading@10
|
5 * modify it under the terms of the GNU Lesser General Public
|
yading@10
|
6 * License as published by the Free Software Foundation; either
|
yading@10
|
7 * version 2.1 of the License, or (at your option) any later version.
|
yading@10
|
8 *
|
yading@10
|
9 * Libav is distributed in the hope that it will be useful,
|
yading@10
|
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
yading@10
|
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
yading@10
|
12 * Lesser General Public License for more details.
|
yading@10
|
13 *
|
yading@10
|
14 * You should have received a copy of the GNU Lesser General Public
|
yading@10
|
15 * License along with Libav; if not, write to the Free Software
|
yading@10
|
16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
yading@10
|
17 */
|
yading@10
|
18
|
yading@10
|
19 /**
|
yading@10
|
20 * @file
|
yading@10
|
21 * Channel split filter
|
yading@10
|
22 *
|
yading@10
|
23 * Split an audio stream into per-channel streams.
|
yading@10
|
24 */
|
yading@10
|
25
|
yading@10
|
26 #include "libavutil/channel_layout.h"
|
yading@10
|
27 #include "libavutil/internal.h"
|
yading@10
|
28 #include "libavutil/opt.h"
|
yading@10
|
29
|
yading@10
|
30 #include "audio.h"
|
yading@10
|
31 #include "avfilter.h"
|
yading@10
|
32 #include "formats.h"
|
yading@10
|
33 #include "internal.h"
|
yading@10
|
34
|
yading@10
|
35 typedef struct ChannelSplitContext {
|
yading@10
|
36 const AVClass *class;
|
yading@10
|
37
|
yading@10
|
38 uint64_t channel_layout;
|
yading@10
|
39 char *channel_layout_str;
|
yading@10
|
40 } ChannelSplitContext;
|
yading@10
|
41
|
yading@10
|
42 #define OFFSET(x) offsetof(ChannelSplitContext, x)
|
yading@10
|
43 #define A AV_OPT_FLAG_AUDIO_PARAM
|
yading@10
|
44 #define F AV_OPT_FLAG_FILTERING_PARAM
|
yading@10
|
45 static const AVOption channelsplit_options[] = {
|
yading@10
|
46 { "channel_layout", "Input channel layout.", OFFSET(channel_layout_str), AV_OPT_TYPE_STRING, { .str = "stereo" }, .flags = A|F },
|
yading@10
|
47 { NULL },
|
yading@10
|
48 };
|
yading@10
|
49
|
yading@10
|
50 AVFILTER_DEFINE_CLASS(channelsplit);
|
yading@10
|
51
|
yading@10
|
52 static int init(AVFilterContext *ctx)
|
yading@10
|
53 {
|
yading@10
|
54 ChannelSplitContext *s = ctx->priv;
|
yading@10
|
55 int nb_channels;
|
yading@10
|
56 int ret = 0, i;
|
yading@10
|
57
|
yading@10
|
58 if (!(s->channel_layout = av_get_channel_layout(s->channel_layout_str))) {
|
yading@10
|
59 av_log(ctx, AV_LOG_ERROR, "Error parsing channel layout '%s'.\n",
|
yading@10
|
60 s->channel_layout_str);
|
yading@10
|
61 ret = AVERROR(EINVAL);
|
yading@10
|
62 goto fail;
|
yading@10
|
63 }
|
yading@10
|
64
|
yading@10
|
65 nb_channels = av_get_channel_layout_nb_channels(s->channel_layout);
|
yading@10
|
66 for (i = 0; i < nb_channels; i++) {
|
yading@10
|
67 uint64_t channel = av_channel_layout_extract_channel(s->channel_layout, i);
|
yading@10
|
68 AVFilterPad pad = { 0 };
|
yading@10
|
69
|
yading@10
|
70 pad.type = AVMEDIA_TYPE_AUDIO;
|
yading@10
|
71 pad.name = av_get_channel_name(channel);
|
yading@10
|
72
|
yading@10
|
73 ff_insert_outpad(ctx, i, &pad);
|
yading@10
|
74 }
|
yading@10
|
75
|
yading@10
|
76 fail:
|
yading@10
|
77 return ret;
|
yading@10
|
78 }
|
yading@10
|
79
|
yading@10
|
80 static int query_formats(AVFilterContext *ctx)
|
yading@10
|
81 {
|
yading@10
|
82 ChannelSplitContext *s = ctx->priv;
|
yading@10
|
83 AVFilterChannelLayouts *in_layouts = NULL;
|
yading@10
|
84 int i;
|
yading@10
|
85
|
yading@10
|
86 ff_set_common_formats (ctx, ff_planar_sample_fmts());
|
yading@10
|
87 ff_set_common_samplerates(ctx, ff_all_samplerates());
|
yading@10
|
88
|
yading@10
|
89 ff_add_channel_layout(&in_layouts, s->channel_layout);
|
yading@10
|
90 ff_channel_layouts_ref(in_layouts, &ctx->inputs[0]->out_channel_layouts);
|
yading@10
|
91
|
yading@10
|
92 for (i = 0; i < ctx->nb_outputs; i++) {
|
yading@10
|
93 AVFilterChannelLayouts *out_layouts = NULL;
|
yading@10
|
94 uint64_t channel = av_channel_layout_extract_channel(s->channel_layout, i);
|
yading@10
|
95
|
yading@10
|
96 ff_add_channel_layout(&out_layouts, channel);
|
yading@10
|
97 ff_channel_layouts_ref(out_layouts, &ctx->outputs[i]->in_channel_layouts);
|
yading@10
|
98 }
|
yading@10
|
99
|
yading@10
|
100 return 0;
|
yading@10
|
101 }
|
yading@10
|
102
|
yading@10
|
103 static int filter_frame(AVFilterLink *inlink, AVFrame *buf)
|
yading@10
|
104 {
|
yading@10
|
105 AVFilterContext *ctx = inlink->dst;
|
yading@10
|
106 int i, ret = 0;
|
yading@10
|
107
|
yading@10
|
108 for (i = 0; i < ctx->nb_outputs; i++) {
|
yading@10
|
109 AVFrame *buf_out = av_frame_clone(buf);
|
yading@10
|
110
|
yading@10
|
111 if (!buf_out) {
|
yading@10
|
112 ret = AVERROR(ENOMEM);
|
yading@10
|
113 break;
|
yading@10
|
114 }
|
yading@10
|
115
|
yading@10
|
116 buf_out->data[0] = buf_out->extended_data[0] = buf_out->extended_data[i];
|
yading@10
|
117 buf_out->channel_layout =
|
yading@10
|
118 av_channel_layout_extract_channel(buf->channel_layout, i);
|
yading@10
|
119 av_frame_set_channels(buf_out, 1);
|
yading@10
|
120
|
yading@10
|
121 ret = ff_filter_frame(ctx->outputs[i], buf_out);
|
yading@10
|
122 if (ret < 0)
|
yading@10
|
123 break;
|
yading@10
|
124 }
|
yading@10
|
125 av_frame_free(&buf);
|
yading@10
|
126 return ret;
|
yading@10
|
127 }
|
yading@10
|
128
|
yading@10
|
129 static const AVFilterPad avfilter_af_channelsplit_inputs[] = {
|
yading@10
|
130 {
|
yading@10
|
131 .name = "default",
|
yading@10
|
132 .type = AVMEDIA_TYPE_AUDIO,
|
yading@10
|
133 .filter_frame = filter_frame,
|
yading@10
|
134 },
|
yading@10
|
135 { NULL }
|
yading@10
|
136 };
|
yading@10
|
137
|
yading@10
|
138 AVFilter avfilter_af_channelsplit = {
|
yading@10
|
139 .name = "channelsplit",
|
yading@10
|
140 .description = NULL_IF_CONFIG_SMALL("Split audio into per-channel streams."),
|
yading@10
|
141 .priv_size = sizeof(ChannelSplitContext),
|
yading@10
|
142 .priv_class = &channelsplit_class,
|
yading@10
|
143
|
yading@10
|
144 .init = init,
|
yading@10
|
145 .query_formats = query_formats,
|
yading@10
|
146
|
yading@10
|
147 .inputs = avfilter_af_channelsplit_inputs,
|
yading@10
|
148 .outputs = NULL,
|
yading@10
|
149
|
yading@10
|
150 .flags = AVFILTER_FLAG_DYNAMIC_OUTPUTS,
|
yading@10
|
151 };
|