xref: /aosp_15_r20/external/webrtc/modules/audio_processing/vad/gmm.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #include "modules/audio_processing/vad/gmm.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include <math.h>
14*d9f75844SAndroid Build Coastguard Worker 
15*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
16*d9f75844SAndroid Build Coastguard Worker 
17*d9f75844SAndroid Build Coastguard Worker static const int kMaxDimension = 10;
18*d9f75844SAndroid Build Coastguard Worker 
RemoveMean(const double * in,const double * mean_vec,int dimension,double * out)19*d9f75844SAndroid Build Coastguard Worker static void RemoveMean(const double* in,
20*d9f75844SAndroid Build Coastguard Worker                        const double* mean_vec,
21*d9f75844SAndroid Build Coastguard Worker                        int dimension,
22*d9f75844SAndroid Build Coastguard Worker                        double* out) {
23*d9f75844SAndroid Build Coastguard Worker   for (int n = 0; n < dimension; ++n)
24*d9f75844SAndroid Build Coastguard Worker     out[n] = in[n] - mean_vec[n];
25*d9f75844SAndroid Build Coastguard Worker }
26*d9f75844SAndroid Build Coastguard Worker 
ComputeExponent(const double * in,const double * covar_inv,int dimension)27*d9f75844SAndroid Build Coastguard Worker static double ComputeExponent(const double* in,
28*d9f75844SAndroid Build Coastguard Worker                               const double* covar_inv,
29*d9f75844SAndroid Build Coastguard Worker                               int dimension) {
30*d9f75844SAndroid Build Coastguard Worker   double q = 0;
31*d9f75844SAndroid Build Coastguard Worker   for (int i = 0; i < dimension; ++i) {
32*d9f75844SAndroid Build Coastguard Worker     double v = 0;
33*d9f75844SAndroid Build Coastguard Worker     for (int j = 0; j < dimension; j++)
34*d9f75844SAndroid Build Coastguard Worker       v += (*covar_inv++) * in[j];
35*d9f75844SAndroid Build Coastguard Worker     q += v * in[i];
36*d9f75844SAndroid Build Coastguard Worker   }
37*d9f75844SAndroid Build Coastguard Worker   q *= -0.5;
38*d9f75844SAndroid Build Coastguard Worker   return q;
39*d9f75844SAndroid Build Coastguard Worker }
40*d9f75844SAndroid Build Coastguard Worker 
EvaluateGmm(const double * x,const GmmParameters & gmm_parameters)41*d9f75844SAndroid Build Coastguard Worker double EvaluateGmm(const double* x, const GmmParameters& gmm_parameters) {
42*d9f75844SAndroid Build Coastguard Worker   if (gmm_parameters.dimension > kMaxDimension) {
43*d9f75844SAndroid Build Coastguard Worker     return -1;  // This is invalid pdf so the caller can check this.
44*d9f75844SAndroid Build Coastguard Worker   }
45*d9f75844SAndroid Build Coastguard Worker   double f = 0;
46*d9f75844SAndroid Build Coastguard Worker   double v[kMaxDimension];
47*d9f75844SAndroid Build Coastguard Worker   const double* mean_vec = gmm_parameters.mean;
48*d9f75844SAndroid Build Coastguard Worker   const double* covar_inv = gmm_parameters.covar_inverse;
49*d9f75844SAndroid Build Coastguard Worker 
50*d9f75844SAndroid Build Coastguard Worker   for (int n = 0; n < gmm_parameters.num_mixtures; n++) {
51*d9f75844SAndroid Build Coastguard Worker     RemoveMean(x, mean_vec, gmm_parameters.dimension, v);
52*d9f75844SAndroid Build Coastguard Worker     double q = ComputeExponent(v, covar_inv, gmm_parameters.dimension) +
53*d9f75844SAndroid Build Coastguard Worker                gmm_parameters.weight[n];
54*d9f75844SAndroid Build Coastguard Worker     f += exp(q);
55*d9f75844SAndroid Build Coastguard Worker     mean_vec += gmm_parameters.dimension;
56*d9f75844SAndroid Build Coastguard Worker     covar_inv += gmm_parameters.dimension * gmm_parameters.dimension;
57*d9f75844SAndroid Build Coastguard Worker   }
58*d9f75844SAndroid Build Coastguard Worker   return f;
59*d9f75844SAndroid Build Coastguard Worker }
60*d9f75844SAndroid Build Coastguard Worker 
61*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
62