xref: /aosp_15_r20/external/liblc3/src/sns.c (revision 49fe348c0058011ee60b6957cdd9d52742df84bc)
1*49fe348cSAndroid Build Coastguard Worker /******************************************************************************
2*49fe348cSAndroid Build Coastguard Worker  *
3*49fe348cSAndroid Build Coastguard Worker  *  Copyright 2022 Google LLC
4*49fe348cSAndroid Build Coastguard Worker  *
5*49fe348cSAndroid Build Coastguard Worker  *  Licensed under the Apache License, Version 2.0 (the "License");
6*49fe348cSAndroid Build Coastguard Worker  *  you may not use this file except in compliance with the License.
7*49fe348cSAndroid Build Coastguard Worker  *  You may obtain a copy of the License at:
8*49fe348cSAndroid Build Coastguard Worker  *
9*49fe348cSAndroid Build Coastguard Worker  *  http://www.apache.org/licenses/LICENSE-2.0
10*49fe348cSAndroid Build Coastguard Worker  *
11*49fe348cSAndroid Build Coastguard Worker  *  Unless required by applicable law or agreed to in writing, software
12*49fe348cSAndroid Build Coastguard Worker  *  distributed under the License is distributed on an "AS IS" BASIS,
13*49fe348cSAndroid Build Coastguard Worker  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*49fe348cSAndroid Build Coastguard Worker  *  See the License for the specific language governing permissions and
15*49fe348cSAndroid Build Coastguard Worker  *  limitations under the License.
16*49fe348cSAndroid Build Coastguard Worker  *
17*49fe348cSAndroid Build Coastguard Worker  ******************************************************************************/
18*49fe348cSAndroid Build Coastguard Worker 
19*49fe348cSAndroid Build Coastguard Worker #include "sns.h"
20*49fe348cSAndroid Build Coastguard Worker #include "tables.h"
21*49fe348cSAndroid Build Coastguard Worker 
22*49fe348cSAndroid Build Coastguard Worker 
23*49fe348cSAndroid Build Coastguard Worker /* ----------------------------------------------------------------------------
24*49fe348cSAndroid Build Coastguard Worker  *  DCT-16
25*49fe348cSAndroid Build Coastguard Worker  * -------------------------------------------------------------------------- */
26*49fe348cSAndroid Build Coastguard Worker 
27*49fe348cSAndroid Build Coastguard Worker /**
28*49fe348cSAndroid Build Coastguard Worker  * Matrix of DCT-16 coefficients
29*49fe348cSAndroid Build Coastguard Worker  *
30*49fe348cSAndroid Build Coastguard Worker  * M[n][k] = 2f cos( Pi k (2n + 1) / 2N )
31*49fe348cSAndroid Build Coastguard Worker  *
32*49fe348cSAndroid Build Coastguard Worker  *   k = [0..N-1], n = [0..N-1], N = 16
33*49fe348cSAndroid Build Coastguard Worker  *   f = sqrt(1/4N) for k=0, sqrt(1/2N) otherwise
34*49fe348cSAndroid Build Coastguard Worker  */
35*49fe348cSAndroid Build Coastguard Worker static const float dct16_m[16][16] = {
36*49fe348cSAndroid Build Coastguard Worker 
37*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01,  3.51850934e-01,  3.46759961e-01,  3.38329500e-01,
38*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01,  3.11806253e-01,  2.93968901e-01,  2.73300467e-01,
39*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01,  2.24291897e-01,  1.96423740e-01,  1.66663915e-01,
40*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01,  1.02631132e-01,  6.89748448e-02,  3.46542923e-02 },
41*49fe348cSAndroid Build Coastguard Worker 
42*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01,  3.38329500e-01,  2.93968901e-01,  2.24291897e-01,
43*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01,  3.46542923e-02, -6.89748448e-02, -1.66663915e-01,
44*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01, -3.11806253e-01, -3.46759961e-01, -3.51850934e-01,
45*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01, -2.73300467e-01, -1.96423740e-01, -1.02631132e-01 },
46*49fe348cSAndroid Build Coastguard Worker 
47*49fe348cSAndroid Build Coastguard Worker      { 2.50000000e-01,  3.11806253e-01,  1.96423740e-01,  3.46542923e-02,
48*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01, -2.73300467e-01, -3.46759961e-01, -3.38329500e-01,
49*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01, -1.02631132e-01,  6.89748448e-02,  2.24291897e-01,
50*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01,  3.51850934e-01,  2.93968901e-01,  1.66663915e-01 },
51*49fe348cSAndroid Build Coastguard Worker 
52*49fe348cSAndroid Build Coastguard Worker      { 2.50000000e-01,  2.73300467e-01,  6.89748448e-02, -1.66663915e-01,
53*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01, -3.38329500e-01, -1.96423740e-01,  3.46542923e-02,
54*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01,  3.51850934e-01,  2.93968901e-01,  1.02631132e-01,
55*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01, -3.11806253e-01, -3.46759961e-01, -2.24291897e-01 },
56*49fe348cSAndroid Build Coastguard Worker 
57*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01,  2.24291897e-01, -6.89748448e-02, -3.11806253e-01,
58*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01, -1.02631132e-01,  1.96423740e-01,  3.51850934e-01,
59*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01, -3.46542923e-02, -2.93968901e-01, -3.38329500e-01,
60*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01,  1.66663915e-01,  3.46759961e-01,  2.73300467e-01 },
61*49fe348cSAndroid Build Coastguard Worker 
62*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01,  1.66663915e-01, -1.96423740e-01, -3.51850934e-01,
63*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01,  2.24291897e-01,  3.46759961e-01,  1.02631132e-01,
64*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01, -3.38329500e-01, -6.89748448e-02,  2.73300467e-01,
65*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01,  3.46542923e-02, -2.93968901e-01, -3.11806253e-01 },
66*49fe348cSAndroid Build Coastguard Worker 
67*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01,  1.02631132e-01, -2.93968901e-01, -2.73300467e-01,
68*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01,  3.51850934e-01,  6.89748448e-02, -3.11806253e-01,
69*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01,  1.66663915e-01,  3.46759961e-01,  3.46542923e-02,
70*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01, -2.24291897e-01,  1.96423740e-01,  3.38329500e-01 },
71*49fe348cSAndroid Build Coastguard Worker 
72*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01,  3.46542923e-02, -3.46759961e-01, -1.02631132e-01,
73*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01,  1.66663915e-01, -2.93968901e-01, -2.24291897e-01,
74*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01,  2.73300467e-01, -1.96423740e-01, -3.11806253e-01,
75*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01,  3.38329500e-01, -6.89748448e-02, -3.51850934e-01 },
76*49fe348cSAndroid Build Coastguard Worker 
77*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -3.46542923e-02, -3.46759961e-01,  1.02631132e-01,
78*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01, -1.66663915e-01, -2.93968901e-01,  2.24291897e-01,
79*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01, -2.73300467e-01, -1.96423740e-01,  3.11806253e-01,
80*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01, -3.38329500e-01, -6.89748448e-02,  3.51850934e-01 },
81*49fe348cSAndroid Build Coastguard Worker 
82*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -1.02631132e-01, -2.93968901e-01,  2.73300467e-01,
83*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01, -3.51850934e-01,  6.89748448e-02,  3.11806253e-01,
84*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01, -1.66663915e-01,  3.46759961e-01, -3.46542923e-02,
85*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01,  2.24291897e-01,  1.96423740e-01, -3.38329500e-01 },
86*49fe348cSAndroid Build Coastguard Worker 
87*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -1.66663915e-01, -1.96423740e-01,  3.51850934e-01,
88*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01, -2.24291897e-01,  3.46759961e-01, -1.02631132e-01,
89*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01,  3.38329500e-01, -6.89748448e-02, -2.73300467e-01,
90*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01, -3.46542923e-02, -2.93968901e-01,  3.11806253e-01 },
91*49fe348cSAndroid Build Coastguard Worker 
92*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -2.24291897e-01, -6.89748448e-02,  3.11806253e-01,
93*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01,  1.02631132e-01,  1.96423740e-01, -3.51850934e-01,
94*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01,  3.46542923e-02, -2.93968901e-01,  3.38329500e-01,
95*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01, -1.66663915e-01,  3.46759961e-01, -2.73300467e-01 },
96*49fe348cSAndroid Build Coastguard Worker 
97*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -2.73300467e-01,  6.89748448e-02,  1.66663915e-01,
98*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01,  3.38329500e-01, -1.96423740e-01, -3.46542923e-02,
99*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01, -3.51850934e-01,  2.93968901e-01, -1.02631132e-01,
100*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01,  3.11806253e-01, -3.46759961e-01,  2.24291897e-01 },
101*49fe348cSAndroid Build Coastguard Worker 
102*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -3.11806253e-01,  1.96423740e-01, -3.46542923e-02,
103*49fe348cSAndroid Build Coastguard Worker       -1.35299025e-01,  2.73300467e-01, -3.46759961e-01,  3.38329500e-01,
104*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01,  1.02631132e-01,  6.89748448e-02, -2.24291897e-01,
105*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01, -3.51850934e-01,  2.93968901e-01, -1.66663915e-01 },
106*49fe348cSAndroid Build Coastguard Worker 
107*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -3.38329500e-01,  2.93968901e-01, -2.24291897e-01,
108*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01, -3.46542923e-02, -6.89748448e-02,  1.66663915e-01,
109*49fe348cSAndroid Build Coastguard Worker       -2.50000000e-01,  3.11806253e-01, -3.46759961e-01,  3.51850934e-01,
110*49fe348cSAndroid Build Coastguard Worker       -3.26640741e-01,  2.73300467e-01, -1.96423740e-01,  1.02631132e-01 },
111*49fe348cSAndroid Build Coastguard Worker 
112*49fe348cSAndroid Build Coastguard Worker     {  2.50000000e-01, -3.51850934e-01,  3.46759961e-01, -3.38329500e-01,
113*49fe348cSAndroid Build Coastguard Worker        3.26640741e-01, -3.11806253e-01,  2.93968901e-01, -2.73300467e-01,
114*49fe348cSAndroid Build Coastguard Worker        2.50000000e-01, -2.24291897e-01,  1.96423740e-01, -1.66663915e-01,
115*49fe348cSAndroid Build Coastguard Worker        1.35299025e-01, -1.02631132e-01,  6.89748448e-02, -3.46542923e-02 },
116*49fe348cSAndroid Build Coastguard Worker 
117*49fe348cSAndroid Build Coastguard Worker };
118*49fe348cSAndroid Build Coastguard Worker 
119*49fe348cSAndroid Build Coastguard Worker /**
120*49fe348cSAndroid Build Coastguard Worker  * Forward DCT-16 transformation
121*49fe348cSAndroid Build Coastguard Worker  * x, y            Input and output 16 values
122*49fe348cSAndroid Build Coastguard Worker  */
dct16_forward(const float * x,float * y)123*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void dct16_forward(const float *x, float *y)
124*49fe348cSAndroid Build Coastguard Worker {
125*49fe348cSAndroid Build Coastguard Worker     for (int i = 0, j; i < 16; i++)
126*49fe348cSAndroid Build Coastguard Worker         for (y[i] = 0, j = 0; j < 16; j++)
127*49fe348cSAndroid Build Coastguard Worker             y[i] += x[j] * dct16_m[j][i];
128*49fe348cSAndroid Build Coastguard Worker }
129*49fe348cSAndroid Build Coastguard Worker 
130*49fe348cSAndroid Build Coastguard Worker /**
131*49fe348cSAndroid Build Coastguard Worker  * Inverse DCT-16 transformation
132*49fe348cSAndroid Build Coastguard Worker  * x, y            Input and output 16 values
133*49fe348cSAndroid Build Coastguard Worker  */
dct16_inverse(const float * x,float * y)134*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void dct16_inverse(const float *x, float *y)
135*49fe348cSAndroid Build Coastguard Worker {
136*49fe348cSAndroid Build Coastguard Worker     for (int i = 0, j; i < 16; i++)
137*49fe348cSAndroid Build Coastguard Worker         for (y[i] = 0, j = 0; j < 16; j++)
138*49fe348cSAndroid Build Coastguard Worker             y[i] += x[j] * dct16_m[i][j];
139*49fe348cSAndroid Build Coastguard Worker }
140*49fe348cSAndroid Build Coastguard Worker 
141*49fe348cSAndroid Build Coastguard Worker 
142*49fe348cSAndroid Build Coastguard Worker /* ----------------------------------------------------------------------------
143*49fe348cSAndroid Build Coastguard Worker  *  Scale factors
144*49fe348cSAndroid Build Coastguard Worker  * -------------------------------------------------------------------------- */
145*49fe348cSAndroid Build Coastguard Worker 
146*49fe348cSAndroid Build Coastguard Worker /**
147*49fe348cSAndroid Build Coastguard Worker  * Scale factors
148*49fe348cSAndroid Build Coastguard Worker  * dt, sr          Duration and samplerate of the frame
149*49fe348cSAndroid Build Coastguard Worker  * nbytes          Size in bytes of the frame
150*49fe348cSAndroid Build Coastguard Worker  * eb              Energy estimation per bands
151*49fe348cSAndroid Build Coastguard Worker  * att             1: Attack detected  0: Otherwise
152*49fe348cSAndroid Build Coastguard Worker  * scf             Output 16 scale factors
153*49fe348cSAndroid Build Coastguard Worker  */
compute_scale_factors(enum lc3_dt dt,enum lc3_srate sr,int nbytes,const float * eb,bool att,float * scf)154*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void compute_scale_factors(
155*49fe348cSAndroid Build Coastguard Worker     enum lc3_dt dt, enum lc3_srate sr, int nbytes,
156*49fe348cSAndroid Build Coastguard Worker     const float *eb, bool att, float *scf)
157*49fe348cSAndroid Build Coastguard Worker {
158*49fe348cSAndroid Build Coastguard Worker     /* Pre-emphasis gain table :
159*49fe348cSAndroid Build Coastguard Worker      * Ge[b] = 10 ^ (b * g_tilt) / 630 , b = [0..63] */
160*49fe348cSAndroid Build Coastguard Worker 
161*49fe348cSAndroid Build Coastguard Worker     static const float ge_14[LC3_MAX_BANDS] = { /* g_tilt = 14 */
162*49fe348cSAndroid Build Coastguard Worker         1.00000000e+00, 1.05250029e+00, 1.10775685e+00, 1.16591440e+00,
163*49fe348cSAndroid Build Coastguard Worker         1.22712524e+00, 1.29154967e+00, 1.35935639e+00, 1.43072299e+00,
164*49fe348cSAndroid Build Coastguard Worker         1.50583635e+00, 1.58489319e+00, 1.66810054e+00, 1.75567629e+00,
165*49fe348cSAndroid Build Coastguard Worker         1.84784980e+00, 1.94486244e+00, 2.04696827e+00, 2.15443469e+00,
166*49fe348cSAndroid Build Coastguard Worker         2.26754313e+00, 2.38658979e+00, 2.51188643e+00, 2.64376119e+00,
167*49fe348cSAndroid Build Coastguard Worker         2.78255940e+00, 2.92864456e+00, 3.08239924e+00, 3.24422608e+00,
168*49fe348cSAndroid Build Coastguard Worker         3.41454887e+00, 3.59381366e+00, 3.78248991e+00, 3.98107171e+00,
169*49fe348cSAndroid Build Coastguard Worker         4.19007911e+00, 4.41005945e+00, 4.64158883e+00, 4.88527357e+00,
170*49fe348cSAndroid Build Coastguard Worker         5.14175183e+00, 5.41169527e+00, 5.69581081e+00, 5.99484250e+00,
171*49fe348cSAndroid Build Coastguard Worker         6.30957344e+00, 6.64082785e+00, 6.98947321e+00, 7.35642254e+00,
172*49fe348cSAndroid Build Coastguard Worker         7.74263683e+00, 8.14912747e+00, 8.57695899e+00, 9.02725178e+00,
173*49fe348cSAndroid Build Coastguard Worker         9.50118507e+00, 1.00000000e+01, 1.05250029e+01, 1.10775685e+01,
174*49fe348cSAndroid Build Coastguard Worker         1.16591440e+01, 1.22712524e+01, 1.29154967e+01, 1.35935639e+01,
175*49fe348cSAndroid Build Coastguard Worker         1.43072299e+01, 1.50583635e+01, 1.58489319e+01, 1.66810054e+01,
176*49fe348cSAndroid Build Coastguard Worker         1.75567629e+01, 1.84784980e+01, 1.94486244e+01, 2.04696827e+01,
177*49fe348cSAndroid Build Coastguard Worker         2.15443469e+01, 2.26754313e+01, 2.38658979e+01, 2.51188643e+01 };
178*49fe348cSAndroid Build Coastguard Worker 
179*49fe348cSAndroid Build Coastguard Worker     static const float ge_18[LC3_MAX_BANDS] = { /* g_tilt = 18 */
180*49fe348cSAndroid Build Coastguard Worker         1.00000000e+00, 1.06800043e+00, 1.14062492e+00, 1.21818791e+00,
181*49fe348cSAndroid Build Coastguard Worker         1.30102522e+00, 1.38949549e+00, 1.48398179e+00, 1.58489319e+00,
182*49fe348cSAndroid Build Coastguard Worker         1.69266662e+00, 1.80776868e+00, 1.93069773e+00, 2.06198601e+00,
183*49fe348cSAndroid Build Coastguard Worker         2.20220195e+00, 2.35195264e+00, 2.51188643e+00, 2.68269580e+00,
184*49fe348cSAndroid Build Coastguard Worker         2.86512027e+00, 3.05994969e+00, 3.26802759e+00, 3.49025488e+00,
185*49fe348cSAndroid Build Coastguard Worker         3.72759372e+00, 3.98107171e+00, 4.25178630e+00, 4.54090961e+00,
186*49fe348cSAndroid Build Coastguard Worker         4.84969343e+00, 5.17947468e+00, 5.53168120e+00, 5.90783791e+00,
187*49fe348cSAndroid Build Coastguard Worker         6.30957344e+00, 6.73862717e+00, 7.19685673e+00, 7.68624610e+00,
188*49fe348cSAndroid Build Coastguard Worker         8.20891416e+00, 8.76712387e+00, 9.36329209e+00, 1.00000000e+01,
189*49fe348cSAndroid Build Coastguard Worker         1.06800043e+01, 1.14062492e+01, 1.21818791e+01, 1.30102522e+01,
190*49fe348cSAndroid Build Coastguard Worker         1.38949549e+01, 1.48398179e+01, 1.58489319e+01, 1.69266662e+01,
191*49fe348cSAndroid Build Coastguard Worker         1.80776868e+01, 1.93069773e+01, 2.06198601e+01, 2.20220195e+01,
192*49fe348cSAndroid Build Coastguard Worker         2.35195264e+01, 2.51188643e+01, 2.68269580e+01, 2.86512027e+01,
193*49fe348cSAndroid Build Coastguard Worker         3.05994969e+01, 3.26802759e+01, 3.49025488e+01, 3.72759372e+01,
194*49fe348cSAndroid Build Coastguard Worker         3.98107171e+01, 4.25178630e+01, 4.54090961e+01, 4.84969343e+01,
195*49fe348cSAndroid Build Coastguard Worker         5.17947468e+01, 5.53168120e+01, 5.90783791e+01, 6.30957344e+01 };
196*49fe348cSAndroid Build Coastguard Worker 
197*49fe348cSAndroid Build Coastguard Worker     static const float ge_22[LC3_MAX_BANDS] = { /* g_tilt = 22 */
198*49fe348cSAndroid Build Coastguard Worker         1.00000000e+00, 1.08372885e+00, 1.17446822e+00, 1.27280509e+00,
199*49fe348cSAndroid Build Coastguard Worker         1.37937560e+00, 1.49486913e+00, 1.62003281e+00, 1.75567629e+00,
200*49fe348cSAndroid Build Coastguard Worker         1.90267705e+00, 2.06198601e+00, 2.23463373e+00, 2.42173704e+00,
201*49fe348cSAndroid Build Coastguard Worker         2.62450630e+00, 2.84425319e+00, 3.08239924e+00, 3.34048498e+00,
202*49fe348cSAndroid Build Coastguard Worker         3.62017995e+00, 3.92329345e+00, 4.25178630e+00, 4.60778348e+00,
203*49fe348cSAndroid Build Coastguard Worker         4.99358789e+00, 5.41169527e+00, 5.86481029e+00, 6.35586411e+00,
204*49fe348cSAndroid Build Coastguard Worker         6.88803330e+00, 7.46476041e+00, 8.08977621e+00, 8.76712387e+00,
205*49fe348cSAndroid Build Coastguard Worker         9.50118507e+00, 1.02967084e+01, 1.11588399e+01, 1.20931568e+01,
206*49fe348cSAndroid Build Coastguard Worker         1.31057029e+01, 1.42030283e+01, 1.53922315e+01, 1.66810054e+01,
207*49fe348cSAndroid Build Coastguard Worker         1.80776868e+01, 1.95913107e+01, 2.12316686e+01, 2.30093718e+01,
208*49fe348cSAndroid Build Coastguard Worker         2.49359200e+01, 2.70237760e+01, 2.92864456e+01, 3.17385661e+01,
209*49fe348cSAndroid Build Coastguard Worker         3.43959997e+01, 3.72759372e+01, 4.03970086e+01, 4.37794036e+01,
210*49fe348cSAndroid Build Coastguard Worker         4.74450028e+01, 5.14175183e+01, 5.57226480e+01, 6.03882412e+01,
211*49fe348cSAndroid Build Coastguard Worker         6.54444792e+01, 7.09240702e+01, 7.68624610e+01, 8.32980665e+01,
212*49fe348cSAndroid Build Coastguard Worker         9.02725178e+01, 9.78309319e+01, 1.06022203e+02, 1.14899320e+02,
213*49fe348cSAndroid Build Coastguard Worker         1.24519708e+02, 1.34945600e+02, 1.46244440e+02, 1.58489319e+02 };
214*49fe348cSAndroid Build Coastguard Worker 
215*49fe348cSAndroid Build Coastguard Worker     static const float ge_26[LC3_MAX_BANDS] = { /* g_tilt = 26 */
216*49fe348cSAndroid Build Coastguard Worker         1.00000000e+00, 1.09968890e+00, 1.20931568e+00, 1.32987103e+00,
217*49fe348cSAndroid Build Coastguard Worker         1.46244440e+00, 1.60823388e+00, 1.76855694e+00, 1.94486244e+00,
218*49fe348cSAndroid Build Coastguard Worker         2.13874364e+00, 2.35195264e+00, 2.58641621e+00, 2.84425319e+00,
219*49fe348cSAndroid Build Coastguard Worker         3.12779366e+00, 3.43959997e+00, 3.78248991e+00, 4.15956216e+00,
220*49fe348cSAndroid Build Coastguard Worker         4.57422434e+00, 5.03022373e+00, 5.53168120e+00, 6.08312841e+00,
221*49fe348cSAndroid Build Coastguard Worker         6.68954879e+00, 7.35642254e+00, 8.08977621e+00, 8.89623710e+00,
222*49fe348cSAndroid Build Coastguard Worker         9.78309319e+00, 1.07583590e+01, 1.18308480e+01, 1.30102522e+01,
223*49fe348cSAndroid Build Coastguard Worker         1.43072299e+01, 1.57335019e+01, 1.73019574e+01, 1.90267705e+01,
224*49fe348cSAndroid Build Coastguard Worker         2.09235283e+01, 2.30093718e+01, 2.53031508e+01, 2.78255940e+01,
225*49fe348cSAndroid Build Coastguard Worker         3.05994969e+01, 3.36499270e+01, 3.70044512e+01, 4.06933843e+01,
226*49fe348cSAndroid Build Coastguard Worker         4.47500630e+01, 4.92111475e+01, 5.41169527e+01, 5.95118121e+01,
227*49fe348cSAndroid Build Coastguard Worker         6.54444792e+01, 7.19685673e+01, 7.91430346e+01, 8.70327166e+01,
228*49fe348cSAndroid Build Coastguard Worker         9.57089124e+01, 1.05250029e+02, 1.15742288e+02, 1.27280509e+02,
229*49fe348cSAndroid Build Coastguard Worker         1.39968963e+02, 1.53922315e+02, 1.69266662e+02, 1.86140669e+02,
230*49fe348cSAndroid Build Coastguard Worker         2.04696827e+02, 2.25102829e+02, 2.47543082e+02, 2.72220379e+02,
231*49fe348cSAndroid Build Coastguard Worker         2.99357729e+02, 3.29200372e+02, 3.62017995e+02, 3.98107171e+02 };
232*49fe348cSAndroid Build Coastguard Worker 
233*49fe348cSAndroid Build Coastguard Worker     static const float ge_30[LC3_MAX_BANDS] = { /* g_tilt = 30 */
234*49fe348cSAndroid Build Coastguard Worker         1.00000000e+00, 1.11588399e+00, 1.24519708e+00, 1.38949549e+00,
235*49fe348cSAndroid Build Coastguard Worker         1.55051578e+00, 1.73019574e+00, 1.93069773e+00, 2.15443469e+00,
236*49fe348cSAndroid Build Coastguard Worker         2.40409918e+00, 2.68269580e+00, 2.99357729e+00, 3.34048498e+00,
237*49fe348cSAndroid Build Coastguard Worker         3.72759372e+00, 4.15956216e+00, 4.64158883e+00, 5.17947468e+00,
238*49fe348cSAndroid Build Coastguard Worker         5.77969288e+00, 6.44946677e+00, 7.19685673e+00, 8.03085722e+00,
239*49fe348cSAndroid Build Coastguard Worker         8.96150502e+00, 1.00000000e+01, 1.11588399e+01, 1.24519708e+01,
240*49fe348cSAndroid Build Coastguard Worker         1.38949549e+01, 1.55051578e+01, 1.73019574e+01, 1.93069773e+01,
241*49fe348cSAndroid Build Coastguard Worker         2.15443469e+01, 2.40409918e+01, 2.68269580e+01, 2.99357729e+01,
242*49fe348cSAndroid Build Coastguard Worker         3.34048498e+01, 3.72759372e+01, 4.15956216e+01, 4.64158883e+01,
243*49fe348cSAndroid Build Coastguard Worker         5.17947468e+01, 5.77969288e+01, 6.44946677e+01, 7.19685673e+01,
244*49fe348cSAndroid Build Coastguard Worker         8.03085722e+01, 8.96150502e+01, 1.00000000e+02, 1.11588399e+02,
245*49fe348cSAndroid Build Coastguard Worker         1.24519708e+02, 1.38949549e+02, 1.55051578e+02, 1.73019574e+02,
246*49fe348cSAndroid Build Coastguard Worker         1.93069773e+02, 2.15443469e+02, 2.40409918e+02, 2.68269580e+02,
247*49fe348cSAndroid Build Coastguard Worker         2.99357729e+02, 3.34048498e+02, 3.72759372e+02, 4.15956216e+02,
248*49fe348cSAndroid Build Coastguard Worker         4.64158883e+02, 5.17947468e+02, 5.77969288e+02, 6.44946677e+02,
249*49fe348cSAndroid Build Coastguard Worker         7.19685673e+02, 8.03085722e+02, 8.96150502e+02, 1.00000000e+03 };
250*49fe348cSAndroid Build Coastguard Worker 
251*49fe348cSAndroid Build Coastguard Worker #if LC3_PLUS_HR
252*49fe348cSAndroid Build Coastguard Worker 
253*49fe348cSAndroid Build Coastguard Worker     static const float ge_34[LC3_MAX_BANDS] = { /* g_tilt = 34 */
254*49fe348cSAndroid Build Coastguard Worker         1.00000000e+00, 1.13231759e+00, 1.28214312e+00, 1.45179321e+00,
255*49fe348cSAndroid Build Coastguard Worker         1.64389099e+00, 1.86140669e+00, 2.10770353e+00, 2.38658979e+00,
256*49fe348cSAndroid Build Coastguard Worker         2.70237760e+00, 3.05994969e+00, 3.46483486e+00, 3.92329345e+00,
257*49fe348cSAndroid Build Coastguard Worker         4.44241419e+00, 5.03022373e+00, 5.69581081e+00, 6.44946677e+00,
258*49fe348cSAndroid Build Coastguard Worker         7.30284467e+00, 8.26913948e+00, 9.36329209e+00, 1.06022203e+01,
259*49fe348cSAndroid Build Coastguard Worker         1.20050806e+01, 1.35935639e+01, 1.53922315e+01, 1.74288945e+01,
260*49fe348cSAndroid Build Coastguard Worker         1.97350438e+01, 2.23463373e+01, 2.53031508e+01, 2.86512027e+01,
261*49fe348cSAndroid Build Coastguard Worker         3.24422608e+01, 3.67349426e+01, 4.15956216e+01, 4.70994540e+01,
262*49fe348cSAndroid Build Coastguard Worker         5.33315403e+01, 6.03882412e+01, 6.83786677e+01, 7.74263683e+01,
263*49fe348cSAndroid Build Coastguard Worker         8.76712387e+01, 9.92716858e+01, 1.12407076e+02, 1.27280509e+02,
264*49fe348cSAndroid Build Coastguard Worker         1.44121960e+02, 1.63191830e+02, 1.84784980e+02, 2.09235283e+02,
265*49fe348cSAndroid Build Coastguard Worker         2.36920791e+02, 2.68269580e+02, 3.03766364e+02, 3.43959997e+02,
266*49fe348cSAndroid Build Coastguard Worker         3.89471955e+02, 4.41005945e+02, 4.99358789e+02, 5.65432741e+02,
267*49fe348cSAndroid Build Coastguard Worker         6.40249439e+02, 7.24965701e+02, 8.20891416e+02, 9.29509790e+02,
268*49fe348cSAndroid Build Coastguard Worker         1.05250029e+03, 1.19176459e+03, 1.34945600e+03, 1.52801277e+03,
269*49fe348cSAndroid Build Coastguard Worker         1.73019574e+03, 1.95913107e+03, 2.21835857e+03, 2.51188643e+03 };
270*49fe348cSAndroid Build Coastguard Worker 
271*49fe348cSAndroid Build Coastguard Worker #endif /* LC3_PLUS_HR */
272*49fe348cSAndroid Build Coastguard Worker 
273*49fe348cSAndroid Build Coastguard Worker     static const float *ge_table[LC3_NUM_SRATE] = {
274*49fe348cSAndroid Build Coastguard Worker         [LC3_SRATE_8K    ] = ge_14, [LC3_SRATE_16K   ] = ge_18,
275*49fe348cSAndroid Build Coastguard Worker         [LC3_SRATE_24K   ] = ge_22, [LC3_SRATE_32K   ] = ge_26,
276*49fe348cSAndroid Build Coastguard Worker         [LC3_SRATE_48K   ] = ge_30,
277*49fe348cSAndroid Build Coastguard Worker 
278*49fe348cSAndroid Build Coastguard Worker #if LC3_PLUS_HR
279*49fe348cSAndroid Build Coastguard Worker         [LC3_SRATE_48K_HR] = ge_30, [LC3_SRATE_96K_HR] = ge_34,
280*49fe348cSAndroid Build Coastguard Worker #endif /* LC3_PLUS_HR */
281*49fe348cSAndroid Build Coastguard Worker 
282*49fe348cSAndroid Build Coastguard Worker     };
283*49fe348cSAndroid Build Coastguard Worker 
284*49fe348cSAndroid Build Coastguard Worker     float e[LC3_MAX_BANDS];
285*49fe348cSAndroid Build Coastguard Worker 
286*49fe348cSAndroid Build Coastguard Worker     /* --- Copy and padding --- */
287*49fe348cSAndroid Build Coastguard Worker 
288*49fe348cSAndroid Build Coastguard Worker     int nb = lc3_num_bands[dt][sr];
289*49fe348cSAndroid Build Coastguard Worker     int n4 = nb < 32 ? 32 % nb : 0;
290*49fe348cSAndroid Build Coastguard Worker     int n2 = nb < 32 ? nb - n4 : LC3_MAX_BANDS - nb;
291*49fe348cSAndroid Build Coastguard Worker 
292*49fe348cSAndroid Build Coastguard Worker     for (int i4 = 0; i4 < n4; i4++)
293*49fe348cSAndroid Build Coastguard Worker         e[4*i4 + 0] = e[4*i4 + 1] =
294*49fe348cSAndroid Build Coastguard Worker         e[4*i4 + 2] = e[4*i4 + 3] = eb[i4];
295*49fe348cSAndroid Build Coastguard Worker 
296*49fe348cSAndroid Build Coastguard Worker     for (int i2 = n4; i2 < n4+n2; i2++)
297*49fe348cSAndroid Build Coastguard Worker         e[2*(n4+i2) + 0] = e[2*(n4+i2) + 1] = eb[i2];
298*49fe348cSAndroid Build Coastguard Worker 
299*49fe348cSAndroid Build Coastguard Worker     memcpy(e + 4*n4 + 2*n2, eb + n4 + n2, (nb - n4 - n2) * sizeof(float));
300*49fe348cSAndroid Build Coastguard Worker 
301*49fe348cSAndroid Build Coastguard Worker     /* --- Smoothing, pre-emphasis and logarithm --- */
302*49fe348cSAndroid Build Coastguard Worker 
303*49fe348cSAndroid Build Coastguard Worker     const float *ge = ge_table[sr];
304*49fe348cSAndroid Build Coastguard Worker 
305*49fe348cSAndroid Build Coastguard Worker     float e0 = e[0], e1 = e[0], e2;
306*49fe348cSAndroid Build Coastguard Worker     float e_sum = 0;
307*49fe348cSAndroid Build Coastguard Worker 
308*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < LC3_MAX_BANDS-1; ) {
309*49fe348cSAndroid Build Coastguard Worker         e[i] = (e0 * 0.25f + e1 * 0.5f + (e2 = e[i+1]) * 0.25f) * ge[i];
310*49fe348cSAndroid Build Coastguard Worker         e_sum += e[i++];
311*49fe348cSAndroid Build Coastguard Worker 
312*49fe348cSAndroid Build Coastguard Worker         e[i] = (e1 * 0.25f + e2 * 0.5f + (e0 = e[i+1]) * 0.25f) * ge[i];
313*49fe348cSAndroid Build Coastguard Worker         e_sum += e[i++];
314*49fe348cSAndroid Build Coastguard Worker 
315*49fe348cSAndroid Build Coastguard Worker         e[i] = (e2 * 0.25f + e0 * 0.5f + (e1 = e[i+1]) * 0.25f) * ge[i];
316*49fe348cSAndroid Build Coastguard Worker         e_sum += e[i++];
317*49fe348cSAndroid Build Coastguard Worker     }
318*49fe348cSAndroid Build Coastguard Worker 
319*49fe348cSAndroid Build Coastguard Worker     e[LC3_MAX_BANDS-1] = (e0 * 0.25f + e1 * 0.75f) * ge[LC3_MAX_BANDS-1];
320*49fe348cSAndroid Build Coastguard Worker     e_sum += e[LC3_MAX_BANDS-1];
321*49fe348cSAndroid Build Coastguard Worker 
322*49fe348cSAndroid Build Coastguard Worker     float noise_floor = fmaxf(e_sum * (1e-4f / 64), 0x1p-32f);
323*49fe348cSAndroid Build Coastguard Worker 
324*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < LC3_MAX_BANDS; i++)
325*49fe348cSAndroid Build Coastguard Worker         e[i] = lc3_log2f(fmaxf(e[i], noise_floor)) * 0.5f;
326*49fe348cSAndroid Build Coastguard Worker 
327*49fe348cSAndroid Build Coastguard Worker     /* --- Grouping & scaling --- */
328*49fe348cSAndroid Build Coastguard Worker 
329*49fe348cSAndroid Build Coastguard Worker     float scf_sum;
330*49fe348cSAndroid Build Coastguard Worker 
331*49fe348cSAndroid Build Coastguard Worker     scf[0] = (e[0] + e[4]) * 1.f/12 +
332*49fe348cSAndroid Build Coastguard Worker              (e[0] + e[3]) * 2.f/12 +
333*49fe348cSAndroid Build Coastguard Worker              (e[1] + e[2]) * 3.f/12  ;
334*49fe348cSAndroid Build Coastguard Worker     scf_sum = scf[0];
335*49fe348cSAndroid Build Coastguard Worker 
336*49fe348cSAndroid Build Coastguard Worker     for (int i = 1; i < 15; i++) {
337*49fe348cSAndroid Build Coastguard Worker         scf[i] = (e[4*i-1] + e[4*i+4]) * 1.f/12 +
338*49fe348cSAndroid Build Coastguard Worker                  (e[4*i  ] + e[4*i+3]) * 2.f/12 +
339*49fe348cSAndroid Build Coastguard Worker                  (e[4*i+1] + e[4*i+2]) * 3.f/12  ;
340*49fe348cSAndroid Build Coastguard Worker         scf_sum += scf[i];
341*49fe348cSAndroid Build Coastguard Worker     }
342*49fe348cSAndroid Build Coastguard Worker 
343*49fe348cSAndroid Build Coastguard Worker     scf[15] = (e[59] + e[63]) * 1.f/12 +
344*49fe348cSAndroid Build Coastguard Worker               (e[60] + e[63]) * 2.f/12 +
345*49fe348cSAndroid Build Coastguard Worker               (e[61] + e[62]) * 3.f/12  ;
346*49fe348cSAndroid Build Coastguard Worker     scf_sum += scf[15];
347*49fe348cSAndroid Build Coastguard Worker 
348*49fe348cSAndroid Build Coastguard Worker     float cf = lc3_hr(sr) ? 0.6f : 0.85f;
349*49fe348cSAndroid Build Coastguard Worker     if (lc3_hr(sr) && 8 * nbytes >
350*49fe348cSAndroid Build Coastguard Worker             (dt < LC3_DT_10M ? 1150 * (int)(1 + dt) : 4400))
351*49fe348cSAndroid Build Coastguard Worker         cf *= dt < LC3_DT_10M ? 0.25f : 0.35f;
352*49fe348cSAndroid Build Coastguard Worker 
353*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 16; i++)
354*49fe348cSAndroid Build Coastguard Worker         scf[i] = cf * (scf[i] - scf_sum * 1.f/16);
355*49fe348cSAndroid Build Coastguard Worker 
356*49fe348cSAndroid Build Coastguard Worker     /* --- Attack handling --- */
357*49fe348cSAndroid Build Coastguard Worker 
358*49fe348cSAndroid Build Coastguard Worker     if (!att)
359*49fe348cSAndroid Build Coastguard Worker         return;
360*49fe348cSAndroid Build Coastguard Worker 
361*49fe348cSAndroid Build Coastguard Worker     float s0, s1 = scf[0], s2 = scf[1], s3 = scf[2], s4 = scf[3];
362*49fe348cSAndroid Build Coastguard Worker     float sn = s1 + s2;
363*49fe348cSAndroid Build Coastguard Worker 
364*49fe348cSAndroid Build Coastguard Worker     scf[0] = (sn += s3) * 1.f/3;
365*49fe348cSAndroid Build Coastguard Worker     scf[1] = (sn += s4) * 1.f/4;
366*49fe348cSAndroid Build Coastguard Worker     scf_sum = scf[0] + scf[1];
367*49fe348cSAndroid Build Coastguard Worker 
368*49fe348cSAndroid Build Coastguard Worker     for (int i = 2; i < 14; i++, sn -= s0) {
369*49fe348cSAndroid Build Coastguard Worker         s0 = s1, s1 = s2, s2 = s3, s3 = s4, s4 = scf[i+2];
370*49fe348cSAndroid Build Coastguard Worker         scf[i] = (sn += s4) * 1.f/5;
371*49fe348cSAndroid Build Coastguard Worker         scf_sum += scf[i];
372*49fe348cSAndroid Build Coastguard Worker     }
373*49fe348cSAndroid Build Coastguard Worker 
374*49fe348cSAndroid Build Coastguard Worker     scf[14] = (sn      ) * 1.f/4;
375*49fe348cSAndroid Build Coastguard Worker     scf[15] = (sn -= s1) * 1.f/3;
376*49fe348cSAndroid Build Coastguard Worker     scf_sum += scf[14] + scf[15];
377*49fe348cSAndroid Build Coastguard Worker 
378*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 16; i++)
379*49fe348cSAndroid Build Coastguard Worker         scf[i] = (dt == LC3_DT_7M5 ? 0.3f : 0.5f) *
380*49fe348cSAndroid Build Coastguard Worker                  (scf[i] - scf_sum * 1.f/16);
381*49fe348cSAndroid Build Coastguard Worker }
382*49fe348cSAndroid Build Coastguard Worker 
383*49fe348cSAndroid Build Coastguard Worker /**
384*49fe348cSAndroid Build Coastguard Worker  * Codebooks
385*49fe348cSAndroid Build Coastguard Worker  * scf             Input 16 scale factors
386*49fe348cSAndroid Build Coastguard Worker  * lf/hfcb_idx     Output the low and high frequency codebooks index
387*49fe348cSAndroid Build Coastguard Worker  */
resolve_codebooks(const float * scf,int * lfcb_idx,int * hfcb_idx)388*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void resolve_codebooks(
389*49fe348cSAndroid Build Coastguard Worker     const float *scf, int *lfcb_idx, int *hfcb_idx)
390*49fe348cSAndroid Build Coastguard Worker {
391*49fe348cSAndroid Build Coastguard Worker     float dlfcb_max = 0, dhfcb_max = 0;
392*49fe348cSAndroid Build Coastguard Worker     *lfcb_idx = *hfcb_idx = 0;
393*49fe348cSAndroid Build Coastguard Worker 
394*49fe348cSAndroid Build Coastguard Worker     for (int icb = 0; icb < 32; icb++) {
395*49fe348cSAndroid Build Coastguard Worker         const float *lfcb = lc3_sns_lfcb[icb];
396*49fe348cSAndroid Build Coastguard Worker         const float *hfcb = lc3_sns_hfcb[icb];
397*49fe348cSAndroid Build Coastguard Worker         float dlfcb = 0, dhfcb = 0;
398*49fe348cSAndroid Build Coastguard Worker 
399*49fe348cSAndroid Build Coastguard Worker         for (int i = 0; i < 8; i++) {
400*49fe348cSAndroid Build Coastguard Worker             dlfcb += (scf[  i] - lfcb[i]) * (scf[  i] - lfcb[i]);
401*49fe348cSAndroid Build Coastguard Worker             dhfcb += (scf[8+i] - hfcb[i]) * (scf[8+i] - hfcb[i]);
402*49fe348cSAndroid Build Coastguard Worker         }
403*49fe348cSAndroid Build Coastguard Worker 
404*49fe348cSAndroid Build Coastguard Worker         if (icb == 0 || dlfcb < dlfcb_max)
405*49fe348cSAndroid Build Coastguard Worker             *lfcb_idx = icb, dlfcb_max = dlfcb;
406*49fe348cSAndroid Build Coastguard Worker 
407*49fe348cSAndroid Build Coastguard Worker         if (icb == 0 || dhfcb < dhfcb_max)
408*49fe348cSAndroid Build Coastguard Worker             *hfcb_idx = icb, dhfcb_max = dhfcb;
409*49fe348cSAndroid Build Coastguard Worker     }
410*49fe348cSAndroid Build Coastguard Worker }
411*49fe348cSAndroid Build Coastguard Worker 
412*49fe348cSAndroid Build Coastguard Worker /**
413*49fe348cSAndroid Build Coastguard Worker  * Unit energy normalize pulse configuration
414*49fe348cSAndroid Build Coastguard Worker  * c               Pulse configuration
415*49fe348cSAndroid Build Coastguard Worker  * cn              Normalized pulse configuration
416*49fe348cSAndroid Build Coastguard Worker  */
normalize(const int * c,float * cn)417*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void normalize(const int *c, float *cn)
418*49fe348cSAndroid Build Coastguard Worker {
419*49fe348cSAndroid Build Coastguard Worker     int c2_sum = 0;
420*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 16; i++)
421*49fe348cSAndroid Build Coastguard Worker         c2_sum += c[i] * c[i];
422*49fe348cSAndroid Build Coastguard Worker 
423*49fe348cSAndroid Build Coastguard Worker     float c_norm = 1.f / sqrtf(c2_sum);
424*49fe348cSAndroid Build Coastguard Worker 
425*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 16; i++)
426*49fe348cSAndroid Build Coastguard Worker         cn[i] = c[i] * c_norm;
427*49fe348cSAndroid Build Coastguard Worker }
428*49fe348cSAndroid Build Coastguard Worker 
429*49fe348cSAndroid Build Coastguard Worker /**
430*49fe348cSAndroid Build Coastguard Worker  * Sub-procedure of `quantize()`, add unit pulse
431*49fe348cSAndroid Build Coastguard Worker  * x, y, n         Transformed residual, and vector of pulses with length
432*49fe348cSAndroid Build Coastguard Worker  * start, end      Current number of pulses, limit to reach
433*49fe348cSAndroid Build Coastguard Worker  * corr, energy    Correlation (x,y) and y energy, updated at output
434*49fe348cSAndroid Build Coastguard Worker  */
add_pulse(const float * x,int * y,int n,int start,int end,float * corr,float * energy)435*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void add_pulse(const float *x, int *y, int n,
436*49fe348cSAndroid Build Coastguard Worker     int start, int end, float *corr, float *energy)
437*49fe348cSAndroid Build Coastguard Worker {
438*49fe348cSAndroid Build Coastguard Worker     for (int k = start; k < end; k++) {
439*49fe348cSAndroid Build Coastguard Worker         float best_c2 = (*corr + x[0]) * (*corr + x[0]);
440*49fe348cSAndroid Build Coastguard Worker         float best_e = *energy + 2*y[0] + 1;
441*49fe348cSAndroid Build Coastguard Worker         int nbest = 0;
442*49fe348cSAndroid Build Coastguard Worker 
443*49fe348cSAndroid Build Coastguard Worker         for (int i = 1; i < n; i++) {
444*49fe348cSAndroid Build Coastguard Worker             float c2 = (*corr + x[i]) * (*corr + x[i]);
445*49fe348cSAndroid Build Coastguard Worker             float e  = *energy + 2*y[i] + 1;
446*49fe348cSAndroid Build Coastguard Worker 
447*49fe348cSAndroid Build Coastguard Worker             if (c2 * best_e > e * best_c2)
448*49fe348cSAndroid Build Coastguard Worker                 best_c2 = c2, best_e = e, nbest = i;
449*49fe348cSAndroid Build Coastguard Worker         }
450*49fe348cSAndroid Build Coastguard Worker 
451*49fe348cSAndroid Build Coastguard Worker         *corr += x[nbest];
452*49fe348cSAndroid Build Coastguard Worker         *energy += 2*y[nbest] + 1;
453*49fe348cSAndroid Build Coastguard Worker         y[nbest]++;
454*49fe348cSAndroid Build Coastguard Worker     }
455*49fe348cSAndroid Build Coastguard Worker }
456*49fe348cSAndroid Build Coastguard Worker 
457*49fe348cSAndroid Build Coastguard Worker /**
458*49fe348cSAndroid Build Coastguard Worker  * Quantization of codebooks residual
459*49fe348cSAndroid Build Coastguard Worker  * scf             Input 16 scale factors, output quantized version
460*49fe348cSAndroid Build Coastguard Worker  * lf/hfcb_idx     Codebooks index
461*49fe348cSAndroid Build Coastguard Worker  * c, cn           Output 4 pulse configurations candidates, normalized
462*49fe348cSAndroid Build Coastguard Worker  * shape/gain_idx  Output selected shape/gain indexes
463*49fe348cSAndroid Build Coastguard Worker  */
quantize(const float * scf,int lfcb_idx,int hfcb_idx,int (* c)[16],float (* cn)[16],int * shape_idx,int * gain_idx)464*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void quantize(const float *scf, int lfcb_idx, int hfcb_idx,
465*49fe348cSAndroid Build Coastguard Worker     int (*c)[16], float (*cn)[16], int *shape_idx, int *gain_idx)
466*49fe348cSAndroid Build Coastguard Worker {
467*49fe348cSAndroid Build Coastguard Worker     /* --- Residual --- */
468*49fe348cSAndroid Build Coastguard Worker 
469*49fe348cSAndroid Build Coastguard Worker     const float *lfcb = lc3_sns_lfcb[lfcb_idx];
470*49fe348cSAndroid Build Coastguard Worker     const float *hfcb = lc3_sns_hfcb[hfcb_idx];
471*49fe348cSAndroid Build Coastguard Worker     float r[16], x[16];
472*49fe348cSAndroid Build Coastguard Worker 
473*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 8; i++) {
474*49fe348cSAndroid Build Coastguard Worker         r[  i] = scf[  i] - lfcb[i];
475*49fe348cSAndroid Build Coastguard Worker         r[8+i] = scf[8+i] - hfcb[i];
476*49fe348cSAndroid Build Coastguard Worker     }
477*49fe348cSAndroid Build Coastguard Worker 
478*49fe348cSAndroid Build Coastguard Worker     dct16_forward(r, x);
479*49fe348cSAndroid Build Coastguard Worker 
480*49fe348cSAndroid Build Coastguard Worker     /* --- Shape 3 candidate ---
481*49fe348cSAndroid Build Coastguard Worker      * Project to or below pyramid N = 16, K = 6,
482*49fe348cSAndroid Build Coastguard Worker      * then add unit pulses until you reach K = 6, over N = 16 */
483*49fe348cSAndroid Build Coastguard Worker 
484*49fe348cSAndroid Build Coastguard Worker     float xm[16];
485*49fe348cSAndroid Build Coastguard Worker     float xm_sum = 0;
486*49fe348cSAndroid Build Coastguard Worker 
487*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 16; i++) {
488*49fe348cSAndroid Build Coastguard Worker         xm[i] = fabsf(x[i]);
489*49fe348cSAndroid Build Coastguard Worker         xm_sum += xm[i];
490*49fe348cSAndroid Build Coastguard Worker     }
491*49fe348cSAndroid Build Coastguard Worker 
492*49fe348cSAndroid Build Coastguard Worker     float proj_factor = (6 - 1) / fmaxf(xm_sum, 1e-31f);
493*49fe348cSAndroid Build Coastguard Worker     float corr = 0, energy = 0;
494*49fe348cSAndroid Build Coastguard Worker     int npulses = 0;
495*49fe348cSAndroid Build Coastguard Worker 
496*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 16; i++) {
497*49fe348cSAndroid Build Coastguard Worker         c[3][i] = floorf(xm[i] * proj_factor);
498*49fe348cSAndroid Build Coastguard Worker         npulses += c[3][i];
499*49fe348cSAndroid Build Coastguard Worker         corr    += c[3][i] * xm[i];
500*49fe348cSAndroid Build Coastguard Worker         energy  += c[3][i] * c[3][i];
501*49fe348cSAndroid Build Coastguard Worker     }
502*49fe348cSAndroid Build Coastguard Worker 
503*49fe348cSAndroid Build Coastguard Worker     add_pulse(xm, c[3], 16, npulses, 6, &corr, &energy);
504*49fe348cSAndroid Build Coastguard Worker     npulses = 6;
505*49fe348cSAndroid Build Coastguard Worker 
506*49fe348cSAndroid Build Coastguard Worker     /* --- Shape 2 candidate ---
507*49fe348cSAndroid Build Coastguard Worker      * Add unit pulses until you reach K = 8 on shape 3 */
508*49fe348cSAndroid Build Coastguard Worker 
509*49fe348cSAndroid Build Coastguard Worker     memcpy(c[2], c[3], sizeof(c[2]));
510*49fe348cSAndroid Build Coastguard Worker 
511*49fe348cSAndroid Build Coastguard Worker     add_pulse(xm, c[2], 16, npulses, 8, &corr, &energy);
512*49fe348cSAndroid Build Coastguard Worker     npulses = 8;
513*49fe348cSAndroid Build Coastguard Worker 
514*49fe348cSAndroid Build Coastguard Worker     /* --- Shape 1 candidate ---
515*49fe348cSAndroid Build Coastguard Worker      * Remove any unit pulses from shape 2 that are not part of 0 to 9
516*49fe348cSAndroid Build Coastguard Worker      * Update energy and correlation terms accordingly
517*49fe348cSAndroid Build Coastguard Worker      * Add unit pulses until you reach K = 10, over N = 10 */
518*49fe348cSAndroid Build Coastguard Worker 
519*49fe348cSAndroid Build Coastguard Worker     memcpy(c[1], c[2], sizeof(c[1]));
520*49fe348cSAndroid Build Coastguard Worker 
521*49fe348cSAndroid Build Coastguard Worker     for (int i = 10; i < 16; i++) {
522*49fe348cSAndroid Build Coastguard Worker         c[1][i] = 0;
523*49fe348cSAndroid Build Coastguard Worker         npulses -= c[2][i];
524*49fe348cSAndroid Build Coastguard Worker         corr    -= c[2][i] * xm[i];
525*49fe348cSAndroid Build Coastguard Worker         energy  -= c[2][i] * c[2][i];
526*49fe348cSAndroid Build Coastguard Worker     }
527*49fe348cSAndroid Build Coastguard Worker 
528*49fe348cSAndroid Build Coastguard Worker     add_pulse(xm, c[1], 10, npulses, 10, &corr, &energy);
529*49fe348cSAndroid Build Coastguard Worker     npulses = 10;
530*49fe348cSAndroid Build Coastguard Worker 
531*49fe348cSAndroid Build Coastguard Worker     /* --- Shape 0 candidate ---
532*49fe348cSAndroid Build Coastguard Worker      * Add unit pulses until you reach K = 1, on shape 1 */
533*49fe348cSAndroid Build Coastguard Worker 
534*49fe348cSAndroid Build Coastguard Worker     memcpy(c[0], c[1], sizeof(c[0]));
535*49fe348cSAndroid Build Coastguard Worker 
536*49fe348cSAndroid Build Coastguard Worker     add_pulse(xm + 10, c[0] + 10, 6, 0, 1, &corr, &energy);
537*49fe348cSAndroid Build Coastguard Worker 
538*49fe348cSAndroid Build Coastguard Worker     /* --- Add sign and unit energy normalize --- */
539*49fe348cSAndroid Build Coastguard Worker 
540*49fe348cSAndroid Build Coastguard Worker     for (int j = 0; j < 16; j++)
541*49fe348cSAndroid Build Coastguard Worker         for (int i = 0; i < 4; i++)
542*49fe348cSAndroid Build Coastguard Worker             c[i][j] = x[j] < 0 ? -c[i][j] : c[i][j];
543*49fe348cSAndroid Build Coastguard Worker 
544*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 4; i++)
545*49fe348cSAndroid Build Coastguard Worker         normalize(c[i], cn[i]);
546*49fe348cSAndroid Build Coastguard Worker 
547*49fe348cSAndroid Build Coastguard Worker     /* --- Determe shape & gain index ---
548*49fe348cSAndroid Build Coastguard Worker      * Search the Mean Square Error, within (shape, gain) combinations */
549*49fe348cSAndroid Build Coastguard Worker 
550*49fe348cSAndroid Build Coastguard Worker     float mse_min = FLT_MAX;
551*49fe348cSAndroid Build Coastguard Worker     *shape_idx = *gain_idx = 0;
552*49fe348cSAndroid Build Coastguard Worker 
553*49fe348cSAndroid Build Coastguard Worker     for (int ic = 0; ic < 4; ic++) {
554*49fe348cSAndroid Build Coastguard Worker         const struct lc3_sns_vq_gains *cgains = lc3_sns_vq_gains + ic;
555*49fe348cSAndroid Build Coastguard Worker         float cmse_min = FLT_MAX;
556*49fe348cSAndroid Build Coastguard Worker         int cgain_idx = 0;
557*49fe348cSAndroid Build Coastguard Worker 
558*49fe348cSAndroid Build Coastguard Worker         for (int ig = 0; ig < cgains->count; ig++) {
559*49fe348cSAndroid Build Coastguard Worker             float g = cgains->v[ig];
560*49fe348cSAndroid Build Coastguard Worker 
561*49fe348cSAndroid Build Coastguard Worker             float mse = 0;
562*49fe348cSAndroid Build Coastguard Worker             for (int i = 0; i < 16; i++)
563*49fe348cSAndroid Build Coastguard Worker                 mse += (x[i] - g * cn[ic][i]) * (x[i] - g * cn[ic][i]);
564*49fe348cSAndroid Build Coastguard Worker 
565*49fe348cSAndroid Build Coastguard Worker             if (mse < cmse_min) {
566*49fe348cSAndroid Build Coastguard Worker                 cgain_idx = ig,
567*49fe348cSAndroid Build Coastguard Worker                 cmse_min = mse;
568*49fe348cSAndroid Build Coastguard Worker             }
569*49fe348cSAndroid Build Coastguard Worker         }
570*49fe348cSAndroid Build Coastguard Worker 
571*49fe348cSAndroid Build Coastguard Worker         if (cmse_min < mse_min) {
572*49fe348cSAndroid Build Coastguard Worker             *shape_idx = ic, *gain_idx = cgain_idx;
573*49fe348cSAndroid Build Coastguard Worker             mse_min = cmse_min;
574*49fe348cSAndroid Build Coastguard Worker         }
575*49fe348cSAndroid Build Coastguard Worker     }
576*49fe348cSAndroid Build Coastguard Worker }
577*49fe348cSAndroid Build Coastguard Worker 
578*49fe348cSAndroid Build Coastguard Worker /**
579*49fe348cSAndroid Build Coastguard Worker  * Unquantization of codebooks residual
580*49fe348cSAndroid Build Coastguard Worker  * lf/hfcb_idx     Low and high frequency codebooks index
581*49fe348cSAndroid Build Coastguard Worker  * c               Table of normalized pulse configuration
582*49fe348cSAndroid Build Coastguard Worker  * shape/gain      Selected shape/gain indexes
583*49fe348cSAndroid Build Coastguard Worker  * scf             Return unquantized scale factors
584*49fe348cSAndroid Build Coastguard Worker  */
unquantize(int lfcb_idx,int hfcb_idx,const float * c,int shape,int gain,float * scf)585*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void unquantize(int lfcb_idx, int hfcb_idx,
586*49fe348cSAndroid Build Coastguard Worker     const float *c, int shape, int gain, float *scf)
587*49fe348cSAndroid Build Coastguard Worker {
588*49fe348cSAndroid Build Coastguard Worker     const float *lfcb = lc3_sns_lfcb[lfcb_idx];
589*49fe348cSAndroid Build Coastguard Worker     const float *hfcb = lc3_sns_hfcb[hfcb_idx];
590*49fe348cSAndroid Build Coastguard Worker     float g = lc3_sns_vq_gains[shape].v[gain];
591*49fe348cSAndroid Build Coastguard Worker 
592*49fe348cSAndroid Build Coastguard Worker     dct16_inverse(c, scf);
593*49fe348cSAndroid Build Coastguard Worker 
594*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 8; i++)
595*49fe348cSAndroid Build Coastguard Worker         scf[i] = lfcb[i] + g * scf[i];
596*49fe348cSAndroid Build Coastguard Worker 
597*49fe348cSAndroid Build Coastguard Worker     for (int i = 8; i < 16; i++)
598*49fe348cSAndroid Build Coastguard Worker         scf[i] = hfcb[i-8] + g * scf[i];
599*49fe348cSAndroid Build Coastguard Worker }
600*49fe348cSAndroid Build Coastguard Worker 
601*49fe348cSAndroid Build Coastguard Worker /**
602*49fe348cSAndroid Build Coastguard Worker  * Sub-procedure of `sns_enumerate()`, enumeration of a vector
603*49fe348cSAndroid Build Coastguard Worker  * c, n            Table of pulse configuration, and length
604*49fe348cSAndroid Build Coastguard Worker  * idx, ls         Return enumeration set
605*49fe348cSAndroid Build Coastguard Worker  */
enum_mvpq(const int * c,int n,int * idx,bool * ls)606*49fe348cSAndroid Build Coastguard Worker static void enum_mvpq(const int *c, int n, int *idx, bool *ls)
607*49fe348cSAndroid Build Coastguard Worker {
608*49fe348cSAndroid Build Coastguard Worker     int ci, i;
609*49fe348cSAndroid Build Coastguard Worker 
610*49fe348cSAndroid Build Coastguard Worker     /* --- Scan for 1st significant coeff --- */
611*49fe348cSAndroid Build Coastguard Worker 
612*49fe348cSAndroid Build Coastguard Worker     for (i = 0, c += n; (ci = *(--c)) == 0 && i < 15; i++);
613*49fe348cSAndroid Build Coastguard Worker 
614*49fe348cSAndroid Build Coastguard Worker     *idx = 0;
615*49fe348cSAndroid Build Coastguard Worker     *ls = ci < 0;
616*49fe348cSAndroid Build Coastguard Worker 
617*49fe348cSAndroid Build Coastguard Worker     /* --- Scan remaining coefficients --- */
618*49fe348cSAndroid Build Coastguard Worker 
619*49fe348cSAndroid Build Coastguard Worker     unsigned j = LC3_ABS(ci);
620*49fe348cSAndroid Build Coastguard Worker 
621*49fe348cSAndroid Build Coastguard Worker     for (i++; i < n; i++, j += LC3_ABS(ci)) {
622*49fe348cSAndroid Build Coastguard Worker 
623*49fe348cSAndroid Build Coastguard Worker         if ((ci = *(--c)) != 0) {
624*49fe348cSAndroid Build Coastguard Worker             *idx = (*idx << 1) | *ls;
625*49fe348cSAndroid Build Coastguard Worker             *ls = ci < 0;
626*49fe348cSAndroid Build Coastguard Worker         }
627*49fe348cSAndroid Build Coastguard Worker 
628*49fe348cSAndroid Build Coastguard Worker         *idx += lc3_sns_mpvq_offsets[i][LC3_MIN(j, 10)];
629*49fe348cSAndroid Build Coastguard Worker     }
630*49fe348cSAndroid Build Coastguard Worker }
631*49fe348cSAndroid Build Coastguard Worker 
632*49fe348cSAndroid Build Coastguard Worker /**
633*49fe348cSAndroid Build Coastguard Worker  * Sub-procedure of `sns_deenumerate()`, deenumeration of a vector
634*49fe348cSAndroid Build Coastguard Worker  * idx, ls         Enumeration set
635*49fe348cSAndroid Build Coastguard Worker  * npulses         Number of pulses in the set
636*49fe348cSAndroid Build Coastguard Worker  * c, n            Table of pulses configuration, and length
637*49fe348cSAndroid Build Coastguard Worker  */
deenum_mvpq(int idx,bool ls,int npulses,int * c,int n)638*49fe348cSAndroid Build Coastguard Worker static void deenum_mvpq(int idx, bool ls, int npulses, int *c, int n)
639*49fe348cSAndroid Build Coastguard Worker {
640*49fe348cSAndroid Build Coastguard Worker     int i;
641*49fe348cSAndroid Build Coastguard Worker 
642*49fe348cSAndroid Build Coastguard Worker     /* --- Scan for coefficients --- */
643*49fe348cSAndroid Build Coastguard Worker 
644*49fe348cSAndroid Build Coastguard Worker     for (i = n-1; i >= 0 && idx; i--) {
645*49fe348cSAndroid Build Coastguard Worker 
646*49fe348cSAndroid Build Coastguard Worker         int ci = 0;
647*49fe348cSAndroid Build Coastguard Worker 
648*49fe348cSAndroid Build Coastguard Worker         for (ci = 0; idx < lc3_sns_mpvq_offsets[i][npulses - ci]; ci++);
649*49fe348cSAndroid Build Coastguard Worker         idx -= lc3_sns_mpvq_offsets[i][npulses - ci];
650*49fe348cSAndroid Build Coastguard Worker 
651*49fe348cSAndroid Build Coastguard Worker         *(c++) = ls ? -ci : ci;
652*49fe348cSAndroid Build Coastguard Worker         npulses -= ci;
653*49fe348cSAndroid Build Coastguard Worker         if (ci > 0) {
654*49fe348cSAndroid Build Coastguard Worker             ls = idx & 1;
655*49fe348cSAndroid Build Coastguard Worker             idx >>= 1;
656*49fe348cSAndroid Build Coastguard Worker         }
657*49fe348cSAndroid Build Coastguard Worker     }
658*49fe348cSAndroid Build Coastguard Worker 
659*49fe348cSAndroid Build Coastguard Worker     /* --- Set last significant --- */
660*49fe348cSAndroid Build Coastguard Worker 
661*49fe348cSAndroid Build Coastguard Worker     int ci = npulses;
662*49fe348cSAndroid Build Coastguard Worker 
663*49fe348cSAndroid Build Coastguard Worker     if (i-- >= 0)
664*49fe348cSAndroid Build Coastguard Worker         *(c++) = ls ? -ci : ci;
665*49fe348cSAndroid Build Coastguard Worker 
666*49fe348cSAndroid Build Coastguard Worker     while (i-- >= 0)
667*49fe348cSAndroid Build Coastguard Worker         *(c++) = 0;
668*49fe348cSAndroid Build Coastguard Worker }
669*49fe348cSAndroid Build Coastguard Worker 
670*49fe348cSAndroid Build Coastguard Worker /**
671*49fe348cSAndroid Build Coastguard Worker  * SNS Enumeration of PVQ configuration
672*49fe348cSAndroid Build Coastguard Worker  * shape           Selected shape index
673*49fe348cSAndroid Build Coastguard Worker  * c               Selected pulse configuration
674*49fe348cSAndroid Build Coastguard Worker  * idx_a, ls_a     Return enumeration set A
675*49fe348cSAndroid Build Coastguard Worker  * idx_b, ls_b     Return enumeration set B (shape = 0)
676*49fe348cSAndroid Build Coastguard Worker  */
enumerate(int shape,const int * c,int * idx_a,bool * ls_a,int * idx_b,bool * ls_b)677*49fe348cSAndroid Build Coastguard Worker static void enumerate(int shape, const int *c,
678*49fe348cSAndroid Build Coastguard Worker     int *idx_a, bool *ls_a, int *idx_b, bool *ls_b)
679*49fe348cSAndroid Build Coastguard Worker {
680*49fe348cSAndroid Build Coastguard Worker     enum_mvpq(c, shape < 2 ? 10 : 16, idx_a, ls_a);
681*49fe348cSAndroid Build Coastguard Worker 
682*49fe348cSAndroid Build Coastguard Worker     if (shape == 0)
683*49fe348cSAndroid Build Coastguard Worker         enum_mvpq(c + 10, 6, idx_b, ls_b);
684*49fe348cSAndroid Build Coastguard Worker }
685*49fe348cSAndroid Build Coastguard Worker 
686*49fe348cSAndroid Build Coastguard Worker /**
687*49fe348cSAndroid Build Coastguard Worker  * SNS Deenumeration of PVQ configuration
688*49fe348cSAndroid Build Coastguard Worker  * shape           Selected shape index
689*49fe348cSAndroid Build Coastguard Worker  * idx_a, ls_a     enumeration set A
690*49fe348cSAndroid Build Coastguard Worker  * idx_b, ls_b     enumeration set B (shape = 0)
691*49fe348cSAndroid Build Coastguard Worker  * c               Return pulse configuration
692*49fe348cSAndroid Build Coastguard Worker  */
deenumerate(int shape,int idx_a,bool ls_a,int idx_b,bool ls_b,int * c)693*49fe348cSAndroid Build Coastguard Worker static void deenumerate(int shape,
694*49fe348cSAndroid Build Coastguard Worker     int idx_a, bool ls_a, int idx_b, bool ls_b, int *c)
695*49fe348cSAndroid Build Coastguard Worker {
696*49fe348cSAndroid Build Coastguard Worker     int npulses_a = (const int []){ 10, 10, 8, 6 }[shape];
697*49fe348cSAndroid Build Coastguard Worker 
698*49fe348cSAndroid Build Coastguard Worker     deenum_mvpq(idx_a, ls_a, npulses_a, c, shape < 2 ? 10 : 16);
699*49fe348cSAndroid Build Coastguard Worker 
700*49fe348cSAndroid Build Coastguard Worker     if (shape == 0)
701*49fe348cSAndroid Build Coastguard Worker         deenum_mvpq(idx_b, ls_b, 1, c + 10, 6);
702*49fe348cSAndroid Build Coastguard Worker     else if (shape == 1)
703*49fe348cSAndroid Build Coastguard Worker         memset(c + 10, 0, 6 * sizeof(*c));
704*49fe348cSAndroid Build Coastguard Worker }
705*49fe348cSAndroid Build Coastguard Worker 
706*49fe348cSAndroid Build Coastguard Worker 
707*49fe348cSAndroid Build Coastguard Worker /* ----------------------------------------------------------------------------
708*49fe348cSAndroid Build Coastguard Worker  *  Filtering
709*49fe348cSAndroid Build Coastguard Worker  * -------------------------------------------------------------------------- */
710*49fe348cSAndroid Build Coastguard Worker 
711*49fe348cSAndroid Build Coastguard Worker /**
712*49fe348cSAndroid Build Coastguard Worker  * Spectral shaping
713*49fe348cSAndroid Build Coastguard Worker  * dt, sr          Duration and samplerate of the frame
714*49fe348cSAndroid Build Coastguard Worker  * scf_q           Quantized scale factors
715*49fe348cSAndroid Build Coastguard Worker  * inv             True on inverse shaping, False otherwise
716*49fe348cSAndroid Build Coastguard Worker  * x               Spectral coefficients
717*49fe348cSAndroid Build Coastguard Worker  * y               Return shapped coefficients
718*49fe348cSAndroid Build Coastguard Worker  *
719*49fe348cSAndroid Build Coastguard Worker  * `x` and `y` can be the same buffer
720*49fe348cSAndroid Build Coastguard Worker  */
spectral_shaping(enum lc3_dt dt,enum lc3_srate sr,const float * scf_q,bool inv,const float * x,float * y)721*49fe348cSAndroid Build Coastguard Worker LC3_HOT static void spectral_shaping(enum lc3_dt dt, enum lc3_srate sr,
722*49fe348cSAndroid Build Coastguard Worker     const float *scf_q, bool inv, const float *x, float *y)
723*49fe348cSAndroid Build Coastguard Worker {
724*49fe348cSAndroid Build Coastguard Worker     /* --- Interpolate scale factors --- */
725*49fe348cSAndroid Build Coastguard Worker 
726*49fe348cSAndroid Build Coastguard Worker     float scf[LC3_MAX_BANDS];
727*49fe348cSAndroid Build Coastguard Worker     float s0, s1 = inv ? -scf_q[0] : scf_q[0];
728*49fe348cSAndroid Build Coastguard Worker 
729*49fe348cSAndroid Build Coastguard Worker     scf[0] = scf[1] = s1;
730*49fe348cSAndroid Build Coastguard Worker     for (int i = 0; i < 15; i++) {
731*49fe348cSAndroid Build Coastguard Worker         s0 = s1, s1 = inv ? -scf_q[i+1] : scf_q[i+1];
732*49fe348cSAndroid Build Coastguard Worker         scf[4*i+2] = s0 + 0.125f * (s1 - s0);
733*49fe348cSAndroid Build Coastguard Worker         scf[4*i+3] = s0 + 0.375f * (s1 - s0);
734*49fe348cSAndroid Build Coastguard Worker         scf[4*i+4] = s0 + 0.625f * (s1 - s0);
735*49fe348cSAndroid Build Coastguard Worker         scf[4*i+5] = s0 + 0.875f * (s1 - s0);
736*49fe348cSAndroid Build Coastguard Worker     }
737*49fe348cSAndroid Build Coastguard Worker     scf[62] = s1 + 0.125f * (s1 - s0);
738*49fe348cSAndroid Build Coastguard Worker     scf[63] = s1 + 0.375f * (s1 - s0);
739*49fe348cSAndroid Build Coastguard Worker 
740*49fe348cSAndroid Build Coastguard Worker     int nb = lc3_num_bands[dt][sr];
741*49fe348cSAndroid Build Coastguard Worker     int n4 = nb < 32 ? 32 % nb : 0;
742*49fe348cSAndroid Build Coastguard Worker     int n2 = nb < 32 ? nb - n4 : LC3_MAX_BANDS - nb;
743*49fe348cSAndroid Build Coastguard Worker 
744*49fe348cSAndroid Build Coastguard Worker     for (int i4 = 0; i4 < n4; i4++)
745*49fe348cSAndroid Build Coastguard Worker         scf[i4] = 0.25f * (scf[4*i4+0] + scf[4*i4+1] +
746*49fe348cSAndroid Build Coastguard Worker                            scf[4*i4+2] + scf[4*i4+3]);
747*49fe348cSAndroid Build Coastguard Worker 
748*49fe348cSAndroid Build Coastguard Worker     for (int i2 = n4; i2 < n4+n2; i2++)
749*49fe348cSAndroid Build Coastguard Worker         scf[i2] = 0.5f * (scf[2*(n4+i2)] + scf[2*(n4+i2)+1]);
750*49fe348cSAndroid Build Coastguard Worker 
751*49fe348cSAndroid Build Coastguard Worker     memmove(scf + n4 + n2, scf + 4*n4 + 2*n2, (nb - n4 - n2) * sizeof(float));
752*49fe348cSAndroid Build Coastguard Worker 
753*49fe348cSAndroid Build Coastguard Worker     /* --- Spectral shaping --- */
754*49fe348cSAndroid Build Coastguard Worker 
755*49fe348cSAndroid Build Coastguard Worker     const int *lim = lc3_band_lim[dt][sr];
756*49fe348cSAndroid Build Coastguard Worker 
757*49fe348cSAndroid Build Coastguard Worker     for (int i = 0, ib = 0; ib < nb; ib++) {
758*49fe348cSAndroid Build Coastguard Worker         float g_sns = lc3_exp2f(-scf[ib]);
759*49fe348cSAndroid Build Coastguard Worker 
760*49fe348cSAndroid Build Coastguard Worker         for ( ; i < lim[ib+1]; i++)
761*49fe348cSAndroid Build Coastguard Worker             y[i] = x[i] * g_sns;
762*49fe348cSAndroid Build Coastguard Worker     }
763*49fe348cSAndroid Build Coastguard Worker }
764*49fe348cSAndroid Build Coastguard Worker 
765*49fe348cSAndroid Build Coastguard Worker 
766*49fe348cSAndroid Build Coastguard Worker /* ----------------------------------------------------------------------------
767*49fe348cSAndroid Build Coastguard Worker  *  Interface
768*49fe348cSAndroid Build Coastguard Worker  * -------------------------------------------------------------------------- */
769*49fe348cSAndroid Build Coastguard Worker 
770*49fe348cSAndroid Build Coastguard Worker /**
771*49fe348cSAndroid Build Coastguard Worker  * SNS analysis
772*49fe348cSAndroid Build Coastguard Worker  */
lc3_sns_analyze(enum lc3_dt dt,enum lc3_srate sr,int nbytes,const float * eb,bool att,struct lc3_sns_data * data,const float * x,float * y)773*49fe348cSAndroid Build Coastguard Worker void lc3_sns_analyze(
774*49fe348cSAndroid Build Coastguard Worker     enum lc3_dt dt, enum lc3_srate sr, int nbytes,
775*49fe348cSAndroid Build Coastguard Worker     const float *eb, bool att, struct lc3_sns_data *data,
776*49fe348cSAndroid Build Coastguard Worker     const float *x, float *y)
777*49fe348cSAndroid Build Coastguard Worker {
778*49fe348cSAndroid Build Coastguard Worker     /* Processing steps :
779*49fe348cSAndroid Build Coastguard Worker      * - Determine 16 scale factors from bands energy estimation
780*49fe348cSAndroid Build Coastguard Worker      * - Get codebooks indexes that match thoses scale factors
781*49fe348cSAndroid Build Coastguard Worker      * - Quantize the residual with the selected codebook
782*49fe348cSAndroid Build Coastguard Worker      * - The pulse configuration `c[]` is enumerated
783*49fe348cSAndroid Build Coastguard Worker      * - Finally shape the spectrum coefficients accordingly */
784*49fe348cSAndroid Build Coastguard Worker 
785*49fe348cSAndroid Build Coastguard Worker     float scf[16], cn[4][16];
786*49fe348cSAndroid Build Coastguard Worker     int c[4][16];
787*49fe348cSAndroid Build Coastguard Worker 
788*49fe348cSAndroid Build Coastguard Worker     compute_scale_factors(dt, sr, nbytes, eb, att, scf);
789*49fe348cSAndroid Build Coastguard Worker 
790*49fe348cSAndroid Build Coastguard Worker     resolve_codebooks(scf, &data->lfcb, &data->hfcb);
791*49fe348cSAndroid Build Coastguard Worker 
792*49fe348cSAndroid Build Coastguard Worker     quantize(scf, data->lfcb, data->hfcb,
793*49fe348cSAndroid Build Coastguard Worker         c, cn, &data->shape, &data->gain);
794*49fe348cSAndroid Build Coastguard Worker 
795*49fe348cSAndroid Build Coastguard Worker     unquantize(data->lfcb, data->hfcb,
796*49fe348cSAndroid Build Coastguard Worker         cn[data->shape], data->shape, data->gain, scf);
797*49fe348cSAndroid Build Coastguard Worker 
798*49fe348cSAndroid Build Coastguard Worker     enumerate(data->shape, c[data->shape],
799*49fe348cSAndroid Build Coastguard Worker         &data->idx_a, &data->ls_a, &data->idx_b, &data->ls_b);
800*49fe348cSAndroid Build Coastguard Worker 
801*49fe348cSAndroid Build Coastguard Worker     spectral_shaping(dt, sr, scf, false, x, y);
802*49fe348cSAndroid Build Coastguard Worker }
803*49fe348cSAndroid Build Coastguard Worker 
804*49fe348cSAndroid Build Coastguard Worker /**
805*49fe348cSAndroid Build Coastguard Worker  * SNS synthesis
806*49fe348cSAndroid Build Coastguard Worker  */
lc3_sns_synthesize(enum lc3_dt dt,enum lc3_srate sr,const lc3_sns_data_t * data,const float * x,float * y)807*49fe348cSAndroid Build Coastguard Worker void lc3_sns_synthesize(
808*49fe348cSAndroid Build Coastguard Worker     enum lc3_dt dt, enum lc3_srate sr,
809*49fe348cSAndroid Build Coastguard Worker     const lc3_sns_data_t *data, const float *x, float *y)
810*49fe348cSAndroid Build Coastguard Worker {
811*49fe348cSAndroid Build Coastguard Worker     float scf[16], cn[16];
812*49fe348cSAndroid Build Coastguard Worker     int c[16];
813*49fe348cSAndroid Build Coastguard Worker 
814*49fe348cSAndroid Build Coastguard Worker     deenumerate(data->shape,
815*49fe348cSAndroid Build Coastguard Worker         data->idx_a, data->ls_a, data->idx_b, data->ls_b, c);
816*49fe348cSAndroid Build Coastguard Worker 
817*49fe348cSAndroid Build Coastguard Worker     normalize(c, cn);
818*49fe348cSAndroid Build Coastguard Worker 
819*49fe348cSAndroid Build Coastguard Worker     unquantize(data->lfcb, data->hfcb, cn, data->shape, data->gain, scf);
820*49fe348cSAndroid Build Coastguard Worker 
821*49fe348cSAndroid Build Coastguard Worker     spectral_shaping(dt, sr, scf, true, x, y);
822*49fe348cSAndroid Build Coastguard Worker }
823*49fe348cSAndroid Build Coastguard Worker 
824*49fe348cSAndroid Build Coastguard Worker /**
825*49fe348cSAndroid Build Coastguard Worker  * Return number of bits coding the bitstream data
826*49fe348cSAndroid Build Coastguard Worker  */
lc3_sns_get_nbits(void)827*49fe348cSAndroid Build Coastguard Worker int lc3_sns_get_nbits(void)
828*49fe348cSAndroid Build Coastguard Worker {
829*49fe348cSAndroid Build Coastguard Worker     return 38;
830*49fe348cSAndroid Build Coastguard Worker }
831*49fe348cSAndroid Build Coastguard Worker 
832*49fe348cSAndroid Build Coastguard Worker /**
833*49fe348cSAndroid Build Coastguard Worker  * Put bitstream data
834*49fe348cSAndroid Build Coastguard Worker  */
lc3_sns_put_data(lc3_bits_t * bits,const struct lc3_sns_data * data)835*49fe348cSAndroid Build Coastguard Worker void lc3_sns_put_data(lc3_bits_t *bits, const struct lc3_sns_data *data)
836*49fe348cSAndroid Build Coastguard Worker {
837*49fe348cSAndroid Build Coastguard Worker     /* --- Codebooks --- */
838*49fe348cSAndroid Build Coastguard Worker 
839*49fe348cSAndroid Build Coastguard Worker     lc3_put_bits(bits, data->lfcb, 5);
840*49fe348cSAndroid Build Coastguard Worker     lc3_put_bits(bits, data->hfcb, 5);
841*49fe348cSAndroid Build Coastguard Worker 
842*49fe348cSAndroid Build Coastguard Worker     /* --- Shape, gain and vectors --- *
843*49fe348cSAndroid Build Coastguard Worker      * Write MSB bit of shape index, next LSB bits of shape and gain,
844*49fe348cSAndroid Build Coastguard Worker      * and MVPQ vectors indexes are muxed */
845*49fe348cSAndroid Build Coastguard Worker 
846*49fe348cSAndroid Build Coastguard Worker     int shape_msb = data->shape >> 1;
847*49fe348cSAndroid Build Coastguard Worker     lc3_put_bit(bits, shape_msb);
848*49fe348cSAndroid Build Coastguard Worker 
849*49fe348cSAndroid Build Coastguard Worker     if (shape_msb == 0) {
850*49fe348cSAndroid Build Coastguard Worker         const int size_a = 2390004;
851*49fe348cSAndroid Build Coastguard Worker         int submode = data->shape & 1;
852*49fe348cSAndroid Build Coastguard Worker 
853*49fe348cSAndroid Build Coastguard Worker         int mux_high = submode == 0 ?
854*49fe348cSAndroid Build Coastguard Worker             2 * (data->idx_b + 1) + data->ls_b : data->gain & 1;
855*49fe348cSAndroid Build Coastguard Worker         int mux_code = mux_high * size_a + data->idx_a;
856*49fe348cSAndroid Build Coastguard Worker 
857*49fe348cSAndroid Build Coastguard Worker         lc3_put_bits(bits, data->gain >> submode, 1);
858*49fe348cSAndroid Build Coastguard Worker         lc3_put_bits(bits, data->ls_a, 1);
859*49fe348cSAndroid Build Coastguard Worker         lc3_put_bits(bits, mux_code, 25);
860*49fe348cSAndroid Build Coastguard Worker 
861*49fe348cSAndroid Build Coastguard Worker     } else {
862*49fe348cSAndroid Build Coastguard Worker         const int size_a = 15158272;
863*49fe348cSAndroid Build Coastguard Worker         int submode = data->shape & 1;
864*49fe348cSAndroid Build Coastguard Worker 
865*49fe348cSAndroid Build Coastguard Worker         int mux_code = submode == 0 ?
866*49fe348cSAndroid Build Coastguard Worker             data->idx_a : size_a + 2 * data->idx_a + (data->gain & 1);
867*49fe348cSAndroid Build Coastguard Worker 
868*49fe348cSAndroid Build Coastguard Worker         lc3_put_bits(bits, data->gain >> submode, 2);
869*49fe348cSAndroid Build Coastguard Worker         lc3_put_bits(bits, data->ls_a, 1);
870*49fe348cSAndroid Build Coastguard Worker         lc3_put_bits(bits, mux_code, 24);
871*49fe348cSAndroid Build Coastguard Worker     }
872*49fe348cSAndroid Build Coastguard Worker }
873*49fe348cSAndroid Build Coastguard Worker 
874*49fe348cSAndroid Build Coastguard Worker /**
875*49fe348cSAndroid Build Coastguard Worker  * Get bitstream data
876*49fe348cSAndroid Build Coastguard Worker  */
lc3_sns_get_data(lc3_bits_t * bits,struct lc3_sns_data * data)877*49fe348cSAndroid Build Coastguard Worker int lc3_sns_get_data(lc3_bits_t *bits, struct lc3_sns_data *data)
878*49fe348cSAndroid Build Coastguard Worker {
879*49fe348cSAndroid Build Coastguard Worker     /* --- Codebooks --- */
880*49fe348cSAndroid Build Coastguard Worker 
881*49fe348cSAndroid Build Coastguard Worker     *data = (struct lc3_sns_data){
882*49fe348cSAndroid Build Coastguard Worker         .lfcb = lc3_get_bits(bits, 5),
883*49fe348cSAndroid Build Coastguard Worker         .hfcb = lc3_get_bits(bits, 5)
884*49fe348cSAndroid Build Coastguard Worker     };
885*49fe348cSAndroid Build Coastguard Worker 
886*49fe348cSAndroid Build Coastguard Worker     /* --- Shape, gain and vectors --- */
887*49fe348cSAndroid Build Coastguard Worker 
888*49fe348cSAndroid Build Coastguard Worker     int shape_msb = lc3_get_bit(bits);
889*49fe348cSAndroid Build Coastguard Worker     data->gain = lc3_get_bits(bits, 1 + shape_msb);
890*49fe348cSAndroid Build Coastguard Worker     data->ls_a = lc3_get_bit(bits);
891*49fe348cSAndroid Build Coastguard Worker 
892*49fe348cSAndroid Build Coastguard Worker     int mux_code = lc3_get_bits(bits, 25 - shape_msb);
893*49fe348cSAndroid Build Coastguard Worker 
894*49fe348cSAndroid Build Coastguard Worker     if (shape_msb == 0) {
895*49fe348cSAndroid Build Coastguard Worker         const int size_a = 2390004;
896*49fe348cSAndroid Build Coastguard Worker 
897*49fe348cSAndroid Build Coastguard Worker         if (mux_code >= size_a * 14)
898*49fe348cSAndroid Build Coastguard Worker             return -1;
899*49fe348cSAndroid Build Coastguard Worker 
900*49fe348cSAndroid Build Coastguard Worker         data->idx_a = mux_code % size_a;
901*49fe348cSAndroid Build Coastguard Worker         mux_code = mux_code / size_a;
902*49fe348cSAndroid Build Coastguard Worker 
903*49fe348cSAndroid Build Coastguard Worker         data->shape = (mux_code < 2);
904*49fe348cSAndroid Build Coastguard Worker 
905*49fe348cSAndroid Build Coastguard Worker         if (data->shape == 0) {
906*49fe348cSAndroid Build Coastguard Worker             data->idx_b = (mux_code - 2) / 2;
907*49fe348cSAndroid Build Coastguard Worker             data->ls_b  = (mux_code - 2) % 2;
908*49fe348cSAndroid Build Coastguard Worker         } else {
909*49fe348cSAndroid Build Coastguard Worker             data->gain = (data->gain << 1) + (mux_code % 2);
910*49fe348cSAndroid Build Coastguard Worker         }
911*49fe348cSAndroid Build Coastguard Worker 
912*49fe348cSAndroid Build Coastguard Worker     } else {
913*49fe348cSAndroid Build Coastguard Worker         const int size_a = 15158272;
914*49fe348cSAndroid Build Coastguard Worker 
915*49fe348cSAndroid Build Coastguard Worker         if (mux_code >= size_a + 1549824)
916*49fe348cSAndroid Build Coastguard Worker             return -1;
917*49fe348cSAndroid Build Coastguard Worker 
918*49fe348cSAndroid Build Coastguard Worker         data->shape = 2 + (mux_code >= size_a);
919*49fe348cSAndroid Build Coastguard Worker         if (data->shape == 2) {
920*49fe348cSAndroid Build Coastguard Worker             data->idx_a = mux_code;
921*49fe348cSAndroid Build Coastguard Worker         } else {
922*49fe348cSAndroid Build Coastguard Worker             mux_code -= size_a;
923*49fe348cSAndroid Build Coastguard Worker             data->idx_a = mux_code / 2;
924*49fe348cSAndroid Build Coastguard Worker             data->gain = (data->gain << 1) + (mux_code % 2);
925*49fe348cSAndroid Build Coastguard Worker         }
926*49fe348cSAndroid Build Coastguard Worker     }
927*49fe348cSAndroid Build Coastguard Worker 
928*49fe348cSAndroid Build Coastguard Worker     return 0;
929*49fe348cSAndroid Build Coastguard Worker }
930