annotate ffmpeg/libavfilter/af_channelsplit.c @ 13:844d341cf643 tip

Back up before ISMIR
author Yading Song <yading.song@eecs.qmul.ac.uk>
date Thu, 31 Oct 2013 13:17:06 +0000
parents 6840f77b83aa
children
rev   line source
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 };