aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRobert Swain <robert.swain@gmail.com>2009-09-28 15:35:13 +0000
committerRobert Swain <robert.swain@gmail.com>2009-09-28 15:35:13 +0000
commite22da6b63704c85e7da71866db2837c7c73a144a (patch)
tree2a3074a6c6b74ce3df7cf8e2f923c347218d214a
parentd2cdf20147d1bfde627b5548a659470172d03dbd (diff)
downloadffmpeg-e22da6b63704c85e7da71866db2837c7c73a144a.tar.gz
AAC: Add channel layout support for channel configuration as defined in the
specification Originally committed as revision 20067 to svn://svn.ffmpeg.org/ffmpeg/trunk
-rw-r--r--libavcodec/aac.c28
-rw-r--r--libavcodec/aacdectab.h23
2 files changed, 46 insertions, 5 deletions
diff --git a/libavcodec/aac.c b/libavcodec/aac.c
index 235cce5394..c659436a88 100644
--- a/libavcodec/aac.c
+++ b/libavcodec/aac.c
@@ -104,7 +104,6 @@ static VLC vlc_spectral[11];
static ChannelElement *get_che(AACContext *ac, int type, int elem_id)
{
- static const int8_t tags_per_config[16] = { 0, 1, 1, 2, 3, 3, 4, 5, 0, 0, 0, 0, 0, 0, 0, 0 };
if (ac->tag_che_map[type][elem_id]) {
return ac->tag_che_map[type][elem_id];
}
@@ -171,6 +170,26 @@ static int output_configure(AACContext *ac,
memcpy(che_pos, new_che_pos, 4 * MAX_ELEM_ID * sizeof(new_che_pos[0][0]));
+ if (channel_config) {
+ for (i = 0; i < tags_per_config[channel_config]; i++) {
+ const int id = aac_channel_layout_map[channel_config - 1][i][1];
+ type = aac_channel_layout_map[channel_config - 1][i][0];
+
+ if (!ac->che[type][id] && !(ac->che[type][id] = av_mallocz(sizeof(ChannelElement))))
+ return AVERROR(ENOMEM);
+
+ if (type != TYPE_CCE) {
+ ac->output_data[channels++] = ac->che[type][id]->ch[0].ret;
+ if (type == TYPE_CPE)
+ ac->output_data[channels++] = ac->che[type][id]->ch[1].ret;
+ }
+ }
+
+ memset(ac->tag_che_map, 0, 4 * MAX_ELEM_ID * sizeof(ac->che[0][0]));
+ ac->tags_mapped = 0;
+
+ avctx->channel_layout = aac_channel_layout[channel_config - 1];
+ } else {
/* Allocate or free elements depending on if they are in the
* current program configuration.
*
@@ -196,12 +215,10 @@ static int output_configure(AACContext *ac,
}
}
- if (channel_config) {
- memset(ac->tag_che_map, 0, 4 * MAX_ELEM_ID * sizeof(ac->che[0][0]));
- ac->tags_mapped = 0;
- } else {
memcpy(ac->tag_che_map, ac->che, 4 * MAX_ELEM_ID * sizeof(ac->che[0][0]));
ac->tags_mapped = 4 * MAX_ELEM_ID;
+
+ avctx->channel_layout = 0;
}
avctx->channels = channels;
@@ -1802,4 +1819,5 @@ AVCodec aac_decoder = {
.sample_fmts = (const enum SampleFormat[]) {
SAMPLE_FMT_S16,SAMPLE_FMT_NONE
},
+ .channel_layouts = aac_channel_layout,
};
diff --git a/libavcodec/aacdectab.h b/libavcodec/aacdectab.h
index b6d80f20c1..f632ca20da 100644
--- a/libavcodec/aacdectab.h
+++ b/libavcodec/aacdectab.h
@@ -69,4 +69,27 @@ static const float * const tns_tmp2_map[4] = {
};
// @}
+static const int8_t tags_per_config[16] = { 0, 1, 1, 2, 3, 3, 4, 5, 0, 0, 0, 0, 0, 0, 0, 0 };
+
+static const uint8_t aac_channel_layout_map[7][5][2] = {
+ { { TYPE_SCE, 0 }, },
+ { { TYPE_CPE, 0 }, },
+ { { TYPE_CPE, 0 }, { TYPE_SCE, 0 }, },
+ { { TYPE_CPE, 0 }, { TYPE_SCE, 0 }, { TYPE_SCE, 1 }, },
+ { { TYPE_CPE, 0 }, { TYPE_SCE, 0 }, { TYPE_CPE, 1 }, },
+ { { TYPE_CPE, 0 }, { TYPE_SCE, 0 }, { TYPE_LFE, 0 }, { TYPE_CPE, 1 }, },
+ { { TYPE_CPE, 0 }, { TYPE_SCE, 0 }, { TYPE_LFE, 0 }, { TYPE_CPE, 2 }, { TYPE_CPE, 1 }, },
+};
+
+static const int64_t aac_channel_layout[8] = {
+ CH_LAYOUT_MONO,
+ CH_LAYOUT_STEREO,
+ CH_LAYOUT_SURROUND,
+ CH_LAYOUT_4POINT0,
+ CH_LAYOUT_5POINT0_BACK,
+ CH_LAYOUT_5POINT1_BACK,
+ CH_LAYOUT_7POINT1_WIDE,
+ 0,
+};
+
#endif /* AVCODEC_AACDECTAB_H */