/**
* @file rdhmmdef_mpdf.c
*
*
* @brief HTK %HMM 定義ファイルの読み込み:ガウス混合分布
*
*
*
* @brief Read HTK %HMM definition file: Gaussian mixture PDF
*
*
* @author Akinobu LEE
* @date Wed Feb 16 01:43:43 2005
*
* $Revision: 1.4 $
*
*/
/*
* Copyright (c) 1991-2012 Kawahara Lab., Kyoto University
* Copyright (c) 2000-2005 Shikano Lab., Nara Institute of Science and Technology
* Copyright (c) 2005-2012 Julius project team, Nagoya Institute of Technology
* All rights reserved
*/
#include
#include
extern char *rdhmmdef_token; ///< Current token
/**
* Allocate a new data area and return it.
*
* @return pointer to newly allocated data.
*/
static HTK_HMM_PDF *
mpdf_new(HTK_HMM_INFO *hmm)
{
HTK_HMM_PDF *new;
new = (HTK_HMM_PDF *)mybmalloc2(sizeof(HTK_HMM_PDF), &(hmm->mroot));
new->name = NULL;
new->tmix = FALSE;
new->stream_id = -1;
new->mix_num = 0;
new->b = NULL;
new->bweight = NULL;
new->next = NULL;
return(new);
}
/**
* Add a new data to the global structure.
*
* @param hmm [i/o] %HMM definition data to store it
* @param new [in] new data to be added
*/
void
mpdf_add(HTK_HMM_INFO *hmm, HTK_HMM_PDF *new)
{
HTK_HMM_PDF *match;
/* link data structure */
new->next = hmm->pdfstart;
hmm->pdfstart = new;
if (new->name != NULL) {
/* add index to search index tree */
if (hmm->pdf_root == NULL) {
hmm->pdf_root = aptree_make_root_node(new, &(hmm->mroot));
} else {
match = aptree_search_data(new->name, hmm->pdf_root);
if (match != NULL && strmatch(match->name, new->name)) {
jlog("Error: rdhmmdef_dens: ~m \"%s\" is already defined\n", new->name);
rderr(NULL);
} else {
aptree_add_entry(new->name, new, match->name, &(hmm->pdf_root), &(hmm->mroot));
}
}
}
}
/**
* Look up a data macro by the name.
*
* @param hmm [in] %HMM definition data
* @param keyname [in] macro name to find
*
* @return pointer to the found data, or NULL if not found.
*/
HTK_HMM_PDF *
mpdf_lookup(HTK_HMM_INFO *hmm, char *keyname)
{
HTK_HMM_PDF *d;
d = aptree_search_data(keyname, hmm->pdf_root);
if (d != NULL && strmatch(d->name, keyname)) {
return d;
} else {
return NULL;
}
}
/**
* @brief Read one new data and returns the pointer
*
* If a sub-component of this data is directly defined at here, they
* will be read from here and assigned to this data. If a sub-component
* is not defined here but a macro name referencing to the component previously
* defined in other place, the data will be searched by the macro name and
* the pointer to the found component will be assigned to this model.
*
* @param fp [in] file pointer
* @param hmm [in] %HMM definition data
* @param mix_num [in] num of Gaussians to be read, or -1 if not specified
*
* @return pointer to the newly read data.
*/
static HTK_HMM_PDF *
mpdf_read(FILE *fp, HTK_HMM_INFO *hmm, int mix_num)
{
HTK_HMM_PDF *new;
int i, mid;
boolean no_nummixes;
new = mpdf_new(hmm);
/* allow inside pdf */
if (currentis("STREAM")) {
read_token(fp);
NoTokErr("missing STREAM value");
new->stream_id = atoi(rdhmmdef_token) - 1;
read_token(fp);
}
/* allow in stream definition */
if (mix_num == -1) {
no_nummixes = TRUE;
} else {
no_nummixes = FALSE;
}
if (currentis("NUMMIXES")) {
read_token(fp);
new->mix_num = atoi(rdhmmdef_token);
if (mix_num != -1 && new->mix_num != mix_num) {
jlog("Error: rdhmmdef_mpdf: exists both in mpdf definition and its referer, and the values are different (%d != %d)\n", new->mix_num, mix_num);
rderr(NULL);
}
read_token(fp);
no_nummixes = FALSE;
} else {
if (mix_num != -1) {
new->mix_num = mix_num;
} else {
/* no NumMixes, assume single gaussian */
new->mix_num = 1;
}
}
if (currentis("TMIX")) {
read_token(fp);
/* read in TMIX */
tmix_read(fp, new, hmm);
/* mark this */
new->tmix = TRUE;
} else {
new->b = (HTK_HMM_Dens **) mybmalloc2(sizeof(HTK_HMM_Dens *) * new->mix_num, &(hmm->mroot));
new->bweight = (PROB *) mybmalloc2(sizeof(PROB) * new->mix_num, &(hmm->mroot));
for (i=0;imix_num;i++) {
new->b[i] = NULL;
new->bweight[i] = LOG_ZERO;
}
if (no_nummixes) { /* no NumMixes */
mid = 0;
new->bweight[mid] = 0.0;
new->b[mid] = get_dens_data(fp, hmm);
} else {
for (;;) {
if (!currentis("MIXTURE")) break;
read_token(fp);
NoTokErr("missing MIXTURE id");
mid = atoi(rdhmmdef_token) - 1;
read_token(fp);
NoTokErr("missing MIXTURE weight");
new->bweight[mid] = (PROB)log(atof(rdhmmdef_token));
read_token(fp);
new->b[mid] = get_dens_data(fp, hmm);
}
}
new->tmix = FALSE;
}
return (new);
}
/**
* @brief Return a pointer to the data located at the current point.
*
* If the current point is a macro reference, the pointer to the
* already defined data will be searched and returned.
* Otherwise, the definition of the data will be read from the current
* point and pointer to the newly allocated data will be returned.
*
* @param fp [in] file pointer
* @param hmm [i/o] %HMM definition data
* @param mix_num [in] num of Gaussians to be read, or -1 if not specified
* @param stream_id [in] stream ID, or -1 if not specified yet
*
* @return pointer to the data located at the current point.
*/
HTK_HMM_PDF *
get_mpdf_data(FILE *fp, HTK_HMM_INFO *hmm, int mix_num, short stream_id)
{
HTK_HMM_PDF *tmp = NULL;
if (currentis("~p")) {
/* macro reference: lookup and return the pointer */
read_token(fp);
NoTokErr("missing macro name");
tmp = mpdf_lookup(hmm, rdhmmdef_token);
if (tmp == NULL) {
jlog("Error: rdhmmdef_mpdf: ~p \"%s\" not defined\n", rdhmmdef_token);
rderr(NULL);
}
if (mix_num != -1 && tmp->mix_num != mix_num) {
jlog("Error: rdhmmdef_mpdf: mixture num in ~p \"%s\" definition and referer is different (%d != %d)\n", rdhmmdef_token, tmp->mix_num, mix_num);
rderr(NULL);
}
if (tmp->stream_id != stream_id) {
jlog("Error: rdhmmdef_mpdf: stream number in ~p \"%s\" definition and referer is different (%d != %d)\n", rdhmmdef_token, tmp->stream_id + 1, stream_id + 1);
rderr(NULL);
}
read_token(fp);
} else if (currentis("NUMMIXES")||currentis("MIXTURE")||currentis("TMIX")||currentis("MEAN")||currentis("~m")||currentis("RCLASS")) {
/* definition: define density data, and return the pointer */
tmp = mpdf_read(fp, hmm, mix_num);
if (tmp->stream_id == -1) {
tmp->stream_id = stream_id;
} else if (tmp->stream_id != stream_id) {
jlog("Error: rdhmmdef_mpdf: stream number exist in inline mpdf definition and referer is different (%d != %d)\n", rdhmmdef_token, tmp->stream_id + 1, stream_id + 1);
rderr(NULL);
}
tmp->name = NULL; /* no name */
mpdf_add(hmm, tmp);
} else {
rderr("syntax error: not mixture pdf data");
}
return tmp;
}
/**
* Read a new data and store it as a macro.
*
* @param name [in] macro name
* @param fp [in] file pointer
* @param hmm [i/o] %HMM definition data
*/
void
def_mpdf_macro(char *name, FILE *fp, HTK_HMM_INFO *hmm)
{
HTK_HMM_PDF *new;
/* read in data and return newly malloced data */
new = mpdf_read(fp, hmm, -1);
if (new->stream_id == -1) {
jlog("Error: rdhmmdef_pdf: definition of ~p \"%s\" has no \n", name);
rderr(NULL);
}
/* register it to the grobal HMM structure */
new->name = name;
mpdf_add(hmm, new);
}
/* end of file */