mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-01-24 13:56:33 +02:00
Add multi-channel correlation support for ALS.
Originally committed as revision 21074 to svn://svn.ffmpeg.org/ffmpeg/trunk
This commit is contained in:
parent
516841ef63
commit
e38215f2a4
@ -109,6 +109,17 @@ static const uint8_t ltp_gain_values [4][4] = {
|
||||
};
|
||||
|
||||
|
||||
/** Inter-channel weighting factors for multi-channel correlation.
|
||||
* To be indexed by the Rice coded indices.
|
||||
*/
|
||||
static const int16_t mcc_weightings[] = {
|
||||
204, 192, 179, 166, 153, 140, 128, 115,
|
||||
102, 89, 76, 64, 51, 38, 25, 12,
|
||||
0, -12, -25, -38, -51, -64, -76, -89,
|
||||
-102, -115, -128, -140, -153, -166, -179, -192
|
||||
};
|
||||
|
||||
|
||||
enum RA_Flag {
|
||||
RA_FLAG_NONE,
|
||||
RA_FLAG_FRAMES,
|
||||
@ -142,6 +153,16 @@ typedef struct {
|
||||
} ALSSpecificConfig;
|
||||
|
||||
|
||||
typedef struct {
|
||||
int stop_flag;
|
||||
int master_channel;
|
||||
int time_diff_flag;
|
||||
int time_diff_sign;
|
||||
int time_diff_index;
|
||||
int weighting[6];
|
||||
} ALSChannelData;
|
||||
|
||||
|
||||
typedef struct {
|
||||
AVCodecContext *avctx;
|
||||
ALSSpecificConfig sconf;
|
||||
@ -155,8 +176,13 @@ typedef struct {
|
||||
int *ltp_lag; ///< contains ltp lag values for all channels
|
||||
int **ltp_gain; ///< gain values for ltp 5-tap filter for a channel
|
||||
int *ltp_gain_buffer; ///< contains all gain values for ltp 5-tap filter
|
||||
int32_t *quant_cof; ///< quantized parcor coefficients
|
||||
int32_t *lpc_cof; ///< coefficients of the direct form prediction filter
|
||||
int32_t **quant_cof; ///< quantized parcor coefficients for a channel
|
||||
int32_t *quant_cof_buffer; ///< contains all quantized parcor coefficients
|
||||
int32_t **lpc_cof; ///< coefficients of the direct form prediction filter for a channel
|
||||
int32_t *lpc_cof_buffer; ///< contains all coefficients of the direct form prediction filter
|
||||
ALSChannelData **chan_data; ///< channel data for multi-channel correlation
|
||||
ALSChannelData *chan_data_buffer; ///< contains channel data for all channels
|
||||
int *reverted_channels; ///< stores a flag for each reverted channel
|
||||
int32_t *prev_raw_samples; ///< contains unshifted raw samples from the previous block
|
||||
int32_t **raw_samples; ///< decoded raw samples for each channel
|
||||
int32_t *raw_buffer; ///< contains all decoded raw samples including carryover samples
|
||||
@ -275,13 +301,6 @@ static av_cold int read_specific_config(ALSDecContext *ctx)
|
||||
|
||||
ctx->cur_frame_length = sconf->frame_length;
|
||||
|
||||
// allocate quantized parcor coefficient buffer
|
||||
if (!(ctx->quant_cof = av_malloc(sizeof(*ctx->quant_cof) * sconf->max_order)) ||
|
||||
!(ctx->lpc_cof = av_malloc(sizeof(*ctx->lpc_cof) * sconf->max_order))) {
|
||||
av_log(avctx, AV_LOG_ERROR, "Allocating buffer memory failed.\n");
|
||||
return AVERROR(ENOMEM);
|
||||
}
|
||||
|
||||
// read channel config
|
||||
if (sconf->chan_config)
|
||||
sconf->chan_config_info = get_bits(&gb, 16);
|
||||
@ -368,7 +387,6 @@ static int check_specific_config(ALSDecContext *ctx)
|
||||
|
||||
MISSING_ERR(sconf->floating, "Floating point decoding", -1);
|
||||
MISSING_ERR(sconf->bgmc, "BGMC entropy decoding", -1);
|
||||
MISSING_ERR(sconf->mc_coding, "Multi-channel correlation", -1);
|
||||
MISSING_ERR(sconf->rlslms, "Adaptive RLS-LMS prediction", -1);
|
||||
MISSING_ERR(sconf->chan_sort, "Channel sorting", 0);
|
||||
|
||||
@ -895,8 +913,8 @@ static int decode_blocks_ind(ALSDecContext *ctx, unsigned int ra_frame,
|
||||
bd.use_ltp = ctx->use_ltp;
|
||||
bd.ltp_lag = ctx->ltp_lag;
|
||||
bd.ltp_gain = ctx->ltp_gain[0];
|
||||
bd.quant_cof = ctx->quant_cof;
|
||||
bd.lpc_cof = ctx->lpc_cof;
|
||||
bd.quant_cof = ctx->quant_cof[0];
|
||||
bd.lpc_cof = ctx->lpc_cof[0];
|
||||
bd.prev_raw_samples = ctx->prev_raw_samples;
|
||||
bd.raw_samples = ctx->raw_samples[c];
|
||||
|
||||
@ -935,8 +953,8 @@ static int decode_blocks(ALSDecContext *ctx, unsigned int ra_frame,
|
||||
bd[0].use_ltp = ctx->use_ltp;
|
||||
bd[0].ltp_lag = ctx->ltp_lag;
|
||||
bd[0].ltp_gain = ctx->ltp_gain[0];
|
||||
bd[0].quant_cof = ctx->quant_cof;
|
||||
bd[0].lpc_cof = ctx->lpc_cof;
|
||||
bd[0].quant_cof = ctx->quant_cof[0];
|
||||
bd[0].lpc_cof = ctx->lpc_cof[0];
|
||||
bd[0].prev_raw_samples = ctx->prev_raw_samples;
|
||||
bd[0].js_blocks = *js_blocks;
|
||||
|
||||
@ -944,8 +962,8 @@ static int decode_blocks(ALSDecContext *ctx, unsigned int ra_frame,
|
||||
bd[1].use_ltp = ctx->use_ltp;
|
||||
bd[1].ltp_lag = ctx->ltp_lag;
|
||||
bd[1].ltp_gain = ctx->ltp_gain[0];
|
||||
bd[1].quant_cof = ctx->quant_cof;
|
||||
bd[1].lpc_cof = ctx->lpc_cof;
|
||||
bd[1].quant_cof = ctx->quant_cof[0];
|
||||
bd[1].lpc_cof = ctx->lpc_cof[0];
|
||||
bd[1].prev_raw_samples = ctx->prev_raw_samples;
|
||||
bd[1].js_blocks = *(js_blocks + 1);
|
||||
|
||||
@ -999,6 +1017,134 @@ static int decode_blocks(ALSDecContext *ctx, unsigned int ra_frame,
|
||||
}
|
||||
|
||||
|
||||
/** Reads the channel data.
|
||||
*/
|
||||
static int read_channel_data(ALSDecContext *ctx, ALSChannelData *cd, int c)
|
||||
{
|
||||
GetBitContext *gb = &ctx->gb;
|
||||
ALSChannelData *current = cd;
|
||||
unsigned int channels = ctx->avctx->channels;
|
||||
int entries = 0;
|
||||
|
||||
while (entries < channels && !(current->stop_flag = get_bits1(gb))) {
|
||||
current->master_channel = get_bits_long(gb, av_ceil_log2(channels));
|
||||
|
||||
if (current->master_channel >= channels) {
|
||||
av_log(ctx->avctx, AV_LOG_ERROR, "Invalid master channel!\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (current->master_channel != c) {
|
||||
current->time_diff_flag = get_bits1(gb);
|
||||
current->weighting[0] = mcc_weightings[av_clip(decode_rice(gb, 1) + 16, 0, 32)];
|
||||
current->weighting[1] = mcc_weightings[av_clip(decode_rice(gb, 2) + 14, 0, 32)];
|
||||
current->weighting[2] = mcc_weightings[av_clip(decode_rice(gb, 1) + 16, 0, 32)];
|
||||
|
||||
if (current->time_diff_flag) {
|
||||
current->weighting[3] = mcc_weightings[av_clip(decode_rice(gb, 1) + 16, 0, 32)];
|
||||
current->weighting[4] = mcc_weightings[av_clip(decode_rice(gb, 1) + 16, 0, 32)];
|
||||
current->weighting[5] = mcc_weightings[av_clip(decode_rice(gb, 1) + 16, 0, 32)];
|
||||
|
||||
current->time_diff_sign = get_bits1(gb);
|
||||
current->time_diff_index = get_bits(gb, ctx->ltp_lag_length - 3) + 3;
|
||||
}
|
||||
}
|
||||
|
||||
current++;
|
||||
entries++;
|
||||
}
|
||||
|
||||
if (entries == channels) {
|
||||
av_log(ctx->avctx, AV_LOG_ERROR, "Damaged channel data!\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
align_get_bits(gb);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
/** Recursively reverts the inter-channel correlation for a block.
|
||||
*/
|
||||
static int revert_channel_correlation(ALSDecContext *ctx, ALSBlockData *bd,
|
||||
ALSChannelData **cd, int *reverted,
|
||||
unsigned int offset, int c)
|
||||
{
|
||||
ALSChannelData *ch = cd[c];
|
||||
unsigned int dep = 0;
|
||||
unsigned int channels = ctx->avctx->channels;
|
||||
|
||||
if (reverted[c])
|
||||
return 0;
|
||||
|
||||
reverted[c] = 1;
|
||||
|
||||
while (dep < channels && !ch[dep].stop_flag) {
|
||||
revert_channel_correlation(ctx, bd, cd, reverted, offset,
|
||||
ch[dep].master_channel);
|
||||
|
||||
dep++;
|
||||
}
|
||||
|
||||
if (dep == channels) {
|
||||
av_log(ctx->avctx, AV_LOG_WARNING, "Invalid channel correlation!\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
bd->use_ltp = ctx->use_ltp + c;
|
||||
bd->ltp_lag = ctx->ltp_lag + c;
|
||||
bd->ltp_gain = ctx->ltp_gain[c];
|
||||
bd->lpc_cof = ctx->lpc_cof[c];
|
||||
bd->quant_cof = ctx->quant_cof[c];
|
||||
bd->raw_samples = ctx->raw_samples[c] + offset;
|
||||
|
||||
dep = 0;
|
||||
while (!ch[dep].stop_flag) {
|
||||
unsigned int smp;
|
||||
unsigned int begin = 1;
|
||||
unsigned int end = bd->block_length - 1;
|
||||
int64_t y;
|
||||
int32_t *master = ctx->raw_samples[ch[dep].master_channel] + offset;
|
||||
|
||||
if (ch[dep].time_diff_flag) {
|
||||
int t = ch[dep].time_diff_index;
|
||||
|
||||
if (ch[dep].time_diff_sign) {
|
||||
t = -t;
|
||||
begin -= t;
|
||||
} else {
|
||||
end -= t;
|
||||
}
|
||||
|
||||
for (smp = begin; smp < end; smp++) {
|
||||
y = (1 << 6) +
|
||||
MUL64(ch[dep].weighting[0], master[smp - 1 ]) +
|
||||
MUL64(ch[dep].weighting[1], master[smp ]) +
|
||||
MUL64(ch[dep].weighting[2], master[smp + 1 ]) +
|
||||
MUL64(ch[dep].weighting[3], master[smp - 1 + t]) +
|
||||
MUL64(ch[dep].weighting[4], master[smp + t]) +
|
||||
MUL64(ch[dep].weighting[5], master[smp + 1 + t]);
|
||||
|
||||
bd->raw_samples[smp] += y >> 7;
|
||||
}
|
||||
} else {
|
||||
for (smp = begin; smp < end; smp++) {
|
||||
y = (1 << 6) +
|
||||
MUL64(ch[dep].weighting[0], master[smp - 1]) +
|
||||
MUL64(ch[dep].weighting[1], master[smp ]) +
|
||||
MUL64(ch[dep].weighting[2], master[smp + 1]);
|
||||
|
||||
bd->raw_samples[smp] += y >> 7;
|
||||
}
|
||||
}
|
||||
|
||||
dep++;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
/** Reads the frame data.
|
||||
*/
|
||||
static int read_frame_data(ALSDecContext *ctx, unsigned int ra_frame)
|
||||
@ -1058,13 +1204,68 @@ static int read_frame_data(ALSDecContext *ctx, unsigned int ra_frame)
|
||||
sizeof(*ctx->raw_samples[c]) * sconf->max_order);
|
||||
}
|
||||
} else { // multi-channel coding
|
||||
ALSBlockData bd;
|
||||
int b;
|
||||
int *reverted_channels = ctx->reverted_channels;
|
||||
unsigned int offset = 0;
|
||||
|
||||
for (c = 0; c < avctx->channels; c++)
|
||||
if (ctx->chan_data[c] < ctx->chan_data_buffer) {
|
||||
av_log(ctx->avctx, AV_LOG_ERROR, "Invalid channel data!\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
memset(&bd, 0, sizeof(ALSBlockData));
|
||||
memset(reverted_channels, 0, sizeof(*reverted_channels) * avctx->channels);
|
||||
|
||||
bd.ra_block = ra_frame;
|
||||
bd.prev_raw_samples = ctx->prev_raw_samples;
|
||||
|
||||
get_block_sizes(ctx, div_blocks, &bs_info);
|
||||
|
||||
// TODO: multi channel coding might use a temporary buffer instead as
|
||||
// the actual channel is not known when read_block-data is called
|
||||
if (decode_blocks_ind(ctx, ra_frame, 0, div_blocks, js_blocks))
|
||||
return -1;
|
||||
// TODO: read_channel_data
|
||||
for (b = 0; b < ctx->num_blocks; b++) {
|
||||
bd.shift_lsbs = 0;
|
||||
bd.block_length = div_blocks[b];
|
||||
|
||||
for (c = 0; c < avctx->channels; c++) {
|
||||
bd.use_ltp = ctx->use_ltp + c;
|
||||
bd.ltp_lag = ctx->ltp_lag + c;
|
||||
bd.ltp_gain = ctx->ltp_gain[c];
|
||||
bd.lpc_cof = ctx->lpc_cof[c];
|
||||
bd.quant_cof = ctx->quant_cof[c];
|
||||
bd.raw_samples = ctx->raw_samples[c] + offset;
|
||||
bd.raw_other = NULL;
|
||||
|
||||
read_block(ctx, &bd);
|
||||
if (read_channel_data(ctx, ctx->chan_data[c], c))
|
||||
return -1;
|
||||
}
|
||||
|
||||
for (c = 0; c < avctx->channels; c++)
|
||||
if (revert_channel_correlation(ctx, &bd, ctx->chan_data,
|
||||
reverted_channels, offset, c))
|
||||
return -1;
|
||||
|
||||
for (c = 0; c < avctx->channels; c++) {
|
||||
bd.use_ltp = ctx->use_ltp + c;
|
||||
bd.ltp_lag = ctx->ltp_lag + c;
|
||||
bd.ltp_gain = ctx->ltp_gain[c];
|
||||
bd.lpc_cof = ctx->lpc_cof[c];
|
||||
bd.quant_cof = ctx->quant_cof[c];
|
||||
bd.raw_samples = ctx->raw_samples[c] + offset;
|
||||
decode_block(ctx, &bd);
|
||||
}
|
||||
|
||||
memset(reverted_channels, 0, avctx->channels * sizeof(*reverted_channels));
|
||||
offset += div_blocks[b];
|
||||
bd.ra_block = 0;
|
||||
}
|
||||
|
||||
// store carryover raw samples
|
||||
for (c = 0; c < avctx->channels; c++)
|
||||
memmove(ctx->raw_samples[c] - sconf->max_order,
|
||||
ctx->raw_samples[c] - sconf->max_order + sconf->frame_length,
|
||||
sizeof(*ctx->raw_samples[c]) * sconf->max_order);
|
||||
}
|
||||
|
||||
// TODO: read_diff_float_data
|
||||
@ -1156,9 +1357,14 @@ static av_cold int decode_end(AVCodecContext *avctx)
|
||||
av_freep(&ctx->ltp_gain_buffer);
|
||||
av_freep(&ctx->quant_cof);
|
||||
av_freep(&ctx->lpc_cof);
|
||||
av_freep(&ctx->quant_cof_buffer);
|
||||
av_freep(&ctx->lpc_cof_buffer);
|
||||
av_freep(&ctx->prev_raw_samples);
|
||||
av_freep(&ctx->raw_samples);
|
||||
av_freep(&ctx->raw_buffer);
|
||||
av_freep(&ctx->chan_data);
|
||||
av_freep(&ctx->chan_data_buffer);
|
||||
av_freep(&ctx->reverted_channels);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -1207,6 +1413,25 @@ static av_cold int decode_init(AVCodecContext *avctx)
|
||||
// allocate quantized parcor coefficient buffer
|
||||
num_buffers = sconf->mc_coding ? avctx->channels : 1;
|
||||
|
||||
ctx->quant_cof = av_malloc(sizeof(*ctx->quant_cof) * num_buffers);
|
||||
ctx->lpc_cof = av_malloc(sizeof(*ctx->lpc_cof) * num_buffers);
|
||||
ctx->quant_cof_buffer = av_malloc(sizeof(*ctx->quant_cof_buffer) *
|
||||
num_buffers * sconf->max_order);
|
||||
ctx->lpc_cof_buffer = av_malloc(sizeof(*ctx->lpc_cof_buffer) *
|
||||
num_buffers * sconf->max_order);
|
||||
|
||||
if (!ctx->quant_cof || !ctx->lpc_cof ||
|
||||
!ctx->quant_cof_buffer || !ctx->lpc_cof_buffer) {
|
||||
av_log(avctx, AV_LOG_ERROR, "Allocating buffer memory failed.\n");
|
||||
return AVERROR(ENOMEM);
|
||||
}
|
||||
|
||||
// assign quantized parcor coefficient buffers
|
||||
for (c = 0; c < num_buffers; c++) {
|
||||
ctx->quant_cof[c] = ctx->quant_cof_buffer + c * sconf->max_order;
|
||||
ctx->lpc_cof[c] = ctx->lpc_cof_buffer + c * sconf->max_order;
|
||||
}
|
||||
|
||||
// allocate and assign lag and gain data buffer for ltp mode
|
||||
ctx->use_ltp = av_mallocz(sizeof(*ctx->use_ltp) * num_buffers);
|
||||
ctx->ltp_lag = av_malloc (sizeof(*ctx->ltp_lag) * num_buffers);
|
||||
@ -1224,6 +1449,29 @@ static av_cold int decode_init(AVCodecContext *avctx)
|
||||
for (c = 0; c < num_buffers; c++)
|
||||
ctx->ltp_gain[c] = ctx->ltp_gain_buffer + c * 5;
|
||||
|
||||
// allocate and assign channel data buffer for mcc mode
|
||||
if (sconf->mc_coding) {
|
||||
ctx->chan_data_buffer = av_malloc(sizeof(*ctx->chan_data_buffer) *
|
||||
num_buffers);
|
||||
ctx->chan_data = av_malloc(sizeof(ALSChannelData) *
|
||||
num_buffers);
|
||||
ctx->reverted_channels = av_malloc(sizeof(*ctx->reverted_channels) *
|
||||
num_buffers);
|
||||
|
||||
if (!ctx->chan_data_buffer || !ctx->chan_data || !ctx->reverted_channels) {
|
||||
av_log(avctx, AV_LOG_ERROR, "Allocating buffer memory failed.\n");
|
||||
decode_end(avctx);
|
||||
return AVERROR(ENOMEM);
|
||||
}
|
||||
|
||||
for (c = 0; c < num_buffers; c++)
|
||||
ctx->chan_data[c] = ctx->chan_data_buffer + c;
|
||||
} else {
|
||||
ctx->chan_data = NULL;
|
||||
ctx->chan_data_buffer = NULL;
|
||||
ctx->reverted_channels = NULL;
|
||||
}
|
||||
|
||||
avctx->frame_size = sconf->frame_length;
|
||||
channel_size = sconf->frame_length + sconf->max_order;
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user