| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <assert.h> |
| #include <math.h> |
| #ifdef HAVE_CONFIG_H |
| #include <config.h> |
| #endif |
|
|
| #ifdef _MSC_VER |
| #pragma warning (disable: 4244) |
| #endif |
|
|
| #include <pocketsphinx.h> |
|
|
| #include "util/ckd_alloc.h" |
| #include "util/strfuncs.h" |
| #include "feat/cmn.h" |
| #include "feat/feat.h" |
|
|
| |
| const char *cmn_type_str[] = { |
| "none", |
| "batch", |
| "live" |
| }; |
| const char *cmn_alt_type_str[] = { |
| "none", |
| "current", |
| "prior" |
| }; |
| static const int n_cmn_type_str = sizeof(cmn_type_str)/sizeof(cmn_type_str[0]); |
|
|
| cmn_type_t |
| cmn_type_from_str(const char *str) |
| { |
| int i; |
|
|
| for (i = 0; i < n_cmn_type_str; ++i) { |
| if (0 == strcmp(str, cmn_type_str[i]) || 0 == strcmp(str, cmn_alt_type_str[i])) |
| return (cmn_type_t)i; |
| } |
| E_FATAL("Unknown CMN type '%s'\n", str); |
| return CMN_NONE; |
| } |
|
|
| const char * |
| cmn_update_repr(cmn_t *cmn) |
| { |
| char *ptr; |
| int i, len; |
| |
| len = 0; |
| for (i = 0; i < cmn->veclen; ++i) { |
| int nbytes = snprintf(NULL, 0, "%g,", MFCC2FLOAT(cmn->cmn_mean[i])); |
| if (nbytes <= 0) { |
| E_ERROR_SYSTEM("Failed to format %g for cmninit", MFCC2FLOAT(cmn->cmn_mean[i])); |
| return NULL; |
| } |
| len += nbytes; |
| } |
| len++; |
| if (cmn->repr) |
| ckd_free(cmn->repr); |
| ptr = cmn->repr = ckd_malloc(len); |
| if (ptr == NULL) { |
| E_ERROR_SYSTEM("Failed to allocate %d bytes for cmn string", len); |
| return NULL; |
| } |
| for (i = 0; i < cmn->veclen; ++i) |
| ptr += snprintf(ptr, cmn->repr + len - ptr, "%g,", |
| MFCC2FLOAT(cmn->cmn_mean[i])); |
| *--ptr = '\0'; |
|
|
| return cmn->repr; |
| } |
|
|
| int |
| cmn_set_repr(cmn_t *cmn, const char *repr) |
| { |
| char *c, *cc, *vallist; |
| int32 nvals; |
|
|
| E_INFO("Update from < %s >\n", cmn->repr); |
| memset(cmn->cmn_mean, 0, sizeof(cmn->cmn_mean[0]) * cmn->veclen); |
| memset(cmn->sum, 0, sizeof(cmn->sum[0]) * cmn->veclen); |
| vallist = ckd_salloc(repr); |
| c = vallist; |
| nvals = 0; |
| while (nvals < cmn->veclen |
| && (cc = strchr(c, ',')) != NULL) { |
| *cc = '\0'; |
| cmn->cmn_mean[nvals] = FLOAT2MFCC(atof_c(c)); |
| cmn->sum[nvals] = cmn->cmn_mean[nvals] * CMN_WIN; |
| c = cc + 1; |
| ++nvals; |
| } |
| if (nvals < cmn->veclen && *c != '\0') { |
| cmn->cmn_mean[nvals] = FLOAT2MFCC(atof_c(c)); |
| cmn->sum[nvals] = cmn->cmn_mean[nvals] * CMN_WIN; |
| } |
| ckd_free(vallist); |
| cmn->nframe = CMN_WIN; |
| E_INFO("Update to < %s >\n", cmn_update_repr(cmn)); |
| return 0; |
| } |
|
|
| cmn_t * |
| cmn_init(int32 veclen) |
| { |
| cmn_t *cmn; |
| cmn = (cmn_t *) ckd_calloc(1, sizeof(cmn_t)); |
| cmn->refcount = 1; |
| cmn->veclen = veclen; |
| cmn->cmn_mean = (mfcc_t *) ckd_calloc(veclen, sizeof(mfcc_t)); |
| cmn->cmn_var = (mfcc_t *) ckd_calloc(veclen, sizeof(mfcc_t)); |
| cmn->sum = (mfcc_t *) ckd_calloc(veclen, sizeof(mfcc_t)); |
| cmn->nframe = 0; |
| cmn_update_repr(cmn); |
|
|
| return cmn; |
| } |
|
|
|
|
| void |
| cmn(cmn_t *cmn, mfcc_t ** mfc, int32 varnorm, int32 n_frame) |
| { |
| mfcc_t *mfcp; |
| mfcc_t t; |
| int32 i, f; |
|
|
| assert(mfc != NULL); |
|
|
| if (n_frame <= 0) |
| return; |
|
|
| |
| memset(cmn->cmn_mean, 0, cmn->veclen * sizeof(mfcc_t)); |
| memset(cmn->sum, 0, cmn->veclen * sizeof(mfcc_t)); |
|
|
| |
| for (f = 0, cmn->nframe = 0; f < n_frame; f++) { |
| mfcp = mfc[f]; |
|
|
| |
| if (mfcp[0] < 0) |
| continue; |
|
|
| for (i = 0; i < cmn->veclen; i++) { |
| cmn->sum[i] += mfcp[i]; |
| } |
|
|
| cmn->nframe++; |
| } |
|
|
| for (i = 0; i < cmn->veclen; i++) { |
| cmn->cmn_mean[i] = cmn->sum[i] / cmn->nframe; |
| } |
|
|
| E_INFO("CMN: %s\n", cmn_update_repr(cmn)); |
| if (!varnorm) { |
| |
| for (f = 0; f < n_frame; f++) { |
| mfcp = mfc[f]; |
| for (i = 0; i < cmn->veclen; i++) |
| mfcp[i] -= cmn->cmn_mean[i]; |
| } |
| } |
| else { |
| |
| |
| memset(cmn->cmn_var, 0, cmn->veclen * sizeof(mfcc_t)); |
|
|
| for (f = 0; f < n_frame; f++) { |
| mfcp = mfc[f]; |
|
|
| for (i = 0; i < cmn->veclen; i++) { |
| t = mfcp[i] - cmn->cmn_mean[i]; |
| cmn->cmn_var[i] += MFCCMUL(t, t); |
| } |
| } |
| for (i = 0; i < cmn->veclen; i++) |
| |
| cmn->cmn_var[i] = FLOAT2MFCC(sqrt((float64)n_frame / MFCC2FLOAT(cmn->cmn_var[i]))); |
|
|
| for (f = 0; f < n_frame; f++) { |
| mfcp = mfc[f]; |
| for (i = 0; i < cmn->veclen; i++) |
| mfcp[i] = MFCCMUL((mfcp[i] - cmn->cmn_mean[i]), cmn->cmn_var[i]); |
| } |
| } |
| } |
|
|
| int |
| cmn_free(cmn_t * cmn) |
| { |
| if (cmn == NULL) |
| return 0; |
| if (--cmn->refcount > 0) |
| return cmn->refcount; |
| if (cmn->cmn_var) |
| ckd_free(cmn->cmn_var); |
| if (cmn->cmn_mean) |
| ckd_free(cmn->cmn_mean); |
| if (cmn->sum) |
| ckd_free(cmn->sum); |
| if (cmn->repr) |
| ckd_free(cmn->repr); |
| ckd_free(cmn); |
| return 0; |
| } |
|
|
| cmn_t * |
| cmn_retain(cmn_t *cmn) |
| { |
| ++cmn->refcount; |
| return cmn; |
| } |
|
|