From e2a09db92b5e9de9375100fe639d032b1edced54 Mon Sep 17 00:00:00 2001 From: Jean-Marc Valin <jean-marc.valin@octasic.com> Date: Wed, 2 Mar 2011 17:54:43 -0500 Subject: [PATCH] Implementing decoder-side support for redundant mode switching (bemasc's idea) --- src/opus_decoder.c | 22 +++++++++++++++++----- src/opus_decoder.h | 1 + src/opus_encoder.c | 13 +++++++++++++ 3 files changed, 31 insertions(+), 5 deletions(-) diff --git a/src/opus_decoder.c b/src/opus_decoder.c index 41671ebef..1ebc3ea2e 100644 --- a/src/opus_decoder.c +++ b/src/opus_decoder.c @@ -83,6 +83,8 @@ int opus_decode(OpusDecoder *st, const unsigned char *data, int audiosize; int mode; int transition=0; + int start_band; + int redundancy; /* Payloads of 1 (2 including ToC) or 0 trigger the PLC/DTX */ if (len<=2) @@ -130,7 +132,8 @@ int opus_decode(OpusDecoder *st, const unsigned char *data, && !(mode == MODE_HYBRID && st->prev_mode == MODE_SILK_ONLY)) { transition = 1; - opus_decode(st, NULL, 0, pcm_transition, IMAX(480, audiosize), 0); + if (mode == MODE_CELT_ONLY && !st->prev_redundancy) + opus_decode(st, NULL, 0, pcm_transition, IMAX(480, audiosize), 0); } if (audiosize > frame_size) { @@ -185,13 +188,21 @@ int opus_decode(OpusDecoder *st, const unsigned char *data, pcm[i] = 0; } + start_band = 0; if (mode == MODE_HYBRID) { - /* This should be adjusted based on the SILK bandwidth */ - celt_decoder_ctl(st->celt_dec, CELT_SET_START_BAND(17)); - } else { - celt_decoder_ctl(st->celt_dec, CELT_SET_START_BAND(0)); + /* Check if we have a redundant 0-8 kHz band */ + redundancy = ec_dec_bit_logp(&dec, 12); + if (!redundancy) + start_band = 17; } + celt_decoder_ctl(st->celt_dec, CELT_SET_START_BAND(start_band)); + + if (redundancy) + transition = 0; + + if (transition && mode != MODE_CELT_ONLY) + opus_decode(st, NULL, 0, pcm_transition, IMAX(480, audiosize), 0); if (mode != MODE_SILK_ONLY) { @@ -242,6 +253,7 @@ int opus_decode(OpusDecoder *st, const unsigned char *data, #endif st->prev_mode = mode; + st->prev_redundancy = redundancy; return celt_ret<0 ? celt_ret : audiosize; } diff --git a/src/opus_decoder.h b/src/opus_decoder.h index 1b0569c22..037c87b72 100644 --- a/src/opus_decoder.h +++ b/src/opus_decoder.h @@ -41,6 +41,7 @@ struct OpusDecoder { /* Sampling rate (at the API level) */ int Fs; int prev_mode; + int prev_redundancy; #ifdef OPUS_TEST_RANGE_CODER_STATE int rangeFinal; diff --git a/src/opus_encoder.c b/src/opus_encoder.c index 2334aa2bd..2c2e589fc 100644 --- a/src/opus_encoder.c +++ b/src/opus_encoder.c @@ -95,6 +95,8 @@ int opus_encode(OpusEncoder *st, const short *pcm, int frame_size, int silk_internal_bandwidth; int bytes_target; int prefill=0; + int start_band; + int redundancy = 0; bytes_target = st->bitrate_bps * frame_size / (st->Fs * 8) - 1; @@ -213,6 +215,17 @@ int opus_encode(OpusEncoder *st, const short *pcm, int frame_size, } else { celt_encoder_ctl(st->celt_enc, CELT_SET_PREDICTION(2)); } + + start_band = 0; + if (st->mode == MODE_HYBRID) + { + /* Check if we have a redundant 0-8 kHz band */ + ec_enc_bit_logp(&enc, redundancy, 12); + if (!redundancy) + start_band = 17; + } + celt_encoder_ctl(st->celt_enc, CELT_SET_START_BAND(start_band)); + if (st->mode == MODE_HYBRID) { int len; -- GitLab