19a19cd78SMatthias Ringwald /****************************************************************************** 29a19cd78SMatthias Ringwald * 3*4930cef6SMatthias 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 */ 123*4930cef6SMatthias 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 */ 134*4930cef6SMatthias 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 1499a19cd78SMatthias Ringwald * eb Energy estimation per bands 1509a19cd78SMatthias Ringwald * att 1: Attack detected 0: Otherwise 1519a19cd78SMatthias Ringwald * scf Output 16 scale factors 1529a19cd78SMatthias Ringwald */ 153*4930cef6SMatthias Ringwald LC3_HOT static void compute_scale_factors( 154*4930cef6SMatthias Ringwald enum lc3_dt dt, enum lc3_srate sr, 1559a19cd78SMatthias Ringwald const float *eb, bool att, float *scf) 1569a19cd78SMatthias Ringwald { 1579a19cd78SMatthias Ringwald /* Pre-emphasis gain table : 1589a19cd78SMatthias Ringwald * Ge[b] = 10 ^ (b * g_tilt) / 630 , b = [0..63] */ 1599a19cd78SMatthias Ringwald 1609a19cd78SMatthias Ringwald static const float ge_table[LC3_NUM_SRATE][LC3_NUM_BANDS] = { 1619a19cd78SMatthias Ringwald 1629a19cd78SMatthias Ringwald [LC3_SRATE_8K] = { /* g_tilt = 14 */ 1639a19cd78SMatthias Ringwald 1.00000000e+00, 1.05250029e+00, 1.10775685e+00, 1.16591440e+00, 1649a19cd78SMatthias Ringwald 1.22712524e+00, 1.29154967e+00, 1.35935639e+00, 1.43072299e+00, 1659a19cd78SMatthias Ringwald 1.50583635e+00, 1.58489319e+00, 1.66810054e+00, 1.75567629e+00, 1669a19cd78SMatthias Ringwald 1.84784980e+00, 1.94486244e+00, 2.04696827e+00, 2.15443469e+00, 1679a19cd78SMatthias Ringwald 2.26754313e+00, 2.38658979e+00, 2.51188643e+00, 2.64376119e+00, 1689a19cd78SMatthias Ringwald 2.78255940e+00, 2.92864456e+00, 3.08239924e+00, 3.24422608e+00, 1699a19cd78SMatthias Ringwald 3.41454887e+00, 3.59381366e+00, 3.78248991e+00, 3.98107171e+00, 1709a19cd78SMatthias Ringwald 4.19007911e+00, 4.41005945e+00, 4.64158883e+00, 4.88527357e+00, 1719a19cd78SMatthias Ringwald 5.14175183e+00, 5.41169527e+00, 5.69581081e+00, 5.99484250e+00, 1729a19cd78SMatthias Ringwald 6.30957344e+00, 6.64082785e+00, 6.98947321e+00, 7.35642254e+00, 1739a19cd78SMatthias Ringwald 7.74263683e+00, 8.14912747e+00, 8.57695899e+00, 9.02725178e+00, 1749a19cd78SMatthias Ringwald 9.50118507e+00, 1.00000000e+01, 1.05250029e+01, 1.10775685e+01, 1759a19cd78SMatthias Ringwald 1.16591440e+01, 1.22712524e+01, 1.29154967e+01, 1.35935639e+01, 1769a19cd78SMatthias Ringwald 1.43072299e+01, 1.50583635e+01, 1.58489319e+01, 1.66810054e+01, 1779a19cd78SMatthias Ringwald 1.75567629e+01, 1.84784980e+01, 1.94486244e+01, 2.04696827e+01, 1789a19cd78SMatthias Ringwald 2.15443469e+01, 2.26754313e+01, 2.38658979e+01, 2.51188643e+01 }, 1799a19cd78SMatthias Ringwald 1809a19cd78SMatthias Ringwald [LC3_SRATE_16K] = { /* g_tilt = 18 */ 1819a19cd78SMatthias Ringwald 1.00000000e+00, 1.06800043e+00, 1.14062492e+00, 1.21818791e+00, 1829a19cd78SMatthias Ringwald 1.30102522e+00, 1.38949549e+00, 1.48398179e+00, 1.58489319e+00, 1839a19cd78SMatthias Ringwald 1.69266662e+00, 1.80776868e+00, 1.93069773e+00, 2.06198601e+00, 1849a19cd78SMatthias Ringwald 2.20220195e+00, 2.35195264e+00, 2.51188643e+00, 2.68269580e+00, 1859a19cd78SMatthias Ringwald 2.86512027e+00, 3.05994969e+00, 3.26802759e+00, 3.49025488e+00, 1869a19cd78SMatthias Ringwald 3.72759372e+00, 3.98107171e+00, 4.25178630e+00, 4.54090961e+00, 1879a19cd78SMatthias Ringwald 4.84969343e+00, 5.17947468e+00, 5.53168120e+00, 5.90783791e+00, 1889a19cd78SMatthias Ringwald 6.30957344e+00, 6.73862717e+00, 7.19685673e+00, 7.68624610e+00, 1899a19cd78SMatthias Ringwald 8.20891416e+00, 8.76712387e+00, 9.36329209e+00, 1.00000000e+01, 1909a19cd78SMatthias Ringwald 1.06800043e+01, 1.14062492e+01, 1.21818791e+01, 1.30102522e+01, 1919a19cd78SMatthias Ringwald 1.38949549e+01, 1.48398179e+01, 1.58489319e+01, 1.69266662e+01, 1929a19cd78SMatthias Ringwald 1.80776868e+01, 1.93069773e+01, 2.06198601e+01, 2.20220195e+01, 1939a19cd78SMatthias Ringwald 2.35195264e+01, 2.51188643e+01, 2.68269580e+01, 2.86512027e+01, 1949a19cd78SMatthias Ringwald 3.05994969e+01, 3.26802759e+01, 3.49025488e+01, 3.72759372e+01, 1959a19cd78SMatthias Ringwald 3.98107171e+01, 4.25178630e+01, 4.54090961e+01, 4.84969343e+01, 1969a19cd78SMatthias Ringwald 5.17947468e+01, 5.53168120e+01, 5.90783791e+01, 6.30957344e+01 }, 1979a19cd78SMatthias Ringwald 1989a19cd78SMatthias Ringwald [LC3_SRATE_24K] = { /* g_tilt = 22 */ 1999a19cd78SMatthias Ringwald 1.00000000e+00, 1.08372885e+00, 1.17446822e+00, 1.27280509e+00, 2009a19cd78SMatthias Ringwald 1.37937560e+00, 1.49486913e+00, 1.62003281e+00, 1.75567629e+00, 2019a19cd78SMatthias Ringwald 1.90267705e+00, 2.06198601e+00, 2.23463373e+00, 2.42173704e+00, 2029a19cd78SMatthias Ringwald 2.62450630e+00, 2.84425319e+00, 3.08239924e+00, 3.34048498e+00, 2039a19cd78SMatthias Ringwald 3.62017995e+00, 3.92329345e+00, 4.25178630e+00, 4.60778348e+00, 2049a19cd78SMatthias Ringwald 4.99358789e+00, 5.41169527e+00, 5.86481029e+00, 6.35586411e+00, 2059a19cd78SMatthias Ringwald 6.88803330e+00, 7.46476041e+00, 8.08977621e+00, 8.76712387e+00, 2069a19cd78SMatthias Ringwald 9.50118507e+00, 1.02967084e+01, 1.11588399e+01, 1.20931568e+01, 2079a19cd78SMatthias Ringwald 1.31057029e+01, 1.42030283e+01, 1.53922315e+01, 1.66810054e+01, 2089a19cd78SMatthias Ringwald 1.80776868e+01, 1.95913107e+01, 2.12316686e+01, 2.30093718e+01, 2099a19cd78SMatthias Ringwald 2.49359200e+01, 2.70237760e+01, 2.92864456e+01, 3.17385661e+01, 2109a19cd78SMatthias Ringwald 3.43959997e+01, 3.72759372e+01, 4.03970086e+01, 4.37794036e+01, 2119a19cd78SMatthias Ringwald 4.74450028e+01, 5.14175183e+01, 5.57226480e+01, 6.03882412e+01, 2129a19cd78SMatthias Ringwald 6.54444792e+01, 7.09240702e+01, 7.68624610e+01, 8.32980665e+01, 2139a19cd78SMatthias Ringwald 9.02725178e+01, 9.78309319e+01, 1.06022203e+02, 1.14899320e+02, 2149a19cd78SMatthias Ringwald 1.24519708e+02, 1.34945600e+02, 1.46244440e+02, 1.58489319e+02 }, 2159a19cd78SMatthias Ringwald 2169a19cd78SMatthias Ringwald [LC3_SRATE_32K] = { /* g_tilt = 26 */ 2179a19cd78SMatthias Ringwald 1.00000000e+00, 1.09968890e+00, 1.20931568e+00, 1.32987103e+00, 2189a19cd78SMatthias Ringwald 1.46244440e+00, 1.60823388e+00, 1.76855694e+00, 1.94486244e+00, 2199a19cd78SMatthias Ringwald 2.13874364e+00, 2.35195264e+00, 2.58641621e+00, 2.84425319e+00, 2209a19cd78SMatthias Ringwald 3.12779366e+00, 3.43959997e+00, 3.78248991e+00, 4.15956216e+00, 2219a19cd78SMatthias Ringwald 4.57422434e+00, 5.03022373e+00, 5.53168120e+00, 6.08312841e+00, 2229a19cd78SMatthias Ringwald 6.68954879e+00, 7.35642254e+00, 8.08977621e+00, 8.89623710e+00, 2239a19cd78SMatthias Ringwald 9.78309319e+00, 1.07583590e+01, 1.18308480e+01, 1.30102522e+01, 2249a19cd78SMatthias Ringwald 1.43072299e+01, 1.57335019e+01, 1.73019574e+01, 1.90267705e+01, 2259a19cd78SMatthias Ringwald 2.09235283e+01, 2.30093718e+01, 2.53031508e+01, 2.78255940e+01, 2269a19cd78SMatthias Ringwald 3.05994969e+01, 3.36499270e+01, 3.70044512e+01, 4.06933843e+01, 2279a19cd78SMatthias Ringwald 4.47500630e+01, 4.92111475e+01, 5.41169527e+01, 5.95118121e+01, 2289a19cd78SMatthias Ringwald 6.54444792e+01, 7.19685673e+01, 7.91430346e+01, 8.70327166e+01, 2299a19cd78SMatthias Ringwald 9.57089124e+01, 1.05250029e+02, 1.15742288e+02, 1.27280509e+02, 2309a19cd78SMatthias Ringwald 1.39968963e+02, 1.53922315e+02, 1.69266662e+02, 1.86140669e+02, 2319a19cd78SMatthias Ringwald 2.04696827e+02, 2.25102829e+02, 2.47543082e+02, 2.72220379e+02, 2329a19cd78SMatthias Ringwald 2.99357729e+02, 3.29200372e+02, 3.62017995e+02, 3.98107171e+02 }, 2339a19cd78SMatthias Ringwald 2349a19cd78SMatthias Ringwald [LC3_SRATE_48K] = { /* g_tilt = 30 */ 2359a19cd78SMatthias Ringwald 1.00000000e+00, 1.11588399e+00, 1.24519708e+00, 1.38949549e+00, 2369a19cd78SMatthias Ringwald 1.55051578e+00, 1.73019574e+00, 1.93069773e+00, 2.15443469e+00, 2379a19cd78SMatthias Ringwald 2.40409918e+00, 2.68269580e+00, 2.99357729e+00, 3.34048498e+00, 2389a19cd78SMatthias Ringwald 3.72759372e+00, 4.15956216e+00, 4.64158883e+00, 5.17947468e+00, 2399a19cd78SMatthias Ringwald 5.77969288e+00, 6.44946677e+00, 7.19685673e+00, 8.03085722e+00, 2409a19cd78SMatthias Ringwald 8.96150502e+00, 1.00000000e+01, 1.11588399e+01, 1.24519708e+01, 2419a19cd78SMatthias Ringwald 1.38949549e+01, 1.55051578e+01, 1.73019574e+01, 1.93069773e+01, 2429a19cd78SMatthias Ringwald 2.15443469e+01, 2.40409918e+01, 2.68269580e+01, 2.99357729e+01, 2439a19cd78SMatthias Ringwald 3.34048498e+01, 3.72759372e+01, 4.15956216e+01, 4.64158883e+01, 2449a19cd78SMatthias Ringwald 5.17947468e+01, 5.77969288e+01, 6.44946677e+01, 7.19685673e+01, 2459a19cd78SMatthias Ringwald 8.03085722e+01, 8.96150502e+01, 1.00000000e+02, 1.11588399e+02, 2469a19cd78SMatthias Ringwald 1.24519708e+02, 1.38949549e+02, 1.55051578e+02, 1.73019574e+02, 2479a19cd78SMatthias Ringwald 1.93069773e+02, 2.15443469e+02, 2.40409918e+02, 2.68269580e+02, 2489a19cd78SMatthias Ringwald 2.99357729e+02, 3.34048498e+02, 3.72759372e+02, 4.15956216e+02, 2499a19cd78SMatthias Ringwald 4.64158883e+02, 5.17947468e+02, 5.77969288e+02, 6.44946677e+02, 2509a19cd78SMatthias Ringwald 7.19685673e+02, 8.03085722e+02, 8.96150502e+02, 1.00000000e+03 }, 2519a19cd78SMatthias Ringwald }; 2529a19cd78SMatthias Ringwald 2539a19cd78SMatthias Ringwald float e[LC3_NUM_BANDS]; 2549a19cd78SMatthias Ringwald 2559a19cd78SMatthias Ringwald /* --- Copy and padding --- */ 2569a19cd78SMatthias Ringwald 2579a19cd78SMatthias Ringwald int nb = LC3_MIN(lc3_band_lim[dt][sr][LC3_NUM_BANDS], LC3_NUM_BANDS); 2589a19cd78SMatthias Ringwald int n2 = LC3_NUM_BANDS - nb; 2599a19cd78SMatthias Ringwald 2609a19cd78SMatthias Ringwald for (int i2 = 0; i2 < n2; i2++) 2619a19cd78SMatthias Ringwald e[2*i2 + 0] = e[2*i2 + 1] = eb[i2]; 2629a19cd78SMatthias Ringwald 2639a19cd78SMatthias Ringwald memcpy(e + 2*n2, eb + n2, (nb - n2) * sizeof(float)); 2649a19cd78SMatthias Ringwald 2659a19cd78SMatthias Ringwald /* --- Smoothing, pre-emphasis and logarithm --- */ 2669a19cd78SMatthias Ringwald 2679a19cd78SMatthias Ringwald const float *ge = ge_table[sr]; 2689a19cd78SMatthias Ringwald 2699a19cd78SMatthias Ringwald float e0 = e[0], e1 = e[0], e2; 2709a19cd78SMatthias Ringwald float e_sum = 0; 2719a19cd78SMatthias Ringwald 2729a19cd78SMatthias Ringwald for (int i = 0; i < LC3_NUM_BANDS-1; ) { 273*4930cef6SMatthias Ringwald e[i] = (e0 * 0.25f + e1 * 0.5f + (e2 = e[i+1]) * 0.25f) * ge[i]; 2749a19cd78SMatthias Ringwald e_sum += e[i++]; 2759a19cd78SMatthias Ringwald 276*4930cef6SMatthias Ringwald e[i] = (e1 * 0.25f + e2 * 0.5f + (e0 = e[i+1]) * 0.25f) * ge[i]; 2779a19cd78SMatthias Ringwald e_sum += e[i++]; 2789a19cd78SMatthias Ringwald 279*4930cef6SMatthias Ringwald e[i] = (e2 * 0.25f + e0 * 0.5f + (e1 = e[i+1]) * 0.25f) * ge[i]; 2809a19cd78SMatthias Ringwald e_sum += e[i++]; 2819a19cd78SMatthias Ringwald } 2829a19cd78SMatthias Ringwald 283*4930cef6SMatthias Ringwald e[LC3_NUM_BANDS-1] = (e0 * 0.25f + e1 * 0.75f) * ge[LC3_NUM_BANDS-1]; 2849a19cd78SMatthias Ringwald e_sum += e[LC3_NUM_BANDS-1]; 2859a19cd78SMatthias Ringwald 286*4930cef6SMatthias Ringwald float noise_floor = fmaxf(e_sum * (1e-4f / 64), 0x1p-32f); 2879a19cd78SMatthias Ringwald 2889a19cd78SMatthias Ringwald for (int i = 0; i < LC3_NUM_BANDS; i++) 289*4930cef6SMatthias Ringwald e[i] = fast_log2f(fmaxf(e[i], noise_floor)) * 0.5f; 2909a19cd78SMatthias Ringwald 2919a19cd78SMatthias Ringwald /* --- Grouping & scaling --- */ 2929a19cd78SMatthias Ringwald 2939a19cd78SMatthias Ringwald float scf_sum; 2949a19cd78SMatthias Ringwald 295*4930cef6SMatthias Ringwald scf[0] = (e[0] + e[4]) * 1.f/12 + 296*4930cef6SMatthias Ringwald (e[0] + e[3]) * 2.f/12 + 297*4930cef6SMatthias Ringwald (e[1] + e[2]) * 3.f/12 ; 2989a19cd78SMatthias Ringwald scf_sum = scf[0]; 2999a19cd78SMatthias Ringwald 3009a19cd78SMatthias Ringwald for (int i = 1; i < 15; i++) { 301*4930cef6SMatthias Ringwald scf[i] = (e[4*i-1] + e[4*i+4]) * 1.f/12 + 302*4930cef6SMatthias Ringwald (e[4*i ] + e[4*i+3]) * 2.f/12 + 303*4930cef6SMatthias Ringwald (e[4*i+1] + e[4*i+2]) * 3.f/12 ; 3049a19cd78SMatthias Ringwald scf_sum += scf[i]; 3059a19cd78SMatthias Ringwald } 3069a19cd78SMatthias Ringwald 307*4930cef6SMatthias Ringwald scf[15] = (e[59] + e[63]) * 1.f/12 + 308*4930cef6SMatthias Ringwald (e[60] + e[63]) * 2.f/12 + 309*4930cef6SMatthias Ringwald (e[61] + e[62]) * 3.f/12 ; 3109a19cd78SMatthias Ringwald scf_sum += scf[15]; 3119a19cd78SMatthias Ringwald 3129a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) 313*4930cef6SMatthias Ringwald scf[i] = 0.85f * (scf[i] - scf_sum * 1.f/16); 3149a19cd78SMatthias Ringwald 3159a19cd78SMatthias Ringwald /* --- Attack handling --- */ 3169a19cd78SMatthias Ringwald 3179a19cd78SMatthias Ringwald if (!att) 3189a19cd78SMatthias Ringwald return; 3199a19cd78SMatthias Ringwald 3209a19cd78SMatthias Ringwald float s0, s1 = scf[0], s2 = scf[1], s3 = scf[2], s4 = scf[3]; 3219a19cd78SMatthias Ringwald float sn = s1 + s2; 3229a19cd78SMatthias Ringwald 323*4930cef6SMatthias Ringwald scf[0] = (sn += s3) * 1.f/3; 324*4930cef6SMatthias Ringwald scf[1] = (sn += s4) * 1.f/4; 3259a19cd78SMatthias Ringwald scf_sum = scf[0] + scf[1]; 3269a19cd78SMatthias Ringwald 3279a19cd78SMatthias Ringwald for (int i = 2; i < 14; i++, sn -= s0) { 3289a19cd78SMatthias Ringwald s0 = s1, s1 = s2, s2 = s3, s3 = s4, s4 = scf[i+2]; 329*4930cef6SMatthias Ringwald scf[i] = (sn += s4) * 1.f/5; 3309a19cd78SMatthias Ringwald scf_sum += scf[i]; 3319a19cd78SMatthias Ringwald } 3329a19cd78SMatthias Ringwald 333*4930cef6SMatthias Ringwald scf[14] = (sn ) * 1.f/4; 334*4930cef6SMatthias Ringwald scf[15] = (sn -= s1) * 1.f/3; 3359a19cd78SMatthias Ringwald scf_sum += scf[14] + scf[15]; 3369a19cd78SMatthias Ringwald 3379a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) 338*4930cef6SMatthias Ringwald scf[i] = (dt == LC3_DT_7M5 ? 0.3f : 0.5f) * 339*4930cef6SMatthias Ringwald (scf[i] - scf_sum * 1.f/16); 3409a19cd78SMatthias Ringwald } 3419a19cd78SMatthias Ringwald 3429a19cd78SMatthias Ringwald /** 3439a19cd78SMatthias Ringwald * Codebooks 3449a19cd78SMatthias Ringwald * scf Input 16 scale factors 3459a19cd78SMatthias Ringwald * lf/hfcb_idx Output the low and high frequency codebooks index 3469a19cd78SMatthias Ringwald */ 347*4930cef6SMatthias Ringwald LC3_HOT static void resolve_codebooks( 348*4930cef6SMatthias Ringwald const float *scf, int *lfcb_idx, int *hfcb_idx) 3499a19cd78SMatthias Ringwald { 3509a19cd78SMatthias Ringwald float dlfcb_max = 0, dhfcb_max = 0; 3519a19cd78SMatthias Ringwald *lfcb_idx = *hfcb_idx = 0; 3529a19cd78SMatthias Ringwald 3539a19cd78SMatthias Ringwald for (int icb = 0; icb < 32; icb++) { 3549a19cd78SMatthias Ringwald const float *lfcb = lc3_sns_lfcb[icb]; 3559a19cd78SMatthias Ringwald const float *hfcb = lc3_sns_hfcb[icb]; 3569a19cd78SMatthias Ringwald float dlfcb = 0, dhfcb = 0; 3579a19cd78SMatthias Ringwald 3589a19cd78SMatthias Ringwald for (int i = 0; i < 8; i++) { 3599a19cd78SMatthias Ringwald dlfcb += (scf[ i] - lfcb[i]) * (scf[ i] - lfcb[i]); 3609a19cd78SMatthias Ringwald dhfcb += (scf[8+i] - hfcb[i]) * (scf[8+i] - hfcb[i]); 3619a19cd78SMatthias Ringwald } 3629a19cd78SMatthias Ringwald 3639a19cd78SMatthias Ringwald if (icb == 0 || dlfcb < dlfcb_max) 3649a19cd78SMatthias Ringwald *lfcb_idx = icb, dlfcb_max = dlfcb; 3659a19cd78SMatthias Ringwald 3669a19cd78SMatthias Ringwald if (icb == 0 || dhfcb < dhfcb_max) 3679a19cd78SMatthias Ringwald *hfcb_idx = icb, dhfcb_max = dhfcb; 3689a19cd78SMatthias Ringwald } 3699a19cd78SMatthias Ringwald } 3709a19cd78SMatthias Ringwald 3719a19cd78SMatthias Ringwald /** 3729a19cd78SMatthias Ringwald * Unit energy normalize pulse configuration 3739a19cd78SMatthias Ringwald * c Pulse configuration 3749a19cd78SMatthias Ringwald * cn Normalized pulse configuration 3759a19cd78SMatthias Ringwald */ 376*4930cef6SMatthias Ringwald LC3_HOT static void normalize(const int *c, float *cn) 3779a19cd78SMatthias Ringwald { 3789a19cd78SMatthias Ringwald int c2_sum = 0; 3799a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) 3809a19cd78SMatthias Ringwald c2_sum += c[i] * c[i]; 3819a19cd78SMatthias Ringwald 3829a19cd78SMatthias Ringwald float c_norm = 1.f / sqrtf(c2_sum); 3839a19cd78SMatthias Ringwald 3849a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) 3859a19cd78SMatthias Ringwald cn[i] = c[i] * c_norm; 3869a19cd78SMatthias Ringwald } 3879a19cd78SMatthias Ringwald 3889a19cd78SMatthias Ringwald /** 3899a19cd78SMatthias Ringwald * Sub-procedure of `quantize()`, add unit pulse 3909a19cd78SMatthias Ringwald * x, y, n Transformed residual, and vector of pulses with length 3919a19cd78SMatthias Ringwald * start, end Current number of pulses, limit to reach 3929a19cd78SMatthias Ringwald * corr, energy Correlation (x,y) and y energy, updated at output 3939a19cd78SMatthias Ringwald */ 394*4930cef6SMatthias Ringwald LC3_HOT static void add_pulse(const float *x, int *y, int n, 3959a19cd78SMatthias Ringwald int start, int end, float *corr, float *energy) 3969a19cd78SMatthias Ringwald { 3979a19cd78SMatthias Ringwald for (int k = start; k < end; k++) { 3989a19cd78SMatthias Ringwald float best_c2 = (*corr + x[0]) * (*corr + x[0]); 3999a19cd78SMatthias Ringwald float best_e = *energy + 2*y[0] + 1; 4009a19cd78SMatthias Ringwald int nbest = 0; 4019a19cd78SMatthias Ringwald 4029a19cd78SMatthias Ringwald for (int i = 1; i < n; i++) { 4039a19cd78SMatthias Ringwald float c2 = (*corr + x[i]) * (*corr + x[i]); 4049a19cd78SMatthias Ringwald float e = *energy + 2*y[i] + 1; 4059a19cd78SMatthias Ringwald 4069a19cd78SMatthias Ringwald if (c2 * best_e > e * best_c2) 4079a19cd78SMatthias Ringwald best_c2 = c2, best_e = e, nbest = i; 4089a19cd78SMatthias Ringwald } 4099a19cd78SMatthias Ringwald 4109a19cd78SMatthias Ringwald *corr += x[nbest]; 4119a19cd78SMatthias Ringwald *energy += 2*y[nbest] + 1; 4129a19cd78SMatthias Ringwald y[nbest]++; 4139a19cd78SMatthias Ringwald } 4149a19cd78SMatthias Ringwald } 4159a19cd78SMatthias Ringwald 4169a19cd78SMatthias Ringwald /** 4179a19cd78SMatthias Ringwald * Quantization of codebooks residual 4189a19cd78SMatthias Ringwald * scf Input 16 scale factors, output quantized version 4199a19cd78SMatthias Ringwald * lf/hfcb_idx Codebooks index 4209a19cd78SMatthias Ringwald * c, cn Output 4 pulse configurations candidates, normalized 4219a19cd78SMatthias Ringwald * shape/gain_idx Output selected shape/gain indexes 4229a19cd78SMatthias Ringwald */ 423*4930cef6SMatthias Ringwald LC3_HOT static void quantize(const float *scf, int lfcb_idx, int hfcb_idx, 4249a19cd78SMatthias Ringwald int (*c)[16], float (*cn)[16], int *shape_idx, int *gain_idx) 4259a19cd78SMatthias Ringwald { 4269a19cd78SMatthias Ringwald /* --- Residual --- */ 4279a19cd78SMatthias Ringwald 4289a19cd78SMatthias Ringwald const float *lfcb = lc3_sns_lfcb[lfcb_idx]; 4299a19cd78SMatthias Ringwald const float *hfcb = lc3_sns_hfcb[hfcb_idx]; 4309a19cd78SMatthias Ringwald float r[16], x[16]; 4319a19cd78SMatthias Ringwald 4329a19cd78SMatthias Ringwald for (int i = 0; i < 8; i++) { 4339a19cd78SMatthias Ringwald r[ i] = scf[ i] - lfcb[i]; 4349a19cd78SMatthias Ringwald r[8+i] = scf[8+i] - hfcb[i]; 4359a19cd78SMatthias Ringwald } 4369a19cd78SMatthias Ringwald 4379a19cd78SMatthias Ringwald dct16_forward(r, x); 4389a19cd78SMatthias Ringwald 4399a19cd78SMatthias Ringwald /* --- Shape 3 candidate --- 4409a19cd78SMatthias Ringwald * Project to or below pyramid N = 16, K = 6, 4419a19cd78SMatthias Ringwald * then add unit pulses until you reach K = 6, over N = 16 */ 4429a19cd78SMatthias Ringwald 4439a19cd78SMatthias Ringwald float xm[16]; 4449a19cd78SMatthias Ringwald float xm_sum = 0; 4459a19cd78SMatthias Ringwald 4469a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) { 4479a19cd78SMatthias Ringwald xm[i] = fabsf(x[i]); 4489a19cd78SMatthias Ringwald xm_sum += xm[i]; 4499a19cd78SMatthias Ringwald } 4509a19cd78SMatthias Ringwald 451*4930cef6SMatthias Ringwald float proj_factor = (6 - 1) / fmaxf(xm_sum, 1e-31f); 4529a19cd78SMatthias Ringwald float corr = 0, energy = 0; 4539a19cd78SMatthias Ringwald int npulses = 0; 4549a19cd78SMatthias Ringwald 4559a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) { 4569a19cd78SMatthias Ringwald c[3][i] = floorf(xm[i] * proj_factor); 4579a19cd78SMatthias Ringwald npulses += c[3][i]; 4589a19cd78SMatthias Ringwald corr += c[3][i] * xm[i]; 4599a19cd78SMatthias Ringwald energy += c[3][i] * c[3][i]; 4609a19cd78SMatthias Ringwald } 4619a19cd78SMatthias Ringwald 4629a19cd78SMatthias Ringwald add_pulse(xm, c[3], 16, npulses, 6, &corr, &energy); 4639a19cd78SMatthias Ringwald npulses = 6; 4649a19cd78SMatthias Ringwald 4659a19cd78SMatthias Ringwald /* --- Shape 2 candidate --- 4669a19cd78SMatthias Ringwald * Add unit pulses until you reach K = 8 on shape 3 */ 4679a19cd78SMatthias Ringwald 4689a19cd78SMatthias Ringwald memcpy(c[2], c[3], sizeof(c[2])); 4699a19cd78SMatthias Ringwald 4709a19cd78SMatthias Ringwald add_pulse(xm, c[2], 16, npulses, 8, &corr, &energy); 4719a19cd78SMatthias Ringwald npulses = 8; 4729a19cd78SMatthias Ringwald 4739a19cd78SMatthias Ringwald /* --- Shape 1 candidate --- 4749a19cd78SMatthias Ringwald * Remove any unit pulses from shape 2 that are not part of 0 to 9 4759a19cd78SMatthias Ringwald * Update energy and correlation terms accordingly 4769a19cd78SMatthias Ringwald * Add unit pulses until you reach K = 10, over N = 10 */ 4779a19cd78SMatthias Ringwald 4789a19cd78SMatthias Ringwald memcpy(c[1], c[2], sizeof(c[1])); 4799a19cd78SMatthias Ringwald 4809a19cd78SMatthias Ringwald for (int i = 10; i < 16; i++) { 4819a19cd78SMatthias Ringwald c[1][i] = 0; 4829a19cd78SMatthias Ringwald npulses -= c[2][i]; 4839a19cd78SMatthias Ringwald corr -= c[2][i] * xm[i]; 4849a19cd78SMatthias Ringwald energy -= c[2][i] * c[2][i]; 4859a19cd78SMatthias Ringwald } 4869a19cd78SMatthias Ringwald 4879a19cd78SMatthias Ringwald add_pulse(xm, c[1], 10, npulses, 10, &corr, &energy); 4889a19cd78SMatthias Ringwald npulses = 10; 4899a19cd78SMatthias Ringwald 4909a19cd78SMatthias Ringwald /* --- Shape 0 candidate --- 4919a19cd78SMatthias Ringwald * Add unit pulses until you reach K = 1, on shape 1 */ 4929a19cd78SMatthias Ringwald 4939a19cd78SMatthias Ringwald memcpy(c[0], c[1], sizeof(c[0])); 4949a19cd78SMatthias Ringwald 4959a19cd78SMatthias Ringwald add_pulse(xm + 10, c[0] + 10, 6, 0, 1, &corr, &energy); 4969a19cd78SMatthias Ringwald 4979a19cd78SMatthias Ringwald /* --- Add sign and unit energy normalize --- */ 4989a19cd78SMatthias Ringwald 4999a19cd78SMatthias Ringwald for (int j = 0; j < 16; j++) 5009a19cd78SMatthias Ringwald for (int i = 0; i < 4; i++) 5019a19cd78SMatthias Ringwald c[i][j] = x[j] < 0 ? -c[i][j] : c[i][j]; 5029a19cd78SMatthias Ringwald 5039a19cd78SMatthias Ringwald for (int i = 0; i < 4; i++) 5049a19cd78SMatthias Ringwald normalize(c[i], cn[i]); 5059a19cd78SMatthias Ringwald 5069a19cd78SMatthias Ringwald /* --- Determe shape & gain index --- 5079a19cd78SMatthias Ringwald * Search the Mean Square Error, within (shape, gain) combinations */ 5089a19cd78SMatthias Ringwald 5099a19cd78SMatthias Ringwald float mse_min = INFINITY; 5109a19cd78SMatthias Ringwald *shape_idx = *gain_idx = 0; 5119a19cd78SMatthias Ringwald 5129a19cd78SMatthias Ringwald for (int ic = 0; ic < 4; ic++) { 5139a19cd78SMatthias Ringwald const struct lc3_sns_vq_gains *cgains = lc3_sns_vq_gains + ic; 5149a19cd78SMatthias Ringwald float cmse_min = INFINITY; 5159a19cd78SMatthias Ringwald int cgain_idx = 0; 5169a19cd78SMatthias Ringwald 5179a19cd78SMatthias Ringwald for (int ig = 0; ig < cgains->count; ig++) { 5189a19cd78SMatthias Ringwald float g = cgains->v[ig]; 5199a19cd78SMatthias Ringwald 5209a19cd78SMatthias Ringwald float mse = 0; 5219a19cd78SMatthias Ringwald for (int i = 0; i < 16; i++) 5229a19cd78SMatthias Ringwald mse += (x[i] - g * cn[ic][i]) * (x[i] - g * cn[ic][i]); 5239a19cd78SMatthias Ringwald 5249a19cd78SMatthias Ringwald if (mse < cmse_min) { 5259a19cd78SMatthias Ringwald cgain_idx = ig, 5269a19cd78SMatthias Ringwald cmse_min = mse; 5279a19cd78SMatthias Ringwald } 5289a19cd78SMatthias Ringwald } 5299a19cd78SMatthias Ringwald 5309a19cd78SMatthias Ringwald if (cmse_min < mse_min) { 5319a19cd78SMatthias Ringwald *shape_idx = ic, *gain_idx = cgain_idx; 5329a19cd78SMatthias Ringwald mse_min = cmse_min; 5339a19cd78SMatthias Ringwald } 5349a19cd78SMatthias Ringwald } 5359a19cd78SMatthias Ringwald } 5369a19cd78SMatthias Ringwald 5379a19cd78SMatthias Ringwald /** 5389a19cd78SMatthias Ringwald * Unquantization of codebooks residual 5399a19cd78SMatthias Ringwald * lf/hfcb_idx Low and high frequency codebooks index 5409a19cd78SMatthias Ringwald * c Table of normalized pulse configuration 5419a19cd78SMatthias Ringwald * shape/gain Selected shape/gain indexes 5429a19cd78SMatthias Ringwald * scf Return unquantized scale factors 5439a19cd78SMatthias Ringwald */ 544*4930cef6SMatthias Ringwald LC3_HOT static void unquantize(int lfcb_idx, int hfcb_idx, 5459a19cd78SMatthias Ringwald const float *c, int shape, int gain, float *scf) 5469a19cd78SMatthias Ringwald { 5479a19cd78SMatthias Ringwald const float *lfcb = lc3_sns_lfcb[lfcb_idx]; 5489a19cd78SMatthias Ringwald const float *hfcb = lc3_sns_hfcb[hfcb_idx]; 5499a19cd78SMatthias Ringwald float g = lc3_sns_vq_gains[shape].v[gain]; 5509a19cd78SMatthias Ringwald 5519a19cd78SMatthias Ringwald dct16_inverse(c, scf); 5529a19cd78SMatthias Ringwald 5539a19cd78SMatthias Ringwald for (int i = 0; i < 8; i++) 5549a19cd78SMatthias Ringwald scf[i] = lfcb[i] + g * scf[i]; 5559a19cd78SMatthias Ringwald 5569a19cd78SMatthias Ringwald for (int i = 8; i < 16; i++) 5579a19cd78SMatthias Ringwald scf[i] = hfcb[i-8] + g * scf[i]; 5589a19cd78SMatthias Ringwald } 5599a19cd78SMatthias Ringwald 5609a19cd78SMatthias Ringwald /** 5619a19cd78SMatthias Ringwald * Sub-procedure of `sns_enumerate()`, enumeration of a vector 5629a19cd78SMatthias Ringwald * c, n Table of pulse configuration, and length 5639a19cd78SMatthias Ringwald * idx, ls Return enumeration set 5649a19cd78SMatthias Ringwald */ 5659a19cd78SMatthias Ringwald static void enum_mvpq(const int *c, int n, int *idx, bool *ls) 5669a19cd78SMatthias Ringwald { 5679a19cd78SMatthias Ringwald int ci, i, j; 5689a19cd78SMatthias Ringwald 5699a19cd78SMatthias Ringwald /* --- Scan for 1st significant coeff --- */ 5709a19cd78SMatthias Ringwald 5719a19cd78SMatthias Ringwald for (i = 0, c += n; (ci = *(--c)) == 0 ; i++); 5729a19cd78SMatthias Ringwald 5739a19cd78SMatthias Ringwald *idx = 0; 5749a19cd78SMatthias Ringwald *ls = ci < 0; 5759a19cd78SMatthias Ringwald 5769a19cd78SMatthias Ringwald /* --- Scan remaining coefficients --- */ 5779a19cd78SMatthias Ringwald 5789a19cd78SMatthias Ringwald for (i++, j = LC3_ABS(ci); i < n; i++, j += LC3_ABS(ci)) { 5799a19cd78SMatthias Ringwald 5809a19cd78SMatthias Ringwald if ((ci = *(--c)) != 0) { 5819a19cd78SMatthias Ringwald *idx = (*idx << 1) | *ls; 5829a19cd78SMatthias Ringwald *ls = ci < 0; 5839a19cd78SMatthias Ringwald } 5849a19cd78SMatthias Ringwald 5859a19cd78SMatthias Ringwald *idx += lc3_sns_mpvq_offsets[i][j]; 5869a19cd78SMatthias Ringwald } 5879a19cd78SMatthias Ringwald } 5889a19cd78SMatthias Ringwald 5899a19cd78SMatthias Ringwald /** 5909a19cd78SMatthias Ringwald * Sub-procedure of `sns_deenumerate()`, deenumeration of a vector 5919a19cd78SMatthias Ringwald * idx, ls Enumeration set 5929a19cd78SMatthias Ringwald * npulses Number of pulses in the set 5939a19cd78SMatthias Ringwald * c, n Table of pulses configuration, and length 5949a19cd78SMatthias Ringwald */ 5959a19cd78SMatthias Ringwald static void deenum_mvpq(int idx, bool ls, int npulses, int *c, int n) 5969a19cd78SMatthias Ringwald { 5979a19cd78SMatthias Ringwald int i; 5989a19cd78SMatthias Ringwald 5999a19cd78SMatthias Ringwald /* --- Scan for coefficients --- */ 6009a19cd78SMatthias Ringwald 6019a19cd78SMatthias Ringwald for (i = n-1; i >= 0 && idx; i--) { 6029a19cd78SMatthias Ringwald 6039a19cd78SMatthias Ringwald int ci = 0; 6049a19cd78SMatthias Ringwald 6059a19cd78SMatthias Ringwald for (ci = 0; idx < lc3_sns_mpvq_offsets[i][npulses - ci]; ci++); 6069a19cd78SMatthias Ringwald idx -= lc3_sns_mpvq_offsets[i][npulses - ci]; 6079a19cd78SMatthias Ringwald 6089a19cd78SMatthias Ringwald *(c++) = ls ? -ci : ci; 6099a19cd78SMatthias Ringwald npulses -= ci; 6109a19cd78SMatthias Ringwald if (ci > 0) { 6119a19cd78SMatthias Ringwald ls = idx & 1; 6129a19cd78SMatthias Ringwald idx >>= 1; 6139a19cd78SMatthias Ringwald } 6149a19cd78SMatthias Ringwald } 6159a19cd78SMatthias Ringwald 6169a19cd78SMatthias Ringwald /* --- Set last significant --- */ 6179a19cd78SMatthias Ringwald 6189a19cd78SMatthias Ringwald int ci = npulses; 6199a19cd78SMatthias Ringwald 6209a19cd78SMatthias Ringwald if (i-- >= 0) 6219a19cd78SMatthias Ringwald *(c++) = ls ? -ci : ci; 6229a19cd78SMatthias Ringwald 6239a19cd78SMatthias Ringwald while (i-- >= 0) 6249a19cd78SMatthias Ringwald *(c++) = 0; 6259a19cd78SMatthias Ringwald } 6269a19cd78SMatthias Ringwald 6279a19cd78SMatthias Ringwald /** 6289a19cd78SMatthias Ringwald * SNS Enumeration of PVQ configuration 6299a19cd78SMatthias Ringwald * shape Selected shape index 6309a19cd78SMatthias Ringwald * c Selected pulse configuration 6319a19cd78SMatthias Ringwald * idx_a, ls_a Return enumeration set A 6329a19cd78SMatthias Ringwald * idx_b, ls_b Return enumeration set B (shape = 0) 6339a19cd78SMatthias Ringwald */ 6349a19cd78SMatthias Ringwald static void enumerate(int shape, const int *c, 6359a19cd78SMatthias Ringwald int *idx_a, bool *ls_a, int *idx_b, bool *ls_b) 6369a19cd78SMatthias Ringwald { 6379a19cd78SMatthias Ringwald enum_mvpq(c, shape < 2 ? 10 : 16, idx_a, ls_a); 6389a19cd78SMatthias Ringwald 6399a19cd78SMatthias Ringwald if (shape == 0) 6409a19cd78SMatthias Ringwald enum_mvpq(c + 10, 6, idx_b, ls_b); 6419a19cd78SMatthias Ringwald } 6429a19cd78SMatthias Ringwald 6439a19cd78SMatthias Ringwald /** 6449a19cd78SMatthias Ringwald * SNS Deenumeration of PVQ configuration 6459a19cd78SMatthias Ringwald * shape Selected shape index 6469a19cd78SMatthias Ringwald * idx_a, ls_a enumeration set A 6479a19cd78SMatthias Ringwald * idx_b, ls_b enumeration set B (shape = 0) 6489a19cd78SMatthias Ringwald * c Return pulse configuration 6499a19cd78SMatthias Ringwald */ 6509a19cd78SMatthias Ringwald static void deenumerate(int shape, 6519a19cd78SMatthias Ringwald int idx_a, bool ls_a, int idx_b, bool ls_b, int *c) 6529a19cd78SMatthias Ringwald { 6539a19cd78SMatthias Ringwald int npulses_a = (const int []){ 10, 10, 8, 6 }[shape]; 6549a19cd78SMatthias Ringwald 6559a19cd78SMatthias Ringwald deenum_mvpq(idx_a, ls_a, npulses_a, c, shape < 2 ? 10 : 16); 6569a19cd78SMatthias Ringwald 6579a19cd78SMatthias Ringwald if (shape == 0) 6589a19cd78SMatthias Ringwald deenum_mvpq(idx_b, ls_b, 1, c + 10, 6); 6599a19cd78SMatthias Ringwald else if (shape == 1) 6609a19cd78SMatthias Ringwald memset(c + 10, 0, 6 * sizeof(*c)); 6619a19cd78SMatthias Ringwald } 6629a19cd78SMatthias Ringwald 6639a19cd78SMatthias Ringwald 6649a19cd78SMatthias Ringwald /* ---------------------------------------------------------------------------- 6659a19cd78SMatthias Ringwald * Filtering 6669a19cd78SMatthias Ringwald * -------------------------------------------------------------------------- */ 6679a19cd78SMatthias Ringwald 6689a19cd78SMatthias Ringwald /** 6699a19cd78SMatthias Ringwald * Spectral shaping 6709a19cd78SMatthias Ringwald * dt, sr Duration and samplerate of the frame 6719a19cd78SMatthias Ringwald * scf_q Quantized scale factors 6729a19cd78SMatthias Ringwald * inv True on inverse shaping, False otherwise 6739a19cd78SMatthias Ringwald * x Spectral coefficients 6749a19cd78SMatthias Ringwald * y Return shapped coefficients 6759a19cd78SMatthias Ringwald * 6769a19cd78SMatthias Ringwald * `x` and `y` can be the same buffer 6779a19cd78SMatthias Ringwald */ 678*4930cef6SMatthias Ringwald LC3_HOT static void spectral_shaping(enum lc3_dt dt, enum lc3_srate sr, 6799a19cd78SMatthias Ringwald const float *scf_q, bool inv, const float *x, float *y) 6809a19cd78SMatthias Ringwald { 6819a19cd78SMatthias Ringwald /* --- Interpolate scale factors --- */ 6829a19cd78SMatthias Ringwald 6839a19cd78SMatthias Ringwald float scf[LC3_NUM_BANDS]; 6849a19cd78SMatthias Ringwald float s0, s1 = inv ? -scf_q[0] : scf_q[0]; 6859a19cd78SMatthias Ringwald 6869a19cd78SMatthias Ringwald scf[0] = scf[1] = s1; 6879a19cd78SMatthias Ringwald for (int i = 0; i < 15; i++) { 6889a19cd78SMatthias Ringwald s0 = s1, s1 = inv ? -scf_q[i+1] : scf_q[i+1]; 689*4930cef6SMatthias Ringwald scf[4*i+2] = s0 + 0.125f * (s1 - s0); 690*4930cef6SMatthias Ringwald scf[4*i+3] = s0 + 0.375f * (s1 - s0); 691*4930cef6SMatthias Ringwald scf[4*i+4] = s0 + 0.625f * (s1 - s0); 692*4930cef6SMatthias Ringwald scf[4*i+5] = s0 + 0.875f * (s1 - s0); 6939a19cd78SMatthias Ringwald } 694*4930cef6SMatthias Ringwald scf[62] = s1 + 0.125f * (s1 - s0); 695*4930cef6SMatthias Ringwald scf[63] = s1 + 0.375f * (s1 - s0); 6969a19cd78SMatthias Ringwald 6979a19cd78SMatthias Ringwald int nb = LC3_MIN(lc3_band_lim[dt][sr][LC3_NUM_BANDS], LC3_NUM_BANDS); 6989a19cd78SMatthias Ringwald int n2 = LC3_NUM_BANDS - nb; 6999a19cd78SMatthias Ringwald 7009a19cd78SMatthias Ringwald for (int i2 = 0; i2 < n2; i2++) 701*4930cef6SMatthias Ringwald scf[i2] = 0.5f * (scf[2*i2] + scf[2*i2+1]); 7029a19cd78SMatthias Ringwald 7039a19cd78SMatthias Ringwald if (n2 > 0) 7049a19cd78SMatthias Ringwald memmove(scf + n2, scf + 2*n2, (nb - n2) * sizeof(float)); 7059a19cd78SMatthias Ringwald 7069a19cd78SMatthias Ringwald /* --- Spectral shaping --- */ 7079a19cd78SMatthias Ringwald 7089a19cd78SMatthias Ringwald const int *lim = lc3_band_lim[dt][sr]; 7099a19cd78SMatthias Ringwald 7109a19cd78SMatthias Ringwald for (int i = 0, ib = 0; ib < nb; ib++) { 711*4930cef6SMatthias Ringwald float g_sns = fast_exp2f(-scf[ib]); 7129a19cd78SMatthias Ringwald 7139a19cd78SMatthias Ringwald for ( ; i < lim[ib+1]; i++) 7149a19cd78SMatthias Ringwald y[i] = x[i] * g_sns; 7159a19cd78SMatthias Ringwald } 7169a19cd78SMatthias Ringwald } 7179a19cd78SMatthias Ringwald 7189a19cd78SMatthias Ringwald 7199a19cd78SMatthias Ringwald /* ---------------------------------------------------------------------------- 7209a19cd78SMatthias Ringwald * Interface 7219a19cd78SMatthias Ringwald * -------------------------------------------------------------------------- */ 7229a19cd78SMatthias Ringwald 7239a19cd78SMatthias Ringwald /** 7249a19cd78SMatthias Ringwald * SNS analysis 7259a19cd78SMatthias Ringwald */ 7269a19cd78SMatthias Ringwald void lc3_sns_analyze(enum lc3_dt dt, enum lc3_srate sr, 7279a19cd78SMatthias Ringwald const float *eb, bool att, struct lc3_sns_data *data, 7289a19cd78SMatthias Ringwald const float *x, float *y) 7299a19cd78SMatthias Ringwald { 7309a19cd78SMatthias Ringwald /* Processing steps : 7319a19cd78SMatthias Ringwald * - Determine 16 scale factors from bands energy estimation 7329a19cd78SMatthias Ringwald * - Get codebooks indexes that match thoses scale factors 7339a19cd78SMatthias Ringwald * - Quantize the residual with the selected codebook 7349a19cd78SMatthias Ringwald * - The pulse configuration `c[]` is enumerated 7359a19cd78SMatthias Ringwald * - Finally shape the spectrum coefficients accordingly */ 7369a19cd78SMatthias Ringwald 7379a19cd78SMatthias Ringwald float scf[16], cn[4][16]; 7389a19cd78SMatthias Ringwald int c[4][16]; 7399a19cd78SMatthias Ringwald 7409a19cd78SMatthias Ringwald compute_scale_factors(dt, sr, eb, att, scf); 7419a19cd78SMatthias Ringwald 7429a19cd78SMatthias Ringwald resolve_codebooks(scf, &data->lfcb, &data->hfcb); 7439a19cd78SMatthias Ringwald 7449a19cd78SMatthias Ringwald quantize(scf, data->lfcb, data->hfcb, 7459a19cd78SMatthias Ringwald c, cn, &data->shape, &data->gain); 7469a19cd78SMatthias Ringwald 7479a19cd78SMatthias Ringwald unquantize(data->lfcb, data->hfcb, 7489a19cd78SMatthias Ringwald cn[data->shape], data->shape, data->gain, scf); 7499a19cd78SMatthias Ringwald 7509a19cd78SMatthias Ringwald enumerate(data->shape, c[data->shape], 7519a19cd78SMatthias Ringwald &data->idx_a, &data->ls_a, &data->idx_b, &data->ls_b); 7529a19cd78SMatthias Ringwald 7539a19cd78SMatthias Ringwald spectral_shaping(dt, sr, scf, false, x, y); 7549a19cd78SMatthias Ringwald } 7559a19cd78SMatthias Ringwald 7569a19cd78SMatthias Ringwald /** 7579a19cd78SMatthias Ringwald * SNS synthesis 7589a19cd78SMatthias Ringwald */ 7599a19cd78SMatthias Ringwald void lc3_sns_synthesize(enum lc3_dt dt, enum lc3_srate sr, 7609a19cd78SMatthias Ringwald const lc3_sns_data_t *data, const float *x, float *y) 7619a19cd78SMatthias Ringwald { 7629a19cd78SMatthias Ringwald float scf[16], cn[16]; 7639a19cd78SMatthias Ringwald int c[16]; 7649a19cd78SMatthias Ringwald 7659a19cd78SMatthias Ringwald deenumerate(data->shape, 7669a19cd78SMatthias Ringwald data->idx_a, data->ls_a, data->idx_b, data->ls_b, c); 7679a19cd78SMatthias Ringwald 7689a19cd78SMatthias Ringwald normalize(c, cn); 7699a19cd78SMatthias Ringwald 7709a19cd78SMatthias Ringwald unquantize(data->lfcb, data->hfcb, cn, data->shape, data->gain, scf); 7719a19cd78SMatthias Ringwald 7729a19cd78SMatthias Ringwald spectral_shaping(dt, sr, scf, true, x, y); 7739a19cd78SMatthias Ringwald } 7749a19cd78SMatthias Ringwald 7759a19cd78SMatthias Ringwald /** 7769a19cd78SMatthias Ringwald * Return number of bits coding the bitstream data 7779a19cd78SMatthias Ringwald */ 7789a19cd78SMatthias Ringwald int lc3_sns_get_nbits(void) 7799a19cd78SMatthias Ringwald { 7809a19cd78SMatthias Ringwald return 38; 7819a19cd78SMatthias Ringwald } 7829a19cd78SMatthias Ringwald 7839a19cd78SMatthias Ringwald /** 7849a19cd78SMatthias Ringwald * Put bitstream data 7859a19cd78SMatthias Ringwald */ 7869a19cd78SMatthias Ringwald void lc3_sns_put_data(lc3_bits_t *bits, const struct lc3_sns_data *data) 7879a19cd78SMatthias Ringwald { 7889a19cd78SMatthias Ringwald /* --- Codebooks --- */ 7899a19cd78SMatthias Ringwald 7909a19cd78SMatthias Ringwald lc3_put_bits(bits, data->lfcb, 5); 7919a19cd78SMatthias Ringwald lc3_put_bits(bits, data->hfcb, 5); 7929a19cd78SMatthias Ringwald 7939a19cd78SMatthias Ringwald /* --- Shape, gain and vectors --- * 7949a19cd78SMatthias Ringwald * Write MSB bit of shape index, next LSB bits of shape and gain, 7959a19cd78SMatthias Ringwald * and MVPQ vectors indexes are muxed */ 7969a19cd78SMatthias Ringwald 7979a19cd78SMatthias Ringwald int shape_msb = data->shape >> 1; 7989a19cd78SMatthias Ringwald lc3_put_bit(bits, shape_msb); 7999a19cd78SMatthias Ringwald 8009a19cd78SMatthias Ringwald if (shape_msb == 0) { 8019a19cd78SMatthias Ringwald const int size_a = 2390004; 8029a19cd78SMatthias Ringwald int submode = data->shape & 1; 8039a19cd78SMatthias Ringwald 8049a19cd78SMatthias Ringwald int mux_high = submode == 0 ? 8059a19cd78SMatthias Ringwald 2 * (data->idx_b + 1) + data->ls_b : data->gain & 1; 8069a19cd78SMatthias Ringwald int mux_code = mux_high * size_a + data->idx_a; 8079a19cd78SMatthias Ringwald 8089a19cd78SMatthias Ringwald lc3_put_bits(bits, data->gain >> submode, 1); 8099a19cd78SMatthias Ringwald lc3_put_bits(bits, data->ls_a, 1); 8109a19cd78SMatthias Ringwald lc3_put_bits(bits, mux_code, 25); 8119a19cd78SMatthias Ringwald 8129a19cd78SMatthias Ringwald } else { 8139a19cd78SMatthias Ringwald const int size_a = 15158272; 8149a19cd78SMatthias Ringwald int submode = data->shape & 1; 8159a19cd78SMatthias Ringwald 8169a19cd78SMatthias Ringwald int mux_code = submode == 0 ? 8179a19cd78SMatthias Ringwald data->idx_a : size_a + 2 * data->idx_a + (data->gain & 1); 8189a19cd78SMatthias Ringwald 8199a19cd78SMatthias Ringwald lc3_put_bits(bits, data->gain >> submode, 2); 8209a19cd78SMatthias Ringwald lc3_put_bits(bits, data->ls_a, 1); 8219a19cd78SMatthias Ringwald lc3_put_bits(bits, mux_code, 24); 8229a19cd78SMatthias Ringwald } 8239a19cd78SMatthias Ringwald } 8249a19cd78SMatthias Ringwald 8259a19cd78SMatthias Ringwald /** 8269a19cd78SMatthias Ringwald * Get bitstream data 8279a19cd78SMatthias Ringwald */ 8289a19cd78SMatthias Ringwald int lc3_sns_get_data(lc3_bits_t *bits, struct lc3_sns_data *data) 8299a19cd78SMatthias Ringwald { 8309a19cd78SMatthias Ringwald /* --- Codebooks --- */ 8319a19cd78SMatthias Ringwald 8329a19cd78SMatthias Ringwald *data = (struct lc3_sns_data){ 8339a19cd78SMatthias Ringwald .lfcb = lc3_get_bits(bits, 5), 8349a19cd78SMatthias Ringwald .hfcb = lc3_get_bits(bits, 5) 8359a19cd78SMatthias Ringwald }; 8369a19cd78SMatthias Ringwald 8379a19cd78SMatthias Ringwald /* --- Shape, gain and vectors --- */ 8389a19cd78SMatthias Ringwald 8399a19cd78SMatthias Ringwald int shape_msb = lc3_get_bit(bits); 8409a19cd78SMatthias Ringwald data->gain = lc3_get_bits(bits, 1 + shape_msb); 8419a19cd78SMatthias Ringwald data->ls_a = lc3_get_bit(bits); 8429a19cd78SMatthias Ringwald 8439a19cd78SMatthias Ringwald int mux_code = lc3_get_bits(bits, 25 - shape_msb); 8449a19cd78SMatthias Ringwald 8459a19cd78SMatthias Ringwald if (shape_msb == 0) { 8469a19cd78SMatthias Ringwald const int size_a = 2390004; 8479a19cd78SMatthias Ringwald 8489a19cd78SMatthias Ringwald if (mux_code >= size_a * 14) 8499a19cd78SMatthias Ringwald return -1; 8509a19cd78SMatthias Ringwald 8519a19cd78SMatthias Ringwald data->idx_a = mux_code % size_a; 8529a19cd78SMatthias Ringwald mux_code = mux_code / size_a; 8539a19cd78SMatthias Ringwald 8549a19cd78SMatthias Ringwald data->shape = (mux_code < 2); 8559a19cd78SMatthias Ringwald 8569a19cd78SMatthias Ringwald if (data->shape == 0) { 8579a19cd78SMatthias Ringwald data->idx_b = (mux_code - 2) / 2; 8589a19cd78SMatthias Ringwald data->ls_b = (mux_code - 2) % 2; 8599a19cd78SMatthias Ringwald } else { 8609a19cd78SMatthias Ringwald data->gain = (data->gain << 1) + (mux_code % 2); 8619a19cd78SMatthias Ringwald } 8629a19cd78SMatthias Ringwald 8639a19cd78SMatthias Ringwald } else { 8649a19cd78SMatthias Ringwald const int size_a = 15158272; 8659a19cd78SMatthias Ringwald 8669a19cd78SMatthias Ringwald if (mux_code >= size_a + 1549824) 8679a19cd78SMatthias Ringwald return -1; 8689a19cd78SMatthias Ringwald 8699a19cd78SMatthias Ringwald data->shape = 2 + (mux_code >= size_a); 8709a19cd78SMatthias Ringwald if (data->shape == 2) { 8719a19cd78SMatthias Ringwald data->idx_a = mux_code; 8729a19cd78SMatthias Ringwald } else { 8739a19cd78SMatthias Ringwald mux_code -= size_a; 8749a19cd78SMatthias Ringwald data->idx_a = mux_code / 2; 8759a19cd78SMatthias Ringwald data->gain = (data->gain << 1) + (mux_code % 2); 8769a19cd78SMatthias Ringwald } 8779a19cd78SMatthias Ringwald } 8789a19cd78SMatthias Ringwald 8799a19cd78SMatthias Ringwald return 0; 8809a19cd78SMatthias Ringwald } 881