mirror of
https://git.ffmpeg.org/ffmpeg.git
synced 2024-12-28 02:12:28 +00:00
2eab5a1f54
This is achieved by using function pointers for AAC SBR functions. This unfortunately necessitated to use void* in ff_aac_sbr_apply(_fixed). Fixes ticket #10999. Reviewed-by: Lynne <dev@lynne.ee> Signed-off-by: Andreas Rheinhardt <andreas.rheinhardt@outlook.com>
449 lines
16 KiB
C
449 lines
16 KiB
C
/*
|
|
* AAC decoder
|
|
* Copyright (c) 2005-2006 Oded Shimon ( ods15 ods15 dyndns org )
|
|
* Copyright (c) 2006-2007 Maxim Gavrilov ( maxim.gavrilov gmail com )
|
|
* Copyright (c) 2008-2013 Alex Converse <alex.converse@gmail.com>
|
|
*
|
|
* AAC LATM decoder
|
|
* Copyright (c) 2008-2010 Paul Kendall <paul@kcbbs.gen.nz>
|
|
* Copyright (c) 2010 Janne Grunau <janne-libav@jannau.net>
|
|
*
|
|
* AAC decoder fixed-point implementation
|
|
* Copyright (c) 2013
|
|
* MIPS Technologies, Inc., California.
|
|
*
|
|
* This file is part of FFmpeg.
|
|
*
|
|
* FFmpeg is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* FFmpeg is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with FFmpeg; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
/**
|
|
* linear congruential pseudorandom number generator
|
|
*
|
|
* @param previous_val pointer to the current state of the generator
|
|
*
|
|
* @return Returns a 32-bit pseudorandom integer
|
|
*/
|
|
static av_always_inline int lcg_random(unsigned previous_val)
|
|
{
|
|
union { unsigned u; int s; } v = { previous_val * 1664525u + 1013904223 };
|
|
return v.s;
|
|
}
|
|
|
|
/**
|
|
* Decode spectral data; reference: table 4.50.
|
|
* Dequantize and scale spectral data; reference: 4.6.3.3.
|
|
*
|
|
* @param coef array of dequantized, scaled spectral data
|
|
* @param sf array of scalefactors or intensity stereo positions
|
|
* @param pulse_present set if pulses are present
|
|
* @param pulse pointer to pulse data struct
|
|
* @param band_type array of the used band type
|
|
*
|
|
* @return Returns error status. 0 - OK, !0 - error
|
|
*/
|
|
static int AAC_RENAME(decode_spectrum_and_dequant)(AACDecContext *ac,
|
|
GetBitContext *gb,
|
|
const Pulse *pulse,
|
|
SingleChannelElement *sce)
|
|
{
|
|
int i, k, g, idx = 0;
|
|
INTFLOAT *coef = sce->AAC_RENAME(coeffs);
|
|
IndividualChannelStream *ics = &sce->ics;
|
|
const int c = 1024 / ics->num_windows;
|
|
const uint16_t *offsets = ics->swb_offset;
|
|
const INTFLOAT *sf = sce->AAC_RENAME(sf);
|
|
const enum BandType *band_type = sce->band_type;
|
|
INTFLOAT *coef_base = coef;
|
|
|
|
for (g = 0; g < ics->num_windows; g++)
|
|
memset(coef + g * 128 + offsets[ics->max_sfb], 0,
|
|
sizeof(INTFLOAT) * (c - offsets[ics->max_sfb]));
|
|
|
|
for (g = 0; g < ics->num_window_groups; g++) {
|
|
unsigned g_len = ics->group_len[g];
|
|
|
|
for (i = 0; i < ics->max_sfb; i++, idx++) {
|
|
const unsigned cbt_m1 = band_type[idx] - 1;
|
|
INTFLOAT *cfo = coef + offsets[i];
|
|
int off_len = offsets[i + 1] - offsets[i];
|
|
int group;
|
|
|
|
if (cbt_m1 >= INTENSITY_BT2 - 1) {
|
|
for (group = 0; group < (AAC_SIGNE)g_len; group++, cfo+=128) {
|
|
memset(cfo, 0, off_len * sizeof(*cfo));
|
|
}
|
|
} else if (cbt_m1 == NOISE_BT - 1) {
|
|
for (group = 0; group < (AAC_SIGNE)g_len; group++, cfo+=128) {
|
|
INTFLOAT band_energy;
|
|
#if USE_FIXED
|
|
for (k = 0; k < off_len; k++) {
|
|
ac->random_state = lcg_random(ac->random_state);
|
|
cfo[k] = ac->random_state >> 3;
|
|
}
|
|
|
|
band_energy = ac->fdsp->scalarproduct_fixed(cfo, cfo, off_len);
|
|
band_energy = fixed_sqrt(band_energy, 31);
|
|
noise_scale(cfo, sf[idx], band_energy, off_len);
|
|
#else
|
|
float scale;
|
|
|
|
for (k = 0; k < off_len; k++) {
|
|
ac->random_state = lcg_random(ac->random_state);
|
|
cfo[k] = ac->random_state;
|
|
}
|
|
|
|
band_energy = ac->fdsp->scalarproduct_float(cfo, cfo, off_len);
|
|
scale = sf[idx] / sqrtf(band_energy);
|
|
ac->fdsp->vector_fmul_scalar(cfo, cfo, scale, off_len);
|
|
#endif /* USE_FIXED */
|
|
}
|
|
} else {
|
|
#if !USE_FIXED
|
|
const float *vq = ff_aac_codebook_vector_vals[cbt_m1];
|
|
#endif /* !USE_FIXED */
|
|
const VLCElem *vlc_tab = ff_vlc_spectral[cbt_m1];
|
|
OPEN_READER(re, gb);
|
|
|
|
switch (cbt_m1 >> 1) {
|
|
case 0:
|
|
for (group = 0; group < (AAC_SIGNE)g_len; group++, cfo+=128) {
|
|
INTFLOAT *cf = cfo;
|
|
int len = off_len;
|
|
|
|
do {
|
|
int code;
|
|
unsigned cb_idx;
|
|
|
|
UPDATE_CACHE(re, gb);
|
|
GET_VLC(code, re, gb, vlc_tab, 8, 2);
|
|
cb_idx = code;
|
|
#if USE_FIXED
|
|
cf = DEC_SQUAD(cf, cb_idx);
|
|
#else
|
|
cf = VMUL4(cf, vq, cb_idx, sf + idx);
|
|
#endif /* USE_FIXED */
|
|
} while (len -= 4);
|
|
}
|
|
break;
|
|
|
|
case 1:
|
|
for (group = 0; group < (AAC_SIGNE)g_len; group++, cfo+=128) {
|
|
INTFLOAT *cf = cfo;
|
|
int len = off_len;
|
|
|
|
do {
|
|
int code;
|
|
unsigned nnz;
|
|
unsigned cb_idx;
|
|
uint32_t bits;
|
|
|
|
UPDATE_CACHE(re, gb);
|
|
GET_VLC(code, re, gb, vlc_tab, 8, 2);
|
|
cb_idx = code;
|
|
nnz = cb_idx >> 8 & 15;
|
|
bits = nnz ? GET_CACHE(re, gb) : 0;
|
|
LAST_SKIP_BITS(re, gb, nnz);
|
|
#if USE_FIXED
|
|
cf = DEC_UQUAD(cf, cb_idx, bits);
|
|
#else
|
|
cf = VMUL4S(cf, vq, cb_idx, bits, sf + idx);
|
|
#endif /* USE_FIXED */
|
|
} while (len -= 4);
|
|
}
|
|
break;
|
|
|
|
case 2:
|
|
for (group = 0; group < (AAC_SIGNE)g_len; group++, cfo+=128) {
|
|
INTFLOAT *cf = cfo;
|
|
int len = off_len;
|
|
|
|
do {
|
|
int code;
|
|
unsigned cb_idx;
|
|
|
|
UPDATE_CACHE(re, gb);
|
|
GET_VLC(code, re, gb, vlc_tab, 8, 2);
|
|
cb_idx = code;
|
|
#if USE_FIXED
|
|
cf = DEC_SPAIR(cf, cb_idx);
|
|
#else
|
|
cf = VMUL2(cf, vq, cb_idx, sf + idx);
|
|
#endif /* USE_FIXED */
|
|
} while (len -= 2);
|
|
}
|
|
break;
|
|
|
|
case 3:
|
|
case 4:
|
|
for (group = 0; group < (AAC_SIGNE)g_len; group++, cfo+=128) {
|
|
INTFLOAT *cf = cfo;
|
|
int len = off_len;
|
|
|
|
do {
|
|
int code;
|
|
unsigned nnz;
|
|
unsigned cb_idx;
|
|
unsigned sign;
|
|
|
|
UPDATE_CACHE(re, gb);
|
|
GET_VLC(code, re, gb, vlc_tab, 8, 2);
|
|
cb_idx = code;
|
|
nnz = cb_idx >> 8 & 15;
|
|
sign = nnz ? SHOW_UBITS(re, gb, nnz) << (cb_idx >> 12) : 0;
|
|
LAST_SKIP_BITS(re, gb, nnz);
|
|
#if USE_FIXED
|
|
cf = DEC_UPAIR(cf, cb_idx, sign);
|
|
#else
|
|
cf = VMUL2S(cf, vq, cb_idx, sign, sf + idx);
|
|
#endif /* USE_FIXED */
|
|
} while (len -= 2);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
for (group = 0; group < (AAC_SIGNE)g_len; group++, cfo+=128) {
|
|
#if USE_FIXED
|
|
int *icf = cfo;
|
|
int v;
|
|
#else
|
|
float *cf = cfo;
|
|
uint32_t *icf = (uint32_t *) cf;
|
|
#endif /* USE_FIXED */
|
|
int len = off_len;
|
|
|
|
do {
|
|
int code;
|
|
unsigned nzt, nnz;
|
|
unsigned cb_idx;
|
|
uint32_t bits;
|
|
int j;
|
|
|
|
UPDATE_CACHE(re, gb);
|
|
GET_VLC(code, re, gb, vlc_tab, 8, 2);
|
|
cb_idx = code;
|
|
|
|
if (cb_idx == 0x0000) {
|
|
*icf++ = 0;
|
|
*icf++ = 0;
|
|
continue;
|
|
}
|
|
|
|
nnz = cb_idx >> 12;
|
|
nzt = cb_idx >> 8;
|
|
bits = SHOW_UBITS(re, gb, nnz) << (32-nnz);
|
|
LAST_SKIP_BITS(re, gb, nnz);
|
|
|
|
for (j = 0; j < 2; j++) {
|
|
if (nzt & 1<<j) {
|
|
uint32_t b;
|
|
int n;
|
|
/* The total length of escape_sequence must be < 22 bits according
|
|
to the specification (i.e. max is 111111110xxxxxxxxxxxx). */
|
|
UPDATE_CACHE(re, gb);
|
|
b = GET_CACHE(re, gb);
|
|
b = 31 - av_log2(~b);
|
|
|
|
if (b > 8) {
|
|
av_log(ac->avctx, AV_LOG_ERROR, "error in spectral data, ESC overflow\n");
|
|
return AVERROR_INVALIDDATA;
|
|
}
|
|
|
|
SKIP_BITS(re, gb, b + 1);
|
|
b += 4;
|
|
n = (1 << b) + SHOW_UBITS(re, gb, b);
|
|
LAST_SKIP_BITS(re, gb, b);
|
|
#if USE_FIXED
|
|
v = n;
|
|
if (bits & 1U<<31)
|
|
v = -v;
|
|
*icf++ = v;
|
|
#else
|
|
*icf++ = ff_cbrt_tab[n] | (bits & 1U<<31);
|
|
#endif /* USE_FIXED */
|
|
bits <<= 1;
|
|
} else {
|
|
#if USE_FIXED
|
|
v = cb_idx & 15;
|
|
if (bits & 1U<<31)
|
|
v = -v;
|
|
*icf++ = v;
|
|
#else
|
|
unsigned v = ((const uint32_t*)vq)[cb_idx & 15];
|
|
*icf++ = (bits & 1U<<31) | v;
|
|
#endif /* USE_FIXED */
|
|
bits <<= !!v;
|
|
}
|
|
cb_idx >>= 4;
|
|
}
|
|
} while (len -= 2);
|
|
#if !USE_FIXED
|
|
ac->fdsp->vector_fmul_scalar(cfo, cfo, sf[idx], off_len);
|
|
#endif /* !USE_FIXED */
|
|
}
|
|
}
|
|
|
|
CLOSE_READER(re, gb);
|
|
}
|
|
}
|
|
coef += g_len << 7;
|
|
}
|
|
|
|
if (pulse) {
|
|
idx = 0;
|
|
for (i = 0; i < pulse->num_pulse; i++) {
|
|
INTFLOAT co = coef_base[ pulse->pos[i] ];
|
|
while (offsets[idx + 1] <= pulse->pos[i])
|
|
idx++;
|
|
if (band_type[idx] != NOISE_BT && sf[idx]) {
|
|
INTFLOAT ico = -pulse->amp[i];
|
|
#if USE_FIXED
|
|
if (co) {
|
|
ico = co + (co > 0 ? -ico : ico);
|
|
}
|
|
coef_base[ pulse->pos[i] ] = ico;
|
|
#else
|
|
if (co) {
|
|
co /= sf[idx];
|
|
ico = co / sqrtf(sqrtf(fabsf(co))) + (co > 0 ? -ico : ico);
|
|
}
|
|
coef_base[ pulse->pos[i] ] = cbrtf(fabsf(ico)) * ico * sf[idx];
|
|
#endif /* USE_FIXED */
|
|
}
|
|
}
|
|
}
|
|
#if USE_FIXED
|
|
coef = coef_base;
|
|
idx = 0;
|
|
for (g = 0; g < ics->num_window_groups; g++) {
|
|
unsigned g_len = ics->group_len[g];
|
|
|
|
for (i = 0; i < ics->max_sfb; i++, idx++) {
|
|
const unsigned cbt_m1 = band_type[idx] - 1;
|
|
int *cfo = coef + offsets[i];
|
|
int off_len = offsets[i + 1] - offsets[i];
|
|
int group;
|
|
|
|
if (cbt_m1 < NOISE_BT - 1) {
|
|
for (group = 0; group < (int)g_len; group++, cfo+=128) {
|
|
vector_pow43(cfo, off_len);
|
|
subband_scale(cfo, cfo, sf[idx], 34, off_len, ac->avctx);
|
|
}
|
|
}
|
|
}
|
|
coef += g_len << 7;
|
|
}
|
|
#endif /* USE_FIXED */
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Decode coupling_channel_element; reference: table 4.8.
|
|
*
|
|
* @return Returns error status. 0 - OK, !0 - error
|
|
*/
|
|
static int AAC_RENAME(decode_cce)(AACDecContext *ac, GetBitContext *gb, ChannelElement *che)
|
|
{
|
|
int num_gain = 0;
|
|
int c, g, sfb, ret;
|
|
int sign;
|
|
INTFLOAT scale;
|
|
SingleChannelElement *sce = &che->ch[0];
|
|
ChannelCoupling *coup = &che->coup;
|
|
|
|
coup->coupling_point = 2 * get_bits1(gb);
|
|
coup->num_coupled = get_bits(gb, 3);
|
|
for (c = 0; c <= coup->num_coupled; c++) {
|
|
num_gain++;
|
|
coup->type[c] = get_bits1(gb) ? TYPE_CPE : TYPE_SCE;
|
|
coup->id_select[c] = get_bits(gb, 4);
|
|
if (coup->type[c] == TYPE_CPE) {
|
|
coup->ch_select[c] = get_bits(gb, 2);
|
|
if (coup->ch_select[c] == 3)
|
|
num_gain++;
|
|
} else
|
|
coup->ch_select[c] = 2;
|
|
}
|
|
coup->coupling_point += get_bits1(gb) || (coup->coupling_point >> 1);
|
|
|
|
sign = get_bits(gb, 1);
|
|
#if USE_FIXED
|
|
scale = get_bits(gb, 2);
|
|
#else
|
|
scale = cce_scale[get_bits(gb, 2)];
|
|
#endif
|
|
|
|
if ((ret = ff_aac_decode_ics(ac, sce, gb, 0, 0)))
|
|
return ret;
|
|
|
|
for (c = 0; c < num_gain; c++) {
|
|
int idx = 0;
|
|
int cge = 1;
|
|
int gain = 0;
|
|
INTFLOAT gain_cache = FIXR10(1.);
|
|
if (c) {
|
|
cge = coup->coupling_point == AFTER_IMDCT ? 1 : get_bits1(gb);
|
|
gain = cge ? get_vlc2(gb, ff_vlc_scalefactors, 7, 3) - 60: 0;
|
|
gain_cache = GET_GAIN(scale, gain);
|
|
#if USE_FIXED
|
|
if ((abs(gain_cache)-1024) >> 3 > 30)
|
|
return AVERROR(ERANGE);
|
|
#endif
|
|
}
|
|
if (coup->coupling_point == AFTER_IMDCT) {
|
|
coup->gain[c][0] = gain_cache;
|
|
} else {
|
|
for (g = 0; g < sce->ics.num_window_groups; g++) {
|
|
for (sfb = 0; sfb < sce->ics.max_sfb; sfb++, idx++) {
|
|
if (sce->band_type[idx] != ZERO_BT) {
|
|
if (!cge) {
|
|
int t = get_vlc2(gb, ff_vlc_scalefactors, 7, 3) - 60;
|
|
if (t) {
|
|
int s = 1;
|
|
t = gain += t;
|
|
if (sign) {
|
|
s -= 2 * (t & 0x1);
|
|
t >>= 1;
|
|
}
|
|
gain_cache = GET_GAIN(scale, t) * s;
|
|
#if USE_FIXED
|
|
if ((abs(gain_cache)-1024) >> 3 > 30)
|
|
return AVERROR(ERANGE);
|
|
#endif
|
|
}
|
|
}
|
|
coup->gain[c][idx] = gain_cache;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static av_cold void AAC_RENAME(aac_proc_init)(AACDecProc *aac_proc)
|
|
{
|
|
#define SET(member) aac_proc->member = AAC_RENAME(member)
|
|
SET(decode_spectrum_and_dequant);
|
|
SET(decode_cce);
|
|
#undef SET
|
|
#define SET(member) aac_proc->member = AV_JOIN(ff_aac_, AAC_RENAME(member));
|
|
SET(sbr_ctx_alloc_init);
|
|
SET(sbr_decode_extension);
|
|
SET(sbr_apply);
|
|
SET(sbr_ctx_close);
|
|
#undef SET
|
|
}
|