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