C++程序  |  169行  |  5.92 KB

/******************************************************************************
 *
 * Copyright (C) 2018 The Android Open Source Project
 *
 * 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.
 *
 *****************************************************************************
 * Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
*/
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include "impd_type_def.h"
#include "impd_error_standards.h"
#include "impd_drc_extr_delta_coded_info.h"
#include "impd_drc_common.h"
#include "impd_drc_struct.h"
#include "impd_drc_filter_bank.h"
#include "impd_drc_multi_band.h"
#include "impd_drc_rom.h"

IA_ERRORCODE impd_fcenter_norm_sb_init(WORD32 num_subbands,
                                       FLOAT32* fcenter_norm_subband) {
  WORD32 s;
  for (s = 0; s < num_subbands; s++) {
    fcenter_norm_subband[s] = (s + 0.5f) / (2.0f * num_subbands);
  }
  return (0);
}

IA_ERRORCODE impd_generate_slope(WORD32 num_sub_bands,
                                 FLOAT32* fcenter_norm_subband,
                                 FLOAT32 fcross_norm_lo, FLOAT32 fcross_norm_hi,
                                 FLOAT32* response) {
  WORD32 i;
  FLOAT32 filter_slope = -24.0f;
  FLOAT32 inv_log10_2 = 3.32192809f;
  FLOAT32 norm = 0.05f * filter_slope * inv_log10_2;

  for (i = 0; i < num_sub_bands; i++) {
    if (fcenter_norm_subband[i] < fcross_norm_lo) {
      response[i] = (FLOAT32)pow(
          10.0, norm * log10(fcross_norm_lo / fcenter_norm_subband[i]));
    } else if (fcenter_norm_subband[i] < fcross_norm_hi) {
      response[i] = 1.0f;
    } else {
      response[i] = (FLOAT32)pow(
          10.0, norm * log10(fcenter_norm_subband[i] / fcross_norm_hi));
    }
  }
  return (0);
}

IA_ERRORCODE impd_generate_overlap_weights(
    WORD32 num_drc_bands, WORD32 drc_band_type,
    ia_gain_params_struct* gain_params, WORD32 dec_subband_count,
    ia_group_overlap_params_struct* pstr_group_overlap_params) {
  FLOAT32 fcenter_norm_subband[AUDIO_CODEC_SUBBAND_COUNT_MAX];
  FLOAT32 w_norm[AUDIO_CODEC_SUBBAND_COUNT_MAX];
  FLOAT32 fcross_norm_lo, fcross_norm_hi;
  WORD32 err, b, s, start_subband_index = 0, stop_sub_band_index = 0;
  err = impd_fcenter_norm_sb_init(dec_subband_count, fcenter_norm_subband);

  if (drc_band_type == 1) {
    fcross_norm_lo = 0.0f;
    for (b = 0; b < num_drc_bands; b++) {
      if (b < num_drc_bands - 1) {
        fcross_norm_hi =
            normal_cross_freq[gain_params[b + 1].crossover_freq_idx]
                .f_cross_norm;
      } else {
        fcross_norm_hi = 0.5f;
      }
      impd_generate_slope(
          dec_subband_count, fcenter_norm_subband, fcross_norm_lo,
          fcross_norm_hi,
          pstr_group_overlap_params->str_band_overlap_params[b].overlap_weight);

      fcross_norm_lo = fcross_norm_hi;
    }
    for (s = 0; s < dec_subband_count; s++) {
      w_norm[s] = pstr_group_overlap_params->str_band_overlap_params[0]
                      .overlap_weight[s];
      for (b = 1; b < num_drc_bands; b++) {
        w_norm[s] += pstr_group_overlap_params->str_band_overlap_params[b]
                         .overlap_weight[s];
      }
    }

    for (s = 0; s < dec_subband_count; s++) {
      for (b = 0; b < num_drc_bands; b++) {
        pstr_group_overlap_params->str_band_overlap_params[b]
            .overlap_weight[s] /= w_norm[s];
      }
    }
  } else {
    start_subband_index = 0;
    for (b = 0; b < num_drc_bands; b++) {
      if (b < num_drc_bands - 1) {
        stop_sub_band_index = gain_params[b + 1].start_subband_index - 1;
      } else {
        stop_sub_band_index = dec_subband_count - 1;
      }
      for (s = 0; s < dec_subband_count; s++) {
        if (s >= start_subband_index && s <= stop_sub_band_index) {
          pstr_group_overlap_params->str_band_overlap_params[b]
              .overlap_weight[s] = 1.0;
        } else {
          pstr_group_overlap_params->str_band_overlap_params[b]
              .overlap_weight[s] = 0.0;
        }
      }
      start_subband_index = stop_sub_band_index + 1;
    }
  }

  return (0);
}

IA_ERRORCODE impd_init_overlap_weight(
    ia_uni_drc_coeffs_struct* str_p_loc_drc_coefficients_uni_drc,
    ia_drc_instructions_struct* str_drc_instruction_str,
    WORD32 sub_band_domain_mode,
    ia_overlap_params_struct* pstr_overlap_params) {
  WORD32 err = 0, g;
  WORD32 dec_subband_count = 0;
  switch (sub_band_domain_mode) {
    case SUBBAND_DOMAIN_MODE_QMF64:
      dec_subband_count = AUDIO_CODEC_SUBBAND_COUNT_QMF64;
      break;
    case SUBBAND_DOMAIN_MODE_QMF71:
      dec_subband_count = AUDIO_CODEC_SUBBAND_COUNT_QMF71;
      break;
    case SUBBAND_DOMAIN_MODE_STFT256:
      dec_subband_count = AUDIO_CODEC_SUBBAND_COUNT_STFT256;
      break;
  }

  for (g = 0; g < str_drc_instruction_str->num_drc_ch_groups; g++) {
    if (str_drc_instruction_str->band_count_of_ch_group[g] > 1) {
      err = impd_generate_overlap_weights(
          str_drc_instruction_str->band_count_of_ch_group[g],
          str_p_loc_drc_coefficients_uni_drc
              ->gain_set_params[str_drc_instruction_str
                                    ->gain_set_index_for_channel_group[g]]
              .drc_band_type,
          str_p_loc_drc_coefficients_uni_drc
              ->gain_set_params[str_drc_instruction_str
                                    ->gain_set_index_for_channel_group[g]]
              .gain_params,
          dec_subband_count,
          &(pstr_overlap_params->str_group_overlap_params[g]));
      if (err) return (err);
    }
  }

  return (0);
}