/** * @file rdhmmdef_data.c * * * @brief HTK %HMM 定義ファイルの読み込み:%HMM モデル * * * * @brief Read HTK %HMM definition file: %HMM model * * * @author Akinobu LEE * @date Wed Feb 16 01:12:19 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 and return it. * * @return pointer to newly allocated data. */ HTK_HMM_Data * htk_hmmdata_new(HTK_HMM_INFO *hmminfo) { HTK_HMM_Data *new; new = (HTK_HMM_Data *)mybmalloc2(sizeof(HTK_HMM_Data), &(hmminfo->mroot)); new->name = NULL; new->state_num = 0; new->s = NULL; new->tr = 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 htk_hmmdata_add(HTK_HMM_INFO *hmm, HTK_HMM_Data *new) { HTK_HMM_Data *match; /* link data structure */ new->next = hmm->start; hmm->start = new; if (new->name == NULL) { /* HMM must have a name */ rderr("HMM has no name"); } else { /* add index to search index tree */ if (hmm->physical_root == NULL) { hmm->physical_root = aptree_make_root_node(new, &(hmm->mroot)); } else { match = aptree_search_data(new->name, hmm->physical_root); if (match != NULL && strmatch(match->name, new->name)) { /* HMM of the same name should not be defined */ jlog("Error: rdhmmdef_data: HMM \"%s\" is defined more than twice\n", new->name); rderr(NULL); } else { aptree_add_entry(new->name, new, match->name, &(hmm->physical_root), &(hmm->mroot)); } } } } /** * @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 * * @return pointer to the newly read data. */ static HTK_HMM_Data * htk_hmmdata_read(FILE *fp, HTK_HMM_INFO *hmm) { HTK_HMM_Data *new; int i; short sid; new = htk_hmmdata_new(hmm); /* begin tag */ if (!currentis("BEGINHMM")) rderr(" not found"); read_token(fp); /* read global opt if any */ /* read_global_opt(fp, &(new->opt)); */ /* num of state */ if (!currentis("NUMSTATES")) rderr(" not found"); read_token(fp); NoTokErr("state num not found\n"); new->state_num = atoi(rdhmmdef_token); read_token(fp); /* malloc state */ new->s = (HTK_HMM_State **)mybmalloc2(sizeof(HTK_HMM_State *) * new->state_num, &(hmm->mroot)); for(i=0;istate_num;i++) { new->s[i] = NULL; } /* read/set each state info */ for (;;) { if (!currentis("STATE")) break; read_token(fp); NoTokErr("STATE id not found"); sid = atoi(rdhmmdef_token) - 1; read_token(fp); new->s[sid] = get_state_data(fp, hmm); } /* read/set transition info */ new->tr = get_trans_data(fp, hmm); if ((new->tr)->statenum != new->state_num) { rderr("# of transition != # of state"); } /* read/set duration */ /* end tag */ if (!currentis("ENDHMM")) rderr(" not found"); read_token(fp); return(new); } /** * 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_HMM(char *name, FILE *fp, HTK_HMM_INFO *hmm) { HTK_HMM_Data *new; /* read in HMM model data from fp, and return newly malloced HTK_HMM_Data */ new = htk_hmmdata_read(fp, hmm); /* set name and add the new data to the main structure */ new->name = name; htk_hmmdata_add(hmm, new); }