xref: /aosp_15_r20/external/liblc3/src/bwdet.c (revision 49fe348c0058011ee60b6957cdd9d52742df84bc)
1*49fe348cSAndroid Build Coastguard Worker /******************************************************************************
2*49fe348cSAndroid Build Coastguard Worker  *
3*49fe348cSAndroid Build Coastguard Worker  *  Copyright 2022 Google LLC
4*49fe348cSAndroid Build Coastguard Worker  *
5*49fe348cSAndroid Build Coastguard Worker  *  Licensed under the Apache License, Version 2.0 (the "License");
6*49fe348cSAndroid Build Coastguard Worker  *  you may not use this file except in compliance with the License.
7*49fe348cSAndroid Build Coastguard Worker  *  You may obtain a copy of the License at:
8*49fe348cSAndroid Build Coastguard Worker  *
9*49fe348cSAndroid Build Coastguard Worker  *  http://www.apache.org/licenses/LICENSE-2.0
10*49fe348cSAndroid Build Coastguard Worker  *
11*49fe348cSAndroid Build Coastguard Worker  *  Unless required by applicable law or agreed to in writing, software
12*49fe348cSAndroid Build Coastguard Worker  *  distributed under the License is distributed on an "AS IS" BASIS,
13*49fe348cSAndroid Build Coastguard Worker  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*49fe348cSAndroid Build Coastguard Worker  *  See the License for the specific language governing permissions and
15*49fe348cSAndroid Build Coastguard Worker  *  limitations under the License.
16*49fe348cSAndroid Build Coastguard Worker  *
17*49fe348cSAndroid Build Coastguard Worker  ******************************************************************************/
18*49fe348cSAndroid Build Coastguard Worker 
19*49fe348cSAndroid Build Coastguard Worker #include "bwdet.h"
20*49fe348cSAndroid Build Coastguard Worker 
21*49fe348cSAndroid Build Coastguard Worker 
22*49fe348cSAndroid Build Coastguard Worker /**
23*49fe348cSAndroid Build Coastguard Worker  * Bandwidth detector
24*49fe348cSAndroid Build Coastguard Worker  */
lc3_bwdet_run(enum lc3_dt dt,enum lc3_srate sr,const float * e)25*49fe348cSAndroid Build Coastguard Worker enum lc3_bandwidth lc3_bwdet_run(
26*49fe348cSAndroid Build Coastguard Worker     enum lc3_dt dt, enum lc3_srate sr, const float *e)
27*49fe348cSAndroid Build Coastguard Worker {
28*49fe348cSAndroid Build Coastguard Worker     /* Bandwidth regions */
29*49fe348cSAndroid Build Coastguard Worker 
30*49fe348cSAndroid Build Coastguard Worker     struct region { int is : 8; int ie : 8; };
31*49fe348cSAndroid Build Coastguard Worker 
32*49fe348cSAndroid Build Coastguard Worker #if LC3_PLUS
33*49fe348cSAndroid Build Coastguard Worker 
34*49fe348cSAndroid Build Coastguard Worker     static const struct region bws_table_2m5[][LC3_BANDWIDTH_SWB+1] = {
35*49fe348cSAndroid Build Coastguard Worker         { { 24, 34+1 } },
36*49fe348cSAndroid Build Coastguard Worker         { { 24, 32+1 }, { 35, 39+1 } },
37*49fe348cSAndroid Build Coastguard Worker         { { 24, 31+1 }, { 33, 38+1 }, { 39, 42+1 } },
38*49fe348cSAndroid Build Coastguard Worker         { { 22, 29+1 }, { 31, 35+1 }, { 37, 40+1 }, { 41, 43+1 } },
39*49fe348cSAndroid Build Coastguard Worker     };
40*49fe348cSAndroid Build Coastguard Worker 
41*49fe348cSAndroid Build Coastguard Worker     static const struct region bws_table_5m[][LC3_BANDWIDTH_SWB+1] = {
42*49fe348cSAndroid Build Coastguard Worker         { { 39, 49+1 } },
43*49fe348cSAndroid Build Coastguard Worker         { { 35, 44+1 }, { 47, 51+1 } },
44*49fe348cSAndroid Build Coastguard Worker         { { 34, 42+1 }, { 44, 49+1 }, { 50, 53+1 } },
45*49fe348cSAndroid Build Coastguard Worker         { { 32, 40+1 }, { 42, 46+1 }, { 48, 51+1 }, { 52, 54+1 } },
46*49fe348cSAndroid Build Coastguard Worker     };
47*49fe348cSAndroid Build Coastguard Worker 
48*49fe348cSAndroid Build Coastguard Worker #endif /* LC3_PLUS */
49*49fe348cSAndroid Build Coastguard Worker 
50*49fe348cSAndroid Build Coastguard Worker     static const struct region bws_table_7m5[][LC3_BANDWIDTH_SWB+1] = {
51*49fe348cSAndroid Build Coastguard Worker         { { 51, 63+1 } },
52*49fe348cSAndroid Build Coastguard Worker         { { 45, 55+1 }, { 58, 63+1 } },
53*49fe348cSAndroid Build Coastguard Worker         { { 42, 51+1 }, { 53, 58+1 }, { 60, 63+1 } },
54*49fe348cSAndroid Build Coastguard Worker         { { 40, 48+1 }, { 51, 55+1 }, { 57, 60+1 }, { 61, 63+1 } },
55*49fe348cSAndroid Build Coastguard Worker     };
56*49fe348cSAndroid Build Coastguard Worker 
57*49fe348cSAndroid Build Coastguard Worker     static const struct region bws_table_10m[][LC3_BANDWIDTH_SWB+1] = {
58*49fe348cSAndroid Build Coastguard Worker         { { 53, 63+1 } },
59*49fe348cSAndroid Build Coastguard Worker         { { 47, 56+1 }, { 59, 63+1 } },
60*49fe348cSAndroid Build Coastguard Worker         { { 44, 52+1 }, { 54, 59+1 }, { 60, 63+1 } },
61*49fe348cSAndroid Build Coastguard Worker         { { 41, 49+1 }, { 51, 55+1 }, { 57, 60+1 }, { 61, 63+1 } },
62*49fe348cSAndroid Build Coastguard Worker     };
63*49fe348cSAndroid Build Coastguard Worker 
64*49fe348cSAndroid Build Coastguard Worker     static const struct region (*bws_table[])[LC3_BANDWIDTH_SWB+1] = {
65*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_2M5] = LC3_IF_PLUS(bws_table_2m5,  NULL),
66*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_5M ] = LC3_IF_PLUS(bws_table_5m ,  NULL),
67*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_7M5] = bws_table_7m5,
68*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_10M] = bws_table_10m,
69*49fe348cSAndroid Build Coastguard Worker     };
70*49fe348cSAndroid Build Coastguard Worker 
71*49fe348cSAndroid Build Coastguard Worker     static const int l_table[LC3_NUM_DT][LC3_BANDWIDTH_SWB+1] = {
72*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_2M5] = { 4, 4, 3, 1 },
73*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_5M ] = { 4, 4, 3, 1 },
74*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_7M5] = { 4, 4, 3, 2 },
75*49fe348cSAndroid Build Coastguard Worker         [LC3_DT_10M] = { 4, 4, 3, 1 },
76*49fe348cSAndroid Build Coastguard Worker     };
77*49fe348cSAndroid Build Coastguard Worker 
78*49fe348cSAndroid Build Coastguard Worker     /* --- Stage 1 ---
79*49fe348cSAndroid Build Coastguard Worker      * Determine bw0 candidate */
80*49fe348cSAndroid Build Coastguard Worker 
81*49fe348cSAndroid Build Coastguard Worker     enum lc3_bandwidth bw0 = LC3_BANDWIDTH_NB;
82*49fe348cSAndroid Build Coastguard Worker     enum lc3_bandwidth bwn = (enum lc3_bandwidth)sr;
83*49fe348cSAndroid Build Coastguard Worker 
84*49fe348cSAndroid Build Coastguard Worker     if (bwn <= bw0 || lc3_hr(sr))
85*49fe348cSAndroid Build Coastguard Worker         return bwn;
86*49fe348cSAndroid Build Coastguard Worker 
87*49fe348cSAndroid Build Coastguard Worker     const struct region *bwr = bws_table[dt][bwn-1];
88*49fe348cSAndroid Build Coastguard Worker 
89*49fe348cSAndroid Build Coastguard Worker     for (enum lc3_bandwidth bw = bw0; bw < bwn; bw++) {
90*49fe348cSAndroid Build Coastguard Worker         int i = bwr[bw].is, ie = bwr[bw].ie;
91*49fe348cSAndroid Build Coastguard Worker         int n = ie - i;
92*49fe348cSAndroid Build Coastguard Worker 
93*49fe348cSAndroid Build Coastguard Worker         float se = e[i];
94*49fe348cSAndroid Build Coastguard Worker         for (i++; i < ie; i++)
95*49fe348cSAndroid Build Coastguard Worker             se += e[i];
96*49fe348cSAndroid Build Coastguard Worker 
97*49fe348cSAndroid Build Coastguard Worker         if (se >= (10 << (bw == LC3_BANDWIDTH_NB)) * n)
98*49fe348cSAndroid Build Coastguard Worker             bw0 = bw + 1;
99*49fe348cSAndroid Build Coastguard Worker     }
100*49fe348cSAndroid Build Coastguard Worker 
101*49fe348cSAndroid Build Coastguard Worker     /* --- Stage 2 ---
102*49fe348cSAndroid Build Coastguard Worker      * Detect drop above cut-off frequency.
103*49fe348cSAndroid Build Coastguard Worker      * The Tc condition (13) is precalculated, as
104*49fe348cSAndroid Build Coastguard Worker      * Tc[] = 10 ^ (n / 10) , n = { 15, 23, 20, 20 } */
105*49fe348cSAndroid Build Coastguard Worker 
106*49fe348cSAndroid Build Coastguard Worker     int hold = bw0 >= bwn;
107*49fe348cSAndroid Build Coastguard Worker 
108*49fe348cSAndroid Build Coastguard Worker     if (!hold) {
109*49fe348cSAndroid Build Coastguard Worker         int i0 = bwr[bw0].is, l = l_table[dt][bw0];
110*49fe348cSAndroid Build Coastguard Worker         float tc = (const float []){
111*49fe348cSAndroid Build Coastguard Worker              31.62277660, 199.52623150, 100, 100 }[bw0];
112*49fe348cSAndroid Build Coastguard Worker 
113*49fe348cSAndroid Build Coastguard Worker         for (int i = i0 - l + 1; !hold && i <= i0 + 1; i++) {
114*49fe348cSAndroid Build Coastguard Worker             hold = e[i-l] > tc * e[i];
115*49fe348cSAndroid Build Coastguard Worker         }
116*49fe348cSAndroid Build Coastguard Worker 
117*49fe348cSAndroid Build Coastguard Worker     }
118*49fe348cSAndroid Build Coastguard Worker 
119*49fe348cSAndroid Build Coastguard Worker     return hold ? bw0 : bwn;
120*49fe348cSAndroid Build Coastguard Worker }
121*49fe348cSAndroid Build Coastguard Worker 
122*49fe348cSAndroid Build Coastguard Worker /**
123*49fe348cSAndroid Build Coastguard Worker  * Return number of bits coding the bandwidth value
124*49fe348cSAndroid Build Coastguard Worker  */
lc3_bwdet_get_nbits(enum lc3_srate sr)125*49fe348cSAndroid Build Coastguard Worker int lc3_bwdet_get_nbits(enum lc3_srate sr)
126*49fe348cSAndroid Build Coastguard Worker {
127*49fe348cSAndroid Build Coastguard Worker     return lc3_hr(sr) ? 0 : (sr > 0) + (sr > 1) + (sr > 3);
128*49fe348cSAndroid Build Coastguard Worker }
129*49fe348cSAndroid Build Coastguard Worker 
130*49fe348cSAndroid Build Coastguard Worker /**
131*49fe348cSAndroid Build Coastguard Worker  * Put bandwidth indication
132*49fe348cSAndroid Build Coastguard Worker  */
lc3_bwdet_put_bw(lc3_bits_t * bits,enum lc3_srate sr,enum lc3_bandwidth bw)133*49fe348cSAndroid Build Coastguard Worker void lc3_bwdet_put_bw(lc3_bits_t *bits,
134*49fe348cSAndroid Build Coastguard Worker     enum lc3_srate sr, enum lc3_bandwidth bw)
135*49fe348cSAndroid Build Coastguard Worker {
136*49fe348cSAndroid Build Coastguard Worker     int nbits_bw = lc3_bwdet_get_nbits(sr);
137*49fe348cSAndroid Build Coastguard Worker     if (nbits_bw > 0)
138*49fe348cSAndroid Build Coastguard Worker         lc3_put_bits(bits, bw, nbits_bw);
139*49fe348cSAndroid Build Coastguard Worker }
140*49fe348cSAndroid Build Coastguard Worker 
141*49fe348cSAndroid Build Coastguard Worker /**
142*49fe348cSAndroid Build Coastguard Worker  * Get bandwidth indication
143*49fe348cSAndroid Build Coastguard Worker  */
lc3_bwdet_get_bw(lc3_bits_t * bits,enum lc3_srate sr,enum lc3_bandwidth * bw)144*49fe348cSAndroid Build Coastguard Worker int lc3_bwdet_get_bw(lc3_bits_t *bits,
145*49fe348cSAndroid Build Coastguard Worker     enum lc3_srate sr, enum lc3_bandwidth *bw)
146*49fe348cSAndroid Build Coastguard Worker {
147*49fe348cSAndroid Build Coastguard Worker     enum lc3_bandwidth max_bw = (enum lc3_bandwidth)sr;
148*49fe348cSAndroid Build Coastguard Worker     int nbits_bw = lc3_bwdet_get_nbits(sr);
149*49fe348cSAndroid Build Coastguard Worker 
150*49fe348cSAndroid Build Coastguard Worker     *bw = nbits_bw <= 0 ? max_bw :
151*49fe348cSAndroid Build Coastguard Worker         (enum lc3_bandwidth)lc3_get_bits(bits, nbits_bw);
152*49fe348cSAndroid Build Coastguard Worker 
153*49fe348cSAndroid Build Coastguard Worker     return *bw > max_bw ? (*bw = max_bw), -1 : 0;
154*49fe348cSAndroid Build Coastguard Worker }
155