| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | #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; |
| | } |
| |
|