/** * @file rdhmmdef_trans.c * * * @brief HTK %HMM 定義ファイルの読み込み:遷移行列 * * 遷移確率はファイル読み込み終了後に log10 に変換されます. * * * * @brief Read HTK %HMM definition file: transition matrix * * The transition probabilities will be converted to log10 scale * after finished reading the while %HMM definition file. * * * @author Akinobu LEE * @date Wed Feb 16 03:50:55 2005 * * $Revision: 1.5 $ * */ /* * 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_Trans * trans_new(HTK_HMM_INFO *hmm) { HTK_HMM_Trans *new; new = (HTK_HMM_Trans *)mybmalloc2(sizeof(HTK_HMM_Trans), &(hmm->mroot)); new->name = (char *)NULL; new->statenum = 0; new->a = (PROB **)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 trans_add(HTK_HMM_INFO *hmm, HTK_HMM_Trans *new) { HTK_HMM_Trans *match; /* link data structure */ new->next = hmm->trstart; hmm->trstart = new; if (new->name != NULL) { /* add index to search index tree */ if (hmm->tr_root == NULL) { hmm->tr_root = aptree_make_root_node(new, &(hmm->mroot)); } else { match = aptree_search_data(new->name, hmm->tr_root); if (match != NULL && strmatch(match->name,new->name)) { jlog("Error: rdhmmdef_trans: ~t \"%s\" is already defined\n", new->name); rderr(NULL); } else { aptree_add_entry(new->name, new, match->name, &(hmm->tr_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. */ static HTK_HMM_Trans * trans_lookup(HTK_HMM_INFO *hmm, char *keyname) { HTK_HMM_Trans *t; t = aptree_search_data(keyname, hmm->tr_root); if (t != NULL && strmatch(t->name, keyname)) { return t; } 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 [i/o] %HMM definition data * * @return pointer to the newly read data. */ static HTK_HMM_Trans * trans_read(FILE *fp, HTK_HMM_INFO *hmm) { HTK_HMM_Trans *new; int i,j; PROB prob; PROB *atmp; /* read tag */ if (!currentis("TRANSP")) rderr(" not found"); /* not match */ read_token(fp); /* read statenum */ new = trans_new(hmm); NoTokErr("missing TRANSP state num"); new->statenum = atoi(rdhmmdef_token); read_token(fp); /* allocate array */ new->a = (PROB **)mybmalloc2(sizeof(PROB *) * new->statenum, &(hmm->mroot)); atmp = (PROB *)mybmalloc2(sizeof(PROB) * new->statenum * new->statenum, &(hmm->mroot)); new->a[0] = &(atmp[0]); for (i=1;istatenum;i++) { new->a[i] = &(atmp[i*new->statenum]); } /* begin reading transition prob */ for (i=0;istatenum; i++) { for (j=0;jstatenum; j++) { NoTokErr("missing some TRANSP value"); prob = (PROB)atof(rdhmmdef_token); new->a[i][j] = prob; read_token(fp); } } 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 * * @return pointer to the data located at the current point. */ HTK_HMM_Trans * get_trans_data(FILE *fp, HTK_HMM_INFO *hmm) { HTK_HMM_Trans *tmp; if (currentis("TRANSP")) { /* definition: define transition data, and return the pointer */ tmp = trans_read(fp, hmm); tmp->name = NULL; /* no name */ trans_add(hmm, tmp); return(tmp); } else if (currentis("~t")) { /* macro reference: lookup and return the pointer */ read_token(fp); NoTokErr("missing TRANSP macro name"); tmp = trans_lookup(hmm, rdhmmdef_token); if (tmp == NULL) { jlog("Error: rdhmmdef_trans: ~t \"%s\" not defined\n", rdhmmdef_token); rderr(NULL); } read_token(fp); return(tmp); } else { rderr("no transition data"); return(NULL); } } /** * 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_trans_macro(char *name, FILE *fp, HTK_HMM_INFO *hmm) { HTK_HMM_Trans *new; /* read in data and return newly malloced data */ new = trans_read(fp, hmm); /* register it to the grobal HMM structure */ new->name = name; trans_add(hmm, new); }