/******************************************************************************
 *
 *  Copyright (C) 1999-2012 Broadcom Corporation
 *
 *  Licensed under the Apache License, Version 2.0 (the "License");
 *  you may not use this file except in compliance with the License.
 *  You may obtain a copy of the License at:
 *
 *  http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing, software
 *  distributed under the License is distributed on an "AS IS" BASIS,
 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *  See the License for the specific language governing permissions and
 *  limitations under the License.
 *
 ******************************************************************************/

/******************************************************************************
 *
 *  This file contains the code for bit allocation algorithm. It calculates
 *  the number of bits required for the encoded stream of data.
 *
 ******************************************************************************/

/*Includes*/
#include "sbc_enc_func_declare.h"
#include "sbc_encoder.h"

/*global arrays*/
extern const int16_t sbc_enc_as16Offset4[4][4];
extern const int16_t sbc_enc_as16Offset8[4][8];

/****************************************************************************
* BitAlloc - Calculates the required number of bits for the given scale factor
* and the number of subbands.
*
* RETURNS : N/A
*/

void sbc_enc_bit_alloc_ste(SBC_ENC_PARAMS* pstrCodecParams) {
  /* CAUTIOM -> mips optim for arm 32 require to use int32_t instead of int16_t
   */
  /* Do not change variable type or name */
  int32_t s32MaxBitNeed; /*to store the max bits needed per sb*/
  int32_t s32BitCount;   /*the used number of bits*/
  int32_t s32SliceCount; /*to store hwo many slices can be put in bitpool*/
  int32_t s32BitSlice;   /*number of bitslices in bitpool*/
  int32_t s32Sb;         /*counter for sub-band*/
  int32_t s32Ch;         /*counter for channel*/
  int16_t* ps16BitNeed;  /*temp memory to store required number of bits*/
  int32_t s32Loudness;   /*used in Loudness calculation*/
  int16_t *ps16GenBufPtr, *pas16ScaleFactor;
  int16_t* ps16GenArrPtr;
  int16_t* ps16GenTabPtr;
  int32_t s32NumOfSubBands = pstrCodecParams->s16NumOfSubBands;
  int32_t s32BitPool = pstrCodecParams->s16BitPool;

  /* bitneed values are derived from scale factor */
  if (pstrCodecParams->s16AllocationMethod == SBC_SNR) {
    ps16BitNeed = pstrCodecParams->as16ScaleFactor;
    s32MaxBitNeed = pstrCodecParams->s16MaxBitNeed;
  } else {
    ps16BitNeed = pstrCodecParams->s16ScartchMemForBitAlloc;
    pas16ScaleFactor = pstrCodecParams->as16ScaleFactor;
    s32MaxBitNeed = 0;
    ps16GenBufPtr = ps16BitNeed;
    for (s32Ch = 0; s32Ch < 2; s32Ch++) {
      if (s32NumOfSubBands == 4) {
        ps16GenTabPtr =
            (int16_t*)sbc_enc_as16Offset4[pstrCodecParams->s16SamplingFreq];
      } else {
        ps16GenTabPtr =
            (int16_t*)sbc_enc_as16Offset8[pstrCodecParams->s16SamplingFreq];
      }

      for (s32Sb = 0; s32Sb < s32NumOfSubBands; s32Sb++) {
        if (*pas16ScaleFactor == 0)
          *ps16GenBufPtr = -5;
        else {
          s32Loudness = (int32_t)(*pas16ScaleFactor - *ps16GenTabPtr);

          if (s32Loudness > 0)
            *ps16GenBufPtr = (int16_t)(s32Loudness >> 1);
          else
            *ps16GenBufPtr = (int16_t)s32Loudness;
        }

        if (*ps16GenBufPtr > s32MaxBitNeed) s32MaxBitNeed = *ps16GenBufPtr;
        pas16ScaleFactor++;
        ps16GenBufPtr++;
        ps16GenTabPtr++;
      }
    }
  }

  /* iterative process to find out hwo many bitslices fit into the bitpool */
  s32BitSlice = s32MaxBitNeed + 1;
  s32BitCount = s32BitPool;
  s32SliceCount = 0;
  do {
    s32BitSlice--;
    s32BitCount -= s32SliceCount;
    s32SliceCount = 0;
    ps16GenBufPtr = ps16BitNeed;

    for (s32Sb = 0; s32Sb < 2 * s32NumOfSubBands; s32Sb++) {
      if ((*ps16GenBufPtr >= s32BitSlice + 1) &&
          (*ps16GenBufPtr < s32BitSlice + 16)) {
        if (*(ps16GenBufPtr) == s32BitSlice + 1)
          s32SliceCount += 2;
        else
          s32SliceCount++;
      }
      ps16GenBufPtr++;
    }
  } while (s32BitCount - s32SliceCount > 0);

  if (s32BitCount - s32SliceCount == 0) {
    s32BitCount -= s32SliceCount;
    s32BitSlice--;
  }

  /* Bits are distributed until the last bitslice is reached */
  ps16GenBufPtr = ps16BitNeed;
  ps16GenArrPtr = pstrCodecParams->as16Bits;
  for (s32Ch = 0; s32Ch < 2; s32Ch++) {
    for (s32Sb = 0; s32Sb < s32NumOfSubBands; s32Sb++) {
      if (*ps16GenBufPtr < s32BitSlice + 2)
        *ps16GenArrPtr = 0;
      else
        *ps16GenArrPtr = ((*(ps16GenBufPtr)-s32BitSlice) < 16)
                             ? (int16_t)(*(ps16GenBufPtr)-s32BitSlice)
                             : 16;
      ps16GenBufPtr++;
      ps16GenArrPtr++;
    }
  }

  /* the remaining bits are allocated starting at subband 0 */
  s32Ch = 0;
  s32Sb = 0;
  ps16GenBufPtr = ps16BitNeed;
  ps16GenArrPtr -= 2 * s32NumOfSubBands;

  while ((s32BitCount > 0) && (s32Sb < s32NumOfSubBands)) {
    if ((*(ps16GenArrPtr) >= 2) && (*(ps16GenArrPtr) < 16)) {
      (*(ps16GenArrPtr))++;
      s32BitCount--;
    } else if ((*ps16GenBufPtr == s32BitSlice + 1) && (s32BitCount > 1)) {
      *(ps16GenArrPtr) = 2;
      s32BitCount -= 2;
    }
    if (s32Ch == 1) {
      s32Ch = 0;
      s32Sb++;
      ps16GenBufPtr = ps16BitNeed + s32Sb;
      ps16GenArrPtr = pstrCodecParams->as16Bits + s32Sb;

    } else {
      s32Ch = 1;
      ps16GenBufPtr = ps16BitNeed + s32NumOfSubBands + s32Sb;
      ps16GenArrPtr = pstrCodecParams->as16Bits + s32NumOfSubBands + s32Sb;
    }
  }

  s32Ch = 0;
  s32Sb = 0;
  ps16GenArrPtr = pstrCodecParams->as16Bits;

  while ((s32BitCount > 0) && (s32Sb < s32NumOfSubBands)) {
    if (*(ps16GenArrPtr) < 16) {
      (*(ps16GenArrPtr))++;
      s32BitCount--;
    }
    if (s32Ch == 1) {
      s32Ch = 0;
      s32Sb++;
      ps16GenArrPtr = pstrCodecParams->as16Bits + s32Sb;
    } else {
      s32Ch = 1;
      ps16GenArrPtr = pstrCodecParams->as16Bits + s32NumOfSubBands + s32Sb;
    }
  }
}

/*End of BitAlloc() function*/