19a19cd78SMatthias Ringwald /******************************************************************************
29a19cd78SMatthias Ringwald *
34930cef6SMatthias Ringwald * Copyright 2022 Google LLC
49a19cd78SMatthias Ringwald *
59a19cd78SMatthias Ringwald * Licensed under the Apache License, Version 2.0 (the "License");
69a19cd78SMatthias Ringwald * you may not use this file except in compliance with the License.
79a19cd78SMatthias Ringwald * You may obtain a copy of the License at:
89a19cd78SMatthias Ringwald *
99a19cd78SMatthias Ringwald * http://www.apache.org/licenses/LICENSE-2.0
109a19cd78SMatthias Ringwald *
119a19cd78SMatthias Ringwald * Unless required by applicable law or agreed to in writing, software
129a19cd78SMatthias Ringwald * distributed under the License is distributed on an "AS IS" BASIS,
139a19cd78SMatthias Ringwald * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
149a19cd78SMatthias Ringwald * See the License for the specific language governing permissions and
159a19cd78SMatthias Ringwald * limitations under the License.
169a19cd78SMatthias Ringwald *
179a19cd78SMatthias Ringwald ******************************************************************************/
189a19cd78SMatthias Ringwald
199a19cd78SMatthias Ringwald #include "sns.h"
209a19cd78SMatthias Ringwald #include "tables.h"
219a19cd78SMatthias Ringwald
229a19cd78SMatthias Ringwald
239a19cd78SMatthias Ringwald /* ----------------------------------------------------------------------------
249a19cd78SMatthias Ringwald * DCT-16
259a19cd78SMatthias Ringwald * -------------------------------------------------------------------------- */
269a19cd78SMatthias Ringwald
279a19cd78SMatthias Ringwald /**
289a19cd78SMatthias Ringwald * Matrix of DCT-16 coefficients
299a19cd78SMatthias Ringwald *
309a19cd78SMatthias Ringwald * M[n][k] = 2f cos( Pi k (2n + 1) / 2N )
319a19cd78SMatthias Ringwald *
329a19cd78SMatthias Ringwald * k = [0..N-1], n = [0..N-1], N = 16
339a19cd78SMatthias Ringwald * f = sqrt(1/4N) for k=0, sqrt(1/2N) otherwise
349a19cd78SMatthias Ringwald */
359a19cd78SMatthias Ringwald static const float dct16_m[16][16] = {
369a19cd78SMatthias Ringwald
379a19cd78SMatthias Ringwald { 2.50000000e-01, 3.51850934e-01, 3.46759961e-01, 3.38329500e-01,
389a19cd78SMatthias Ringwald 3.26640741e-01, 3.11806253e-01, 2.93968901e-01, 2.73300467e-01,
399a19cd78SMatthias Ringwald 2.50000000e-01, 2.24291897e-01, 1.96423740e-01, 1.66663915e-01,
409a19cd78SMatthias Ringwald 1.35299025e-01, 1.02631132e-01, 6.89748448e-02, 3.46542923e-02 },
419a19cd78SMatthias Ringwald
429a19cd78SMatthias Ringwald { 2.50000000e-01, 3.38329500e-01, 2.93968901e-01, 2.24291897e-01,
439a19cd78SMatthias Ringwald 1.35299025e-01, 3.46542923e-02, -6.89748448e-02, -1.66663915e-01,
449a19cd78SMatthias Ringwald -2.50000000e-01, -3.11806253e-01, -3.46759961e-01, -3.51850934e-01,
459a19cd78SMatthias Ringwald -3.26640741e-01, -2.73300467e-01, -1.96423740e-01, -1.02631132e-01 },
469a19cd78SMatthias Ringwald
479a19cd78SMatthias Ringwald { 2.50000000e-01, 3.11806253e-01, 1.96423740e-01, 3.46542923e-02,
489a19cd78SMatthias Ringwald -1.35299025e-01, -2.73300467e-01, -3.46759961e-01, -3.38329500e-01,
499a19cd78SMatthias Ringwald -2.50000000e-01, -1.02631132e-01, 6.89748448e-02, 2.24291897e-01,
509a19cd78SMatthias Ringwald 3.26640741e-01, 3.51850934e-01, 2.93968901e-01, 1.66663915e-01 },
519a19cd78SMatthias Ringwald
529a19cd78SMatthias Ringwald { 2.50000000e-01, 2.73300467e-01, 6.89748448e-02, -1.66663915e-01,
539a19cd78SMatthias Ringwald -3.26640741e-01, -3.38329500e-01, -1.96423740e-01, 3.46542923e-02,
549a19cd78SMatthias Ringwald 2.50000000e-01, 3.51850934e-01, 2.93968901e-01, 1.02631132e-01,
559a19cd78SMatthias Ringwald -1.35299025e-01, -3.11806253e-01, -3.46759961e-01, -2.24291897e-01 },
569a19cd78SMatthias Ringwald
579a19cd78SMatthias Ringwald { 2.50000000e-01, 2.24291897e-01, -6.89748448e-02, -3.11806253e-01,
589a19cd78SMatthias Ringwald -3.26640741e-01, -1.02631132e-01, 1.96423740e-01, 3.51850934e-01,
599a19cd78SMatthias Ringwald 2.50000000e-01, -3.46542923e-02, -2.93968901e-01, -3.38329500e-01,
609a19cd78SMatthias Ringwald -1.35299025e-01, 1.66663915e-01, 3.46759961e-01, 2.73300467e-01 },
619a19cd78SMatthias Ringwald
629a19cd78SMatthias Ringwald { 2.50000000e-01, 1.66663915e-01, -1.96423740e-01, -3.51850934e-01,
639a19cd78SMatthias Ringwald -1.35299025e-01, 2.24291897e-01, 3.46759961e-01, 1.02631132e-01,
649a19cd78SMatthias Ringwald -2.50000000e-01, -3.38329500e-01, -6.89748448e-02, 2.73300467e-01,
659a19cd78SMatthias Ringwald 3.26640741e-01, 3.46542923e-02, -2.93968901e-01, -3.11806253e-01 },
669a19cd78SMatthias Ringwald
679a19cd78SMatthias Ringwald { 2.50000000e-01, 1.02631132e-01, -2.93968901e-01, -2.73300467e-01,
689a19cd78SMatthias Ringwald 1.35299025e-01, 3.51850934e-01, 6.89748448e-02, -3.11806253e-01,
699a19cd78SMatthias Ringwald -2.50000000e-01, 1.66663915e-01, 3.46759961e-01, 3.46542923e-02,
709a19cd78SMatthias Ringwald -3.26640741e-01, -2.24291897e-01, 1.96423740e-01, 3.38329500e-01 },
719a19cd78SMatthias Ringwald
729a19cd78SMatthias Ringwald { 2.50000000e-01, 3.46542923e-02, -3.46759961e-01, -1.02631132e-01,
739a19cd78SMatthias Ringwald 3.26640741e-01, 1.66663915e-01, -2.93968901e-01, -2.24291897e-01,
749a19cd78SMatthias Ringwald 2.50000000e-01, 2.73300467e-01, -1.96423740e-01, -3.11806253e-01,
759a19cd78SMatthias Ringwald 1.35299025e-01, 3.38329500e-01, -6.89748448e-02, -3.51850934e-01 },
769a19cd78SMatthias Ringwald
779a19cd78SMatthias Ringwald { 2.50000000e-01, -3.46542923e-02, -3.46759961e-01, 1.02631132e-01,
789a19cd78SMatthias Ringwald 3.26640741e-01, -1.66663915e-01, -2.93968901e-01, 2.24291897e-01,
799a19cd78SMatthias Ringwald 2.50000000e-01, -2.73300467e-01, -1.96423740e-01, 3.11806253e-01,
809a19cd78SMatthias Ringwald 1.35299025e-01, -3.38329500e-01, -6.89748448e-02, 3.51850934e-01 },
819a19cd78SMatthias Ringwald
829a19cd78SMatthias Ringwald { 2.50000000e-01, -1.02631132e-01, -2.93968901e-01, 2.73300467e-01,
839a19cd78SMatthias Ringwald 1.35299025e-01, -3.51850934e-01, 6.89748448e-02, 3.11806253e-01,
849a19cd78SMatthias Ringwald -2.50000000e-01, -1.66663915e-01, 3.46759961e-01, -3.46542923e-02,
859a19cd78SMatthias Ringwald -3.26640741e-01, 2.24291897e-01, 1.96423740e-01, -3.38329500e-01 },
869a19cd78SMatthias Ringwald
879a19cd78SMatthias Ringwald { 2.50000000e-01, -1.66663915e-01, -1.96423740e-01, 3.51850934e-01,
889a19cd78SMatthias Ringwald -1.35299025e-01, -2.24291897e-01, 3.46759961e-01, -1.02631132e-01,
899a19cd78SMatthias Ringwald -2.50000000e-01, 3.38329500e-01, -6.89748448e-02, -2.73300467e-01,
909a19cd78SMatthias Ringwald 3.26640741e-01, -3.46542923e-02, -2.93968901e-01, 3.11806253e-01 },
919a19cd78SMatthias Ringwald
929a19cd78SMatthias Ringwald { 2.50000000e-01, -2.24291897e-01, -6.89748448e-02, 3.11806253e-01,
939a19cd78SMatthias Ringwald -3.26640741e-01, 1.02631132e-01, 1.96423740e-01, -3.51850934e-01,
949a19cd78SMatthias Ringwald 2.50000000e-01, 3.46542923e-02, -2.93968901e-01, 3.38329500e-01,
959a19cd78SMatthias Ringwald -1.35299025e-01, -1.66663915e-01, 3.46759961e-01, -2.73300467e-01 },
969a19cd78SMatthias Ringwald
979a19cd78SMatthias Ringwald { 2.50000000e-01, -2.73300467e-01, 6.89748448e-02, 1.66663915e-01,
989a19cd78SMatthias Ringwald -3.26640741e-01, 3.38329500e-01, -1.96423740e-01, -3.46542923e-02,
999a19cd78SMatthias Ringwald 2.50000000e-01, -3.51850934e-01, 2.93968901e-01, -1.02631132e-01,
1009a19cd78SMatthias Ringwald -1.35299025e-01, 3.11806253e-01, -3.46759961e-01, 2.24291897e-01 },
1019a19cd78SMatthias Ringwald
1029a19cd78SMatthias Ringwald { 2.50000000e-01, -3.11806253e-01, 1.96423740e-01, -3.46542923e-02,
1039a19cd78SMatthias Ringwald -1.35299025e-01, 2.73300467e-01, -3.46759961e-01, 3.38329500e-01,
1049a19cd78SMatthias Ringwald -2.50000000e-01, 1.02631132e-01, 6.89748448e-02, -2.24291897e-01,
1059a19cd78SMatthias Ringwald 3.26640741e-01, -3.51850934e-01, 2.93968901e-01, -1.66663915e-01 },
1069a19cd78SMatthias Ringwald
1079a19cd78SMatthias Ringwald { 2.50000000e-01, -3.38329500e-01, 2.93968901e-01, -2.24291897e-01,
1089a19cd78SMatthias Ringwald 1.35299025e-01, -3.46542923e-02, -6.89748448e-02, 1.66663915e-01,
1099a19cd78SMatthias Ringwald -2.50000000e-01, 3.11806253e-01, -3.46759961e-01, 3.51850934e-01,
1109a19cd78SMatthias Ringwald -3.26640741e-01, 2.73300467e-01, -1.96423740e-01, 1.02631132e-01 },
1119a19cd78SMatthias Ringwald
1129a19cd78SMatthias Ringwald { 2.50000000e-01, -3.51850934e-01, 3.46759961e-01, -3.38329500e-01,
1139a19cd78SMatthias Ringwald 3.26640741e-01, -3.11806253e-01, 2.93968901e-01, -2.73300467e-01,
1149a19cd78SMatthias Ringwald 2.50000000e-01, -2.24291897e-01, 1.96423740e-01, -1.66663915e-01,
1159a19cd78SMatthias Ringwald 1.35299025e-01, -1.02631132e-01, 6.89748448e-02, -3.46542923e-02 },
1169a19cd78SMatthias Ringwald
1179a19cd78SMatthias Ringwald };
1189a19cd78SMatthias Ringwald
1199a19cd78SMatthias Ringwald /**
1209a19cd78SMatthias Ringwald * Forward DCT-16 transformation
1219a19cd78SMatthias Ringwald * x, y Input and output 16 values
1229a19cd78SMatthias Ringwald */
dct16_forward(const float * x,float * y)1234930cef6SMatthias Ringwald LC3_HOT static void dct16_forward(const float *x, float *y)
1249a19cd78SMatthias Ringwald {
1259a19cd78SMatthias Ringwald for (int i = 0, j; i < 16; i++)
1269a19cd78SMatthias Ringwald for (y[i] = 0, j = 0; j < 16; j++)
1279a19cd78SMatthias Ringwald y[i] += x[j] * dct16_m[j][i];
1289a19cd78SMatthias Ringwald }
1299a19cd78SMatthias Ringwald
1309a19cd78SMatthias Ringwald /**
1319a19cd78SMatthias Ringwald * Inverse DCT-16 transformation
1329a19cd78SMatthias Ringwald * x, y Input and output 16 values
1339a19cd78SMatthias Ringwald */
dct16_inverse(const float * x,float * y)1344930cef6SMatthias Ringwald LC3_HOT static void dct16_inverse(const float *x, float *y)
1359a19cd78SMatthias Ringwald {
1369a19cd78SMatthias Ringwald for (int i = 0, j; i < 16; i++)
1379a19cd78SMatthias Ringwald for (y[i] = 0, j = 0; j < 16; j++)
1389a19cd78SMatthias Ringwald y[i] += x[j] * dct16_m[i][j];
1399a19cd78SMatthias Ringwald }
1409a19cd78SMatthias Ringwald
1419a19cd78SMatthias Ringwald
1429a19cd78SMatthias Ringwald /* ----------------------------------------------------------------------------
1439a19cd78SMatthias Ringwald * Scale factors
1449a19cd78SMatthias Ringwald * -------------------------------------------------------------------------- */
1459a19cd78SMatthias Ringwald
1469a19cd78SMatthias Ringwald /**
1479a19cd78SMatthias Ringwald * Scale factors
1489a19cd78SMatthias Ringwald * dt, sr Duration and samplerate of the frame
149*6897da5cSDirk Helbig * nbytes Size in bytes of the frame
1509a19cd78SMatthias Ringwald * eb Energy estimation per bands
1519a19cd78SMatthias Ringwald * att 1: Attack detected 0: Otherwise
1529a19cd78SMatthias Ringwald * scf Output 16 scale factors
1539a19cd78SMatthias Ringwald */
compute_scale_factors(enum lc3_dt dt,enum lc3_srate sr,int nbytes,const float * eb,bool att,float * scf)1544930cef6SMatthias Ringwald LC3_HOT static void compute_scale_factors(
155*6897da5cSDirk Helbig enum lc3_dt dt, enum lc3_srate sr, int nbytes,
1569a19cd78SMatthias Ringwald const float *eb, bool att, float *scf)
1579a19cd78SMatthias Ringwald {
1589a19cd78SMatthias Ringwald /* Pre-emphasis gain table :
1599a19cd78SMatthias Ringwald * Ge[b] = 10 ^ (b * g_tilt) / 630 , b = [0..63] */
1609a19cd78SMatthias Ringwald
161*6897da5cSDirk Helbig static const float ge_14[LC3_MAX_BANDS] = { /* g_tilt = 14 */
1629a19cd78SMatthias Ringwald 1.00000000e+00, 1.05250029e+00, 1.10775685e+00, 1.16591440e+00,
1639a19cd78SMatthias Ringwald 1.22712524e+00, 1.29154967e+00, 1.35935639e+00, 1.43072299e+00,
1649a19cd78SMatthias Ringwald 1.50583635e+00, 1.58489319e+00, 1.66810054e+00, 1.75567629e+00,
1659a19cd78SMatthias Ringwald 1.84784980e+00, 1.94486244e+00, 2.04696827e+00, 2.15443469e+00,
1669a19cd78SMatthias Ringwald 2.26754313e+00, 2.38658979e+00, 2.51188643e+00, 2.64376119e+00,
1679a19cd78SMatthias Ringwald 2.78255940e+00, 2.92864456e+00, 3.08239924e+00, 3.24422608e+00,
1689a19cd78SMatthias Ringwald 3.41454887e+00, 3.59381366e+00, 3.78248991e+00, 3.98107171e+00,
1699a19cd78SMatthias Ringwald 4.19007911e+00, 4.41005945e+00, 4.64158883e+00, 4.88527357e+00,
1709a19cd78SMatthias Ringwald 5.14175183e+00, 5.41169527e+00, 5.69581081e+00, 5.99484250e+00,
1719a19cd78SMatthias Ringwald 6.30957344e+00, 6.64082785e+00, 6.98947321e+00, 7.35642254e+00,
1729a19cd78SMatthias Ringwald 7.74263683e+00, 8.14912747e+00, 8.57695899e+00, 9.02725178e+00,
1739a19cd78SMatthias Ringwald 9.50118507e+00, 1.00000000e+01, 1.05250029e+01, 1.10775685e+01,
1749a19cd78SMatthias Ringwald 1.16591440e+01, 1.22712524e+01, 1.29154967e+01, 1.35935639e+01,
1759a19cd78SMatthias Ringwald 1.43072299e+01, 1.50583635e+01, 1.58489319e+01, 1.66810054e+01,
1769a19cd78SMatthias Ringwald 1.75567629e+01, 1.84784980e+01, 1.94486244e+01, 2.04696827e+01,
177*6897da5cSDirk Helbig 2.15443469e+01, 2.26754313e+01, 2.38658979e+01, 2.51188643e+01 };
1789a19cd78SMatthias Ringwald
179*6897da5cSDirk Helbig static const float ge_18[LC3_MAX_BANDS] = { /* g_tilt = 18 */
1809a19cd78SMatthias Ringwald 1.00000000e+00, 1.06800043e+00, 1.14062492e+00, 1.21818791e+00,
1819a19cd78SMatthias Ringwald 1.30102522e+00, 1.38949549e+00, 1.48398179e+00, 1.58489319e+00,
1829a19cd78SMatthias Ringwald 1.69266662e+00, 1.80776868e+00, 1.93069773e+00, 2.06198601e+00,
1839a19cd78SMatthias Ringwald 2.20220195e+00, 2.35195264e+00, 2.51188643e+00, 2.68269580e+00,
1849a19cd78SMatthias Ringwald 2.86512027e+00, 3.05994969e+00, 3.26802759e+00, 3.49025488e+00,
1859a19cd78SMatthias Ringwald 3.72759372e+00, 3.98107171e+00, 4.25178630e+00, 4.54090961e+00,
1869a19cd78SMatthias Ringwald 4.84969343e+00, 5.17947468e+00, 5.53168120e+00, 5.90783791e+00,
1879a19cd78SMatthias Ringwald 6.30957344e+00, 6.73862717e+00, 7.19685673e+00, 7.68624610e+00,
1889a19cd78SMatthias Ringwald 8.20891416e+00, 8.76712387e+00, 9.36329209e+00, 1.00000000e+01,
1899a19cd78SMatthias Ringwald 1.06800043e+01, 1.14062492e+01, 1.21818791e+01, 1.30102522e+01,
1909a19cd78SMatthias Ringwald 1.38949549e+01, 1.48398179e+01, 1.58489319e+01, 1.69266662e+01,
1919a19cd78SMatthias Ringwald 1.80776868e+01, 1.93069773e+01, 2.06198601e+01, 2.20220195e+01,
1929a19cd78SMatthias Ringwald 2.35195264e+01, 2.51188643e+01, 2.68269580e+01, 2.86512027e+01,
1939a19cd78SMatthias Ringwald 3.05994969e+01, 3.26802759e+01, 3.49025488e+01, 3.72759372e+01,
1949a19cd78SMatthias Ringwald 3.98107171e+01, 4.25178630e+01, 4.54090961e+01, 4.84969343e+01,
195*6897da5cSDirk Helbig 5.17947468e+01, 5.53168120e+01, 5.90783791e+01, 6.30957344e+01 };
1969a19cd78SMatthias Ringwald
197*6897da5cSDirk Helbig static const float ge_22[LC3_MAX_BANDS] = { /* g_tilt = 22 */
1989a19cd78SMatthias Ringwald 1.00000000e+00, 1.08372885e+00, 1.17446822e+00, 1.27280509e+00,
1999a19cd78SMatthias Ringwald 1.37937560e+00, 1.49486913e+00, 1.62003281e+00, 1.75567629e+00,
2009a19cd78SMatthias Ringwald 1.90267705e+00, 2.06198601e+00, 2.23463373e+00, 2.42173704e+00,
2019a19cd78SMatthias Ringwald 2.62450630e+00, 2.84425319e+00, 3.08239924e+00, 3.34048498e+00,
2029a19cd78SMatthias Ringwald 3.62017995e+00, 3.92329345e+00, 4.25178630e+00, 4.60778348e+00,
2039a19cd78SMatthias Ringwald 4.99358789e+00, 5.41169527e+00, 5.86481029e+00, 6.35586411e+00,
2049a19cd78SMatthias Ringwald 6.88803330e+00, 7.46476041e+00, 8.08977621e+00, 8.76712387e+00,
2059a19cd78SMatthias Ringwald 9.50118507e+00, 1.02967084e+01, 1.11588399e+01, 1.20931568e+01,
2069a19cd78SMatthias Ringwald 1.31057029e+01, 1.42030283e+01, 1.53922315e+01, 1.66810054e+01,
2079a19cd78SMatthias Ringwald 1.80776868e+01, 1.95913107e+01, 2.12316686e+01, 2.30093718e+01,
2089a19cd78SMatthias Ringwald 2.49359200e+01, 2.70237760e+01, 2.92864456e+01, 3.17385661e+01,
2099a19cd78SMatthias Ringwald 3.43959997e+01, 3.72759372e+01, 4.03970086e+01, 4.37794036e+01,
2109a19cd78SMatthias Ringwald 4.74450028e+01, 5.14175183e+01, 5.57226480e+01, 6.03882412e+01,
2119a19cd78SMatthias Ringwald 6.54444792e+01, 7.09240702e+01, 7.68624610e+01, 8.32980665e+01,
2129a19cd78SMatthias Ringwald 9.02725178e+01, 9.78309319e+01, 1.06022203e+02, 1.14899320e+02,
213*6897da5cSDirk Helbig 1.24519708e+02, 1.34945600e+02, 1.46244440e+02, 1.58489319e+02 };
2149a19cd78SMatthias Ringwald
215*6897da5cSDirk Helbig static const float ge_26[LC3_MAX_BANDS] = { /* g_tilt = 26 */
2169a19cd78SMatthias Ringwald 1.00000000e+00, 1.09968890e+00, 1.20931568e+00, 1.32987103e+00,
2179a19cd78SMatthias Ringwald 1.46244440e+00, 1.60823388e+00, 1.76855694e+00, 1.94486244e+00,
2189a19cd78SMatthias Ringwald 2.13874364e+00, 2.35195264e+00, 2.58641621e+00, 2.84425319e+00,
2199a19cd78SMatthias Ringwald 3.12779366e+00, 3.43959997e+00, 3.78248991e+00, 4.15956216e+00,
2209a19cd78SMatthias Ringwald 4.57422434e+00, 5.03022373e+00, 5.53168120e+00, 6.08312841e+00,
2219a19cd78SMatthias Ringwald 6.68954879e+00, 7.35642254e+00, 8.08977621e+00, 8.89623710e+00,
2229a19cd78SMatthias Ringwald 9.78309319e+00, 1.07583590e+01, 1.18308480e+01, 1.30102522e+01,
2239a19cd78SMatthias Ringwald 1.43072299e+01, 1.57335019e+01, 1.73019574e+01, 1.90267705e+01,
2249a19cd78SMatthias Ringwald 2.09235283e+01, 2.30093718e+01, 2.53031508e+01, 2.78255940e+01,
2259a19cd78SMatthias Ringwald 3.05994969e+01, 3.36499270e+01, 3.70044512e+01, 4.06933843e+01,
2269a19cd78SMatthias Ringwald 4.47500630e+01, 4.92111475e+01, 5.41169527e+01, 5.95118121e+01,
2279a19cd78SMatthias Ringwald 6.54444792e+01, 7.19685673e+01, 7.91430346e+01, 8.70327166e+01,
2289a19cd78SMatthias Ringwald 9.57089124e+01, 1.05250029e+02, 1.15742288e+02, 1.27280509e+02,
2299a19cd78SMatthias Ringwald 1.39968963e+02, 1.53922315e+02, 1.69266662e+02, 1.86140669e+02,
2309a19cd78SMatthias Ringwald 2.04696827e+02, 2.25102829e+02, 2.47543082e+02, 2.72220379e+02,
231*6897da5cSDirk Helbig 2.99357729e+02, 3.29200372e+02, 3.62017995e+02, 3.98107171e+02 };
2329a19cd78SMatthias Ringwald
233*6897da5cSDirk Helbig static const float ge_30[LC3_MAX_BANDS] = { /* g_tilt = 30 */
2349a19cd78SMatthias Ringwald 1.00000000e+00, 1.11588399e+00, 1.24519708e+00, 1.38949549e+00,
2359a19cd78SMatthias Ringwald 1.55051578e+00, 1.73019574e+00, 1.93069773e+00, 2.15443469e+00,
2369a19cd78SMatthias Ringwald 2.40409918e+00, 2.68269580e+00, 2.99357729e+00, 3.34048498e+00,
2379a19cd78SMatthias Ringwald 3.72759372e+00, 4.15956216e+00, 4.64158883e+00, 5.17947468e+00,
2389a19cd78SMatthias Ringwald 5.77969288e+00, 6.44946677e+00, 7.19685673e+00, 8.03085722e+00,
2399a19cd78SMatthias Ringwald 8.96150502e+00, 1.00000000e+01, 1.11588399e+01, 1.24519708e+01,
2409a19cd78SMatthias Ringwald 1.38949549e+01, 1.55051578e+01, 1.73019574e+01, 1.93069773e+01,
2419a19cd78SMatthias Ringwald 2.15443469e+01, 2.40409918e+01, 2.68269580e+01, 2.99357729e+01,
2429a19cd78SMatthias Ringwald 3.34048498e+01, 3.72759372e+01, 4.15956216e+01, 4.64158883e+01,
2439a19cd78SMatthias Ringwald 5.17947468e+01, 5.77969288e+01, 6.44946677e+01, 7.19685673e+01,
2449a19cd78SMatthias Ringwald 8.03085722e+01, 8.96150502e+01, 1.00000000e+02, 1.11588399e+02,
2459a19cd78SMatthias Ringwald 1.24519708e+02, 1.38949549e+02, 1.55051578e+02, 1.73019574e+02,
2469a19cd78SMatthias Ringwald 1.93069773e+02, 2.15443469e+02, 2.40409918e+02, 2.68269580e+02,
2479a19cd78SMatthias Ringwald 2.99357729e+02, 3.34048498e+02, 3.72759372e+02, 4.15956216e+02,
2489a19cd78SMatthias Ringwald 4.64158883e+02, 5.17947468e+02, 5.77969288e+02, 6.44946677e+02,
249*6897da5cSDirk Helbig 7.19685673e+02, 8.03085722e+02, 8.96150502e+02, 1.00000000e+03 };
250*6897da5cSDirk Helbig
251*6897da5cSDirk Helbig #if LC3_PLUS_HR
252*6897da5cSDirk Helbig
253*6897da5cSDirk Helbig static const float ge_34[LC3_MAX_BANDS] = { /* g_tilt = 34 */
254*6897da5cSDirk Helbig 1.00000000e+00, 1.13231759e+00, 1.28214312e+00, 1.45179321e+00,
255*6897da5cSDirk Helbig 1.64389099e+00, 1.86140669e+00, 2.10770353e+00, 2.38658979e+00,
256*6897da5cSDirk Helbig 2.70237760e+00, 3.05994969e+00, 3.46483486e+00, 3.92329345e+00,
257*6897da5cSDirk Helbig 4.44241419e+00, 5.03022373e+00, 5.69581081e+00, 6.44946677e+00,
258*6897da5cSDirk Helbig 7.30284467e+00, 8.26913948e+00, 9.36329209e+00, 1.06022203e+01,
259*6897da5cSDirk Helbig 1.20050806e+01, 1.35935639e+01, 1.53922315e+01, 1.74288945e+01,
260*6897da5cSDirk Helbig 1.97350438e+01, 2.23463373e+01, 2.53031508e+01, 2.86512027e+01,
261*6897da5cSDirk Helbig 3.24422608e+01, 3.67349426e+01, 4.15956216e+01, 4.70994540e+01,
262*6897da5cSDirk Helbig 5.33315403e+01, 6.03882412e+01, 6.83786677e+01, 7.74263683e+01,
263*6897da5cSDirk Helbig 8.76712387e+01, 9.92716858e+01, 1.12407076e+02, 1.27280509e+02,
264*6897da5cSDirk Helbig 1.44121960e+02, 1.63191830e+02, 1.84784980e+02, 2.09235283e+02,
265*6897da5cSDirk Helbig 2.36920791e+02, 2.68269580e+02, 3.03766364e+02, 3.43959997e+02,
266*6897da5cSDirk Helbig 3.89471955e+02, 4.41005945e+02, 4.99358789e+02, 5.65432741e+02,
267*6897da5cSDirk Helbig 6.40249439e+02, 7.24965701e+02, 8.20891416e+02, 9.29509790e+02,
268*6897da5cSDirk Helbig 1.05250029e+03, 1.19176459e+03, 1.34945600e+03, 1.52801277e+03,
269*6897da5cSDirk Helbig 1.73019574e+03, 1.95913107e+03, 2.21835857e+03, 2.51188643e+03 };
270*6897da5cSDirk Helbig
271*6897da5cSDirk Helbig #endif /* LC3_PLUS_HR */
272*6897da5cSDirk Helbig
273*6897da5cSDirk Helbig static const float *ge_table[LC3_NUM_SRATE] = {
274*6897da5cSDirk Helbig [LC3_SRATE_8K ] = ge_14, [LC3_SRATE_16K ] = ge_18,
275*6897da5cSDirk Helbig [LC3_SRATE_24K ] = ge_22, [LC3_SRATE_32K ] = ge_26,
276*6897da5cSDirk Helbig [LC3_SRATE_48K ] = ge_30,
277*6897da5cSDirk Helbig
278*6897da5cSDirk Helbig #if LC3_PLUS_HR
279*6897da5cSDirk Helbig [LC3_SRATE_48K_HR] = ge_30, [LC3_SRATE_96K_HR] = ge_34,
280*6897da5cSDirk Helbig #endif /* LC3_PLUS_HR */
281*6897da5cSDirk Helbig
2829a19cd78SMatthias Ringwald };
2839a19cd78SMatthias Ringwald
284*6897da5cSDirk Helbig float e[LC3_MAX_BANDS];
2859a19cd78SMatthias Ringwald
2869a19cd78SMatthias Ringwald /* --- Copy and padding --- */
2879a19cd78SMatthias Ringwald
288*6897da5cSDirk Helbig int nb = lc3_num_bands[dt][sr];
289*6897da5cSDirk Helbig int n4 = nb < 32 ? 32 % nb : 0;
290*6897da5cSDirk Helbig int n2 = nb < 32 ? nb - n4 : LC3_MAX_BANDS - nb;
2919a19cd78SMatthias Ringwald
292*6897da5cSDirk Helbig for (int i4 = 0; i4 < n4; i4++)
293*6897da5cSDirk Helbig e[4*i4 + 0] = e[4*i4 + 1] =
294*6897da5cSDirk Helbig e[4*i4 + 2] = e[4*i4 + 3] = eb[i4];
2959a19cd78SMatthias Ringwald
296*6897da5cSDirk Helbig for (int i2 = n4; i2 < n4+n2; i2++)
297*6897da5cSDirk Helbig e[2*(n4+i2) + 0] = e[2*(n4+i2) + 1] = eb[i2];
298*6897da5cSDirk Helbig
299*6897da5cSDirk Helbig memcpy(e + 4*n4 + 2*n2, eb + n4 + n2, (nb - n4 - n2) * sizeof(float));
3009a19cd78SMatthias Ringwald
3019a19cd78SMatthias Ringwald /* --- Smoothing, pre-emphasis and logarithm --- */
3029a19cd78SMatthias Ringwald
3039a19cd78SMatthias Ringwald const float *ge = ge_table[sr];
3049a19cd78SMatthias Ringwald
3059a19cd78SMatthias Ringwald float e0 = e[0], e1 = e[0], e2;
3069a19cd78SMatthias Ringwald float e_sum = 0;
3079a19cd78SMatthias Ringwald
308*6897da5cSDirk Helbig for (int i = 0; i < LC3_MAX_BANDS-1; ) {
3094930cef6SMatthias Ringwald e[i] = (e0 * 0.25f + e1 * 0.5f + (e2 = e[i+1]) * 0.25f) * ge[i];
3109a19cd78SMatthias Ringwald e_sum += e[i++];
3119a19cd78SMatthias Ringwald
3124930cef6SMatthias Ringwald e[i] = (e1 * 0.25f + e2 * 0.5f + (e0 = e[i+1]) * 0.25f) * ge[i];
3139a19cd78SMatthias Ringwald e_sum += e[i++];
3149a19cd78SMatthias Ringwald
3154930cef6SMatthias Ringwald e[i] = (e2 * 0.25f + e0 * 0.5f + (e1 = e[i+1]) * 0.25f) * ge[i];
3169a19cd78SMatthias Ringwald e_sum += e[i++];
3179a19cd78SMatthias Ringwald }
3189a19cd78SMatthias Ringwald
319*6897da5cSDirk Helbig e[LC3_MAX_BANDS-1] = (e0 * 0.25f + e1 * 0.75f) * ge[LC3_MAX_BANDS-1];
320*6897da5cSDirk Helbig e_sum += e[LC3_MAX_BANDS-1];
3219a19cd78SMatthias Ringwald
3224930cef6SMatthias Ringwald float noise_floor = fmaxf(e_sum * (1e-4f / 64), 0x1p-32f);
3239a19cd78SMatthias Ringwald
324*6897da5cSDirk Helbig for (int i = 0; i < LC3_MAX_BANDS; i++)
325*6897da5cSDirk Helbig e[i] = lc3_log2f(fmaxf(e[i], noise_floor)) * 0.5f;
3269a19cd78SMatthias Ringwald
3279a19cd78SMatthias Ringwald /* --- Grouping & scaling --- */
3289a19cd78SMatthias Ringwald
3299a19cd78SMatthias Ringwald float scf_sum;
3309a19cd78SMatthias Ringwald
3314930cef6SMatthias Ringwald scf[0] = (e[0] + e[4]) * 1.f/12 +
3324930cef6SMatthias Ringwald (e[0] + e[3]) * 2.f/12 +
3334930cef6SMatthias Ringwald (e[1] + e[2]) * 3.f/12 ;
3349a19cd78SMatthias Ringwald scf_sum = scf[0];
3359a19cd78SMatthias Ringwald
3369a19cd78SMatthias Ringwald for (int i = 1; i < 15; i++) {
3374930cef6SMatthias Ringwald scf[i] = (e[4*i-1] + e[4*i+4]) * 1.f/12 +
3384930cef6SMatthias Ringwald (e[4*i ] + e[4*i+3]) * 2.f/12 +
3394930cef6SMatthias Ringwald (e[4*i+1] + e[4*i+2]) * 3.f/12 ;
3409a19cd78SMatthias Ringwald scf_sum += scf[i];
3419a19cd78SMatthias Ringwald }
3429a19cd78SMatthias Ringwald
3434930cef6SMatthias Ringwald scf[15] = (e[59] + e[63]) * 1.f/12 +
3444930cef6SMatthias Ringwald (e[60] + e[63]) * 2.f/12 +
3454930cef6SMatthias Ringwald (e[61] + e[62]) * 3.f/12 ;
3469a19cd78SMatthias Ringwald scf_sum += scf[15];
3479a19cd78SMatthias Ringwald
348*6897da5cSDirk Helbig float cf = lc3_hr(sr) ? 0.6f : 0.85f;
349*6897da5cSDirk Helbig if (lc3_hr(sr) && 8 * nbytes >
350*6897da5cSDirk Helbig (dt < LC3_DT_10M ? 1150 * (int)(1 + dt) : 4400))
351*6897da5cSDirk Helbig cf *= dt < LC3_DT_10M ? 0.25f : 0.35f;
352*6897da5cSDirk Helbig
3539a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++)
354*6897da5cSDirk Helbig scf[i] = cf * (scf[i] - scf_sum * 1.f/16);
3559a19cd78SMatthias Ringwald
3569a19cd78SMatthias Ringwald /* --- Attack handling --- */
3579a19cd78SMatthias Ringwald
3589a19cd78SMatthias Ringwald if (!att)
3599a19cd78SMatthias Ringwald return;
3609a19cd78SMatthias Ringwald
3619a19cd78SMatthias Ringwald float s0, s1 = scf[0], s2 = scf[1], s3 = scf[2], s4 = scf[3];
3629a19cd78SMatthias Ringwald float sn = s1 + s2;
3639a19cd78SMatthias Ringwald
3644930cef6SMatthias Ringwald scf[0] = (sn += s3) * 1.f/3;
3654930cef6SMatthias Ringwald scf[1] = (sn += s4) * 1.f/4;
3669a19cd78SMatthias Ringwald scf_sum = scf[0] + scf[1];
3679a19cd78SMatthias Ringwald
3689a19cd78SMatthias Ringwald for (int i = 2; i < 14; i++, sn -= s0) {
3699a19cd78SMatthias Ringwald s0 = s1, s1 = s2, s2 = s3, s3 = s4, s4 = scf[i+2];
3704930cef6SMatthias Ringwald scf[i] = (sn += s4) * 1.f/5;
3719a19cd78SMatthias Ringwald scf_sum += scf[i];
3729a19cd78SMatthias Ringwald }
3739a19cd78SMatthias Ringwald
3744930cef6SMatthias Ringwald scf[14] = (sn ) * 1.f/4;
3754930cef6SMatthias Ringwald scf[15] = (sn -= s1) * 1.f/3;
3769a19cd78SMatthias Ringwald scf_sum += scf[14] + scf[15];
3779a19cd78SMatthias Ringwald
3789a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++)
3794930cef6SMatthias Ringwald scf[i] = (dt == LC3_DT_7M5 ? 0.3f : 0.5f) *
3804930cef6SMatthias Ringwald (scf[i] - scf_sum * 1.f/16);
3819a19cd78SMatthias Ringwald }
3829a19cd78SMatthias Ringwald
3839a19cd78SMatthias Ringwald /**
3849a19cd78SMatthias Ringwald * Codebooks
3859a19cd78SMatthias Ringwald * scf Input 16 scale factors
3869a19cd78SMatthias Ringwald * lf/hfcb_idx Output the low and high frequency codebooks index
3879a19cd78SMatthias Ringwald */
resolve_codebooks(const float * scf,int * lfcb_idx,int * hfcb_idx)3884930cef6SMatthias Ringwald LC3_HOT static void resolve_codebooks(
3894930cef6SMatthias Ringwald const float *scf, int *lfcb_idx, int *hfcb_idx)
3909a19cd78SMatthias Ringwald {
3919a19cd78SMatthias Ringwald float dlfcb_max = 0, dhfcb_max = 0;
3929a19cd78SMatthias Ringwald *lfcb_idx = *hfcb_idx = 0;
3939a19cd78SMatthias Ringwald
3949a19cd78SMatthias Ringwald for (int icb = 0; icb < 32; icb++) {
3959a19cd78SMatthias Ringwald const float *lfcb = lc3_sns_lfcb[icb];
3969a19cd78SMatthias Ringwald const float *hfcb = lc3_sns_hfcb[icb];
3979a19cd78SMatthias Ringwald float dlfcb = 0, dhfcb = 0;
3989a19cd78SMatthias Ringwald
3999a19cd78SMatthias Ringwald for (int i = 0; i < 8; i++) {
4009a19cd78SMatthias Ringwald dlfcb += (scf[ i] - lfcb[i]) * (scf[ i] - lfcb[i]);
4019a19cd78SMatthias Ringwald dhfcb += (scf[8+i] - hfcb[i]) * (scf[8+i] - hfcb[i]);
4029a19cd78SMatthias Ringwald }
4039a19cd78SMatthias Ringwald
4049a19cd78SMatthias Ringwald if (icb == 0 || dlfcb < dlfcb_max)
4059a19cd78SMatthias Ringwald *lfcb_idx = icb, dlfcb_max = dlfcb;
4069a19cd78SMatthias Ringwald
4079a19cd78SMatthias Ringwald if (icb == 0 || dhfcb < dhfcb_max)
4089a19cd78SMatthias Ringwald *hfcb_idx = icb, dhfcb_max = dhfcb;
4099a19cd78SMatthias Ringwald }
4109a19cd78SMatthias Ringwald }
4119a19cd78SMatthias Ringwald
4129a19cd78SMatthias Ringwald /**
4139a19cd78SMatthias Ringwald * Unit energy normalize pulse configuration
4149a19cd78SMatthias Ringwald * c Pulse configuration
4159a19cd78SMatthias Ringwald * cn Normalized pulse configuration
4169a19cd78SMatthias Ringwald */
normalize(const int * c,float * cn)4174930cef6SMatthias Ringwald LC3_HOT static void normalize(const int *c, float *cn)
4189a19cd78SMatthias Ringwald {
4199a19cd78SMatthias Ringwald int c2_sum = 0;
4209a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++)
4219a19cd78SMatthias Ringwald c2_sum += c[i] * c[i];
4229a19cd78SMatthias Ringwald
4239a19cd78SMatthias Ringwald float c_norm = 1.f / sqrtf(c2_sum);
4249a19cd78SMatthias Ringwald
4259a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++)
4269a19cd78SMatthias Ringwald cn[i] = c[i] * c_norm;
4279a19cd78SMatthias Ringwald }
4289a19cd78SMatthias Ringwald
4299a19cd78SMatthias Ringwald /**
4309a19cd78SMatthias Ringwald * Sub-procedure of `quantize()`, add unit pulse
4319a19cd78SMatthias Ringwald * x, y, n Transformed residual, and vector of pulses with length
4329a19cd78SMatthias Ringwald * start, end Current number of pulses, limit to reach
4339a19cd78SMatthias Ringwald * corr, energy Correlation (x,y) and y energy, updated at output
4349a19cd78SMatthias Ringwald */
add_pulse(const float * x,int * y,int n,int start,int end,float * corr,float * energy)4354930cef6SMatthias Ringwald LC3_HOT static void add_pulse(const float *x, int *y, int n,
4369a19cd78SMatthias Ringwald int start, int end, float *corr, float *energy)
4379a19cd78SMatthias Ringwald {
4389a19cd78SMatthias Ringwald for (int k = start; k < end; k++) {
4399a19cd78SMatthias Ringwald float best_c2 = (*corr + x[0]) * (*corr + x[0]);
4409a19cd78SMatthias Ringwald float best_e = *energy + 2*y[0] + 1;
4419a19cd78SMatthias Ringwald int nbest = 0;
4429a19cd78SMatthias Ringwald
4439a19cd78SMatthias Ringwald for (int i = 1; i < n; i++) {
4449a19cd78SMatthias Ringwald float c2 = (*corr + x[i]) * (*corr + x[i]);
4459a19cd78SMatthias Ringwald float e = *energy + 2*y[i] + 1;
4469a19cd78SMatthias Ringwald
4479a19cd78SMatthias Ringwald if (c2 * best_e > e * best_c2)
4489a19cd78SMatthias Ringwald best_c2 = c2, best_e = e, nbest = i;
4499a19cd78SMatthias Ringwald }
4509a19cd78SMatthias Ringwald
4519a19cd78SMatthias Ringwald *corr += x[nbest];
4529a19cd78SMatthias Ringwald *energy += 2*y[nbest] + 1;
4539a19cd78SMatthias Ringwald y[nbest]++;
4549a19cd78SMatthias Ringwald }
4559a19cd78SMatthias Ringwald }
4569a19cd78SMatthias Ringwald
4579a19cd78SMatthias Ringwald /**
4589a19cd78SMatthias Ringwald * Quantization of codebooks residual
4599a19cd78SMatthias Ringwald * scf Input 16 scale factors, output quantized version
4609a19cd78SMatthias Ringwald * lf/hfcb_idx Codebooks index
4619a19cd78SMatthias Ringwald * c, cn Output 4 pulse configurations candidates, normalized
4629a19cd78SMatthias Ringwald * shape/gain_idx Output selected shape/gain indexes
4639a19cd78SMatthias Ringwald */
quantize(const float * scf,int lfcb_idx,int hfcb_idx,int (* c)[16],float (* cn)[16],int * shape_idx,int * gain_idx)4644930cef6SMatthias Ringwald LC3_HOT static void quantize(const float *scf, int lfcb_idx, int hfcb_idx,
4659a19cd78SMatthias Ringwald int (*c)[16], float (*cn)[16], int *shape_idx, int *gain_idx)
4669a19cd78SMatthias Ringwald {
4679a19cd78SMatthias Ringwald /* --- Residual --- */
4689a19cd78SMatthias Ringwald
4699a19cd78SMatthias Ringwald const float *lfcb = lc3_sns_lfcb[lfcb_idx];
4709a19cd78SMatthias Ringwald const float *hfcb = lc3_sns_hfcb[hfcb_idx];
4719a19cd78SMatthias Ringwald float r[16], x[16];
4729a19cd78SMatthias Ringwald
4739a19cd78SMatthias Ringwald for (int i = 0; i < 8; i++) {
4749a19cd78SMatthias Ringwald r[ i] = scf[ i] - lfcb[i];
4759a19cd78SMatthias Ringwald r[8+i] = scf[8+i] - hfcb[i];
4769a19cd78SMatthias Ringwald }
4779a19cd78SMatthias Ringwald
4789a19cd78SMatthias Ringwald dct16_forward(r, x);
4799a19cd78SMatthias Ringwald
4809a19cd78SMatthias Ringwald /* --- Shape 3 candidate ---
4819a19cd78SMatthias Ringwald * Project to or below pyramid N = 16, K = 6,
4829a19cd78SMatthias Ringwald * then add unit pulses until you reach K = 6, over N = 16 */
4839a19cd78SMatthias Ringwald
4849a19cd78SMatthias Ringwald float xm[16];
4859a19cd78SMatthias Ringwald float xm_sum = 0;
4869a19cd78SMatthias Ringwald
4879a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) {
4889a19cd78SMatthias Ringwald xm[i] = fabsf(x[i]);
4899a19cd78SMatthias Ringwald xm_sum += xm[i];
4909a19cd78SMatthias Ringwald }
4919a19cd78SMatthias Ringwald
4924930cef6SMatthias Ringwald float proj_factor = (6 - 1) / fmaxf(xm_sum, 1e-31f);
4939a19cd78SMatthias Ringwald float corr = 0, energy = 0;
4949a19cd78SMatthias Ringwald int npulses = 0;
4959a19cd78SMatthias Ringwald
4969a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) {
4979a19cd78SMatthias Ringwald c[3][i] = floorf(xm[i] * proj_factor);
4989a19cd78SMatthias Ringwald npulses += c[3][i];
4999a19cd78SMatthias Ringwald corr += c[3][i] * xm[i];
5009a19cd78SMatthias Ringwald energy += c[3][i] * c[3][i];
5019a19cd78SMatthias Ringwald }
5029a19cd78SMatthias Ringwald
5039a19cd78SMatthias Ringwald add_pulse(xm, c[3], 16, npulses, 6, &corr, &energy);
5049a19cd78SMatthias Ringwald npulses = 6;
5059a19cd78SMatthias Ringwald
5069a19cd78SMatthias Ringwald /* --- Shape 2 candidate ---
5079a19cd78SMatthias Ringwald * Add unit pulses until you reach K = 8 on shape 3 */
5089a19cd78SMatthias Ringwald
5099a19cd78SMatthias Ringwald memcpy(c[2], c[3], sizeof(c[2]));
5109a19cd78SMatthias Ringwald
5119a19cd78SMatthias Ringwald add_pulse(xm, c[2], 16, npulses, 8, &corr, &energy);
5129a19cd78SMatthias Ringwald npulses = 8;
5139a19cd78SMatthias Ringwald
5149a19cd78SMatthias Ringwald /* --- Shape 1 candidate ---
5159a19cd78SMatthias Ringwald * Remove any unit pulses from shape 2 that are not part of 0 to 9
5169a19cd78SMatthias Ringwald * Update energy and correlation terms accordingly
5179a19cd78SMatthias Ringwald * Add unit pulses until you reach K = 10, over N = 10 */
5189a19cd78SMatthias Ringwald
5199a19cd78SMatthias Ringwald memcpy(c[1], c[2], sizeof(c[1]));
5209a19cd78SMatthias Ringwald
5219a19cd78SMatthias Ringwald for (int i = 10; i < 16; i++) {
5229a19cd78SMatthias Ringwald c[1][i] = 0;
5239a19cd78SMatthias Ringwald npulses -= c[2][i];
5249a19cd78SMatthias Ringwald corr -= c[2][i] * xm[i];
5259a19cd78SMatthias Ringwald energy -= c[2][i] * c[2][i];
5269a19cd78SMatthias Ringwald }
5279a19cd78SMatthias Ringwald
5289a19cd78SMatthias Ringwald add_pulse(xm, c[1], 10, npulses, 10, &corr, &energy);
5299a19cd78SMatthias Ringwald npulses = 10;
5309a19cd78SMatthias Ringwald
5319a19cd78SMatthias Ringwald /* --- Shape 0 candidate ---
5329a19cd78SMatthias Ringwald * Add unit pulses until you reach K = 1, on shape 1 */
5339a19cd78SMatthias Ringwald
5349a19cd78SMatthias Ringwald memcpy(c[0], c[1], sizeof(c[0]));
5359a19cd78SMatthias Ringwald
5369a19cd78SMatthias Ringwald add_pulse(xm + 10, c[0] + 10, 6, 0, 1, &corr, &energy);
5379a19cd78SMatthias Ringwald
5389a19cd78SMatthias Ringwald /* --- Add sign and unit energy normalize --- */
5399a19cd78SMatthias Ringwald
5409a19cd78SMatthias Ringwald for (int j = 0; j < 16; j++)
5419a19cd78SMatthias Ringwald for (int i = 0; i < 4; i++)
5429a19cd78SMatthias Ringwald c[i][j] = x[j] < 0 ? -c[i][j] : c[i][j];
5439a19cd78SMatthias Ringwald
5449a19cd78SMatthias Ringwald for (int i = 0; i < 4; i++)
5459a19cd78SMatthias Ringwald normalize(c[i], cn[i]);
5469a19cd78SMatthias Ringwald
5479a19cd78SMatthias Ringwald /* --- Determe shape & gain index ---
5489a19cd78SMatthias Ringwald * Search the Mean Square Error, within (shape, gain) combinations */
5499a19cd78SMatthias Ringwald
550*6897da5cSDirk Helbig float mse_min = FLT_MAX;
5519a19cd78SMatthias Ringwald *shape_idx = *gain_idx = 0;
5529a19cd78SMatthias Ringwald
5539a19cd78SMatthias Ringwald for (int ic = 0; ic < 4; ic++) {
5549a19cd78SMatthias Ringwald const struct lc3_sns_vq_gains *cgains = lc3_sns_vq_gains + ic;
555*6897da5cSDirk Helbig float cmse_min = FLT_MAX;
5569a19cd78SMatthias Ringwald int cgain_idx = 0;
5579a19cd78SMatthias Ringwald
5589a19cd78SMatthias Ringwald for (int ig = 0; ig < cgains->count; ig++) {
5599a19cd78SMatthias Ringwald float g = cgains->v[ig];
5609a19cd78SMatthias Ringwald
5619a19cd78SMatthias Ringwald float mse = 0;
5629a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++)
5639a19cd78SMatthias Ringwald mse += (x[i] - g * cn[ic][i]) * (x[i] - g * cn[ic][i]);
5649a19cd78SMatthias Ringwald
5659a19cd78SMatthias Ringwald if (mse < cmse_min) {
5669a19cd78SMatthias Ringwald cgain_idx = ig,
5679a19cd78SMatthias Ringwald cmse_min = mse;
5689a19cd78SMatthias Ringwald }
5699a19cd78SMatthias Ringwald }
5709a19cd78SMatthias Ringwald
5719a19cd78SMatthias Ringwald if (cmse_min < mse_min) {
5729a19cd78SMatthias Ringwald *shape_idx = ic, *gain_idx = cgain_idx;
5739a19cd78SMatthias Ringwald mse_min = cmse_min;
5749a19cd78SMatthias Ringwald }
5759a19cd78SMatthias Ringwald }
5769a19cd78SMatthias Ringwald }
5779a19cd78SMatthias Ringwald
5789a19cd78SMatthias Ringwald /**
5799a19cd78SMatthias Ringwald * Unquantization of codebooks residual
5809a19cd78SMatthias Ringwald * lf/hfcb_idx Low and high frequency codebooks index
5819a19cd78SMatthias Ringwald * c Table of normalized pulse configuration
5829a19cd78SMatthias Ringwald * shape/gain Selected shape/gain indexes
5839a19cd78SMatthias Ringwald * scf Return unquantized scale factors
5849a19cd78SMatthias Ringwald */
unquantize(int lfcb_idx,int hfcb_idx,const float * c,int shape,int gain,float * scf)5854930cef6SMatthias Ringwald LC3_HOT static void unquantize(int lfcb_idx, int hfcb_idx,
5869a19cd78SMatthias Ringwald const float *c, int shape, int gain, float *scf)
5879a19cd78SMatthias Ringwald {
5889a19cd78SMatthias Ringwald const float *lfcb = lc3_sns_lfcb[lfcb_idx];
5899a19cd78SMatthias Ringwald const float *hfcb = lc3_sns_hfcb[hfcb_idx];
5909a19cd78SMatthias Ringwald float g = lc3_sns_vq_gains[shape].v[gain];
5919a19cd78SMatthias Ringwald
5929a19cd78SMatthias Ringwald dct16_inverse(c, scf);
5939a19cd78SMatthias Ringwald
5949a19cd78SMatthias Ringwald for (int i = 0; i < 8; i++)
5959a19cd78SMatthias Ringwald scf[i] = lfcb[i] + g * scf[i];
5969a19cd78SMatthias Ringwald
5979a19cd78SMatthias Ringwald for (int i = 8; i < 16; i++)
5989a19cd78SMatthias Ringwald scf[i] = hfcb[i-8] + g * scf[i];
5999a19cd78SMatthias Ringwald }
6009a19cd78SMatthias Ringwald
6019a19cd78SMatthias Ringwald /**
6029a19cd78SMatthias Ringwald * Sub-procedure of `sns_enumerate()`, enumeration of a vector
6039a19cd78SMatthias Ringwald * c, n Table of pulse configuration, and length
6049a19cd78SMatthias Ringwald * idx, ls Return enumeration set
6059a19cd78SMatthias Ringwald */
enum_mvpq(const int * c,int n,int * idx,bool * ls)6069a19cd78SMatthias Ringwald static void enum_mvpq(const int *c, int n, int *idx, bool *ls)
6079a19cd78SMatthias Ringwald {
608*6897da5cSDirk Helbig int ci, i;
6099a19cd78SMatthias Ringwald
6109a19cd78SMatthias Ringwald /* --- Scan for 1st significant coeff --- */
6119a19cd78SMatthias Ringwald
612*6897da5cSDirk Helbig for (i = 0, c += n; (ci = *(--c)) == 0 && i < 15; i++);
6139a19cd78SMatthias Ringwald
6149a19cd78SMatthias Ringwald *idx = 0;
6159a19cd78SMatthias Ringwald *ls = ci < 0;
6169a19cd78SMatthias Ringwald
6179a19cd78SMatthias Ringwald /* --- Scan remaining coefficients --- */
6189a19cd78SMatthias Ringwald
619*6897da5cSDirk Helbig unsigned j = LC3_ABS(ci);
620*6897da5cSDirk Helbig
621*6897da5cSDirk Helbig for (i++; i < n; i++, j += LC3_ABS(ci)) {
6229a19cd78SMatthias Ringwald
6239a19cd78SMatthias Ringwald if ((ci = *(--c)) != 0) {
6249a19cd78SMatthias Ringwald *idx = (*idx << 1) | *ls;
6259a19cd78SMatthias Ringwald *ls = ci < 0;
6269a19cd78SMatthias Ringwald }
6279a19cd78SMatthias Ringwald
628*6897da5cSDirk Helbig *idx += lc3_sns_mpvq_offsets[i][LC3_MIN(j, 10)];
6299a19cd78SMatthias Ringwald }
6309a19cd78SMatthias Ringwald }
6319a19cd78SMatthias Ringwald
6329a19cd78SMatthias Ringwald /**
6339a19cd78SMatthias Ringwald * Sub-procedure of `sns_deenumerate()`, deenumeration of a vector
6349a19cd78SMatthias Ringwald * idx, ls Enumeration set
6359a19cd78SMatthias Ringwald * npulses Number of pulses in the set
6369a19cd78SMatthias Ringwald * c, n Table of pulses configuration, and length
6379a19cd78SMatthias Ringwald */
deenum_mvpq(int idx,bool ls,int npulses,int * c,int n)6389a19cd78SMatthias Ringwald static void deenum_mvpq(int idx, bool ls, int npulses, int *c, int n)
6399a19cd78SMatthias Ringwald {
6409a19cd78SMatthias Ringwald int i;
6419a19cd78SMatthias Ringwald
6429a19cd78SMatthias Ringwald /* --- Scan for coefficients --- */
6439a19cd78SMatthias Ringwald
6449a19cd78SMatthias Ringwald for (i = n-1; i >= 0 && idx; i--) {
6459a19cd78SMatthias Ringwald
6469a19cd78SMatthias Ringwald int ci = 0;
6479a19cd78SMatthias Ringwald
6489a19cd78SMatthias Ringwald for (ci = 0; idx < lc3_sns_mpvq_offsets[i][npulses - ci]; ci++);
6499a19cd78SMatthias Ringwald idx -= lc3_sns_mpvq_offsets[i][npulses - ci];
6509a19cd78SMatthias Ringwald
6519a19cd78SMatthias Ringwald *(c++) = ls ? -ci : ci;
6529a19cd78SMatthias Ringwald npulses -= ci;
6539a19cd78SMatthias Ringwald if (ci > 0) {
6549a19cd78SMatthias Ringwald ls = idx & 1;
6559a19cd78SMatthias Ringwald idx >>= 1;
6569a19cd78SMatthias Ringwald }
6579a19cd78SMatthias Ringwald }
6589a19cd78SMatthias Ringwald
6599a19cd78SMatthias Ringwald /* --- Set last significant --- */
6609a19cd78SMatthias Ringwald
6619a19cd78SMatthias Ringwald int ci = npulses;
6629a19cd78SMatthias Ringwald
6639a19cd78SMatthias Ringwald if (i-- >= 0)
6649a19cd78SMatthias Ringwald *(c++) = ls ? -ci : ci;
6659a19cd78SMatthias Ringwald
6669a19cd78SMatthias Ringwald while (i-- >= 0)
6679a19cd78SMatthias Ringwald *(c++) = 0;
6689a19cd78SMatthias Ringwald }
6699a19cd78SMatthias Ringwald
6709a19cd78SMatthias Ringwald /**
6719a19cd78SMatthias Ringwald * SNS Enumeration of PVQ configuration
6729a19cd78SMatthias Ringwald * shape Selected shape index
6739a19cd78SMatthias Ringwald * c Selected pulse configuration
6749a19cd78SMatthias Ringwald * idx_a, ls_a Return enumeration set A
6759a19cd78SMatthias Ringwald * idx_b, ls_b Return enumeration set B (shape = 0)
6769a19cd78SMatthias Ringwald */
enumerate(int shape,const int * c,int * idx_a,bool * ls_a,int * idx_b,bool * ls_b)6779a19cd78SMatthias Ringwald static void enumerate(int shape, const int *c,
6789a19cd78SMatthias Ringwald int *idx_a, bool *ls_a, int *idx_b, bool *ls_b)
6799a19cd78SMatthias Ringwald {
6809a19cd78SMatthias Ringwald enum_mvpq(c, shape < 2 ? 10 : 16, idx_a, ls_a);
6819a19cd78SMatthias Ringwald
6829a19cd78SMatthias Ringwald if (shape == 0)
6839a19cd78SMatthias Ringwald enum_mvpq(c + 10, 6, idx_b, ls_b);
6849a19cd78SMatthias Ringwald }
6859a19cd78SMatthias Ringwald
6869a19cd78SMatthias Ringwald /**
6879a19cd78SMatthias Ringwald * SNS Deenumeration of PVQ configuration
6889a19cd78SMatthias Ringwald * shape Selected shape index
6899a19cd78SMatthias Ringwald * idx_a, ls_a enumeration set A
6909a19cd78SMatthias Ringwald * idx_b, ls_b enumeration set B (shape = 0)
6919a19cd78SMatthias Ringwald * c Return pulse configuration
6929a19cd78SMatthias Ringwald */
deenumerate(int shape,int idx_a,bool ls_a,int idx_b,bool ls_b,int * c)6939a19cd78SMatthias Ringwald static void deenumerate(int shape,
6949a19cd78SMatthias Ringwald int idx_a, bool ls_a, int idx_b, bool ls_b, int *c)
6959a19cd78SMatthias Ringwald {
6969a19cd78SMatthias Ringwald int npulses_a = (const int []){ 10, 10, 8, 6 }[shape];
6979a19cd78SMatthias Ringwald
6989a19cd78SMatthias Ringwald deenum_mvpq(idx_a, ls_a, npulses_a, c, shape < 2 ? 10 : 16);
6999a19cd78SMatthias Ringwald
7009a19cd78SMatthias Ringwald if (shape == 0)
7019a19cd78SMatthias Ringwald deenum_mvpq(idx_b, ls_b, 1, c + 10, 6);
7029a19cd78SMatthias Ringwald else if (shape == 1)
7039a19cd78SMatthias Ringwald memset(c + 10, 0, 6 * sizeof(*c));
7049a19cd78SMatthias Ringwald }
7059a19cd78SMatthias Ringwald
7069a19cd78SMatthias Ringwald
7079a19cd78SMatthias Ringwald /* ----------------------------------------------------------------------------
7089a19cd78SMatthias Ringwald * Filtering
7099a19cd78SMatthias Ringwald * -------------------------------------------------------------------------- */
7109a19cd78SMatthias Ringwald
7119a19cd78SMatthias Ringwald /**
7129a19cd78SMatthias Ringwald * Spectral shaping
7139a19cd78SMatthias Ringwald * dt, sr Duration and samplerate of the frame
7149a19cd78SMatthias Ringwald * scf_q Quantized scale factors
7159a19cd78SMatthias Ringwald * inv True on inverse shaping, False otherwise
7169a19cd78SMatthias Ringwald * x Spectral coefficients
7179a19cd78SMatthias Ringwald * y Return shapped coefficients
7189a19cd78SMatthias Ringwald *
7199a19cd78SMatthias Ringwald * `x` and `y` can be the same buffer
7209a19cd78SMatthias Ringwald */
spectral_shaping(enum lc3_dt dt,enum lc3_srate sr,const float * scf_q,bool inv,const float * x,float * y)7214930cef6SMatthias Ringwald LC3_HOT static void spectral_shaping(enum lc3_dt dt, enum lc3_srate sr,
7229a19cd78SMatthias Ringwald const float *scf_q, bool inv, const float *x, float *y)
7239a19cd78SMatthias Ringwald {
7249a19cd78SMatthias Ringwald /* --- Interpolate scale factors --- */
7259a19cd78SMatthias Ringwald
726*6897da5cSDirk Helbig float scf[LC3_MAX_BANDS];
7279a19cd78SMatthias Ringwald float s0, s1 = inv ? -scf_q[0] : scf_q[0];
7289a19cd78SMatthias Ringwald
7299a19cd78SMatthias Ringwald scf[0] = scf[1] = s1;
7309a19cd78SMatthias Ringwald for (int i = 0; i < 15; i++) {
7319a19cd78SMatthias Ringwald s0 = s1, s1 = inv ? -scf_q[i+1] : scf_q[i+1];
7324930cef6SMatthias Ringwald scf[4*i+2] = s0 + 0.125f * (s1 - s0);
7334930cef6SMatthias Ringwald scf[4*i+3] = s0 + 0.375f * (s1 - s0);
7344930cef6SMatthias Ringwald scf[4*i+4] = s0 + 0.625f * (s1 - s0);
7354930cef6SMatthias Ringwald scf[4*i+5] = s0 + 0.875f * (s1 - s0);
7369a19cd78SMatthias Ringwald }
7374930cef6SMatthias Ringwald scf[62] = s1 + 0.125f * (s1 - s0);
7384930cef6SMatthias Ringwald scf[63] = s1 + 0.375f * (s1 - s0);
7399a19cd78SMatthias Ringwald
740*6897da5cSDirk Helbig int nb = lc3_num_bands[dt][sr];
741*6897da5cSDirk Helbig int n4 = nb < 32 ? 32 % nb : 0;
742*6897da5cSDirk Helbig int n2 = nb < 32 ? nb - n4 : LC3_MAX_BANDS - nb;
7439a19cd78SMatthias Ringwald
744*6897da5cSDirk Helbig for (int i4 = 0; i4 < n4; i4++)
745*6897da5cSDirk Helbig scf[i4] = 0.25f * (scf[4*i4+0] + scf[4*i4+1] +
746*6897da5cSDirk Helbig scf[4*i4+2] + scf[4*i4+3]);
7479a19cd78SMatthias Ringwald
748*6897da5cSDirk Helbig for (int i2 = n4; i2 < n4+n2; i2++)
749*6897da5cSDirk Helbig scf[i2] = 0.5f * (scf[2*(n4+i2)] + scf[2*(n4+i2)+1]);
750*6897da5cSDirk Helbig
751*6897da5cSDirk Helbig memmove(scf + n4 + n2, scf + 4*n4 + 2*n2, (nb - n4 - n2) * sizeof(float));
7529a19cd78SMatthias Ringwald
7539a19cd78SMatthias Ringwald /* --- Spectral shaping --- */
7549a19cd78SMatthias Ringwald
7559a19cd78SMatthias Ringwald const int *lim = lc3_band_lim[dt][sr];
7569a19cd78SMatthias Ringwald
7579a19cd78SMatthias Ringwald for (int i = 0, ib = 0; ib < nb; ib++) {
758*6897da5cSDirk Helbig float g_sns = lc3_exp2f(-scf[ib]);
7599a19cd78SMatthias Ringwald
7609a19cd78SMatthias Ringwald for ( ; i < lim[ib+1]; i++)
7619a19cd78SMatthias Ringwald y[i] = x[i] * g_sns;
7629a19cd78SMatthias Ringwald }
7639a19cd78SMatthias Ringwald }
7649a19cd78SMatthias Ringwald
7659a19cd78SMatthias Ringwald
7669a19cd78SMatthias Ringwald /* ----------------------------------------------------------------------------
7679a19cd78SMatthias Ringwald * Interface
7689a19cd78SMatthias Ringwald * -------------------------------------------------------------------------- */
7699a19cd78SMatthias Ringwald
7709a19cd78SMatthias Ringwald /**
7719a19cd78SMatthias Ringwald * SNS analysis
7729a19cd78SMatthias Ringwald */
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*6897da5cSDirk Helbig void lc3_sns_analyze(
774*6897da5cSDirk Helbig enum lc3_dt dt, enum lc3_srate sr, int nbytes,
7759a19cd78SMatthias Ringwald const float *eb, bool att, struct lc3_sns_data *data,
7769a19cd78SMatthias Ringwald const float *x, float *y)
7779a19cd78SMatthias Ringwald {
7789a19cd78SMatthias Ringwald /* Processing steps :
7799a19cd78SMatthias Ringwald * - Determine 16 scale factors from bands energy estimation
7809a19cd78SMatthias Ringwald * - Get codebooks indexes that match thoses scale factors
7819a19cd78SMatthias Ringwald * - Quantize the residual with the selected codebook
7829a19cd78SMatthias Ringwald * - The pulse configuration `c[]` is enumerated
7839a19cd78SMatthias Ringwald * - Finally shape the spectrum coefficients accordingly */
7849a19cd78SMatthias Ringwald
7859a19cd78SMatthias Ringwald float scf[16], cn[4][16];
7869a19cd78SMatthias Ringwald int c[4][16];
7879a19cd78SMatthias Ringwald
788*6897da5cSDirk Helbig compute_scale_factors(dt, sr, nbytes, eb, att, scf);
7899a19cd78SMatthias Ringwald
7909a19cd78SMatthias Ringwald resolve_codebooks(scf, &data->lfcb, &data->hfcb);
7919a19cd78SMatthias Ringwald
7929a19cd78SMatthias Ringwald quantize(scf, data->lfcb, data->hfcb,
7939a19cd78SMatthias Ringwald c, cn, &data->shape, &data->gain);
7949a19cd78SMatthias Ringwald
7959a19cd78SMatthias Ringwald unquantize(data->lfcb, data->hfcb,
7969a19cd78SMatthias Ringwald cn[data->shape], data->shape, data->gain, scf);
7979a19cd78SMatthias Ringwald
7989a19cd78SMatthias Ringwald enumerate(data->shape, c[data->shape],
7999a19cd78SMatthias Ringwald &data->idx_a, &data->ls_a, &data->idx_b, &data->ls_b);
8009a19cd78SMatthias Ringwald
8019a19cd78SMatthias Ringwald spectral_shaping(dt, sr, scf, false, x, y);
8029a19cd78SMatthias Ringwald }
8039a19cd78SMatthias Ringwald
8049a19cd78SMatthias Ringwald /**
8059a19cd78SMatthias Ringwald * SNS synthesis
8069a19cd78SMatthias Ringwald */
lc3_sns_synthesize(enum lc3_dt dt,enum lc3_srate sr,const lc3_sns_data_t * data,const float * x,float * y)807*6897da5cSDirk Helbig void lc3_sns_synthesize(
808*6897da5cSDirk Helbig enum lc3_dt dt, enum lc3_srate sr,
8099a19cd78SMatthias Ringwald const lc3_sns_data_t *data, const float *x, float *y)
8109a19cd78SMatthias Ringwald {
8119a19cd78SMatthias Ringwald float scf[16], cn[16];
8129a19cd78SMatthias Ringwald int c[16];
8139a19cd78SMatthias Ringwald
8149a19cd78SMatthias Ringwald deenumerate(data->shape,
8159a19cd78SMatthias Ringwald data->idx_a, data->ls_a, data->idx_b, data->ls_b, c);
8169a19cd78SMatthias Ringwald
8179a19cd78SMatthias Ringwald normalize(c, cn);
8189a19cd78SMatthias Ringwald
8199a19cd78SMatthias Ringwald unquantize(data->lfcb, data->hfcb, cn, data->shape, data->gain, scf);
8209a19cd78SMatthias Ringwald
8219a19cd78SMatthias Ringwald spectral_shaping(dt, sr, scf, true, x, y);
8229a19cd78SMatthias Ringwald }
8239a19cd78SMatthias Ringwald
8249a19cd78SMatthias Ringwald /**
8259a19cd78SMatthias Ringwald * Return number of bits coding the bitstream data
8269a19cd78SMatthias Ringwald */
lc3_sns_get_nbits(void)8279a19cd78SMatthias Ringwald int lc3_sns_get_nbits(void)
8289a19cd78SMatthias Ringwald {
8299a19cd78SMatthias Ringwald return 38;
8309a19cd78SMatthias Ringwald }
8319a19cd78SMatthias Ringwald
8329a19cd78SMatthias Ringwald /**
8339a19cd78SMatthias Ringwald * Put bitstream data
8349a19cd78SMatthias Ringwald */
lc3_sns_put_data(lc3_bits_t * bits,const struct lc3_sns_data * data)8359a19cd78SMatthias Ringwald void lc3_sns_put_data(lc3_bits_t *bits, const struct lc3_sns_data *data)
8369a19cd78SMatthias Ringwald {
8379a19cd78SMatthias Ringwald /* --- Codebooks --- */
8389a19cd78SMatthias Ringwald
8399a19cd78SMatthias Ringwald lc3_put_bits(bits, data->lfcb, 5);
8409a19cd78SMatthias Ringwald lc3_put_bits(bits, data->hfcb, 5);
8419a19cd78SMatthias Ringwald
8429a19cd78SMatthias Ringwald /* --- Shape, gain and vectors --- *
8439a19cd78SMatthias Ringwald * Write MSB bit of shape index, next LSB bits of shape and gain,
8449a19cd78SMatthias Ringwald * and MVPQ vectors indexes are muxed */
8459a19cd78SMatthias Ringwald
8469a19cd78SMatthias Ringwald int shape_msb = data->shape >> 1;
8479a19cd78SMatthias Ringwald lc3_put_bit(bits, shape_msb);
8489a19cd78SMatthias Ringwald
8499a19cd78SMatthias Ringwald if (shape_msb == 0) {
8509a19cd78SMatthias Ringwald const int size_a = 2390004;
8519a19cd78SMatthias Ringwald int submode = data->shape & 1;
8529a19cd78SMatthias Ringwald
8539a19cd78SMatthias Ringwald int mux_high = submode == 0 ?
8549a19cd78SMatthias Ringwald 2 * (data->idx_b + 1) + data->ls_b : data->gain & 1;
8559a19cd78SMatthias Ringwald int mux_code = mux_high * size_a + data->idx_a;
8569a19cd78SMatthias Ringwald
8579a19cd78SMatthias Ringwald lc3_put_bits(bits, data->gain >> submode, 1);
8589a19cd78SMatthias Ringwald lc3_put_bits(bits, data->ls_a, 1);
8599a19cd78SMatthias Ringwald lc3_put_bits(bits, mux_code, 25);
8609a19cd78SMatthias Ringwald
8619a19cd78SMatthias Ringwald } else {
8629a19cd78SMatthias Ringwald const int size_a = 15158272;
8639a19cd78SMatthias Ringwald int submode = data->shape & 1;
8649a19cd78SMatthias Ringwald
8659a19cd78SMatthias Ringwald int mux_code = submode == 0 ?
8669a19cd78SMatthias Ringwald data->idx_a : size_a + 2 * data->idx_a + (data->gain & 1);
8679a19cd78SMatthias Ringwald
8689a19cd78SMatthias Ringwald lc3_put_bits(bits, data->gain >> submode, 2);
8699a19cd78SMatthias Ringwald lc3_put_bits(bits, data->ls_a, 1);
8709a19cd78SMatthias Ringwald lc3_put_bits(bits, mux_code, 24);
8719a19cd78SMatthias Ringwald }
8729a19cd78SMatthias Ringwald }
8739a19cd78SMatthias Ringwald
8749a19cd78SMatthias Ringwald /**
8759a19cd78SMatthias Ringwald * Get bitstream data
8769a19cd78SMatthias Ringwald */
lc3_sns_get_data(lc3_bits_t * bits,struct lc3_sns_data * data)8779a19cd78SMatthias Ringwald int lc3_sns_get_data(lc3_bits_t *bits, struct lc3_sns_data *data)
8789a19cd78SMatthias Ringwald {
8799a19cd78SMatthias Ringwald /* --- Codebooks --- */
8809a19cd78SMatthias Ringwald
8819a19cd78SMatthias Ringwald *data = (struct lc3_sns_data){
8829a19cd78SMatthias Ringwald .lfcb = lc3_get_bits(bits, 5),
8839a19cd78SMatthias Ringwald .hfcb = lc3_get_bits(bits, 5)
8849a19cd78SMatthias Ringwald };
8859a19cd78SMatthias Ringwald
8869a19cd78SMatthias Ringwald /* --- Shape, gain and vectors --- */
8879a19cd78SMatthias Ringwald
8889a19cd78SMatthias Ringwald int shape_msb = lc3_get_bit(bits);
8899a19cd78SMatthias Ringwald data->gain = lc3_get_bits(bits, 1 + shape_msb);
8909a19cd78SMatthias Ringwald data->ls_a = lc3_get_bit(bits);
8919a19cd78SMatthias Ringwald
8929a19cd78SMatthias Ringwald int mux_code = lc3_get_bits(bits, 25 - shape_msb);
8939a19cd78SMatthias Ringwald
8949a19cd78SMatthias Ringwald if (shape_msb == 0) {
8959a19cd78SMatthias Ringwald const int size_a = 2390004;
8969a19cd78SMatthias Ringwald
8979a19cd78SMatthias Ringwald if (mux_code >= size_a * 14)
8989a19cd78SMatthias Ringwald return -1;
8999a19cd78SMatthias Ringwald
9009a19cd78SMatthias Ringwald data->idx_a = mux_code % size_a;
9019a19cd78SMatthias Ringwald mux_code = mux_code / size_a;
9029a19cd78SMatthias Ringwald
9039a19cd78SMatthias Ringwald data->shape = (mux_code < 2);
9049a19cd78SMatthias Ringwald
9059a19cd78SMatthias Ringwald if (data->shape == 0) {
9069a19cd78SMatthias Ringwald data->idx_b = (mux_code - 2) / 2;
9079a19cd78SMatthias Ringwald data->ls_b = (mux_code - 2) % 2;
9089a19cd78SMatthias Ringwald } else {
9099a19cd78SMatthias Ringwald data->gain = (data->gain << 1) + (mux_code % 2);
9109a19cd78SMatthias Ringwald }
9119a19cd78SMatthias Ringwald
9129a19cd78SMatthias Ringwald } else {
9139a19cd78SMatthias Ringwald const int size_a = 15158272;
9149a19cd78SMatthias Ringwald
9159a19cd78SMatthias Ringwald if (mux_code >= size_a + 1549824)
9169a19cd78SMatthias Ringwald return -1;
9179a19cd78SMatthias Ringwald
9189a19cd78SMatthias Ringwald data->shape = 2 + (mux_code >= size_a);
9199a19cd78SMatthias Ringwald if (data->shape == 2) {
9209a19cd78SMatthias Ringwald data->idx_a = mux_code;
9219a19cd78SMatthias Ringwald } else {
9229a19cd78SMatthias Ringwald mux_code -= size_a;
9239a19cd78SMatthias Ringwald data->idx_a = mux_code / 2;
9249a19cd78SMatthias Ringwald data->gain = (data->gain << 1) + (mux_code % 2);
9259a19cd78SMatthias Ringwald }
9269a19cd78SMatthias Ringwald }
9279a19cd78SMatthias Ringwald
9289a19cd78SMatthias Ringwald return 0;
9299a19cd78SMatthias Ringwald }
930