1 /******************************************************************************
2  *                                                                            *
3  * Copyright (C) 2018 The Android Open Source Project
4  *
5  * Licensed under the Apache License, Version 2.0 (the "License");
6  * you may not use this file except in compliance with the License.
7  * You may obtain a copy of the License at:
8  *
9  * http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  *
17  *****************************************************************************
18  * Originally developed and contributed by Ittiam Systems Pvt. Ltd, Bangalore
19 */
20 #include <math.h>
21 #include "ixheaacd_type_def.h"
22 #include "ixheaacd_bitbuffer.h"
23 #include "ixheaacd_config.h"
24 
25 #include "ixheaacd_mps_polyphase.h"
26 
27 #include "ixheaacd_mps_dec.h"
28 #include "ixheaacd_mps_interface.h"
29 
30 #define max(a, b) ((a) > (b) ? (a) : (b))
31 
32 #define min(a, b) ((a) < (b) ? (a) : (b))
33 
34 #define DIR_DIFF_IN 0
35 #define DOWNMIX_IN 1
36 
37 #define LAMDA (4.0f)
38 #define GES_ALPHA (0.99637864f)
39 #define GES_BETA (0.9643691f)
40 
41 extern const WORD32
42     ixheaacd_hybrid_band_71_to_processing_band_20_map[MAX_HYBRID_BANDS_MPS];
43 
ixheaacd_mps_env_init(ia_mps_dec_state_struct * self)44 VOID ixheaacd_mps_env_init(ia_mps_dec_state_struct *self) {
45   WORD32 i;
46   for (i = 0; i < 3; i++) {
47     self->guided_env_shaping.avg_energy_prev[i] = 32768.f * 32768.f;
48   }
49 }
50 
ixheaacd_mps_est_normalized_envelope(ia_mps_dec_state_struct * self,WORD32 inp,WORD32 ch,FLOAT32 * env)51 static VOID ixheaacd_mps_est_normalized_envelope(ia_mps_dec_state_struct *self,
52                                                  WORD32 inp, WORD32 ch,
53                                                  FLOAT32 *env) {
54   FLOAT32 slot_energy[MAX_TIME_SLOTS][MAX_PARAMETER_BANDS] = {{0}};
55   FLOAT32 pb_energy[MAX_PARAMETER_BANDS] = {0};
56   FLOAT32 whitening_weight[MAX_PARAMETER_BANDS];
57   WORD32 ii, jj, param_band;
58 
59   WORD32 k_start = 10;
60   WORD32 k_stop = 18;
61 
62   FLOAT32 total_energy = 0, avg_energy = 0;
63 
64   WORD32 ch_offset;
65 
66   switch (inp) {
67     case DIR_DIFF_IN:
68       ch_offset = 0;
69       for (ii = 0; ii < self->time_slots; ii++) {
70         for (jj = 0; jj < self->hyb_band_count; jj++) {
71           slot_energy[ii]
72                      [ixheaacd_hybrid_band_71_to_processing_band_20_map[jj]] +=
73               ((self->hyb_dir_out[ch][ii][jj].re +
74                 self->hyb_diff_out[ch][ii][jj].re) *
75                (self->hyb_dir_out[ch][ii][jj].re +
76                 self->hyb_diff_out[ch][ii][jj].re)) +
77               ((self->hyb_dir_out[ch][ii][jj].im +
78                 self->hyb_diff_out[ch][ii][jj].im) *
79                (self->hyb_dir_out[ch][ii][jj].im +
80                 self->hyb_diff_out[ch][ii][jj].im));
81         }
82       }
83       break;
84     case DOWNMIX_IN:
85       ch_offset = self->out_ch_count;
86       for (ii = 0; ii < self->time_slots; ii++) {
87         for (jj = 0; jj < self->hyb_band_count; jj++) {
88           slot_energy[ii]
89                      [ixheaacd_hybrid_band_71_to_processing_band_20_map[jj]] +=
90               self->hyb_in[ch][ii][jj].re * self->hyb_in[ch][ii][jj].re +
91               self->hyb_in[ch][ii][jj].im * self->hyb_in[ch][ii][jj].im;
92         }
93       }
94       break;
95     default:
96       ch_offset = 0;
97       break;
98   }
99 
100   for (param_band = k_start; param_band <= k_stop; param_band++)
101     pb_energy[param_band] =
102         self->guided_env_shaping.pb_energy_prev[ch + ch_offset][param_band];
103 
104   avg_energy = self->guided_env_shaping.avg_energy_prev[ch + ch_offset];
105 
106   for (ii = 0; ii < self->time_slots; ii++) {
107     total_energy = 0;
108     for (param_band = k_start; param_band <= k_stop; param_band++) {
109       pb_energy[param_band] = (1 - GES_ALPHA) * slot_energy[ii][param_band] +
110                               GES_ALPHA * pb_energy[param_band];
111 
112       total_energy += slot_energy[ii][param_band];
113     }
114     total_energy /= (k_stop - k_start + 1);
115 
116     total_energy =
117         (1 - GES_ALPHA) * total_energy +
118         GES_ALPHA * self->guided_env_shaping.frame_energy_prev[ch + ch_offset];
119 
120     self->guided_env_shaping.frame_energy_prev[ch + ch_offset] = total_energy;
121 
122     for (param_band = k_start; param_band <= k_stop; param_band++) {
123       whitening_weight[param_band] =
124           total_energy / (pb_energy[param_band] + ABS_THR);
125     }
126 
127     env[ii] = 0;
128     for (param_band = k_start; param_band <= k_stop; param_band++) {
129       env[ii] += slot_energy[ii][param_band] * whitening_weight[param_band];
130     }
131 
132     avg_energy = (1 - GES_BETA) * env[ii] + GES_BETA * avg_energy;
133 
134     env[ii] = (FLOAT32)sqrt(env[ii] / (avg_energy + ABS_THR));
135   }
136 
137   for (param_band = k_start; param_band <= k_stop; param_band++)
138     self->guided_env_shaping.pb_energy_prev[ch + ch_offset][param_band] =
139         pb_energy[param_band];
140 
141   self->guided_env_shaping.avg_energy_prev[ch + ch_offset] = avg_energy;
142 }
143 
ixheaacd_mps_time_env_shaping(ia_mps_dec_state_struct * self)144 VOID ixheaacd_mps_time_env_shaping(ia_mps_dec_state_struct *self) {
145   FLOAT32 dir_energy[MAX_TIME_SLOTS];
146   FLOAT32 dmx_energy[MAX_TIME_SLOTS];
147   WORD32 ch, time_slot, jj;
148 
149   WORD32 band_start;
150   FLOAT32 gain, ratio;
151 
152   FLOAT32 amp_direct = 0;
153   FLOAT32 amp_diff = 0;
154   FLOAT32 amp_ratio;
155 
156   band_start = 6;
157 
158   ixheaacd_mps_est_normalized_envelope(self, DOWNMIX_IN, 0, dmx_energy);
159 
160   for (ch = 0; ch < self->out_ch_count; ch++) {
161     ixheaacd_mps_est_normalized_envelope(self, DIR_DIFF_IN, ch, dir_energy);
162 
163     if (self->temp_shape_enable_ch_ges[ch]) {
164       for (time_slot = 0; time_slot < self->time_slots; time_slot++) {
165         gain = self->env_shape_data[ch][time_slot] * dmx_energy[time_slot] /
166                (dir_energy[time_slot] + 1e-9f);
167 
168         amp_direct = 0;
169         amp_diff = 0;
170 
171         for (jj = band_start; jj < self->hyb_band_count; jj++) {
172           amp_direct += self->hyb_dir_out[ch][time_slot][jj].re *
173                             self->hyb_dir_out[ch][time_slot][jj].re +
174                         self->hyb_dir_out[ch][time_slot][jj].im *
175                             self->hyb_dir_out[ch][time_slot][jj].im;
176 
177           amp_diff += self->hyb_diff_out[ch][time_slot][jj].re *
178                           self->hyb_diff_out[ch][time_slot][jj].re +
179                       self->hyb_diff_out[ch][time_slot][jj].im *
180                           self->hyb_diff_out[ch][time_slot][jj].im;
181         }
182 
183         amp_ratio = (FLOAT32)sqrt(amp_diff / (amp_direct + ABS_THR));
184 
185         ratio = min(max((gain + amp_ratio * (gain - 1)), 1 / LAMDA), LAMDA);
186 
187         for (jj = band_start; jj < self->hyb_band_count; jj++) {
188           self->hyb_dir_out[ch][time_slot][jj].re *= ratio;
189           self->hyb_dir_out[ch][time_slot][jj].im *= ratio;
190         }
191       }
192     }
193   }
194 }