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