summaryrefslogtreecommitdiffstats
path: root/fdk-aac/libSACenc/src/sacenc_nlc_enc.cpp
diff options
context:
space:
mode:
authorMatthias P. Braendli <matthias.braendli@mpb.li>2019-11-11 11:38:02 +0100
committerMatthias P. Braendli <matthias.braendli@mpb.li>2019-11-11 11:38:02 +0100
commit0e5af65c467b2423a0b857ae3ad98c91acc1e190 (patch)
treed07f69550d8886271e44fe79c4dcfb299cafbd38 /fdk-aac/libSACenc/src/sacenc_nlc_enc.cpp
parentefe406d9724f959c8bc2a31802559ca6d41fd897 (diff)
downloadODR-AudioEnc-0e5af65c467b2423a0b857ae3ad98c91acc1e190.tar.gz
ODR-AudioEnc-0e5af65c467b2423a0b857ae3ad98c91acc1e190.tar.bz2
ODR-AudioEnc-0e5af65c467b2423a0b857ae3ad98c91acc1e190.zip
Include patched FDK-AAC in the repository
The initial idea was to get the DAB+ patch into upstream, but since that follows the android source releases, there is no place for a custom DAB+ patch there. So instead of having to maintain a patched fdk-aac that has to have the same .so version as the distribution package on which it is installed, we prefer having a separate fdk-aac-dab library to avoid collision. At that point, there's no reason to keep fdk-aac in a separate repository, as odr-audioenc is the only tool that needs DAB+ encoding support. Including it here simplifies installation, and makes it consistent with toolame-dab, also shipped in this repository. DAB+ decoding support (needed by ODR-SourceCompanion, dablin, etisnoop, welle.io and others) can be done using upstream FDK-AAC.
Diffstat (limited to 'fdk-aac/libSACenc/src/sacenc_nlc_enc.cpp')
-rw-r--r--fdk-aac/libSACenc/src/sacenc_nlc_enc.cpp1442
1 files changed, 1442 insertions, 0 deletions
diff --git a/fdk-aac/libSACenc/src/sacenc_nlc_enc.cpp b/fdk-aac/libSACenc/src/sacenc_nlc_enc.cpp
new file mode 100644
index 0000000..0ba6cc9
--- /dev/null
+++ b/fdk-aac/libSACenc/src/sacenc_nlc_enc.cpp
@@ -0,0 +1,1442 @@
+/* -----------------------------------------------------------------------------
+Software License for The Fraunhofer FDK AAC Codec Library for Android
+
+© Copyright 1995 - 2018 Fraunhofer-Gesellschaft zur Förderung der angewandten
+Forschung e.V. All rights reserved.
+
+ 1. INTRODUCTION
+The Fraunhofer FDK AAC Codec Library for Android ("FDK AAC Codec") is software
+that implements the MPEG Advanced Audio Coding ("AAC") encoding and decoding
+scheme for digital audio. This FDK AAC Codec software is intended to be used on
+a wide variety of Android devices.
+
+AAC's HE-AAC and HE-AAC v2 versions are regarded as today's most efficient
+general perceptual audio codecs. AAC-ELD is considered the best-performing
+full-bandwidth communications codec by independent studies and is widely
+deployed. AAC has been standardized by ISO and IEC as part of the MPEG
+specifications.
+
+Patent licenses for necessary patent claims for the FDK AAC Codec (including
+those of Fraunhofer) may be obtained through Via Licensing
+(www.vialicensing.com) or through the respective patent owners individually for
+the purpose of encoding or decoding bit streams in products that are compliant
+with the ISO/IEC MPEG audio standards. Please note that most manufacturers of
+Android devices already license these patent claims through Via Licensing or
+directly from the patent owners, and therefore FDK AAC Codec software may
+already be covered under those patent licenses when it is used for those
+licensed purposes only.
+
+Commercially-licensed AAC software libraries, including floating-point versions
+with enhanced sound quality, are also available from Fraunhofer. Users are
+encouraged to check the Fraunhofer website for additional applications
+information and documentation.
+
+2. COPYRIGHT LICENSE
+
+Redistribution and use in source and binary forms, with or without modification,
+are permitted without payment of copyright license fees provided that you
+satisfy the following conditions:
+
+You must retain the complete text of this software license in redistributions of
+the FDK AAC Codec or your modifications thereto in source code form.
+
+You must retain the complete text of this software license in the documentation
+and/or other materials provided with redistributions of the FDK AAC Codec or
+your modifications thereto in binary form. You must make available free of
+charge copies of the complete source code of the FDK AAC Codec and your
+modifications thereto to recipients of copies in binary form.
+
+The name of Fraunhofer may not be used to endorse or promote products derived
+from this library without prior written permission.
+
+You may not charge copyright license fees for anyone to use, copy or distribute
+the FDK AAC Codec software or your modifications thereto.
+
+Your modified versions of the FDK AAC Codec must carry prominent notices stating
+that you changed the software and the date of any change. For modified versions
+of the FDK AAC Codec, the term "Fraunhofer FDK AAC Codec Library for Android"
+must be replaced by the term "Third-Party Modified Version of the Fraunhofer FDK
+AAC Codec Library for Android."
+
+3. NO PATENT LICENSE
+
+NO EXPRESS OR IMPLIED LICENSES TO ANY PATENT CLAIMS, including without
+limitation the patents of Fraunhofer, ARE GRANTED BY THIS SOFTWARE LICENSE.
+Fraunhofer provides no warranty of patent non-infringement with respect to this
+software.
+
+You may use this FDK AAC Codec software or modifications thereto only for
+purposes that are authorized by appropriate patent licenses.
+
+4. DISCLAIMER
+
+This FDK AAC Codec software is provided by Fraunhofer on behalf of the copyright
+holders and contributors "AS IS" and WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES,
+including but not limited to the implied warranties of merchantability and
+fitness for a particular purpose. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
+CONTRIBUTORS BE LIABLE for any direct, indirect, incidental, special, exemplary,
+or consequential damages, including but not limited to procurement of substitute
+goods or services; loss of use, data, or profits, or business interruption,
+however caused and on any theory of liability, whether in contract, strict
+liability, or tort (including negligence), arising in any way out of the use of
+this software, even if advised of the possibility of such damage.
+
+5. CONTACT INFORMATION
+
+Fraunhofer Institute for Integrated Circuits IIS
+Attention: Audio and Multimedia Departments - FDK AAC LL
+Am Wolfsmantel 33
+91058 Erlangen, Germany
+
+www.iis.fraunhofer.de/amm
+amm-info@iis.fraunhofer.de
+----------------------------------------------------------------------------- */
+
+/*********************** MPEG surround encoder library *************************
+
+ Author(s): Karsten Linzmeier
+
+ Description: Noiseless Coding
+ Huffman encoder
+
+*******************************************************************************/
+
+/* Includes ******************************************************************/
+#include "sacenc_nlc_enc.h"
+
+#include "genericStds.h"
+#include "fixpoint_math.h"
+
+#include "sacenc_const.h"
+#include "sacenc_huff_tab.h"
+#include "sacenc_paramextract.h"
+
+/* Defines *******************************************************************/
+#define PAIR_SHIFT 4
+#define PAIR_MASK 0xf
+
+#define PBC_MIN_BANDS 5
+
+typedef enum {
+ BACKWARDS = 0x0,
+ FORWARDS = 0x1
+
+} DIRECTION;
+
+typedef enum {
+ DIFF_FREQ = 0x0,
+ DIFF_TIME = 0x1
+
+} DIFF_TYPE;
+
+typedef enum {
+ HUFF_1D = 0x0,
+ HUFF_2D = 0x1
+
+} CODING_SCHEME;
+
+typedef enum {
+ FREQ_PAIR = 0x0,
+ TIME_PAIR = 0x1
+
+} PAIRING;
+
+/* Data Types ****************************************************************/
+
+/* Constants *****************************************************************/
+static const UCHAR lavHuffVal[4] = {0, 2, 6, 7};
+static const UCHAR lavHuffLen[4] = {1, 2, 3, 3};
+
+static const UCHAR lav_step_CLD[] = {0, 0, 0, 0, 1, 1, 2, 2, 3, 3};
+static const UCHAR lav_step_ICC[] = {0, 0, 1, 1, 2, 2, 3, 3};
+
+/* Function / Class Declarations *********************************************/
+
+/* Function / Class Definition ***********************************************/
+static void split_lsb(const SHORT *const in_data, SHORT offset,
+ const INT num_val, SHORT *const out_data_lsb,
+ SHORT *const out_data_msb) {
+ int i;
+
+ for (i = 0; i < num_val; i++) {
+ SHORT val = in_data[i] + offset;
+ if (out_data_lsb != NULL) out_data_lsb[i] = val & 0x0001;
+ if (out_data_msb != NULL) out_data_msb[i] = val >> 1;
+ }
+}
+
+static void apply_lsb_coding(HANDLE_FDK_BITSTREAM strm,
+ const SHORT *const in_data_lsb, const UINT num_lsb,
+ const INT num_val) {
+ int i;
+
+ for (i = 0; i < num_val; i++) {
+ FDKwriteBits(strm, in_data_lsb[i], num_lsb);
+ }
+}
+
+static void calc_diff_freq(const SHORT *const in_data, SHORT *const out_data,
+ const INT num_val) {
+ int i;
+ out_data[0] = in_data[0];
+
+ for (i = 1; i < num_val; i++) {
+ out_data[i] = in_data[i] - in_data[i - 1];
+ }
+}
+
+static void calc_diff_time(const SHORT *const in_data,
+ const SHORT *const prev_data, SHORT *const out_data,
+ const INT num_val) {
+ int i;
+ out_data[0] = in_data[0];
+ out_data[1] = prev_data[0];
+
+ for (i = 0; i < num_val; i++) {
+ out_data[i + 2] = in_data[i] - prev_data[i];
+ }
+}
+
+static INT sym_check(SHORT data[2], const INT lav, SHORT *const pSym_bits) {
+ UCHAR symBits = 0;
+ int sum_val = data[0] + data[1];
+ int diff_val = data[0] - data[1];
+ int num_sbits = 0;
+
+ if (sum_val != 0) {
+ int sum_neg = (sum_val < 0) ? 1 : 0;
+ if (sum_neg) {
+ sum_val = -sum_val;
+ diff_val = -diff_val;
+ }
+ symBits = (symBits << 1) | sum_neg;
+ num_sbits++;
+ }
+
+ if (diff_val != 0) {
+ int diff_neg = (diff_val < 0) ? 1 : 0;
+ if (diff_neg) {
+ diff_val = -diff_val;
+ }
+ symBits = (symBits << 1) | diff_neg;
+ num_sbits++;
+ }
+
+ if (pSym_bits != NULL) {
+ *pSym_bits = symBits;
+ }
+
+ if (sum_val % 2) {
+ data[0] = lav - sum_val / 2;
+ data[1] = lav - diff_val / 2;
+ } else {
+ data[0] = sum_val / 2;
+ data[1] = diff_val / 2;
+ }
+
+ return num_sbits;
+}
+
+static INT ilog2(UINT i) {
+ int l = 0;
+
+ if (i) i--;
+ while (i > 0) {
+ i >>= 1;
+ l++;
+ }
+
+ return l;
+}
+
+static SHORT calc_pcm_bits(const SHORT num_val, const SHORT num_levels) {
+ SHORT num_complete_chunks = 0, rest_chunk_size = 0;
+ SHORT max_grp_len = 0, bits_pcm = 0;
+ int chunk_levels, i;
+
+ switch (num_levels) {
+ case 3:
+ max_grp_len = 5;
+ break;
+ case 6:
+ max_grp_len = 5;
+ break;
+ case 7:
+ max_grp_len = 6;
+ break;
+ case 11:
+ max_grp_len = 2;
+ break;
+ case 13:
+ max_grp_len = 4;
+ break;
+ case 19:
+ max_grp_len = 4;
+ break;
+ case 25:
+ max_grp_len = 3;
+ break;
+ case 51:
+ max_grp_len = 4;
+ break;
+ default:
+ max_grp_len = 1;
+ }
+
+ num_complete_chunks = num_val / max_grp_len;
+ rest_chunk_size = num_val % max_grp_len;
+
+ chunk_levels = 1;
+ for (i = 1; i <= max_grp_len; i++) {
+ chunk_levels *= num_levels;
+ }
+
+ bits_pcm = (SHORT)(ilog2(chunk_levels) * num_complete_chunks);
+ bits_pcm += (SHORT)(ilog2(num_levels) * rest_chunk_size);
+
+ return bits_pcm;
+}
+
+static void apply_pcm_coding(HANDLE_FDK_BITSTREAM strm,
+ const SHORT *const in_data_1,
+ const SHORT *const in_data_2, const SHORT offset,
+ const SHORT num_val, const SHORT num_levels) {
+ SHORT i = 0, j = 0, idx = 0;
+ SHORT max_grp_len = 0, grp_len = 0, next_val = 0;
+ int grp_val = 0, chunk_levels = 0;
+
+ SHORT pcm_chunk_size[7] = {0};
+
+ switch (num_levels) {
+ case 3:
+ max_grp_len = 5;
+ break;
+ case 5:
+ max_grp_len = 3;
+ break;
+ case 6:
+ max_grp_len = 5;
+ break;
+ case 7:
+ max_grp_len = 6;
+ break;
+ case 9:
+ max_grp_len = 5;
+ break;
+ case 11:
+ max_grp_len = 2;
+ break;
+ case 13:
+ max_grp_len = 4;
+ break;
+ case 19:
+ max_grp_len = 4;
+ break;
+ case 25:
+ max_grp_len = 3;
+ break;
+ case 51:
+ max_grp_len = 4;
+ break;
+ default:
+ max_grp_len = 1;
+ }
+
+ chunk_levels = 1;
+ for (i = 1; i <= max_grp_len; i++) {
+ chunk_levels *= num_levels;
+ pcm_chunk_size[i] = ilog2(chunk_levels);
+ }
+
+ for (i = 0; i < num_val; i += max_grp_len) {
+ grp_len = FDKmin(max_grp_len, num_val - i);
+ grp_val = 0;
+ for (j = 0; j < grp_len; j++) {
+ idx = i + j;
+ if (in_data_2 == NULL) {
+ next_val = in_data_1[idx];
+ } else if (in_data_1 == NULL) {
+ next_val = in_data_2[idx];
+ } else {
+ next_val = ((idx % 2) ? in_data_2[idx / 2] : in_data_1[idx / 2]);
+ }
+ next_val += offset;
+ grp_val = grp_val * num_levels + next_val;
+ }
+
+ FDKwriteBits(strm, grp_val, pcm_chunk_size[grp_len]);
+ }
+}
+
+static UINT huff_enc_1D(HANDLE_FDK_BITSTREAM strm, const DATA_TYPE data_type,
+ const INT dim1, SHORT *const in_data,
+ const SHORT num_val, const SHORT p0_flag) {
+ int i, offset = 0;
+ UINT huffBits = 0;
+
+ HUFF_ENTRY part0 = {0};
+ const HUFF_ENTRY *pHuffTab = NULL;
+
+ switch (data_type) {
+ case t_CLD:
+ pHuffTab = fdk_sacenc_huffCLDTab.h1D[dim1];
+ break;
+ case t_ICC:
+ pHuffTab = fdk_sacenc_huffICCTab.h1D[dim1];
+ break;
+ }
+
+ if (p0_flag) {
+ switch (data_type) {
+ case t_CLD:
+ part0 = fdk_sacenc_huffPart0Tab.cld[in_data[0]];
+ break;
+ case t_ICC:
+ part0 = fdk_sacenc_huffPart0Tab.icc[in_data[0]];
+ break;
+ }
+ huffBits += FDKwriteBits(strm, HUFF_VALUE(part0), HUFF_LENGTH(part0));
+ offset = 1;
+ }
+
+ for (i = offset; i < num_val; i++) {
+ int id_sign = 0;
+ int id = in_data[i];
+
+ if (id != 0) {
+ id_sign = 0;
+ if (id < 0) {
+ id = -id;
+ id_sign = 1;
+ }
+ }
+
+ huffBits +=
+ FDKwriteBits(strm, HUFF_VALUE(pHuffTab[id]), HUFF_LENGTH(pHuffTab[id]));
+
+ if (id != 0) {
+ huffBits += FDKwriteBits(strm, id_sign, 1);
+ }
+ } /* for i */
+
+ return huffBits;
+}
+
+static void getHuffEntry(const INT lav, const DATA_TYPE data_type, const INT i,
+ const SHORT tab_idx_2D[2], const SHORT in_data[][2],
+ HUFF_ENTRY *const pEntry, HUFF_ENTRY *const pEscape) {
+ const HUFF_CLD_TAB_2D *pCLD2dTab =
+ &fdk_sacenc_huffCLDTab.h2D[tab_idx_2D[0]][tab_idx_2D[1]];
+ const HUFF_ICC_TAB_2D *pICC2dTab =
+ &fdk_sacenc_huffICCTab.h2D[tab_idx_2D[0]][tab_idx_2D[1]];
+
+ switch (lav) {
+ case 1: {
+ const LAV1_2D *pLav1 = NULL;
+ switch (data_type) {
+ case t_CLD:
+ pLav1 = NULL;
+ break;
+ case t_ICC:
+ pLav1 = &pICC2dTab->lav1;
+ break;
+ }
+ if (pLav1 != NULL) {
+ *pEntry = pLav1->entry[in_data[i][0]][in_data[i][1]];
+ *pEscape = pLav1->escape;
+ }
+ } break;
+ case 3: {
+ const LAV3_2D *pLav3 = NULL;
+ switch (data_type) {
+ case t_CLD:
+ pLav3 = &pCLD2dTab->lav3;
+ break;
+ case t_ICC:
+ pLav3 = &pICC2dTab->lav3;
+ break;
+ }
+ if (pLav3 != NULL) {
+ *pEntry = pLav3->entry[in_data[i][0]][in_data[i][1]];
+ *pEscape = pLav3->escape;
+ }
+ } break;
+ case 5: {
+ const LAV5_2D *pLav5 = NULL;
+ switch (data_type) {
+ case t_CLD:
+ pLav5 = &pCLD2dTab->lav5;
+ break;
+ case t_ICC:
+ pLav5 = &pICC2dTab->lav5;
+ break;
+ }
+ if (pLav5 != NULL) {
+ *pEntry = pLav5->entry[in_data[i][0]][in_data[i][1]];
+ *pEscape = pLav5->escape;
+ }
+ } break;
+ case 7: {
+ const LAV7_2D *pLav7 = NULL;
+ switch (data_type) {
+ case t_CLD:
+ pLav7 = &pCLD2dTab->lav7;
+ break;
+ case t_ICC:
+ pLav7 = &pICC2dTab->lav7;
+ break;
+ }
+ if (pLav7 != NULL) {
+ *pEntry = pLav7->entry[in_data[i][0]][in_data[i][1]];
+ *pEscape = pLav7->escape;
+ }
+ } break;
+ case 9: {
+ const LAV9_2D *pLav9 = NULL;
+ switch (data_type) {
+ case t_CLD:
+ pLav9 = &pCLD2dTab->lav9;
+ break;
+ case t_ICC:
+ pLav9 = NULL;
+ break;
+ }
+ if (pLav9 != NULL) {
+ *pEntry = pLav9->entry[in_data[i][0]][in_data[i][1]];
+ *pEscape = pLav9->escape;
+ }
+ } break;
+ }
+}
+
+static UINT huff_enc_2D(HANDLE_FDK_BITSTREAM strm, const DATA_TYPE data_type,
+ SHORT tab_idx_2D[2], SHORT lav_idx, SHORT in_data[][2],
+ SHORT num_val, SHORT stride, SHORT *p0_data[2]) {
+ SHORT i = 0, lav = 0, num_sbits = 0, sym_bits = 0, escIdx = 0;
+ SHORT esc_data[2][MAXBANDS] = {{0}};
+
+ UINT huffBits = 0;
+
+ const HUFF_ENTRY *pHuffEntry = NULL;
+
+ switch (data_type) {
+ case t_CLD:
+ lav = 2 * lav_idx + 3; /* LAV */
+ pHuffEntry = fdk_sacenc_huffPart0Tab.cld;
+ break;
+ case t_ICC:
+ lav = 2 * lav_idx + 1; /* LAV */
+ pHuffEntry = fdk_sacenc_huffPart0Tab.icc;
+ break;
+ }
+
+ /* Partition 0 */
+ if (p0_data[0] != NULL) {
+ HUFF_ENTRY entry = pHuffEntry[*p0_data[0]];
+ huffBits += FDKwriteBits(strm, HUFF_VALUE(entry), HUFF_LENGTH(entry));
+ }
+ if (p0_data[1] != NULL) {
+ HUFF_ENTRY entry = pHuffEntry[*p0_data[1]];
+ huffBits += FDKwriteBits(strm, HUFF_VALUE(entry), HUFF_LENGTH(entry));
+ }
+
+ for (i = 0; i < num_val; i += stride) {
+ HUFF_ENTRY entry = {0};
+ HUFF_ENTRY escape = {0};
+
+ esc_data[0][escIdx] = in_data[i][0] + lav;
+ esc_data[1][escIdx] = in_data[i][1] + lav;
+
+ num_sbits = sym_check(in_data[i], lav, &sym_bits);
+
+ getHuffEntry(lav, data_type, i, tab_idx_2D, in_data, &entry, &escape);
+
+ huffBits += FDKwriteBits(strm, HUFF_VALUE(entry), HUFF_LENGTH(entry));
+
+ if ((HUFF_VALUE(entry) == HUFF_VALUE(escape)) &&
+ (HUFF_LENGTH(entry) == HUFF_LENGTH(escape))) {
+ escIdx++;
+ } else {
+ huffBits += FDKwriteBits(strm, sym_bits, num_sbits);
+ }
+ } /* for i */
+
+ if (escIdx > 0) {
+ huffBits += calc_pcm_bits(2 * escIdx, (2 * lav + 1));
+ if (strm != NULL) {
+ apply_pcm_coding(strm, esc_data[0], esc_data[1], 0 /*offset*/, 2 * escIdx,
+ (2 * lav + 1));
+ }
+ }
+
+ return huffBits;
+}
+
+static SCHAR get_next_lav_step(const INT lav, const DATA_TYPE data_type) {
+ SCHAR lav_step = 0;
+
+ switch (data_type) {
+ case t_CLD:
+ lav_step = (lav > 9) ? -1 : lav_step_CLD[lav];
+ break;
+ case t_ICC:
+ lav_step = (lav > 7) ? -1 : lav_step_ICC[lav];
+ break;
+ }
+
+ return lav_step;
+}
+
+static INT diff_type_offset(const DIFF_TYPE diff_type) {
+ int offset = 0;
+ switch (diff_type) {
+ case DIFF_FREQ:
+ offset = 0;
+ break;
+ case DIFF_TIME:
+ offset = 2;
+ break;
+ }
+ return offset;
+}
+
+static SHORT calc_huff_bits(SHORT *in_data_1, SHORT *in_data_2,
+ const DATA_TYPE data_type,
+ const DIFF_TYPE diff_type_1,
+ const DIFF_TYPE diff_type_2, const SHORT num_val,
+ SHORT *const lav_idx, SHORT *const cdg_scheme) {
+ SHORT tab_idx_2D[2][2] = {{0}};
+ SHORT tab_idx_1D[2] = {0};
+ SHORT df_rest_flag[2] = {0};
+ SHORT p0_flag[2] = {0};
+
+ SHORT pair_vec[MAXBANDS][2] = {{0}};
+
+ SHORT *p0_data_1[2] = {NULL};
+ SHORT *p0_data_2[2] = {NULL};
+
+ SHORT i = 0;
+ SHORT lav_fp[2] = {0};
+
+ SHORT bit_count_1D = 0;
+ SHORT bit_count_2D_freq = 0;
+ SHORT bit_count_min = 0;
+
+ SHORT num_val_1_short = 0;
+ SHORT num_val_2_short = 0;
+
+ SHORT *in_data_1_short = NULL;
+ SHORT *in_data_2_short = NULL;
+
+ /* 1D Huffman coding */
+ bit_count_1D = 1; /* HUFF_1D */
+
+ num_val_1_short = num_val;
+ num_val_2_short = num_val;
+
+ if (in_data_1 != NULL) {
+ in_data_1_short = in_data_1 + diff_type_offset(diff_type_1);
+ }
+ if (in_data_2 != NULL) {
+ in_data_2_short = in_data_2 + diff_type_offset(diff_type_2);
+ }
+
+ p0_flag[0] = (diff_type_1 == DIFF_FREQ);
+ p0_flag[1] = (diff_type_2 == DIFF_FREQ);
+
+ tab_idx_1D[0] = (diff_type_1 == DIFF_FREQ) ? 0 : 1;
+ tab_idx_1D[1] = (diff_type_2 == DIFF_FREQ) ? 0 : 1;
+
+ if (in_data_1 != NULL) {
+ bit_count_1D += huff_enc_1D(NULL, data_type, tab_idx_1D[0], in_data_1_short,
+ num_val_1_short, p0_flag[0]);
+ }
+ if (in_data_2 != NULL) {
+ bit_count_1D += huff_enc_1D(NULL, data_type, tab_idx_1D[1], in_data_2_short,
+ num_val_2_short, p0_flag[1]);
+ }
+
+ bit_count_min = bit_count_1D;
+ *cdg_scheme = HUFF_1D << PAIR_SHIFT;
+ lav_idx[0] = lav_idx[1] = -1;
+
+ /* Huffman 2D frequency pairs */
+ bit_count_2D_freq = 1; /* HUFF_2D */
+
+ num_val_1_short = num_val;
+ num_val_2_short = num_val;
+
+ if (in_data_1 != NULL) {
+ in_data_1_short = in_data_1 + diff_type_offset(diff_type_1);
+ }
+ if (in_data_2 != NULL) {
+ in_data_2_short = in_data_2 + diff_type_offset(diff_type_2);
+ }
+
+ lav_fp[0] = lav_fp[1] = 0;
+
+ p0_data_1[0] = NULL;
+ p0_data_1[1] = NULL;
+ p0_data_2[0] = NULL;
+ p0_data_2[1] = NULL;
+
+ if (in_data_1 != NULL) {
+ if (diff_type_1 == DIFF_FREQ) {
+ p0_data_1[0] = &in_data_1[0];
+ p0_data_1[1] = NULL;
+
+ num_val_1_short -= 1;
+ in_data_1_short += 1;
+ }
+
+ df_rest_flag[0] = num_val_1_short % 2;
+
+ if (df_rest_flag[0]) num_val_1_short -= 1;
+
+ for (i = 0; i < num_val_1_short - 1; i += 2) {
+ pair_vec[i][0] = in_data_1_short[i];
+ pair_vec[i][1] = in_data_1_short[i + 1];
+
+ lav_fp[0] = FDKmax(lav_fp[0], fAbs(pair_vec[i][0]));
+ lav_fp[0] = FDKmax(lav_fp[0], fAbs(pair_vec[i][1]));
+ }
+
+ tab_idx_2D[0][0] = (diff_type_1 == DIFF_TIME) ? 1 : 0;
+ tab_idx_2D[0][1] = 0;
+
+ tab_idx_1D[0] = (diff_type_1 == DIFF_FREQ) ? 0 : 1;
+
+ lav_fp[0] = get_next_lav_step(lav_fp[0], data_type);
+
+ if (lav_fp[0] != -1) bit_count_2D_freq += lavHuffLen[lav_fp[0]];
+ }
+
+ if (in_data_2 != NULL) {
+ if (diff_type_2 == DIFF_FREQ) {
+ p0_data_2[0] = NULL;
+ p0_data_2[1] = &in_data_2[0];
+
+ num_val_2_short -= 1;
+ in_data_2_short += 1;
+ }
+
+ df_rest_flag[1] = num_val_2_short % 2;
+
+ if (df_rest_flag[1]) num_val_2_short -= 1;
+
+ for (i = 0; i < num_val_2_short - 1; i += 2) {
+ pair_vec[i + 1][0] = in_data_2_short[i];
+ pair_vec[i + 1][1] = in_data_2_short[i + 1];
+
+ lav_fp[1] = FDKmax(lav_fp[1], fAbs(pair_vec[i + 1][0]));
+ lav_fp[1] = FDKmax(lav_fp[1], fAbs(pair_vec[i + 1][1]));
+ }
+
+ tab_idx_2D[1][0] = (diff_type_2 == DIFF_TIME) ? 1 : 0;
+ tab_idx_2D[1][1] = 0;
+
+ tab_idx_1D[1] = (diff_type_2 == DIFF_FREQ) ? 0 : 1;
+
+ lav_fp[1] = get_next_lav_step(lav_fp[1], data_type);
+
+ if (lav_fp[1] != -1) bit_count_2D_freq += lavHuffLen[lav_fp[1]];
+ }
+
+ if ((lav_fp[0] != -1) && (lav_fp[1] != -1)) {
+ if (in_data_1 != NULL) {
+ bit_count_2D_freq +=
+ huff_enc_2D(NULL, data_type, tab_idx_2D[0], lav_fp[0], pair_vec,
+ num_val_1_short, 2, p0_data_1);
+ }
+ if (in_data_2 != NULL) {
+ bit_count_2D_freq +=
+ huff_enc_2D(NULL, data_type, tab_idx_2D[1], lav_fp[1], pair_vec + 1,
+ num_val_2_short, 2, p0_data_2);
+ }
+ if (in_data_1 != NULL) {
+ if (df_rest_flag[0])
+ bit_count_2D_freq +=
+ huff_enc_1D(NULL, data_type, tab_idx_1D[0],
+ in_data_1_short + num_val_1_short, 1, 0);
+ }
+ if (in_data_2 != NULL) {
+ if (df_rest_flag[1])
+ bit_count_2D_freq +=
+ huff_enc_1D(NULL, data_type, tab_idx_1D[1],
+ in_data_2_short + num_val_2_short, 1, 0);
+ }
+
+ if (bit_count_2D_freq < bit_count_min) {
+ bit_count_min = bit_count_2D_freq;
+ *cdg_scheme = HUFF_2D << PAIR_SHIFT | FREQ_PAIR;
+ lav_idx[0] = lav_fp[0];
+ lav_idx[1] = lav_fp[1];
+ }
+ }
+
+ return bit_count_min;
+}
+
+static void apply_huff_coding(HANDLE_FDK_BITSTREAM strm, SHORT *const in_data_1,
+ SHORT *const in_data_2, const DATA_TYPE data_type,
+ const DIFF_TYPE diff_type_1,
+ const DIFF_TYPE diff_type_2, const SHORT num_val,
+ const SHORT *const lav_idx,
+ const SHORT cdg_scheme) {
+ SHORT tab_idx_2D[2][2] = {{0}};
+ SHORT tab_idx_1D[2] = {0};
+ SHORT df_rest_flag[2] = {0};
+ SHORT p0_flag[2] = {0};
+
+ SHORT pair_vec[MAXBANDS][2] = {{0}};
+
+ SHORT *p0_data_1[2] = {NULL};
+ SHORT *p0_data_2[2] = {NULL};
+
+ SHORT i = 0;
+
+ SHORT num_val_1_short = num_val;
+ SHORT num_val_2_short = num_val;
+
+ SHORT *in_data_1_short = NULL;
+ SHORT *in_data_2_short = NULL;
+
+ /* Offset */
+ if (in_data_1 != NULL) {
+ in_data_1_short = in_data_1 + diff_type_offset(diff_type_1);
+ }
+ if (in_data_2 != NULL) {
+ in_data_2_short = in_data_2 + diff_type_offset(diff_type_2);
+ }
+
+ /* Signalize coding scheme */
+ FDKwriteBits(strm, cdg_scheme >> PAIR_SHIFT, 1);
+
+ switch (cdg_scheme >> PAIR_SHIFT) {
+ case HUFF_1D:
+
+ p0_flag[0] = (diff_type_1 == DIFF_FREQ);
+ p0_flag[1] = (diff_type_2 == DIFF_FREQ);
+
+ tab_idx_1D[0] = (diff_type_1 == DIFF_FREQ) ? 0 : 1;
+ tab_idx_1D[1] = (diff_type_2 == DIFF_FREQ) ? 0 : 1;
+
+ if (in_data_1 != NULL) {
+ huff_enc_1D(strm, data_type, tab_idx_1D[0], in_data_1_short,
+ num_val_1_short, p0_flag[0]);
+ }
+ if (in_data_2 != NULL) {
+ huff_enc_1D(strm, data_type, tab_idx_1D[1], in_data_2_short,
+ num_val_2_short, p0_flag[1]);
+ }
+ break; /* HUFF_1D */
+
+ case HUFF_2D:
+
+ switch (cdg_scheme & PAIR_MASK) {
+ case FREQ_PAIR:
+
+ if (in_data_1 != NULL) {
+ if (diff_type_1 == DIFF_FREQ) {
+ p0_data_1[0] = &in_data_1[0];
+ p0_data_1[1] = NULL;
+
+ num_val_1_short -= 1;
+ in_data_1_short += 1;
+ }
+
+ df_rest_flag[0] = num_val_1_short % 2;
+
+ if (df_rest_flag[0]) num_val_1_short -= 1;
+
+ for (i = 0; i < num_val_1_short - 1; i += 2) {
+ pair_vec[i][0] = in_data_1_short[i];
+ pair_vec[i][1] = in_data_1_short[i + 1];
+ }
+
+ tab_idx_2D[0][0] = (diff_type_1 == DIFF_TIME) ? 1 : 0;
+ tab_idx_2D[0][1] = 0;
+
+ tab_idx_1D[0] = (diff_type_1 == DIFF_FREQ) ? 0 : 1;
+ } /* if( in_data_1 != NULL ) */
+
+ if (in_data_2 != NULL) {
+ if (diff_type_2 == DIFF_FREQ) {
+ p0_data_2[0] = NULL;
+ p0_data_2[1] = &in_data_2[0];
+
+ num_val_2_short -= 1;
+ in_data_2_short += 1;
+ }
+
+ df_rest_flag[1] = num_val_2_short % 2;
+
+ if (df_rest_flag[1]) num_val_2_short -= 1;
+
+ for (i = 0; i < num_val_2_short - 1; i += 2) {
+ pair_vec[i + 1][0] = in_data_2_short[i];
+ pair_vec[i + 1][1] = in_data_2_short[i + 1];
+ }
+
+ tab_idx_2D[1][0] = (diff_type_2 == DIFF_TIME) ? 1 : 0;
+ tab_idx_2D[1][1] = 0;
+
+ tab_idx_1D[1] = (diff_type_2 == DIFF_FREQ) ? 0 : 1;
+ } /* if( in_data_2 != NULL ) */
+
+ if (in_data_1 != NULL) {
+ FDKwriteBits(strm, lavHuffVal[lav_idx[0]], lavHuffLen[lav_idx[0]]);
+ huff_enc_2D(strm, data_type, tab_idx_2D[0], lav_idx[0], pair_vec,
+ num_val_1_short, 2, p0_data_1);
+ if (df_rest_flag[0]) {
+ huff_enc_1D(strm, data_type, tab_idx_1D[0],
+ in_data_1_short + num_val_1_short, 1, 0);
+ }
+ }
+ if (in_data_2 != NULL) {
+ FDKwriteBits(strm, lavHuffVal[lav_idx[1]], lavHuffLen[lav_idx[1]]);
+ huff_enc_2D(strm, data_type, tab_idx_2D[1], lav_idx[1],
+ pair_vec + 1, num_val_2_short, 2, p0_data_2);
+ if (df_rest_flag[1]) {
+ huff_enc_1D(strm, data_type, tab_idx_1D[1],
+ in_data_2_short + num_val_2_short, 1, 0);
+ }
+ }
+ break; /* FREQ_PAIR */
+
+ case TIME_PAIR:
+
+ if ((diff_type_1 == DIFF_FREQ) || (diff_type_2 == DIFF_FREQ)) {
+ p0_data_1[0] = &in_data_1[0];
+ p0_data_1[1] = &in_data_2[0];
+
+ in_data_1_short += 1;
+ in_data_2_short += 1;
+
+ num_val_1_short -= 1;
+ }
+
+ for (i = 0; i < num_val_1_short; i++) {
+ pair_vec[i][0] = in_data_1_short[i];
+ pair_vec[i][1] = in_data_2_short[i];
+ }
+
+ tab_idx_2D[0][0] =
+ ((diff_type_1 == DIFF_TIME) || (diff_type_2 == DIFF_TIME)) ? 1
+ : 0;
+ tab_idx_2D[0][1] = 1;
+
+ FDKwriteBits(strm, lavHuffVal[lav_idx[0]], lavHuffLen[lav_idx[0]]);
+
+ huff_enc_2D(strm, data_type, tab_idx_2D[0], lav_idx[0], pair_vec,
+ num_val_1_short, 1, p0_data_1);
+
+ break; /* TIME_PAIR */
+ } /* switch( cdg_scheme & PAIR_MASK ) */
+
+ break; /* HUFF_2D */
+
+ default:
+ break;
+ } /* switch( cdg_scheme >> PAIR_SHIFT ) */
+}
+
+INT fdk_sacenc_ecDataPairEnc(HANDLE_FDK_BITSTREAM strm,
+ SHORT aaInData[][MAXBANDS],
+ SHORT aHistory[MAXBANDS],
+ const DATA_TYPE data_type, const INT setIdx,
+ const INT startBand, const INT dataBands,
+ const INT coarse_flag,
+ const INT independency_flag) {
+ SHORT reset = 0, pb = 0;
+ SHORT quant_levels = 0, quant_offset = 0, num_pcm_val = 0;
+
+ SHORT splitLsb_flag = 0;
+ SHORT pcmCoding_flag = 0;
+
+ SHORT allowDiffTimeBack_flag = !independency_flag || (setIdx > 0);
+
+ SHORT num_lsb_bits = -1;
+ SHORT num_pcm_bits = -1;
+
+ SHORT quant_data_lsb[2][MAXBANDS];
+ SHORT quant_data_msb[2][MAXBANDS];
+
+ SHORT quant_data_hist_lsb[MAXBANDS];
+ SHORT quant_data_hist_msb[MAXBANDS];
+
+ SHORT data_diff_freq[2][MAXBANDS];
+ SHORT data_diff_time[2][MAXBANDS + 2];
+
+ SHORT *p_quant_data_msb[2];
+ SHORT *p_quant_data_hist_msb = NULL;
+
+ SHORT min_bits_all = 0;
+ SHORT min_found = 0;
+
+ SHORT min_bits_df_df = -1;
+ SHORT min_bits_df_dt = -1;
+ SHORT min_bits_dtbw_df = -1;
+ SHORT min_bits_dt_dt = -1;
+
+ SHORT lav_df_df[2] = {-1, -1};
+ SHORT lav_df_dt[2] = {-1, -1};
+ SHORT lav_dtbw_df[2] = {-1, -1};
+ SHORT lav_dt_dt[2] = {-1, -1};
+
+ SHORT coding_scheme_df_df = 0;
+ SHORT coding_scheme_df_dt = 0;
+ SHORT coding_scheme_dtbw_df = 0;
+ SHORT coding_scheme_dt_dt = 0;
+
+ switch (data_type) {
+ case t_CLD:
+ if (coarse_flag) {
+ splitLsb_flag = 0;
+ quant_levels = 15;
+ quant_offset = 7;
+ } else {
+ splitLsb_flag = 0;
+ quant_levels = 31;
+ quant_offset = 15;
+ }
+ break;
+ case t_ICC:
+ if (coarse_flag) {
+ splitLsb_flag = 0;
+ quant_levels = 4;
+ quant_offset = 0;
+ } else {
+ splitLsb_flag = 0;
+ quant_levels = 8;
+ quant_offset = 0;
+ }
+ break;
+ } /* switch( data_type ) */
+
+ /* Split off LSB */
+ if (splitLsb_flag) {
+ split_lsb(aaInData[setIdx] + startBand, quant_offset, dataBands,
+ quant_data_lsb[0], quant_data_msb[0]);
+
+ split_lsb(aaInData[setIdx + 1] + startBand, quant_offset, dataBands,
+ quant_data_lsb[1], quant_data_msb[1]);
+
+ p_quant_data_msb[0] = quant_data_msb[0];
+ p_quant_data_msb[1] = quant_data_msb[1];
+
+ num_lsb_bits = 2 * dataBands;
+ } else if (quant_offset != 0) {
+ for (pb = 0; pb < dataBands; pb++) {
+ quant_data_msb[0][pb] = aaInData[setIdx][startBand + pb] + quant_offset;
+ quant_data_msb[1][pb] =
+ aaInData[setIdx + 1][startBand + pb] + quant_offset;
+ }
+
+ p_quant_data_msb[0] = quant_data_msb[0];
+ p_quant_data_msb[1] = quant_data_msb[1];
+
+ num_lsb_bits = 0;
+ } else {
+ p_quant_data_msb[0] = aaInData[setIdx] + startBand;
+ p_quant_data_msb[1] = aaInData[setIdx + 1] + startBand;
+
+ num_lsb_bits = 0;
+ }
+
+ if (allowDiffTimeBack_flag) {
+ if (splitLsb_flag) {
+ split_lsb(aHistory + startBand, quant_offset, dataBands,
+ quant_data_hist_lsb, quant_data_hist_msb);
+
+ p_quant_data_hist_msb = quant_data_hist_msb;
+ } else if (quant_offset != 0) {
+ for (pb = 0; pb < dataBands; pb++) {
+ quant_data_hist_msb[pb] = aHistory[startBand + pb] + quant_offset;
+ }
+ p_quant_data_hist_msb = quant_data_hist_msb;
+ } else {
+ p_quant_data_hist_msb = aHistory + startBand;
+ }
+ }
+
+ /* Calculate frequency differences */
+ calc_diff_freq(p_quant_data_msb[0], data_diff_freq[0], dataBands);
+
+ calc_diff_freq(p_quant_data_msb[1], data_diff_freq[1], dataBands);
+
+ /* Calculate time differences */
+ if (allowDiffTimeBack_flag) {
+ calc_diff_time(p_quant_data_msb[0], p_quant_data_hist_msb,
+ data_diff_time[0], dataBands);
+ }
+
+ calc_diff_time(p_quant_data_msb[1], p_quant_data_msb[0], data_diff_time[1],
+ dataBands);
+
+ /* Calculate coding scheme with minumum bit consumption */
+
+ /**********************************************************/
+ num_pcm_bits = calc_pcm_bits(2 * dataBands, quant_levels);
+ num_pcm_val = 2 * dataBands;
+
+ /**********************************************************/
+
+ min_bits_all = num_pcm_bits;
+
+ /**********************************************************/
+ /**********************************************************/
+
+ /**********************************************************/
+ min_bits_df_df =
+ calc_huff_bits(data_diff_freq[0], data_diff_freq[1], data_type, DIFF_FREQ,
+ DIFF_FREQ, dataBands, lav_df_df, &coding_scheme_df_df);
+
+ min_bits_df_df += 2;
+
+ min_bits_df_df += num_lsb_bits;
+
+ if (min_bits_df_df < min_bits_all) {
+ min_bits_all = min_bits_df_df;
+ }
+ /**********************************************************/
+
+ /**********************************************************/
+ min_bits_df_dt =
+ calc_huff_bits(data_diff_freq[0], data_diff_time[1], data_type, DIFF_FREQ,
+ DIFF_TIME, dataBands, lav_df_dt, &coding_scheme_df_dt);
+
+ min_bits_df_dt += 2;
+
+ min_bits_df_dt += num_lsb_bits;
+
+ if (min_bits_df_dt < min_bits_all) {
+ min_bits_all = min_bits_df_dt;
+ }
+ /**********************************************************/
+
+ /**********************************************************/
+ /**********************************************************/
+
+ if (allowDiffTimeBack_flag) {
+ /**********************************************************/
+ min_bits_dtbw_df = calc_huff_bits(
+ data_diff_time[0], data_diff_freq[1], data_type, DIFF_TIME, DIFF_FREQ,
+ dataBands, lav_dtbw_df, &coding_scheme_dtbw_df);
+
+ min_bits_dtbw_df += 2;
+
+ min_bits_dtbw_df += num_lsb_bits;
+
+ if (min_bits_dtbw_df < min_bits_all) {
+ min_bits_all = min_bits_dtbw_df;
+ }
+ /**********************************************************/
+
+ /**********************************************************/
+ min_bits_dt_dt = calc_huff_bits(data_diff_time[0], data_diff_time[1],
+ data_type, DIFF_TIME, DIFF_TIME, dataBands,
+ lav_dt_dt, &coding_scheme_dt_dt);
+
+ min_bits_dt_dt += 2;
+
+ min_bits_dt_dt += num_lsb_bits;
+
+ if (min_bits_dt_dt < min_bits_all) {
+ min_bits_all = min_bits_dt_dt;
+ }
+ /**********************************************************/
+
+ } /* if( allowDiffTimeBack_flag ) */
+
+ /***************************/
+ /* Start actual coding now */
+ /***************************/
+
+ /* PCM or Diff/Huff Coding? */
+ pcmCoding_flag = (min_bits_all == num_pcm_bits);
+
+ FDKwriteBits(strm, pcmCoding_flag, 1);
+
+ if (pcmCoding_flag) {
+ /* Grouped PCM Coding */
+ apply_pcm_coding(strm, aaInData[setIdx] + startBand,
+ aaInData[setIdx + 1] + startBand, quant_offset,
+ num_pcm_val, quant_levels);
+ } else {
+ /* Diff/Huff Coding */
+
+ min_found = 0;
+
+ /*******************************************/
+ if (min_bits_all == min_bits_df_df) {
+ FDKwriteBits(strm, DIFF_FREQ, 1);
+ FDKwriteBits(strm, DIFF_FREQ, 1);
+
+ apply_huff_coding(strm, data_diff_freq[0], data_diff_freq[1], data_type,
+ DIFF_FREQ, DIFF_FREQ, dataBands, lav_df_df,
+ coding_scheme_df_df);
+
+ min_found = 1;
+ }
+ /*******************************************/
+
+ /*******************************************/
+ if (!min_found && (min_bits_all == min_bits_df_dt)) {
+ FDKwriteBits(strm, DIFF_FREQ, 1);
+ FDKwriteBits(strm, DIFF_TIME, 1);
+
+ apply_huff_coding(strm, data_diff_freq[0], data_diff_time[1], data_type,
+ DIFF_FREQ, DIFF_TIME, dataBands, lav_df_dt,
+ coding_scheme_df_dt);
+
+ min_found = 1;
+ }
+ /*******************************************/
+
+ /*******************************************/
+ /*******************************************/
+
+ if (allowDiffTimeBack_flag) {
+ /*******************************************/
+ if (!min_found && (min_bits_all == min_bits_dtbw_df)) {
+ FDKwriteBits(strm, DIFF_TIME, 1);
+ FDKwriteBits(strm, DIFF_FREQ, 1);
+
+ apply_huff_coding(strm, data_diff_time[0], data_diff_freq[1], data_type,
+ DIFF_TIME, DIFF_FREQ, dataBands, lav_dtbw_df,
+ coding_scheme_dtbw_df);
+
+ min_found = 1;
+ }
+ /*******************************************/
+
+ /*******************************************/
+ if (!min_found && (min_bits_all == min_bits_dt_dt)) {
+ FDKwriteBits(strm, DIFF_TIME, 1);
+ FDKwriteBits(strm, DIFF_TIME, 1);
+
+ apply_huff_coding(strm, data_diff_time[0], data_diff_time[1], data_type,
+ DIFF_TIME, DIFF_TIME, dataBands, lav_dt_dt,
+ coding_scheme_dt_dt);
+ }
+ /*******************************************/
+
+ } /* if( allowDiffTimeBack_flag ) */
+
+ /* LSB coding */
+ if (splitLsb_flag) {
+ apply_lsb_coding(strm, quant_data_lsb[0], 1, dataBands);
+
+ apply_lsb_coding(strm, quant_data_lsb[1], 1, dataBands);
+ }
+
+ } /* Diff/Huff/LSB coding */
+
+ return reset;
+}
+
+INT fdk_sacenc_ecDataSingleEnc(HANDLE_FDK_BITSTREAM strm,
+ SHORT aaInData[][MAXBANDS],
+ SHORT aHistory[MAXBANDS],
+ const DATA_TYPE data_type, const INT setIdx,
+ const INT startBand, const INT dataBands,
+ const INT coarse_flag,
+ const INT independency_flag) {
+ SHORT reset = 0, pb = 0;
+ SHORT quant_levels = 0, quant_offset = 0, num_pcm_val = 0;
+
+ SHORT splitLsb_flag = 0;
+ SHORT pcmCoding_flag = 0;
+
+ SHORT allowDiffTimeBack_flag = !independency_flag || (setIdx > 0);
+
+ SHORT num_lsb_bits = -1;
+ SHORT num_pcm_bits = -1;
+
+ SHORT quant_data_lsb[MAXBANDS];
+ SHORT quant_data_msb[MAXBANDS];
+
+ SHORT quant_data_hist_lsb[MAXBANDS];
+ SHORT quant_data_hist_msb[MAXBANDS];
+
+ SHORT data_diff_freq[MAXBANDS];
+ SHORT data_diff_time[MAXBANDS + 2];
+
+ SHORT *p_quant_data_msb;
+ SHORT *p_quant_data_hist_msb = NULL;
+
+ SHORT min_bits_all = 0;
+ SHORT min_found = 0;
+
+ SHORT min_bits_df = -1;
+ SHORT min_bits_dt = -1;
+
+ SHORT lav_df[2] = {-1, -1};
+ SHORT lav_dt[2] = {-1, -1};
+
+ SHORT coding_scheme_df = 0;
+ SHORT coding_scheme_dt = 0;
+
+ switch (data_type) {
+ case t_CLD:
+ if (coarse_flag) {
+ splitLsb_flag = 0;
+ quant_levels = 15;
+ quant_offset = 7;
+ } else {
+ splitLsb_flag = 0;
+ quant_levels = 31;
+ quant_offset = 15;
+ }
+ break;
+ case t_ICC:
+ if (coarse_flag) {
+ splitLsb_flag = 0;
+ quant_levels = 4;
+ quant_offset = 0;
+ } else {
+ splitLsb_flag = 0;
+ quant_levels = 8;
+ quant_offset = 0;
+ }
+ break;
+ } /* switch( data_type ) */
+
+ /* Split off LSB */
+ if (splitLsb_flag) {
+ split_lsb(aaInData[setIdx] + startBand, quant_offset, dataBands,
+ quant_data_lsb, quant_data_msb);
+
+ p_quant_data_msb = quant_data_msb;
+ num_lsb_bits = dataBands;
+ } else if (quant_offset != 0) {
+ for (pb = 0; pb < dataBands; pb++) {
+ quant_data_msb[pb] = aaInData[setIdx][startBand + pb] + quant_offset;
+ }
+
+ p_quant_data_msb = quant_data_msb;
+ num_lsb_bits = 0;
+ } else {
+ p_quant_data_msb = aaInData[setIdx] + startBand;
+ num_lsb_bits = 0;
+ }
+
+ if (allowDiffTimeBack_flag) {
+ if (splitLsb_flag) {
+ split_lsb(aHistory + startBand, quant_offset, dataBands,
+ quant_data_hist_lsb, quant_data_hist_msb);
+
+ p_quant_data_hist_msb = quant_data_hist_msb;
+ } else if (quant_offset != 0) {
+ for (pb = 0; pb < dataBands; pb++) {
+ quant_data_hist_msb[pb] = aHistory[startBand + pb] + quant_offset;
+ }
+ p_quant_data_hist_msb = quant_data_hist_msb;
+ } else {
+ p_quant_data_hist_msb = aHistory + startBand;
+ }
+ }
+
+ /* Calculate frequency differences */
+ calc_diff_freq(p_quant_data_msb, data_diff_freq, dataBands);
+
+ /* Calculate time differences */
+ if (allowDiffTimeBack_flag) {
+ calc_diff_time(p_quant_data_msb, p_quant_data_hist_msb, data_diff_time,
+ dataBands);
+ }
+
+ /* Calculate coding scheme with minumum bit consumption */
+
+ /**********************************************************/
+ num_pcm_bits = calc_pcm_bits(dataBands, quant_levels);
+ num_pcm_val = dataBands;
+
+ /**********************************************************/
+
+ min_bits_all = num_pcm_bits;
+
+ /**********************************************************/
+ /**********************************************************/
+
+ /**********************************************************/
+ min_bits_df = calc_huff_bits(data_diff_freq, NULL, data_type, DIFF_FREQ,
+ DIFF_FREQ, dataBands, lav_df, &coding_scheme_df);
+
+ if (allowDiffTimeBack_flag) min_bits_df += 1;
+
+ min_bits_df += num_lsb_bits;
+
+ if (min_bits_df < min_bits_all) {
+ min_bits_all = min_bits_df;
+ }
+ /**********************************************************/
+
+ /**********************************************************/
+ if (allowDiffTimeBack_flag) {
+ min_bits_dt =
+ calc_huff_bits(data_diff_time, NULL, data_type, DIFF_TIME, DIFF_TIME,
+ dataBands, lav_dt, &coding_scheme_dt);
+
+ min_bits_dt += 1;
+ min_bits_dt += num_lsb_bits;
+
+ if (min_bits_dt < min_bits_all) {
+ min_bits_all = min_bits_dt;
+ }
+ } /* if( allowDiffTimeBack_flag ) */
+
+ /***************************/
+ /* Start actual coding now */
+ /***************************/
+
+ /* PCM or Diff/Huff Coding? */
+ pcmCoding_flag = (min_bits_all == num_pcm_bits);
+
+ FDKwriteBits(strm, pcmCoding_flag, 1);
+
+ if (pcmCoding_flag) {
+ /* Grouped PCM Coding */
+ apply_pcm_coding(strm, aaInData[setIdx] + startBand, NULL, quant_offset,
+ num_pcm_val, quant_levels);
+ } else {
+ /* Diff/Huff Coding */
+
+ min_found = 0;
+
+ /*******************************************/
+ if (min_bits_all == min_bits_df) {
+ if (allowDiffTimeBack_flag) {
+ FDKwriteBits(strm, DIFF_FREQ, 1);
+ }
+
+ apply_huff_coding(strm, data_diff_freq, NULL, data_type, DIFF_FREQ,
+ DIFF_FREQ, dataBands, lav_df, coding_scheme_df);
+
+ min_found = 1;
+ } /* if( min_bits_all == min_bits_df ) */
+ /*******************************************/
+
+ /*******************************************/
+ if (allowDiffTimeBack_flag) {
+ /*******************************************/
+ if (!min_found && (min_bits_all == min_bits_dt)) {
+ FDKwriteBits(strm, DIFF_TIME, 1);
+
+ apply_huff_coding(strm, data_diff_time, NULL, data_type, DIFF_TIME,
+ DIFF_TIME, dataBands, lav_dt, coding_scheme_dt);
+ }
+ /*******************************************/
+
+ } /* if( allowDiffTimeBack_flag ) */
+
+ /* LSB coding */
+ if (splitLsb_flag) {
+ apply_lsb_coding(strm, quant_data_lsb, 1, dataBands);
+ }
+
+ } /* Diff/Huff/LSB coding */
+
+ return reset;
+}