Commit 0ef38d51 authored by gpoirier's avatar gpoirier

Allows the AC3 parser to read the frame size and codec parameters from E-AC3 streams,

allowing them to be correctly demuxed.
Currently it only identifies the primary substream, and will skip over any additional
dependent or independent substreams.
Patch by Ian Caulfield % ian P caulfield A gmail P com %
Original thread:
date: Jan 19, 2007 9:55 AM
subject: Re: [Ffmpeg-devel] [PATCH] Correctly parse headers of E-AC3 streams


git-svn-id: file:///var/local/repositories/ffmpeg/trunk@7687 9553f0bf-9b14-0410-a0b8-cfaf0461ba5b
parent 88088273
...@@ -632,9 +632,14 @@ static const int ac3_bitrates[64] = { ...@@ -632,9 +632,14 @@ static const int ac3_bitrates[64] = {
384, 448, 448, 512, 512, 576, 576, 640, 640, 384, 448, 448, 512, 512, 576, 576, 640, 640,
}; };
static const int ac3_channels[8] = { static const uint8_t ac3_channels[8] = {
2, 1, 2, 3, 3, 4, 4, 5 2, 1, 2, 3, 3, 4, 4, 5
}; };
static const uint8_t eac3_blocks[4] = {
1, 2, 3, 6
};
#endif /* CONFIG_AC3_PARSER */ #endif /* CONFIG_AC3_PARSER */
#ifdef CONFIG_AAC_PARSER #ifdef CONFIG_AAC_PARSER
...@@ -653,6 +658,7 @@ static int ac3_sync(const uint8_t *buf, int *channels, int *sample_rate, ...@@ -653,6 +658,7 @@ static int ac3_sync(const uint8_t *buf, int *channels, int *sample_rate,
int *bit_rate, int *samples) int *bit_rate, int *samples)
{ {
unsigned int fscod, frmsizecod, acmod, bsid, lfeon; unsigned int fscod, frmsizecod, acmod, bsid, lfeon;
unsigned int strmtyp, substreamid, frmsiz, fscod2, numblkscod;
GetBitContext bits; GetBitContext bits;
init_get_bits(&bits, buf, AC3_HEADER_SIZE * 8); init_get_bits(&bits, buf, AC3_HEADER_SIZE * 8);
...@@ -660,16 +666,16 @@ static int ac3_sync(const uint8_t *buf, int *channels, int *sample_rate, ...@@ -660,16 +666,16 @@ static int ac3_sync(const uint8_t *buf, int *channels, int *sample_rate,
if(get_bits(&bits, 16) != 0x0b77) if(get_bits(&bits, 16) != 0x0b77)
return 0; return 0;
bsid = show_bits_long(&bits, 29) & 0x1f;
if(bsid <= 8) { /* Normal AC-3 */
skip_bits(&bits, 16); /* crc */ skip_bits(&bits, 16); /* crc */
fscod = get_bits(&bits, 2); fscod = get_bits(&bits, 2);
frmsizecod = get_bits(&bits, 6); frmsizecod = get_bits(&bits, 6);
if(!ac3_sample_rates[fscod]) if(fscod == 3)
return 0; return 0;
bsid = get_bits(&bits, 5); skip_bits(&bits, 5); /* bsid */
if(bsid > 8)
return 0;
skip_bits(&bits, 3); /* bsmod */ skip_bits(&bits, 3); /* bsmod */
acmod = get_bits(&bits, 3); acmod = get_bits(&bits, 3);
if(acmod & 1 && acmod != 1) if(acmod & 1 && acmod != 1)
...@@ -686,6 +692,41 @@ static int ac3_sync(const uint8_t *buf, int *channels, int *sample_rate, ...@@ -686,6 +692,41 @@ static int ac3_sync(const uint8_t *buf, int *channels, int *sample_rate,
*samples = 6 * 256; *samples = 6 * 256;
return ac3_frame_sizes[frmsizecod][fscod] * 2; return ac3_frame_sizes[frmsizecod][fscod] * 2;
} else if (bsid >= 10 && bsid <= 16) { /* Enhanced AC-3 */
strmtyp = get_bits(&bits, 2);
substreamid = get_bits(&bits, 3);
if (strmtyp != 0 || substreamid != 0)
return 0; /* Currently don't support additional streams */
frmsiz = get_bits(&bits, 11) + 1;
fscod = get_bits(&bits, 2);
if (fscod == 3) {
fscod2 = get_bits(&bits, 2);
numblkscod = 3;
if(fscod2 == 3)
return 0;
*sample_rate = ac3_sample_rates[fscod2] / 2;
} else {
numblkscod = get_bits(&bits, 2);
*sample_rate = ac3_sample_rates[fscod];
}
acmod = get_bits(&bits, 3);
lfeon = get_bits1(&bits);
*samples = eac3_blocks[numblkscod] * 256;
*bit_rate = frmsiz * (*sample_rate) * 16 / (*samples);
*channels = ac3_channels[acmod] + lfeon;
return frmsiz * 2;
}
/* Unsupported bitstream version */
return 0;
} }
#endif /* CONFIG_AC3_PARSER */ #endif /* CONFIG_AC3_PARSER */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment