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