/************************************************************************/ /* */ /* Centre for Speech Technology Research */ /* University of Edinburgh, UK */ /* Copyright (c) 1996,1997 */ /* All Rights Reserved. */ /* */ /* Permission is hereby granted, free of charge, to use and distribute */ /* this software and its documentation without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of this work, and to */ /* permit persons to whom this work is furnished to do so, subject to */ /* the following conditions: */ /* 1. The code must retain the above copyright notice, this list of */ /* conditions and the following disclaimer. */ /* 2. Any modifications must be clearly marked as such. */ /* 3. Original authors' names are not deleted. */ /* 4. The authors' names are not used to endorse or promote products */ /* derived from this software without specific prior written */ /* permission. */ /* */ /* THE UNIVERSITY OF EDINBURGH AND THE CONTRIBUTORS TO THIS WORK */ /* DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING */ /* ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT */ /* SHALL THE UNIVERSITY OF EDINBURGH NOR THE CONTRIBUTORS BE LIABLE */ /* FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES */ /* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN */ /* AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, */ /* ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF */ /* THIS SOFTWARE. */ /* */ /************************************************************************/ /* Author: Richard Caley (rjc@cstr.ed.ac.uk) */ /* Date: Fri Feb 28 1997 */ /************************************************************************/ /* */ /* A template class which allows names (const char *s) to be */ /* associated with enums, providing conversion. */ /* */ /************************************************************************/ #include #include #include #include "EST_walloc.h" #include "EST_TNamedEnum.h" // This only takes a void * because I can't manage to get the // parameter declaration in the definition past gcc with the actual type. template void EST_TValuedEnumI::initialise(const void *vdefs) { int n=0; typedef EST_TValuedEnumDefinition defn; const defn *defs = (const defn *)vdefs; for(n=1; defs[n].token != defs[0].token; n++) ; this->ndefinitions = n; this->definitions = new defn[n]; this->definitions[0] = defs[0]; for(n=1; defs[n].token != defs[0].token; n++) this->definitions[n] = defs[n]; this->p_unknown_enum = defs[n].token; this->p_unknown_value = defs[n].values[0]; } template void EST_TValuedEnumI::initialise(const void *vdefs, ENUM (*conv)(const char *)) { int n=0; // const struct EST_TValuedEnumDefinition *defs = (const struct EST_TValuedEnumDefinition *)vdefs; typedef EST_TValuedEnumDefinition _EST_TMPNAME; const _EST_TMPNAME *defs = (const _EST_TMPNAME *)vdefs; // fprintf(stderr, "start setup\n"); for(n=1; strcmp(defs[n].token, defs[0].token) != 0; n++) { //const char *a = defs[0].token; // const char *b = defs[n].token; // fprintf(stderr, ": %d '%s' '%s'\n", n, defs[n].token, defs[0].token); } this->ndefinitions = n; typedef EST_TValuedEnumDefinition defn; this->definitions = new defn[n]; this->definitions[0].token = conv(defs[0].token); for(int i=0; idefinitions[0].values[i] = defs[0].values[i]; this->definitions[0].info = defs[0].info; for(n=1; strcmp(defs[n].token, defs[0].token) != 0; n++) { this->definitions[n].token = conv(defs[n].token); for(int i2=0; i2definitions[n].values[i2] = defs[n].values[i2]; this->definitions[n].info = defs[n].info; } this->p_unknown_enum = conv(defs[n].token); this->p_unknown_value = defs[n].values[0]; } template EST_TValuedEnumI::~EST_TValuedEnumI(void) { if (this->definitions) delete[] this->definitions; } template int EST_TValuedEnumI::n(void) const { return this->ndefinitions; } template VAL EST_TValuedEnumI::value (ENUM token, int n) const { int i; for(i=0; indefinitions; i++) if (this->definitions[i].token == token) return this->definitions[i].values[n]; return this->p_unknown_value; } template INFO &EST_TValuedEnumI::info (ENUM token) const { int i; for(i=0; indefinitions; i++) if (this->definitions[i].token == token) return this->definitions[i].info; cerr << "Fetching info for invalid entry\n"; abort(); static INFO dummyI; return dummyI; } template ENUM EST_TValuedEnumI::nth_token (int n) const { if (n>=0 && n < this->ndefinitions) return this->definitions[n].token; return this->p_unknown_enum; } template ENUM EST_TValuedEnumI::token (VAL value) const { int i,j; for(i=0; indefinitions; i++) for(j=0; jdefinitions[i].values[j] ; j++) if (eq_vals(this->definitions[i].values[j], value)) return this->definitions[i].token; return this->p_unknown_enum; } template EST_read_status EST_TNamedEnum::priv_load(EST_String name, EST_TNamedEnum *definitive) { typedef EST_TValuedEnumDefinition Defn; #define LINE_LENGTH (1024) EST_String line(NULL, 'x', LINE_LENGTH); char *buffer = (char *)line; EST_String tokens[NAMED_ENUM_MAX_SYNONYMS+2]; FILE *file; char quote = '\0'; int have_unknown=0; int n=0; if ((file=fopen(name, "rb"))==NULL) return misc_read_error; if (this->definitions) delete[] this->definitions; this->ndefinitions= -1; this->definitions=NULL; buffer[LINE_LENGTH-1] = 'x'; while (fgets(buffer, LINE_LENGTH, file)) { if ( buffer[LINE_LENGTH-1] != 'x') { cerr << "line too long .. '" << buffer << "'\n"; return wrong_format; } if (this->ndefinitions>=0 && quote != '\0' && buffer[0] == '=') { // definition by number if ( n>= this->ndefinitions) { cerr << "too many definitions\n"; return wrong_format; } int ntokens = split(line, tokens, NAMED_ENUM_MAX_SYNONYMS+2, RXwhite, '"'); this->definitions[n].token = (ENUM)atoi(tokens[0].after(0,1)); for(int i=1; idefinitions[n].values[i-1] = wstrdup(tokens[i].unquote_if_needed(quote)); for(int j=ntokens-1 ; j< NAMED_ENUM_MAX_SYNONYMS; j++) this->definitions[n].values[j]=NULL; n++; } else if (have_unknown && this->ndefinitions>=0 && quote != '\0' && buffer[0] == quote) { // definition by standard name if (!definitive) { cerr << "can't use names in this definition\n"; return wrong_format; } if ( n>= this->ndefinitions) { cerr << "too many definitions\n"; return wrong_format; } int ntokens = split(line, tokens, NAMED_ENUM_MAX_SYNONYMS+2, RXwhite, quote); this->definitions[n].token = definitive->token(tokens[0].unquote(quote)); for(int i=1; idefinitions[n].values[i-1] = wstrdup(tokens[i].unquote_if_needed(quote)); for(int j=ntokens-1 ; j< NAMED_ENUM_MAX_SYNONYMS; j++) this->definitions[n].values[j]=NULL; n++; } else { // parameter int mlen; int eq = line.search("=", 1, mlen); if (eq <0) { cerr << "bad header line '" << line; return wrong_format; } EST_String key(line.before(eq)); if (key == "quote") { quote = line[eq+1]; // cout << "quote = '" << quote << "'\n"; } else if (key == "number") { this->ndefinitions=atoi(line.after(eq,1)); // cout << "n = '" << ndefinitions << "'\n"; this->definitions = new Defn[this->ndefinitions]; for(int i=0; indefinitions; i++) this->definitions[i].values[0] =NULL; n=0; } else if (key == "unknown") { this->p_unknown_enum=(ENUM)atoi(line.after(eq,1)); // cout << "unknown = '" << p_unknown_enum << "'\n"; have_unknown=1; } else { cerr << "bad header line '" << line; return wrong_format; } } } fclose(file); return format_ok; } template EST_write_status EST_TNamedEnum::priv_save(EST_String name, EST_TNamedEnum *definitive, char quote) const { FILE *file; if ((file=fopen(name, "wb"))==NULL) return write_fail; fprintf(file, "unknown=%d\n", this->p_unknown_enum); fprintf(file, "quote=%c\n", quote); fprintf(file, "number=%d\n", this->ndefinitions); for(int i=0; indefinitions; i++) if (this->definitions[i].values[0]) { if (definitive) fprintf(file, "%s ", (const char *)EST_String(definitive->name(this->definitions[i].token)).quote(quote)); else fprintf(file, "=%d ", (int)this->definitions[i].token); for(int j=0; jdefinitions[i].values[j] != NULL; j++) fprintf(file, "%s ", (const char *) EST_String(this->definitions[i].values[j]).quote_if_needed(quote)); fputc('\n', file); } fclose(file); return write_ok; }