Change Translator from Class to Struct, so avoiding use of Classes in eSpeak. Make more functions and variables "extern". git-svn-id: https://espeak.svn.sourceforge.net/svnroot/espeak/trunk@209 d46cf337-b52f-0410-862d-fd96e6ae7743master
@@ -40,7 +40,7 @@ int HashDictionary(const char *string); | |||
static FILE *f_log = NULL; | |||
extern char *dir_dictionary; | |||
int linenum; | |||
static int linenum; | |||
static int error_count; | |||
static int transpose_offset; // transpose character range for LookupDictList() | |||
static int transpose_min; | |||
@@ -48,9 +48,9 @@ static int transpose_max; | |||
static int text_mode = 0; | |||
static int debug_flag = 0; | |||
int hash_counts[N_HASH_DICT]; | |||
char *hash_chains[N_HASH_DICT]; | |||
char letterGroupsDefined[N_LETTER_GROUPS]; | |||
static int hash_counts[N_HASH_DICT]; | |||
static char *hash_chains[N_HASH_DICT]; | |||
static char letterGroupsDefined[N_LETTER_GROUPS]; | |||
MNEM_TAB mnem_flags[] = { | |||
// these in the first group put a value in bits0-3 of dictionary_flags | |||
@@ -136,6 +136,28 @@ int isspace2(unsigned int c) | |||
} | |||
static const char *LookupMnem2(MNEM_TAB *table, int value) | |||
{//======================================================= | |||
while(table->mnem != NULL) | |||
{ | |||
if(table->value == value) | |||
return(table->mnem); | |||
table++; | |||
} | |||
return(""); | |||
} | |||
char *print_dictionary_flags(unsigned int *flags) | |||
{//============================================== | |||
static char buf[20]; | |||
sprintf(buf,"%s 0x%x/%x",LookupMnem2(mnem_flags,(flags[0] & 0xf)+0x40), flags[0], flags[1]); | |||
return(buf); | |||
} | |||
static FILE *fopen_log(const char *fname,const char *access) | |||
{//================================================== | |||
// performs fopen, but produces error message to f_log if it fails | |||
@@ -168,8 +190,8 @@ static const char *lookup_mnem(MNEM_TAB *table, int value) | |||
int compile_line(char *linebuf, char *dict_line, int *hash) | |||
{//======================================================== | |||
static int compile_line(char *linebuf, char *dict_line, int *hash) | |||
{//=============================================================== | |||
// Compile a line in the language_list file | |||
unsigned char c; | |||
char *p; | |||
@@ -448,7 +470,7 @@ step=1; // TEST | |||
{ | |||
// this assumes that the lower case char is the same length as the upper case char | |||
// OK, except for Turkish "I", but use towlower() rather than towlower2() | |||
ix = utf8_in(&c2,p,0); | |||
ix = utf8_in(&c2,p); | |||
if(c2 == 0) | |||
break; | |||
if(iswupper(c2)) | |||
@@ -551,8 +573,8 @@ step=1; // TEST | |||
void compile_dictlist_start(void) | |||
{//============================== | |||
static void compile_dictlist_start(void) | |||
{//===================================== | |||
// initialise dictionary list | |||
int ix; | |||
char *p; | |||
@@ -573,8 +595,8 @@ void compile_dictlist_start(void) | |||
} | |||
void compile_dictlist_end(FILE *f_out) | |||
{//=================================== | |||
static void compile_dictlist_end(FILE *f_out) | |||
{//========================================== | |||
// Write out the compiled dictionary list | |||
int hash; | |||
int length; | |||
@@ -610,8 +632,8 @@ void compile_dictlist_end(FILE *f_out) | |||
int compile_dictlist_file(const char *path, const char* filename) | |||
{//============================================================== | |||
static int compile_dictlist_file(const char *path, const char* filename) | |||
{//===================================================================== | |||
int length; | |||
int hash; | |||
char *p; | |||
@@ -664,26 +686,19 @@ int compile_dictlist_file(const char *path, const char* filename) | |||
char rule_cond[80]; | |||
char rule_pre[80]; | |||
char rule_post[80]; | |||
char rule_match[80]; | |||
char rule_phonemes[80]; | |||
char group_name[LEN_GROUP_NAME+1]; | |||
static char rule_cond[80]; | |||
static char rule_pre[80]; | |||
static char rule_post[80]; | |||
static char rule_match[80]; | |||
static char rule_phonemes[80]; | |||
static char group_name[LEN_GROUP_NAME+1]; | |||
#define N_RULES 2000 // max rules for each group | |||
int hexdigit(char c) | |||
{//================= | |||
if(isdigit(c)) | |||
return(c - '0'); | |||
return(tolower(c) - 'a' + 10); | |||
} | |||
void copy_rule_string(char *string, int &state) | |||
{//============================================ | |||
static void copy_rule_string(char *string, int &state) | |||
{//=================================================== | |||
// state 0: conditional, 1=pre, 2=match, 3=post, 4=phonemes | |||
static char *outbuf[5] = {rule_cond, rule_pre, rule_match, rule_post, rule_phonemes}; | |||
static int next_state[5] = {2,2,4,4,4}; | |||
@@ -892,8 +907,8 @@ void copy_rule_string(char *string, int &state) | |||
char *compile_rule(char *input) | |||
{//============================ | |||
static char *compile_rule(char *input) | |||
{//=================================== | |||
int ix; | |||
unsigned char c; | |||
int wc; | |||
@@ -990,7 +1005,7 @@ char *compile_rule(char *input) | |||
len_name = strlen(group_name); | |||
if((len_name > 0) && (memcmp(rule_match,group_name,len_name) != 0)) | |||
{ | |||
utf8_in(&wc,rule_match,0); | |||
utf8_in(&wc,rule_match); | |||
if((group_name[0] == '9') && IsDigit(wc)) | |||
{ | |||
// numeric group, rule_match starts with a digit, so OK | |||
@@ -1080,8 +1095,8 @@ static int __cdecl rgroup_sorter(RGROUP *a, RGROUP *b) | |||
#ifdef OUTPUT_FORMAT | |||
void print_rule_group(FILE *f_out, int n_rules, char **rules, char *name) | |||
{//====================================================================== | |||
static void print_rule_group(FILE *f_out, int n_rules, char **rules, char *name) | |||
{//============================================================================= | |||
int rule; | |||
int ix; | |||
unsigned char c; | |||
@@ -1187,8 +1202,8 @@ void print_rule_group(FILE *f_out, int n_rules, char **rules, char *name) | |||
//#define LIST_GROUP_INFO | |||
void output_rule_group(FILE *f_out, int n_rules, char **rules, char *name) | |||
{//======================================================================= | |||
static void output_rule_group(FILE *f_out, int n_rules, char **rules, char *name) | |||
{//============================================================================== | |||
int ix; | |||
int len1; | |||
int len2; | |||
@@ -1406,7 +1421,7 @@ static int compile_dictrules(FILE *f_in, FILE *f_out, char *fname_temp) | |||
if(strlen(group_name) > 2) | |||
{ | |||
if(utf8_in(&c,group_name,0) < 2) | |||
if(utf8_in(&c,group_name) < 2) | |||
{ | |||
fprintf(f_log,"%5d: Group name longer than 2 bytes (UTF8)",linenum); | |||
error_count++; | |||
@@ -1442,7 +1457,7 @@ static int compile_dictrules(FILE *f_in, FILE *f_out, char *fname_temp) | |||
ix = 0; | |||
while((unsigned char)(*p) > 0x20) // not space or zero-byte | |||
{ | |||
p += utf8_in(&c,p,0); | |||
p += utf8_in(&c,p); | |||
replace1 += (c << ix); | |||
ix += 16; | |||
} | |||
@@ -1450,7 +1465,7 @@ static int compile_dictrules(FILE *f_in, FILE *f_out, char *fname_temp) | |||
ix = 0; | |||
while((unsigned char)(*p) > 0x20) | |||
{ | |||
p += utf8_in(&c,p,0); | |||
p += utf8_in(&c,p); | |||
replace2 += (c << ix); | |||
ix += 16; | |||
} | |||
@@ -1508,7 +1523,6 @@ static int compile_dictrules(FILE *f_in, FILE *f_out, char *fname_temp) | |||
int CompileDictionary(const char *dsource, const char *dict_name, FILE *log, char *fname_err, int flags) | |||
{//===================================================================================================== | |||
// fname: space to write the filename in case of error | |||
@@ -1572,7 +1586,7 @@ int CompileDictionary(const char *dsource, const char *dict_name, FILE *log, cha | |||
compile_dictlist_start(); | |||
fprintf(f_log,"Using phonemetable: '%s'\n",PhonemeTabName()); | |||
fprintf(f_log,"Using phonemetable: '%s'\n",phoneme_tab_list[phoneme_tab_number].name); | |||
compile_dictlist_file(path,"roots"); | |||
if(translator->langopts.listx) | |||
{ | |||
@@ -1598,7 +1612,7 @@ int CompileDictionary(const char *dsource, const char *dict_name, FILE *log, cha | |||
Write4Bytes(f_out,offset_rules); | |||
fclose(f_out); | |||
translator->LoadDictionary(dict_name,0); | |||
LoadDictionary(translator, dict_name, 0); | |||
return(error_count); | |||
} // end of compile_dictionary |
@@ -109,7 +109,7 @@ void event_init(void) | |||
} | |||
//> | |||
//<event_display | |||
void event_display(espeak_EVENT* event) | |||
static void event_display(espeak_EVENT* event) | |||
{ | |||
ENTER("event_display"); | |||
@@ -35,6 +35,7 @@ | |||
#include "translate.h" | |||
#include "options.h" | |||
extern char word_phonemes[N_WORD_PHONEMES]; // a word translated into phoneme codes | |||
//****************************************************************************************************** | |||
@@ -343,7 +344,7 @@ void Lexicon_De() | |||
// convert word to lower-case | |||
for(ix=0, p=&word2[1];;) | |||
{ | |||
ix += utf8_in(&c,&word[ix],0); | |||
ix += utf8_in(&c,&word[ix]); | |||
c = towlower(c); | |||
p += utf8_out(c,p); | |||
if(c == 0) | |||
@@ -419,9 +420,9 @@ void Lexicon_De() | |||
// translate | |||
memset(&winfo,0,sizeof(winfo)); | |||
translator->TranslateWord(&word2[1],0,&winfo); | |||
TranslateWord(translator,&word2[1],0,&winfo); | |||
DecodePhonemes2(translator->word_phonemes,phonemes); // also need to change some phoneme names | |||
DecodePhonemes2(word_phonemes,phonemes); // also need to change some phoneme names | |||
if(strcmp(phonemes,pronounce2) == 0) | |||
{ | |||
@@ -624,8 +625,8 @@ p_unicode = unicode; | |||
// translate | |||
memset(&winfo,0,sizeof(winfo)); | |||
translator->TranslateWord(&word2[1],0,&winfo); | |||
DecodePhonemes(translator->word_phonemes,phonemes); | |||
TranslateWord(translator, &word2[1],0,&winfo); | |||
DecodePhonemes(word_phonemes,phonemes); | |||
// find the stress position in the translation | |||
max_stress = 0; | |||
@@ -634,7 +635,7 @@ p_unicode = unicode; | |||
check_root = 0; | |||
ph = phoneme_tab[phonPAUSE]; | |||
for(p=translator->word_phonemes; *p != 0; p++) | |||
for(p=word_phonemes; *p != 0; p++) | |||
{ | |||
ph = phoneme_tab[(unsigned int)*p]; | |||
if(ph == NULL) | |||
@@ -866,7 +867,7 @@ void CountWordFreq(wxString path, wcount **hashtab) | |||
n_chars = 0; | |||
for(k=0; k<ix; ) | |||
{ | |||
k += utf8_in(&wc,&buf[k],0); | |||
k += utf8_in(&wc,&buf[k]); | |||
wc = towlower(wc); // convert to lower case | |||
if(iswalpha(wc)) | |||
{ |
@@ -269,7 +269,7 @@ int fifo_is_busy () | |||
//<sleep_until_start_request_or_inactivity | |||
int sleep_until_start_request_or_inactivity() | |||
static int sleep_until_start_request_or_inactivity() | |||
{ | |||
SHOW_TIME("fifo > sleep_until_start_request_or_inactivity > ENTER"); | |||
int a_start_is_required=0; |
@@ -49,7 +49,7 @@ typedef struct { | |||
short pitch2; | |||
} SYLLABLE; | |||
SYLLABLE *syllable_tab; | |||
static SYLLABLE *syllable_tab; | |||
static int tone_pitch_env; /* used to return pitch envelope */ | |||
@@ -706,8 +706,8 @@ static int calc_pitches(int start, int end, int head_tone, int nucleus_tone) | |||
void Translator::CalcPitches_Tone(int clause_tone) | |||
{//=============================================== | |||
static void CalcPitches_Tone(Translator *tr, int clause_tone) | |||
{//========================================================== | |||
// clause_tone: 0=. 1=, 2=?, 3=! 4=none | |||
PHONEME_LIST *p; | |||
int ix; | |||
@@ -750,12 +750,12 @@ void Translator::CalcPitches_Tone(int clause_tone) | |||
phoneme_list[final_stressed].tone = 7; | |||
// language specific, changes to tones | |||
if(translator_name == L('v','i')) | |||
if(tr->translator_name == L('v','i')) | |||
{ | |||
// LANG=vi | |||
p = &phoneme_list[final_stressed]; | |||
if(p->tone_ph == 0) | |||
p->tone_ph = LookupPh("7"); // change default tone (tone 1) to falling tone at end of clause | |||
p->tone_ph = PhonemeCode('7'); // change default tone (tone 1) to falling tone at end of clause | |||
} | |||
@@ -785,18 +785,18 @@ void Translator::CalcPitches_Tone(int clause_tone) | |||
tph = phoneme_tab[tone_ph]; | |||
// Mandarin | |||
if(translator_name == L('z','h')) | |||
if(tr->translator_name == L('z','h')) | |||
{ | |||
if(tone_ph == 0) | |||
{ | |||
if(pause || tone_promoted) | |||
{ | |||
tone_ph = LookupPh("55"); // no previous vowel, use tone 1 | |||
tone_ph = PhonemeCode2('5','5'); // no previous vowel, use tone 1 | |||
tone_promoted = 1; | |||
} | |||
else | |||
{ | |||
tone_ph = LookupPh("11"); // default tone 5 | |||
tone_ph = PhonemeCode2('1','1'); // default tone 5 | |||
} | |||
p->tone_ph = tone_ph; | |||
@@ -820,24 +820,24 @@ void Translator::CalcPitches_Tone(int clause_tone) | |||
if(prevw_tph->mnemonic == 0x343132) // [214] | |||
{ | |||
if(tph->mnemonic == 0x343132) // [214] | |||
prev_p->tone_ph = LookupPh("35"); | |||
prev_p->tone_ph = PhonemeCode2('3','5'); | |||
else | |||
prev_p->tone_ph = LookupPh("21"); | |||
prev_p->tone_ph = PhonemeCode2('2','1'); | |||
} | |||
if((prev_tph->mnemonic == 0x3135) && (tph->mnemonic == 0x3135)) // [51] + [51] | |||
{ | |||
prev_p->tone_ph = LookupPh("53"); | |||
prev_p->tone_ph = PhonemeCode2('5','3'); | |||
} | |||
if(tph->mnemonic == 0x3131) // [11] Tone 5 | |||
{ | |||
// tone 5, change its level depending on the previous tone (across word boundaries) | |||
if(prevw_tph->mnemonic == 0x3535) | |||
p->tone_ph = LookupPh("22"); | |||
p->tone_ph = PhonemeCode2('2','2'); | |||
if(prevw_tph->mnemonic == 0x3533) | |||
p->tone_ph = LookupPh("33"); | |||
p->tone_ph = PhonemeCode2('3','3'); | |||
if(prevw_tph->mnemonic == 0x343132) | |||
p->tone_ph = LookupPh("44"); | |||
p->tone_ph = PhonemeCode2('4','4'); | |||
// tone 5 is unstressed (shorter) | |||
p->tone = 1; // diminished stress | |||
@@ -889,8 +889,8 @@ void Translator::CalcPitches_Tone(int clause_tone) | |||
void Translator::CalcPitches(int clause_type) | |||
{//========================================== | |||
void CalcPitches(Translator *tr, int clause_type) | |||
{//============================================== | |||
// clause_type: 0=. 1=, 2=?, 3=! 4=none | |||
PHONEME_LIST *p; | |||
SYLLABLE *syl; | |||
@@ -942,20 +942,20 @@ void Translator::CalcPitches(int clause_type) | |||
if(langopts.tone_language == 1) | |||
if(tr->langopts.tone_language == 1) | |||
{ | |||
CalcPitches_Tone(clause_type); | |||
CalcPitches_Tone(tr,clause_type); | |||
return; | |||
} | |||
option = langopts.intonation_group; | |||
option = tr->langopts.intonation_group; | |||
if(option >= INTONATION_TYPES) | |||
option = 0; | |||
group_tone = punct_to_tone[option][clause_type]; | |||
group_tone_emph = punct_to_tone[option][5]; // emphatic form of statement | |||
group_tone_comma = punct_to_tone[option][1]; // emphatic form of statement | |||
group_tone = tr->punct_to_tone[option][clause_type]; | |||
group_tone_emph = tr->punct_to_tone[option][5]; // emphatic form of statement | |||
group_tone_comma = tr->punct_to_tone[option][1]; // emphatic form of statement | |||
if(clause_type == 4) | |||
no_tonic = 1; /* incomplete clause, used for abbreviations such as Mr. Dr. Mrs. */ |
@@ -223,7 +223,7 @@ Converts synthesis parameters to a waveform. | |||
*/ | |||
int parwave(klatt_frame_ptr frame) | |||
static int parwave(klatt_frame_ptr frame) | |||
{ | |||
double temp; | |||
double outbypas; | |||
@@ -489,7 +489,7 @@ Initialises all parameters used in parwave, sets resonator internal memory | |||
to zero. | |||
*/ | |||
void reset_resonators() | |||
static void reset_resonators() | |||
{ | |||
int r_ix; | |||
@@ -500,7 +500,7 @@ void reset_resonators() | |||
} | |||
} | |||
void parwave_init() | |||
static void parwave_init() | |||
{ | |||
kt_globals.FLPhz = (950 * kt_globals.samrate) / 10000; | |||
kt_globals.BLPhz = (630 * kt_globals.samrate) / 10000; |
@@ -71,7 +71,7 @@ int flags; | |||
} ACCENTS; | |||
// these are tokens to look up in the *_list file. | |||
ACCENTS accents_tab[] = { | |||
static ACCENTS accents_tab[] = { | |||
{"_lig", 1}, | |||
{"_smc", 1}, // smallcap | |||
{"_tur", 1}, // turned | |||
@@ -126,7 +126,7 @@ static const short non_ascii_tab[] = { | |||
// characters U+00e0 to U+017f | |||
const unsigned short letter_accents_0e0[] = { | |||
static const unsigned short letter_accents_0e0[] = { | |||
LETTER('a',M_GRAVE,0), // U+00e0 | |||
LETTER('a',M_ACUTE,0), | |||
LETTER('a',M_CIRCUMFLEX,0), | |||
@@ -291,7 +291,7 @@ LETTER('s',M_NAME,0), // long-s // U+17f | |||
// characters U+0250 to U+029F | |||
const unsigned short letter_accents_250[] = { | |||
static const unsigned short letter_accents_250[] = { | |||
LETTER('a',M_TURNED,0), // U+250 | |||
LETTER(L_ALPHA,0,0), | |||
LETTER(L_ALPHA,M_TURNED,0), | |||
@@ -383,8 +383,8 @@ LIGATURE('t','s',0), | |||
LIGATURE('t','s',M_CURL), | |||
}; | |||
int Translator::LookupLetter2(unsigned int letter, char *ph_buf) | |||
{//============================================================= | |||
static int LookupLetter2(Translator *tr, unsigned int letter, char *ph_buf) | |||
{//======================================================================== | |||
int len; | |||
char single_letter[10]; | |||
@@ -394,20 +394,20 @@ int Translator::LookupLetter2(unsigned int letter, char *ph_buf) | |||
single_letter[len+2] = ' '; | |||
single_letter[len+3] = 0; | |||
if(Lookup(&single_letter[1],ph_buf) == 0) | |||
if(Lookup(tr, &single_letter[1], ph_buf) == 0) | |||
{ | |||
single_letter[1] = ' '; | |||
if(Lookup(&single_letter[2],ph_buf) == 0) | |||
if(Lookup(tr, &single_letter[2], ph_buf) == 0) | |||
{ | |||
TranslateRules(&single_letter[2], ph_buf, 20, NULL,0,NULL); | |||
TranslateRules(tr, &single_letter[2], ph_buf, 20, NULL,0,NULL); | |||
} | |||
} | |||
return(ph_buf[0]); | |||
} | |||
void Translator::LookupAccentedLetter(unsigned int letter, char *ph_buf) | |||
{//===================================================================== | |||
void LookupAccentedLetter(Translator *tr, unsigned int letter, char *ph_buf) | |||
{//========================================================================= | |||
// lookup the character in the accents table | |||
int accent_data = 0; | |||
int accent1 = 0; | |||
@@ -450,14 +450,14 @@ void Translator::LookupAccentedLetter(unsigned int letter, char *ph_buf) | |||
} | |||
if(Lookup(accents_tab[accent1].name, ph_accent1) != 0) | |||
if(Lookup(tr, accents_tab[accent1].name, ph_accent1) != 0) | |||
{ | |||
if(LookupLetter2(basic_letter, ph_letter1) != 0) | |||
if(LookupLetter2(tr, basic_letter, ph_letter1) != 0) | |||
{ | |||
if(accent2 != 0) | |||
{ | |||
if(Lookup(accents_tab[accent2].name, ph_accent2) == 0) | |||
if(Lookup(tr, accents_tab[accent2].name, ph_accent2) == 0) | |||
{ | |||
// break; | |||
} | |||
@@ -472,7 +472,7 @@ void Translator::LookupAccentedLetter(unsigned int letter, char *ph_buf) | |||
if(letter2 != 0) | |||
{ | |||
//ligature | |||
LookupLetter2(letter2, ph_letter2); | |||
LookupLetter2(tr, letter2, ph_letter2); | |||
sprintf(ph_buf,"%s%c%s%c%s%s",ph_accent1, phonPAUSE_VSHORT, ph_letter1, phonSTRESS_P, ph_letter2, ph_accent2); | |||
} | |||
else | |||
@@ -480,7 +480,7 @@ void Translator::LookupAccentedLetter(unsigned int letter, char *ph_buf) | |||
if(accent1 == 0) | |||
strcpy(ph_buf, ph_letter1); | |||
else | |||
if((langopts.accents & 1) || (accents_tab[accent1].flags & 1)) | |||
if((tr->langopts.accents & 1) || (accents_tab[accent1].flags & 1)) | |||
sprintf(ph_buf,"%s%c%c%s", ph_accent1, phonPAUSE_VSHORT, phonSTRESS_P, ph_letter1); | |||
else | |||
sprintf(ph_buf,"%c%s%c%s%c", phonSTRESS_2, ph_letter1, phonPAUSE_VSHORT, ph_accent1, phonPAUSE_VSHORT); | |||
@@ -492,8 +492,8 @@ void Translator::LookupAccentedLetter(unsigned int letter, char *ph_buf) | |||
void Translator::LookupLetter(unsigned int letter, int next_byte, char *ph_buf1) | |||
{//============================================================================= | |||
void LookupLetter(Translator *tr, unsigned int letter, int next_byte, char *ph_buf1) | |||
{//================================================================================= | |||
int len; | |||
unsigned char *p; | |||
static char single_letter[10] = {0,0}; | |||
@@ -509,16 +509,16 @@ void Translator::LookupLetter(unsigned int letter, int next_byte, char *ph_buf1) | |||
if(next_byte == -1) | |||
{ | |||
// speaking normal text, not individual characters | |||
if(Lookup(&single_letter[2],ph_buf1) != 0) | |||
if(Lookup(tr, &single_letter[2], ph_buf1) != 0) | |||
return; | |||
single_letter[1] = '_'; | |||
if(Lookup(&single_letter[1],ph_buf3) != 0) | |||
if(Lookup(tr, &single_letter[1], ph_buf3) != 0) | |||
return; // the character is specified as _* so ignore it when speaking normal text | |||
// check whether this character is specified for English | |||
SetTranslator2("en"); | |||
if(translator2->Lookup(&single_letter[2], ph_buf3) != 0) | |||
if(Lookup(translator2, &single_letter[2], ph_buf3) != 0) | |||
{ | |||
// yes, switch to English and re-translate the word | |||
sprintf(ph_buf1,"%c",phonSWITCH); | |||
@@ -531,7 +531,7 @@ void Translator::LookupLetter(unsigned int letter, int next_byte, char *ph_buf1) | |||
{ | |||
// lookup space as _&32 etc. | |||
sprintf(&single_letter[1],"_#%d ",letter); | |||
Lookup(&single_letter[1],ph_buf1); | |||
Lookup(tr, &single_letter[1], ph_buf1); | |||
return; | |||
} | |||
@@ -545,18 +545,18 @@ void Translator::LookupLetter(unsigned int letter, int next_byte, char *ph_buf1) | |||
dict_flags[1] = 0; | |||
ptr = &single_letter[1]; | |||
if(Lookup(&single_letter[1],ph_buf3) == 0) | |||
if(Lookup(tr, &single_letter[1], ph_buf3) == 0) | |||
{ | |||
single_letter[1] = ' '; | |||
if(Lookup(&single_letter[2],ph_buf3) == 0) | |||
if(Lookup(tr, &single_letter[2], ph_buf3) == 0) | |||
{ | |||
TranslateRules(&single_letter[2], ph_buf3, sizeof(ph_buf3), NULL,FLAG_NO_TRACE,NULL); | |||
TranslateRules(tr, &single_letter[2], ph_buf3, sizeof(ph_buf3), NULL,FLAG_NO_TRACE,NULL); | |||
} | |||
} | |||
if(ph_buf3[0] == 0) | |||
{ | |||
LookupAccentedLetter(letter, ph_buf3); | |||
LookupAccentedLetter(tr, letter, ph_buf3); | |||
} | |||
if(ph_buf3[0] == 0) | |||
@@ -583,7 +583,7 @@ void Translator::LookupLetter(unsigned int letter, int next_byte, char *ph_buf1) | |||
int Translator::TranslateLetter(char *word, char *phonemes, int control, int word_length) | |||
int TranslateLetter(Translator *tr, char *word, char *phonemes, int control, int word_length) | |||
{//====================================================================================== | |||
// get pronunciation for an isolated letter | |||
// return number of bytes used by the letter | |||
@@ -602,7 +602,7 @@ int Translator::TranslateLetter(char *word, char *phonemes, int control, int wor | |||
ph_buf[0] = 0; | |||
capital[0] = 0; | |||
n_bytes = utf8_in(&letter,word,0); | |||
n_bytes = utf8_in(&letter,word); | |||
if((letter & 0xfff00) == 0x0e000) | |||
{ | |||
@@ -614,12 +614,12 @@ int Translator::TranslateLetter(char *word, char *phonemes, int control, int wor | |||
// include CAPITAL information | |||
if(iswupper(letter)) | |||
{ | |||
Lookup("_cap",capital); | |||
Lookup(tr, "_cap", capital); | |||
} | |||
} | |||
letter = towlower2(letter); | |||
LookupLetter(letter, word[n_bytes], ph_buf); | |||
LookupLetter(tr, letter, word[n_bytes], ph_buf); | |||
if(ph_buf[0] == phonSWITCH) | |||
{ | |||
@@ -627,13 +627,13 @@ int Translator::TranslateLetter(char *word, char *phonemes, int control, int wor | |||
return(0); | |||
} | |||
if((ph_buf[0] == 0) && (translator_name != L('e','n'))) | |||
if((ph_buf[0] == 0) && (tr->translator_name != L('e','n'))) | |||
{ | |||
// speak as English, check whether there is a translation for this character | |||
SetTranslator2("en"); | |||
save_option_phonemes = option_phonemes; | |||
option_phonemes = 0; | |||
translator2->LookupLetter(letter, word[n_bytes], ph_buf); | |||
LookupLetter(translator2, letter, word[n_bytes], ph_buf); | |||
SelectPhonemeTable(voice->phoneme_tab_ix); // revert to original phoneme table | |||
option_phonemes = save_option_phonemes; | |||
@@ -648,10 +648,10 @@ int Translator::TranslateLetter(char *word, char *phonemes, int control, int wor | |||
{ | |||
// character name not found | |||
if(iswalpha(letter)) | |||
Lookup("_?A",ph_buf); | |||
Lookup(tr, "_?A", ph_buf); | |||
if((ph_buf[0]==0) && !iswspace(letter)) | |||
Lookup("_??",ph_buf); | |||
Lookup(tr, "_??", ph_buf); | |||
if(ph_buf[0] != 0) | |||
{ | |||
@@ -662,13 +662,13 @@ int Translator::TranslateLetter(char *word, char *phonemes, int control, int wor | |||
{ | |||
pbuf += strlen(pbuf); | |||
*pbuf++ = phonPAUSE_VSHORT; | |||
LookupLetter(*p2, 0, pbuf); | |||
LookupLetter(tr, *p2, 0, pbuf); | |||
} | |||
} | |||
} | |||
len = strlen(phonemes); | |||
if(langopts.accents & 2) | |||
if(tr->langopts.accents & 2) | |||
sprintf(ph_buf2,"%c%s%s",0xff,ph_buf,capital); | |||
else | |||
sprintf(ph_buf2,"%c%s%s",0xff,capital,ph_buf); // the 0xff marker will be removed or replaced in SetSpellingStress() | |||
@@ -681,8 +681,8 @@ int Translator::TranslateLetter(char *word, char *phonemes, int control, int wor | |||
void Translator::SetSpellingStress(char *phonemes, int control, int n_chars) | |||
{//========================================================================= | |||
void SetSpellingStress(Translator *tr, char *phonemes, int control, int n_chars) | |||
{//============================================================================= | |||
// Individual letter names, reduce the stress of some. | |||
int ix; | |||
unsigned int c; | |||
@@ -707,7 +707,7 @@ void Translator::SetSpellingStress(char *phonemes, int control, int n_chars) | |||
{ | |||
count++; | |||
if(langopts.spelling_stress == 1) | |||
if(tr->langopts.spelling_stress == 1) | |||
{ | |||
// stress on initial letter when spelling | |||
if(count > 1) | |||
@@ -745,7 +745,7 @@ void Translator::SetSpellingStress(char *phonemes, int control, int n_chars) | |||
int Translator::TranslateRoman(char *word, char *ph_out) | |||
int TranslateRoman(Translator *tr, char *word, char *ph_out) | |||
{//===================================================== | |||
int c; | |||
char *p; | |||
@@ -804,29 +804,88 @@ int Translator::TranslateRoman(char *word, char *ph_out) | |||
if(acc < 2) | |||
return(0); | |||
if(acc > langopts.max_roman) | |||
if(acc > tr->langopts.max_roman) | |||
return(0); | |||
Lookup("_roman",ph_roman); // precede by "roman" if _rom is defined in *_list | |||
Lookup(tr, "_roman",ph_roman); // precede by "roman" if _rom is defined in *_list | |||
p = &ph_out[0]; | |||
if((langopts.numbers & NUM_ROMAN_AFTER) == 0) | |||
if((tr->langopts.numbers & NUM_ROMAN_AFTER) == 0) | |||
{ | |||
strcpy(ph_out,ph_roman); | |||
p = &ph_out[strlen(ph_out)]; | |||
} | |||
sprintf(number_chars," %d ",acc); | |||
TranslateNumber(&number_chars[1],p,&flags,0); | |||
TranslateNumber(tr, &number_chars[1], p, &flags, 0); | |||
if(langopts.numbers & NUM_ROMAN_AFTER) | |||
if(tr->langopts.numbers & NUM_ROMAN_AFTER) | |||
strcat(ph_out,ph_roman); | |||
return(1); | |||
} // end of TranslateRoman | |||
int Translator::LookupNum2(int value, int control, char *ph_out) | |||
{//============================================================= | |||
static const char *M_Variant(int value) | |||
{//==================================== | |||
// returns M, or perhaps MA for some cases | |||
if(((value % 100)>20) || ((value % 100)<10)) // but not teens, 10 to 19 | |||
{ | |||
if ((translator->langopts.numbers2 & 0x40) && | |||
((value % 10)>=2) && | |||
((value % 10)<=4)) | |||
{ | |||
// for Polish language - two forms of plural! | |||
return("0MA"); | |||
} | |||
if((translator->langopts.numbers2 & 0x80) && | |||
((value % 10)==1)) | |||
{ | |||
return("1MA"); | |||
} | |||
} | |||
return("0M"); | |||
} | |||
static int LookupThousands(Translator *tr, int value, int thousandplex, char *ph_out) | |||
{//================================================================================== | |||
int found; | |||
char string[12]; | |||
char ph_of[12]; | |||
char ph_thousands[40]; | |||
ph_of[0] = 0; | |||
// first look fora match with the exact value of thousands | |||
sprintf(string,"_%dM%d",value,thousandplex); | |||
if((found = Lookup(tr, string, ph_thousands)) == 0) | |||
{ | |||
if((value % 100) >= 20) | |||
{ | |||
Lookup(tr, "_0of", ph_of); | |||
} | |||
sprintf(string,"_%s%d",M_Variant(value),thousandplex); | |||
if(Lookup(tr, string, ph_thousands) == 0) | |||
{ | |||
// repeat "thousand" if higher order names are not available | |||
sprintf(string,"_%dM1",value); | |||
if((found = Lookup(tr, string, ph_thousands)) == 0) | |||
Lookup(tr, "_0M1", ph_thousands); | |||
} | |||
} | |||
sprintf(ph_out,"%s%s",ph_of,ph_thousands); | |||
return(found); | |||
} | |||
static int LookupNum2(Translator *tr, int value, int control, char *ph_out) | |||
{//======================================================================== | |||
// Lookup a 2 digit number | |||
// control bit 0: use special form of '1' | |||
// control bit 2: use feminine form of '2' | |||
@@ -843,7 +902,7 @@ int Translator::LookupNum2(int value, int control, char *ph_out) | |||
if((value == 1) && (control & 1)) | |||
{ | |||
if(Lookup("_1a",ph_out) != 0) | |||
if(Lookup(tr, "_1a", ph_out) != 0) | |||
return(0); | |||
} | |||
// is there a special pronunciation for this 2-digit number | |||
@@ -851,19 +910,19 @@ int Translator::LookupNum2(int value, int control, char *ph_out) | |||
if(control & 4) | |||
{ | |||
sprintf(string,"_%df",value); | |||
found = Lookup(string,ph_digits); | |||
found = Lookup(tr, string, ph_digits); | |||
} | |||
if(found == 0) | |||
{ | |||
sprintf(string,"_%d",value); | |||
found = Lookup(string,ph_digits); | |||
found = Lookup(tr, string, ph_digits); | |||
} | |||
// no, speak as tens+units | |||
if((control & 2) && (value < 10)) | |||
{ | |||
// speak leading zero | |||
Lookup("_0",ph_tens); | |||
Lookup(tr, "_0", ph_tens); | |||
} | |||
else | |||
{ | |||
@@ -876,12 +935,12 @@ int Translator::LookupNum2(int value, int control, char *ph_out) | |||
if((value % 10) == 0) | |||
{ | |||
sprintf(string,"_%d0",value / 10); | |||
found = Lookup(string,ph_tens); | |||
found = Lookup(tr, string, ph_tens); | |||
} | |||
if(!found) | |||
{ | |||
sprintf(string,"_%dX",value / 10); | |||
Lookup(string,ph_tens); | |||
Lookup(tr, string, ph_tens); | |||
} | |||
if((value % 10) == 0) | |||
@@ -896,19 +955,19 @@ int Translator::LookupNum2(int value, int control, char *ph_out) | |||
{ | |||
// is there a variant form of this number? | |||
sprintf(string,"_%df",units); | |||
found = Lookup(string,ph_digits); | |||
found = Lookup(tr, string, ph_digits); | |||
} | |||
if(found == 0) | |||
{ | |||
sprintf(string,"_%d",units); | |||
Lookup(string,ph_digits); | |||
Lookup(tr, string, ph_digits); | |||
} | |||
} | |||
if(langopts.numbers & 0x30) | |||
if(tr->langopts.numbers & 0x30) | |||
{ | |||
Lookup("_0and",ph_and); | |||
if(langopts.numbers & 0x10) | |||
Lookup(tr, "_0and", ph_and); | |||
if(tr->langopts.numbers & 0x10) | |||
sprintf(ph_out,"%s%s%s",ph_digits,ph_and,ph_tens); | |||
else | |||
sprintf(ph_out,"%s%s%s",ph_tens,ph_and,ph_digits); | |||
@@ -916,7 +975,7 @@ int Translator::LookupNum2(int value, int control, char *ph_out) | |||
} | |||
else | |||
{ | |||
if(langopts.numbers & 0x200) | |||
if(tr->langopts.numbers & 0x200) | |||
{ | |||
// remove vowel from the end of tens if units starts with a vowel (LANG=Italian) | |||
if((ix = strlen(ph_tens)-1) >= 0) | |||
@@ -931,7 +990,7 @@ int Translator::LookupNum2(int value, int control, char *ph_out) | |||
sprintf(ph_out,"%s%s",ph_tens,ph_digits); | |||
} | |||
if(langopts.numbers & 0x100) | |||
if(tr->langopts.numbers & 0x100) | |||
{ | |||
// only one primary stress | |||
found = 0; | |||
@@ -950,8 +1009,8 @@ int Translator::LookupNum2(int value, int control, char *ph_out) | |||
} // end of LookupNum2 | |||
int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thousandplex, int prev_thousands) | |||
{//========================================================================================================= | |||
static int LookupNum3(Translator *tr, int value, char *ph_out, int suppress_null, int thousandplex, int prev_thousands) | |||
{//==================================================================================================================== | |||
// Translate a 3 digit number | |||
int found; | |||
int hundreds; | |||
@@ -974,9 +1033,9 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
ph_thousands[0] = 0; | |||
ph_thousand_and[0] = 0; | |||
Lookup("_0C",ph_100); | |||
Lookup(tr, "_0C", ph_100); | |||
if(((langopts.numbers & 0x0800) != 0) && (hundreds == 19)) | |||
if(((tr->langopts.numbers & 0x0800) != 0) && (hundreds == 19)) | |||
{ | |||
// speak numbers such as 1984 as years: nineteen-eighty-four | |||
// ph_100[0] = 0; // don't say "hundred", we also need to surpess "and" | |||
@@ -986,15 +1045,15 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
{ | |||
ph_digits[0] = 0; | |||
if(LookupThousands(hundreds / 10, thousandplex+1, ph_10T) == 0) | |||
if(LookupThousands(tr, hundreds / 10, thousandplex+1, ph_10T) == 0) | |||
{ | |||
x = 0; | |||
if(langopts.numbers2 & (1 << (thousandplex+1))) | |||
if(tr->langopts.numbers2 & (1 << (thousandplex+1))) | |||
x = 4; | |||
LookupNum2(hundreds/10, x, ph_digits); | |||
LookupNum2(tr, hundreds/10, x, ph_digits); | |||
} | |||
if(langopts.numbers2 & 0x200) | |||
if(tr->langopts.numbers2 & 0x200) | |||
sprintf(ph_thousands,"%s%s%c",ph_10T,ph_digits,phonPAUSE_NOLINK); // say "thousands" before its number, not after | |||
else | |||
sprintf(ph_thousands,"%s%s%c",ph_digits,ph_10T,phonPAUSE_NOLINK); | |||
@@ -1008,9 +1067,9 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
ph_digits[0] = 0; | |||
if(hundreds > 0) | |||
{ | |||
if((langopts.numbers & 0x100000) && (prev_thousands || (ph_thousands[0] != 0))) | |||
if((tr->langopts.numbers & 0x100000) && (prev_thousands || (ph_thousands[0] != 0))) | |||
{ | |||
Lookup("_0and",ph_thousand_and); | |||
Lookup(tr, "_0and", ph_thousand_and); | |||
} | |||
suppress_null = 1; | |||
@@ -1019,12 +1078,12 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
if((value % 1000) == 100) | |||
{ | |||
// is there a special pronunciation for exactly 100 ? | |||
found = Lookup("_1C0",ph_digits); | |||
found = Lookup(tr, "_1C0", ph_digits); | |||
} | |||
if(!found) | |||
{ | |||
sprintf(string,"_%dC",hundreds); | |||
found = Lookup(string,ph_digits); // is there a specific pronunciation for n-hundred ? | |||
found = Lookup(tr, string, ph_digits); // is there a specific pronunciation for n-hundred ? | |||
} | |||
if(found) | |||
@@ -1033,9 +1092,9 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
} | |||
else | |||
{ | |||
if((hundreds > 1) || ((langopts.numbers & 0x400) == 0)) | |||
if((hundreds > 1) || ((tr->langopts.numbers & 0x400) == 0)) | |||
{ | |||
LookupNum2(hundreds,0,ph_digits); | |||
LookupNum2(tr, hundreds, 0, ph_digits); | |||
} | |||
} | |||
} | |||
@@ -1044,11 +1103,11 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
} | |||
ph_hundred_and[0] = 0; | |||
if((langopts.numbers & 0x40) && ((value % 100) != 0)) | |||
if((tr->langopts.numbers & 0x40) && ((value % 100) != 0)) | |||
{ | |||
if((value > 100) || (prev_thousands && (thousandplex==0))) | |||
{ | |||
Lookup("_0and",ph_hundred_and); | |||
Lookup(tr, "_0and", ph_hundred_and); | |||
} | |||
} | |||
@@ -1059,7 +1118,7 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
if(value == 0) | |||
{ | |||
if(suppress_null == 0) | |||
Lookup("_0",buf2); | |||
Lookup(tr, "_0", buf2); | |||
} | |||
else | |||
{ | |||
@@ -1068,13 +1127,13 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
x = 1; // allow "eins" for 1 rather than "ein" | |||
else | |||
{ | |||
if(langopts.numbers2 & (1 << thousandplex)) | |||
if(tr->langopts.numbers2 & (1 << thousandplex)) | |||
x = 4; // use variant (feminine) for before thousands and millions | |||
} | |||
if(LookupNum2(value,x,buf2) != 0) | |||
if(LookupNum2(tr, value, x, buf2) != 0) | |||
{ | |||
if(langopts.numbers & 0x80) | |||
if(tr->langopts.numbers & 0x80) | |||
ph_hundred_and[0] = 0; // don't put 'and' after 'hundred' if there's 'and' between tens and units | |||
} | |||
} | |||
@@ -1086,67 +1145,8 @@ int Translator::LookupNum3(int value, char *ph_out, int suppress_null, int thous | |||
static const char *M_Variant(int value) | |||
{//==================================== | |||
// returns M, or perhaps MA for some cases | |||
if(((value % 100)>20) || ((value % 100)<10)) // but not teens, 10 to 19 | |||
{ | |||
if ((translator->langopts.numbers2 & 0x40) && | |||
((value % 10)>=2) && | |||
((value % 10)<=4)) | |||
{ | |||
// for Polish language - two forms of plural! | |||
return("0MA"); | |||
} | |||
if((translator->langopts.numbers2 & 0x80) && | |||
((value % 10)==1)) | |||
{ | |||
return("1MA"); | |||
} | |||
} | |||
return("0M"); | |||
} | |||
int Translator::LookupThousands(int value, int thousandplex, char *ph_out) | |||
{//======================================================================= | |||
int found; | |||
char string[12]; | |||
char ph_of[12]; | |||
char ph_thousands[40]; | |||
ph_of[0] = 0; | |||
// first look fora match with the exact value of thousands | |||
sprintf(string,"_%dM%d",value,thousandplex); | |||
if((found = Lookup(string,ph_thousands)) == 0) | |||
{ | |||
if((value % 100) >= 20) | |||
{ | |||
Lookup("_0of",ph_of); | |||
} | |||
sprintf(string,"_%s%d",M_Variant(value),thousandplex); | |||
if(Lookup(string,ph_thousands) == 0) | |||
{ | |||
// repeat "thousand" if higher order names are not available | |||
sprintf(string,"_%dM1",value); | |||
if((found = Lookup(string,ph_thousands)) == 0) | |||
Lookup("_0M1",ph_thousands); | |||
} | |||
} | |||
sprintf(ph_out,"%s%s",ph_of,ph_thousands); | |||
return(found); | |||
} | |||
int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, int wflags) | |||
{//========================================================================================= | |||
static int TranslateNumber_1(Translator *tr, char *word, char *ph_out, unsigned int *flags, int wflags) | |||
{//==================================================================================================== | |||
// Number translation with various options | |||
// the "word" may be up to 4 digits | |||
// "words" of 3 digits may be preceded by another number "word" for thousands or millions | |||
@@ -1180,12 +1180,12 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
ph_buf2[0] = 0; | |||
// is there a previous thousands part (as a previous "word") ? | |||
if((n_digits == 3) && (word[-2] == langopts.thousands_sep) && isdigit(word[-3])) | |||
if((n_digits == 3) && (word[-2] == tr->langopts.thousands_sep) && isdigit(word[-3])) | |||
{ | |||
prev_thousands = 1; | |||
} | |||
else | |||
if((langopts.thousands_sep == ' ') || (langopts.numbers & 0x1000)) | |||
if((tr->langopts.thousands_sep == ' ') || (tr->langopts.numbers & 0x1000)) | |||
{ | |||
// thousands groups can be separated by spaces | |||
if((n_digits == 3) && isdigit(word[-2])) | |||
@@ -1194,7 +1194,7 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
} | |||
} | |||
if((word[0] == '0') && (prev_thousands == 0) && (word[1] != langopts.decimal_sep)) | |||
if((word[0] == '0') && (prev_thousands == 0) && (word[1] != tr->langopts.decimal_sep)) | |||
{ | |||
if((n_digits == 2) && (word[3] == ':') && isdigit(word[5]) && isspace(word[7])) | |||
{ | |||
@@ -1206,10 +1206,10 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
} | |||
} | |||
if((langopts.numbers & 0x1000) && (word[n_digits] == ' ')) | |||
if((tr->langopts.numbers & 0x1000) && (word[n_digits] == ' ')) | |||
thousands_inc = 1; | |||
else | |||
if(word[n_digits] == langopts.thousands_sep) | |||
if(word[n_digits] == tr->langopts.thousands_sep) | |||
thousands_inc = 2; | |||
if(thousands_inc > 0) | |||
@@ -1221,7 +1221,7 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
while(isdigit(word[ix]) && isdigit(word[ix+1]) && isdigit(word[ix+2])) | |||
{ | |||
thousandplex++; | |||
if(word[ix+3] == langopts.thousands_sep) | |||
if(word[ix+3] == tr->langopts.thousands_sep) | |||
ix += (3 + thousands_inc); | |||
else | |||
break; | |||
@@ -1233,10 +1233,10 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
suppress_null = 1; | |||
} | |||
if((word[n_digits] == langopts.decimal_sep) && isdigit(word[n_digits+1])) | |||
if((word[n_digits] == tr->langopts.decimal_sep) && isdigit(word[n_digits+1])) | |||
{ | |||
// this "word" ends with a decimal point | |||
Lookup("_dpt",ph_append); | |||
Lookup(tr, "_dpt", ph_append); | |||
decimal_point = 1; | |||
} | |||
else | |||
@@ -1246,7 +1246,7 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
{ | |||
if((thousandplex > 0) && (value < 1000)) | |||
{ | |||
if((suppress_null == 0) && (LookupThousands(value,thousandplex,ph_append))) | |||
if((suppress_null == 0) && (LookupThousands(tr,value,thousandplex,ph_append))) | |||
{ | |||
// found an exact match for N thousand | |||
value = 0; | |||
@@ -1259,21 +1259,21 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
if((thousandplex > 1) && prev_thousands && (prev_value > 0)) | |||
{ | |||
sprintf(string,"_%s%d",M_Variant(value),thousandplex+1); | |||
if(Lookup(string,buf1)==0) | |||
if(Lookup(tr, string, buf1)==0) | |||
{ | |||
// speak this thousandplex if there was no word for the previous thousandplex | |||
sprintf(string,"_0M%d",thousandplex); | |||
Lookup(string,ph_append); | |||
Lookup(tr, string, ph_append); | |||
} | |||
} | |||
if((ph_append[0] == 0) && (word[n_digits] == '.') && (thousandplex == 0)) | |||
{ | |||
Lookup("_.",ph_append); | |||
Lookup(tr, "_.", ph_append); | |||
} | |||
LookupNum3(value, ph_buf, suppress_null, thousandplex, prev_thousands); | |||
if((thousandplex > 0) && (langopts.numbers2 & 0x200)) | |||
LookupNum3(tr, value, ph_buf, suppress_null, thousandplex, prev_thousands); | |||
if((thousandplex > 0) && (tr->langopts.numbers2 & 0x200)) | |||
sprintf(ph_out,"%s%s%s",ph_append,ph_buf2,ph_buf); // say "thousands" before its number | |||
else | |||
sprintf(ph_out,"%s%s%s",ph_buf2,ph_buf,ph_append); | |||
@@ -1290,7 +1290,7 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
if(decimal_count > 1) | |||
{ | |||
max_decimal_count = 2; | |||
switch(langopts.numbers & 0xe000) | |||
switch(tr->langopts.numbers & 0xe000) | |||
{ | |||
case 0x8000: | |||
max_decimal_count = 5; | |||
@@ -1298,14 +1298,14 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
// French/Polish decimal fraction | |||
while(word[n_digits] == '0') | |||
{ | |||
Lookup("_0",buf1); | |||
Lookup(tr, "_0", buf1); | |||
strcat(ph_out,buf1); | |||
decimal_count--; | |||
n_digits++; | |||
} | |||
if((decimal_count <= max_decimal_count) && isdigit(word[n_digits])) | |||
{ | |||
LookupNum3(atoi(&word[n_digits]),buf1,0,0,0); | |||
LookupNum3(tr, atoi(&word[n_digits]), buf1, 0,0,0); | |||
strcat(ph_out,buf1); | |||
n_digits += decimal_count; | |||
} | |||
@@ -1315,13 +1315,13 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
// Italian decimal fractions | |||
if((decimal_count < 4) || ((decimal_count==4) && (word[n_digits] != '0'))) | |||
{ | |||
LookupNum3(atoi(&word[n_digits]),buf1,0,0,0); | |||
LookupNum3(tr, atoi(&word[n_digits]), buf1, 0,0,0); | |||
strcat(ph_out,buf1); | |||
if(word[n_digits]=='0') | |||
{ | |||
// decimal part has leading zeros, so add a "hundredths" or "thousandths" suffix | |||
sprintf(string,"_0Z%d",decimal_count); | |||
Lookup(string,buf1); | |||
Lookup(tr, string, buf1); | |||
strcat(ph_out,buf1); | |||
} | |||
n_digits += decimal_count; | |||
@@ -1332,7 +1332,7 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
// Romanian decimal fractions | |||
if((decimal_count <= 4) && (word[n_digits] != '0')) | |||
{ | |||
LookupNum3(atoi(&word[n_digits]),buf1,0,0,0); | |||
LookupNum3(tr, atoi(&word[n_digits]), buf1, 0,0,0); | |||
strcat(ph_out,buf1); | |||
n_digits += decimal_count; | |||
} | |||
@@ -1343,17 +1343,17 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
while(isdigit(c = word[n_digits]) && (strlen(ph_out) < (N_WORD_PHONEMES - 10))) | |||
{ | |||
value = word[n_digits++] - '0'; | |||
LookupNum2(value, 1, buf1); | |||
LookupNum2(tr, value, 1, buf1); | |||
strcat(ph_out,buf1); | |||
} | |||
// something after the decimal part ? | |||
if(Lookup("_dpt2",buf1)) | |||
if(Lookup(tr, "_dpt2", buf1)) | |||
strcat(ph_out,buf1); | |||
if((c == langopts.decimal_sep) && isdigit(word[n_digits+1])) | |||
if((c == tr->langopts.decimal_sep) && isdigit(word[n_digits+1])) | |||
{ | |||
Lookup("_dpt",buf1); | |||
Lookup(tr, "_dpt", buf1); | |||
strcat(ph_out,buf1); | |||
} | |||
else | |||
@@ -1367,9 +1367,9 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
char *p; | |||
p = &word[n_digits+1]; | |||
p += utf8_in(&next_char,p,0); | |||
if((langopts.numbers & NUM_NOPAUSE) && (next_char == ' ')) | |||
utf8_in(&next_char,p,0); | |||
p += utf8_in(&next_char,p); | |||
if((tr->langopts.numbers & NUM_NOPAUSE) && (next_char == ' ')) | |||
utf8_in(&next_char,p); | |||
if(!iswalpha(next_char)) | |||
strcat(ph_out,str_pause); // don't add pause for 100s, 6th, etc. | |||
@@ -1382,13 +1382,13 @@ int Translator::TranslateNumber_1(char *word, char *ph_out, unsigned int *flags, | |||
int Translator::TranslateNumber(char *word1, char *ph_out, unsigned int *flags, int wflags) | |||
{//======================================================================================= | |||
int TranslateNumber(Translator *tr, char *word1, char *ph_out, unsigned int *flags, int wflags) | |||
{//============================================================================================ | |||
if(option_sayas == SAYAS_DIGITS1) | |||
return(0); // speak digits individually | |||
if((langopts.numbers & 0x3) == 1) | |||
return(TranslateNumber_1(word1,ph_out,flags,wflags)); | |||
if((tr->langopts.numbers & 0x3) == 1) | |||
return(TranslateNumber_1(tr, word1, ph_out, flags, wflags)); | |||
return(0); | |||
} // end of TranslateNumber |
@@ -151,10 +151,16 @@ extern int n_replace_phonemes; | |||
extern REPLACE_PHONEMES replace_phonemes[N_REPLACE_PHONEMES]; | |||
#define PH(c1,c2) (c2<<8)+c1 // combine two characters into an integer for phoneme name | |||
#define PH3(c1,c2,c3) (c3<<16)+(c2<<8)+c1 | |||
#define PhonemeCode2(c1,c2) PhonemeCode((c2<<8)+c1) | |||
int LookupPhonemeString(const char *string); | |||
int PhonemeCode(unsigned int mnem); | |||
char *EncodePhonemes(char *p, char *outptr, unsigned char *bad_phoneme); | |||
void DecodePhonemes(const char *inptr, char *outptr); | |||
const char *PhonemeTabName(void); | |||
int LookupPh(const char *string); | |||
extern const char *WordToString(unsigned int word); | |||
extern PHONEME_TAB_LIST phoneme_tab_list[N_PHONEME_TABS]; | |||
extern int phoneme_tab_number; |
@@ -33,16 +33,25 @@ | |||
const unsigned char pause_phonemes[8] = {0, phonPAUSE_VSHORT, phonPAUSE_SHORT, phonPAUSE, phonPAUSE_LONG, phonGLOTTALSTOP, phonPAUSE_LONG, phonPAUSE_LONG}; | |||
int Translator::ChangePhonemes(PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch) | |||
{//====================================================================================================== | |||
extern int n_ph_list2; | |||
extern PHONEME_LIST2 ph_list2[N_PHONEME_LIST]; // first stage of text->phonemes | |||
static int ChangePhonemes(Translator *tr, PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch) | |||
{//================================================================================================================= | |||
// Called for each phoneme in the phoneme list, to allow a language to make changes | |||
// ph The current phoneme | |||
if(tr->translator_name == L('r','u')) | |||
return(ChangePhonemes_ru(tr, phlist, n_ph, index, ph, ch)); | |||
return(0); | |||
} | |||
int Translator::SubstitutePhonemes(PHONEME_LIST2 *plist_out) | |||
{//========================================================= | |||
static int SubstitutePhonemes(Translator *tr, PHONEME_LIST2 *plist_out) | |||
{//==================================================================== | |||
// Copy the phonemes list and perform any substitutions that are required for the | |||
// current voice | |||
int ix; | |||
@@ -77,7 +86,7 @@ int Translator::SubstitutePhonemes(PHONEME_LIST2 *plist_out) | |||
if((plist2+1)->sourceix || ((next != 0) && (next->type == phPAUSE))) | |||
word_end = 1; // this phoneme is the end of a word | |||
if(langopts.phoneme_change != 0) | |||
if(tr->langopts.phoneme_change != 0) | |||
{ | |||
// this language does changes to phonemes after translation | |||
@@ -134,7 +143,7 @@ int Translator::SubstitutePhonemes(PHONEME_LIST2 *plist_out) | |||
ch.vowel_this = syllable; | |||
ch.vowel_stressed = syllable_stressed; | |||
ChangePhonemes(ph_list2, n_ph_list2, ix, phoneme_tab[ph_list2[ix].phcode], &ch); | |||
ChangePhonemes(tr, ph_list2, n_ph_list2, ix, phoneme_tab[ph_list2[ix].phcode], &ch); | |||
} | |||
// check whether a Voice has specified that we should replace this phoneme | |||
@@ -172,8 +181,8 @@ int Translator::SubstitutePhonemes(PHONEME_LIST2 *plist_out) | |||
void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
{//============================================================================================ | |||
void MakePhonemeList(Translator *tr, int post_pause, int start_sentence) | |||
{//===================================================================== | |||
int ix=0; | |||
int j; | |||
@@ -196,17 +205,18 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
PHONEME_LIST2 *plist2 = &ph_list2_null; | |||
PHONEME_LIST2 *plist2_inserted = NULL; | |||
plist2 = ph_list2; | |||
phlist = phoneme_list; | |||
end_sourceix = ph_list2[n_ph_list2-1].sourceix; | |||
end_sourceix = plist2[n_ph_list2-1].sourceix; | |||
// is the last word of the clause unstressed ? | |||
max_stress = 0; | |||
for(j=n_ph_list2-3; j>=0; j--) | |||
for(j = n_ph_list2-3; j>=0; j--) | |||
{ | |||
// start with the last phoneme (before the terminating pauses) and move forwards | |||
if((ph_list2[j].stress & 0x7f) > max_stress) | |||
max_stress = ph_list2[j].stress & 0x7f; | |||
if(ph_list2[j].sourceix != 0) | |||
if((plist2[j].stress & 0x7f) > max_stress) | |||
max_stress = plist2[j].stress & 0x7f; | |||
if(plist2[j].sourceix != 0) | |||
break; | |||
} | |||
if(max_stress < 4) | |||
@@ -214,12 +224,12 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
// the last word is unstressed, look for a previous word that can be stressed | |||
while(--j >= 0) | |||
{ | |||
if(ph_list2[j].synthflags & SFLAG_PROMOTE_STRESS) // dictionary flags indicated that this stress can be promoted | |||
if(plist2[j].synthflags & SFLAG_PROMOTE_STRESS) // dictionary flags indicated that this stress can be promoted | |||
{ | |||
ph_list2[j].stress = 4; // promote to stressed | |||
plist2[j].stress = 4; // promote to stressed | |||
break; | |||
} | |||
if(ph_list2[j].stress >= 4) | |||
if(plist2[j].stress >= 4) | |||
{ | |||
// found a stressed syllable, so stop looking | |||
break; | |||
@@ -227,7 +237,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
} | |||
} | |||
if((regression = langopts.param[LOPT_REGRESSIVE_VOICING]) != 0) | |||
if((regression = tr->langopts.param[LOPT_REGRESSIVE_VOICING]) != 0) | |||
{ | |||
// set consonant clusters to all voiced or all unvoiced | |||
// Regressive | |||
@@ -236,7 +246,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
for(j=n_ph_list2-1; j>=0; j--) | |||
{ | |||
ph = phoneme_tab[ph_list2[j].phcode]; | |||
ph = phoneme_tab[plist2[j].phcode]; | |||
if(ph == NULL) | |||
continue; | |||
@@ -263,7 +273,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
else | |||
if((voicing==2) && ((ph->phflags & phALTERNATIVE)==phSWITCHVOICING)) | |||
{ | |||
ph_list2[j].phcode = ph->alternative_ph; // change to voiced equivalent | |||
plist2[j].phcode = ph->alternative_ph; // change to voiced equivalent | |||
} | |||
} | |||
else | |||
@@ -276,7 +286,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
else | |||
if((voicing==1) && ((ph->phflags & phALTERNATIVE)==phSWITCHVOICING)) | |||
{ | |||
ph_list2[j].phcode = ph->alternative_ph; // change to unvoiced equivalent | |||
plist2[j].phcode = ph->alternative_ph; // change to unvoiced equivalent | |||
} | |||
} | |||
else | |||
@@ -292,7 +302,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
voicing = 0; | |||
} | |||
} | |||
if((regression & 0x4) && (ph_list2[j].sourceix)) | |||
if((regression & 0x4) && (plist2[j].sourceix)) | |||
{ | |||
// stop propagation at a word boundary | |||
voicing = 0; | |||
@@ -300,13 +310,13 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
} | |||
} | |||
n_ph_list2 = SubstitutePhonemes(ph_list3) - 2; | |||
n_ph_list2 = SubstitutePhonemes(tr,ph_list3) - 2; | |||
// transfer all the phonemes of the clause into phoneme_list | |||
ph = phoneme_tab[phonPAUSE]; | |||
switched_language = 0; | |||
for(j=0; insert_ph || ((j<n_ph_list2) && (ix < N_PHONEME_LIST-3)); j++) | |||
for(j=0; insert_ph || ((j < n_ph_list2) && (ix < N_PHONEME_LIST-3)); j++) | |||
{ | |||
prev = ph; | |||
@@ -371,7 +381,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
{ | |||
// in a sequence of unstressed syllables, reduce alternate syllables to 'diminished' | |||
// stress. But not for the last phoneme of a stressed word | |||
if((langopts.stress_flags & 0x2) || ((word_stress > 3) && ((plist2+1)->sourceix!=0))) | |||
if((tr->langopts.stress_flags & 0x2) || ((word_stress > 3) && ((plist2+1)->sourceix!=0))) | |||
{ | |||
// An unstressed final vowel of a stressed word | |||
unstress_count=1; // try again for next syllable | |||
@@ -444,7 +454,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
plist2->synthflags &= ~SFLAG_SYLLABLE; | |||
} | |||
if(langopts.param[LOPT_REDUCE_T]) | |||
if(tr->langopts.param[LOPT_REDUCE_T]) | |||
{ | |||
if((ph->mnemonic == 't') && (plist2->sourceix == 0) && ((prev->type == phVOWEL) || (prev->mnemonic == 'n'))) | |||
{ | |||
@@ -456,7 +466,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
} | |||
while((ph->reduce_to != 0) && (!(plist2->synthflags & SFLAG_DICTIONARY) || (langopts.param[LOPT_REDUCE] & 1))) | |||
while((ph->reduce_to != 0) && (!(plist2->synthflags & SFLAG_DICTIONARY) || (tr->langopts.param[LOPT_REDUCE] & 1))) | |||
{ | |||
int reduce_level; | |||
int stress_level; | |||
@@ -486,7 +496,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
if(stress_level < reduce_level) | |||
reduce =1; | |||
if((word_stress < 4) && (langopts.param[LOPT_REDUCE] & 0x2) && (stress_level >= word_stress)) | |||
if((word_stress < 4) && (tr->langopts.param[LOPT_REDUCE] & 0x2) && (stress_level >= word_stress)) | |||
{ | |||
// don't reduce the most stressed syllable in an unstressed word | |||
reduce = 0; | |||
@@ -516,10 +526,10 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
{ | |||
int x; | |||
if(langopts.vowel_pause && (ph->type != phPAUSE)) | |||
if(tr->langopts.vowel_pause && (ph->type != phPAUSE)) | |||
{ | |||
if((ph->type != phVOWEL) && (langopts.vowel_pause & 0x200)) | |||
if((ph->type != phVOWEL) && (tr->langopts.vowel_pause & 0x200)) | |||
{ | |||
// add a pause after a word which ends in a consonant | |||
insert_ph = phonPAUSE_NOLINK; | |||
@@ -527,7 +537,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
if(next->type == phVOWEL) | |||
{ | |||
if((x = langopts.vowel_pause & 0x0c) != 0) | |||
if((x = tr->langopts.vowel_pause & 0x0c) != 0) | |||
{ | |||
// break before a word which starts with a vowel | |||
if(x == 0xc) | |||
@@ -536,7 +546,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
insert_ph = phonPAUSE_VSHORT; | |||
} | |||
if((ph->type == phVOWEL) && ((x = langopts.vowel_pause & 0x03) != 0)) | |||
if((ph->type == phVOWEL) && ((x = tr->langopts.vowel_pause & 0x03) != 0)) | |||
{ | |||
// adjacent vowels over a word boundary | |||
if(x == 2) | |||
@@ -545,7 +555,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
insert_ph = phonPAUSE_VSHORT; | |||
} | |||
if(((plist2+1)->stress >= 4) && (langopts.vowel_pause & 0x100)) | |||
if(((plist2+1)->stress >= 4) && (tr->langopts.vowel_pause & 0x100)) | |||
{ | |||
// pause before a words which starts with a stressed vowel | |||
insert_ph = phonPAUSE_SHORT; | |||
@@ -555,7 +565,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
if(plist2 != plist2_inserted) | |||
{ | |||
if((x = (langopts.word_gap & 0x7)) != 0) | |||
if((x = (tr->langopts.word_gap & 0x7)) != 0) | |||
{ | |||
if((x > 1) || ((insert_ph != phonPAUSE_SHORT) && (insert_ph != phonPAUSE_NOLINK))) | |||
{ | |||
@@ -584,7 +594,7 @@ void Translator::MakePhonemeList(int post_pause, int start_sentence) | |||
} | |||
} | |||
else | |||
if(((langopts.word_gap & 8)==0) || ((plist2+1)->sourceix == 0)) | |||
if(((tr->langopts.word_gap & 8)==0) || ((plist2+1)->sourceix == 0)) | |||
{ | |||
// This phoneme can be linked to a following vowel by inserting a linking phoneme | |||
if(next->type == phVOWEL) |
@@ -42,26 +42,26 @@ | |||
#define N_XML_BUF 256 | |||
const char *xmlbase = ""; // base URL from <speak> | |||
static const char *xmlbase = ""; // base URL from <speak> | |||
int namedata_ix=0; | |||
int n_namedata = 0; | |||
static int namedata_ix=0; | |||
static int n_namedata = 0; | |||
char *namedata = NULL; | |||
FILE *f_input = NULL; | |||
int ungot_char2 = 0; | |||
static FILE *f_input = NULL; | |||
static int ungot_char2 = 0; | |||
char *p_textinput; | |||
wchar_t *p_wchar_input; | |||
int ungot_char; | |||
const char *ungot_word = NULL; | |||
int end_of_input; | |||
static int ungot_char; | |||
static const char *ungot_word = NULL; | |||
static int end_of_input; | |||
int ignore_text=0; // set during <sub> ... </sub> to ignore text which has been replaced by an alias | |||
int clear_skipping_text = 0; // next clause should clear the skipping_text flag | |||
static int ignore_text=0; // set during <sub> ... </sub> to ignore text which has been replaced by an alias | |||
static int clear_skipping_text = 0; // next clause should clear the skipping_text flag | |||
int count_characters = 0; | |||
int sayas_mode; | |||
int ssml_ignore_l_angle = 0; | |||
static int sayas_mode; | |||
static int ssml_ignore_l_angle = 0; | |||
static const char *punct_stop = ".:!?"; // pitch fall if followed by space | |||
static const char *punct_close = ")]}>;'\""; // always pitch fall unless followed by alnum | |||
@@ -71,7 +71,7 @@ static const char *tone_punct_on = "\0016T"; // add reverberation, lower pitch | |||
static const char *tone_punct_off = "\001T"; | |||
// punctuations symbols that can end a clause | |||
const unsigned short punct_chars[] = {',','.','?','!',':',';', | |||
static const unsigned short punct_chars[] = {',','.','?','!',':',';', | |||
0x2013, // en-dash | |||
0x2014, // em-dash | |||
0x2026, // elipsis | |||
@@ -159,17 +159,17 @@ typedef struct { | |||
} SSML_STACK; | |||
#define N_SSML_STACK 20 | |||
int n_ssml_stack; | |||
SSML_STACK ssml_stack[N_SSML_STACK]; | |||
static int n_ssml_stack; | |||
static SSML_STACK ssml_stack[N_SSML_STACK]; | |||
char current_voice_id[40] = {0}; | |||
static char current_voice_id[40] = {0}; | |||
#define N_PARAM_STACK 20 | |||
int n_param_stack; | |||
static int n_param_stack; | |||
PARAM_STACK param_stack[N_PARAM_STACK]; | |||
int speech_parameters[N_SPEECH_PARAM]; // current values, from param_stack | |||
static int speech_parameters[N_SPEECH_PARAM]; // current values, from param_stack | |||
const int param_defaults[N_SPEECH_PARAM] = { | |||
0, // silence (internal use) | |||
@@ -511,8 +511,8 @@ static void UngetC(int c) | |||
} | |||
const char *WordToString2(unsigned int word) | |||
{//======================================== | |||
static const char *WordToString2(unsigned int word) | |||
{//================================================ | |||
// Convert a language mnemonic word into a string | |||
int ix; | |||
static char buf[5]; | |||
@@ -529,16 +529,16 @@ const char *WordToString2(unsigned int word) | |||
} | |||
const char *Translator::LookupSpecial(const char *string, char* text_out) | |||
{//====================================================================== | |||
static const char *LookupSpecial(Translator *tr, const char *string, char* text_out) | |||
{//================================================================================= | |||
unsigned int flags[2]; | |||
char phonemes[55]; | |||
char phonemes2[55]; | |||
char *string1 = (char *)string; | |||
if(LookupDictList(&string1,phonemes,flags,0,NULL)) | |||
if(LookupDictList(tr,&string1,phonemes,flags,0,NULL)) | |||
{ | |||
SetWordStress(phonemes,flags[0],-1,0); | |||
SetWordStress(tr, phonemes, flags[0], -1, 0); | |||
DecodePhonemes(phonemes,phonemes2); | |||
sprintf(text_out,"[[%s]]",phonemes2); | |||
option_phoneme_input |= 2; | |||
@@ -548,8 +548,8 @@ const char *Translator::LookupSpecial(const char *string, char* text_out) | |||
} | |||
const char *Translator::LookupCharName(int c) | |||
{//========================================== | |||
static const char *LookupCharName(Translator *tr, int c) | |||
{//===================================================== | |||
// Find the phoneme string (in ascii) to speak the name of character c | |||
// Used for punctuation characters and symbols | |||
@@ -571,28 +571,28 @@ const char *Translator::LookupCharName(int c) | |||
single_letter[2+ix]=0; | |||
string = &single_letter[1]; | |||
if(LookupDictList(&string, phonemes, flags, 0, NULL) == 0) | |||
if(LookupDictList(tr, &string, phonemes, flags, 0, NULL) == 0) | |||
{ | |||
// try _* then * | |||
string = &single_letter[2]; | |||
if(LookupDictList(&string, phonemes, flags, 0, NULL) == 0) | |||
if(LookupDictList(tr, &string, phonemes, flags, 0, NULL) == 0) | |||
{ | |||
// now try the rules | |||
single_letter[1] = ' '; | |||
TranslateRules(&single_letter[2], phonemes, sizeof(phonemes), NULL,0,NULL); | |||
TranslateRules(tr, &single_letter[2], phonemes, sizeof(phonemes), NULL,0,NULL); | |||
} | |||
} | |||
if((phonemes[0] == 0) && (translator_name != L('e','n'))) | |||
if((phonemes[0] == 0) && (tr->translator_name != L('e','n'))) | |||
{ | |||
// not found, try English | |||
SetTranslator2("en"); | |||
string = &single_letter[1]; | |||
single_letter[1] = '_'; | |||
if(translator2->LookupDictList(&string, phonemes, flags, 0, NULL) == 0) | |||
if(LookupDictList(translator2, &string, phonemes, flags, 0, NULL) == 0) | |||
{ | |||
string = &single_letter[2]; | |||
translator2->LookupDictList(&string, phonemes, flags, 0, NULL); | |||
LookupDictList(translator2, &string, phonemes, flags, 0, NULL); | |||
} | |||
if(phonemes[0]) | |||
{ | |||
@@ -608,14 +608,14 @@ const char *Translator::LookupCharName(int c) | |||
{ | |||
if(lang_name) | |||
{ | |||
translator2->SetWordStress(phonemes,flags[0],-1,0); | |||
SetWordStress(translator2, phonemes, flags[0], -1, 0); | |||
DecodePhonemes(phonemes,phonemes2); | |||
sprintf(buf,"[[_^_%s %s _^_%s]]","en",phonemes2,WordToString2(translator_name)); | |||
sprintf(buf,"[[_^_%s %s _^_%s]]","en",phonemes2,WordToString2(tr->translator_name)); | |||
SelectPhonemeTable(voice->phoneme_tab_ix); // revert to original phoneme table | |||
} | |||
else | |||
{ | |||
SetWordStress(phonemes,flags[0],-1,0); | |||
SetWordStress(tr, phonemes, flags[0], -1, 0); | |||
DecodePhonemes(phonemes,phonemes2); | |||
sprintf(buf,"[[%s]] ",phonemes2); | |||
} | |||
@@ -789,8 +789,8 @@ static int LoadSoundFile2(const char *fname) | |||
int Translator::AnnouncePunctuation(int c1, int c2, char *buf, int bufix) | |||
{//====================================================================== | |||
static int AnnouncePunctuation(Translator *tr, int c1, int c2, char *buf, int bufix) | |||
{//================================================================================= | |||
// announce punctuation names | |||
// c1: the punctuation character | |||
// c2: the following character | |||
@@ -809,7 +809,7 @@ int Translator::AnnouncePunctuation(int c1, int c2, char *buf, int bufix) | |||
found = 1; | |||
} | |||
else | |||
if((punctname = LookupCharName(c1)) != NULL) | |||
if((punctname = LookupCharName(tr, c1)) != NULL) | |||
{ | |||
found = 1; | |||
if(bufix==0) | |||
@@ -891,7 +891,7 @@ int Translator::AnnouncePunctuation(int c1, int c2, char *buf, int bufix) | |||
static char ignore_if_self_closing[] = {0,1,1,1,1,0,0,0,0,1,1,0,1,0,1,0,0}; | |||
MNEM_TAB ssmltags[] = { | |||
static MNEM_TAB ssmltags[] = { | |||
{"speak", SSML_SPEAK}, | |||
{"voice", SSML_VOICE}, | |||
{"prosody", SSML_PROSODY}, | |||
@@ -1243,8 +1243,8 @@ static int attr_prosody_value(int param_type, const wchar_t *pw, int *value_out) | |||
} // end of attr_prosody_value | |||
int AddNameData(const char *name, int wide) | |||
{//======================================== | |||
static int AddNameData(const char *name, int wide) | |||
{//=============================================== | |||
// Add the name to the namedata and return its position | |||
int ix; | |||
int len; | |||
@@ -1849,7 +1849,7 @@ terminator=0; // ?? Sentence intonation, but no pause ?? | |||
} // end of ProcessSsmlTag | |||
MNEM_TAB xml_char_mnemonics[] = { | |||
static MNEM_TAB xml_char_mnemonics[] = { | |||
{"gt",'>'}, | |||
{"lt",'<'}, | |||
{"amp", '&'}, | |||
@@ -1859,8 +1859,8 @@ MNEM_TAB xml_char_mnemonics[] = { | |||
{NULL,-1}}; | |||
int Translator::ReadClause(FILE *f_in, char *buf, short *charix, int n_buf, int *tone_type) | |||
{//======================================================================================== | |||
int ReadClause(Translator *tr, FILE *f_in, char *buf, short *charix, int n_buf, int *tone_type) | |||
{//============================================================================================ | |||
/* Find the end of the current clause. | |||
Write the clause into buf | |||
@@ -1903,8 +1903,8 @@ int Translator::ReadClause(FILE *f_in, char *buf, short *charix, int n_buf, int | |||
clear_skipping_text = 0; | |||
} | |||
clause_upper_count = 0; | |||
clause_lower_count = 0; | |||
tr->clause_upper_count = 0; | |||
tr->clause_lower_count = 0; | |||
end_of_input = 0; | |||
*tone_type = 0; | |||
@@ -2158,7 +2158,7 @@ f_input = f_in; // for GetC etc | |||
{ | |||
char *p_word; | |||
if(translator_name == 0x6a626f) | |||
if(tr->translator_name == 0x6a626f) | |||
{ | |||
// language jbo : lojban | |||
// treat "i" or ".i" as end-of-sentence | |||
@@ -2181,12 +2181,12 @@ f_input = f_in; // for GetC etc | |||
if(iswupper(c1)) | |||
{ | |||
clause_upper_count++; | |||
tr->clause_upper_count++; | |||
if((option_capitals == 2) && (sayas_mode == 0) && !iswupper(cprev)) | |||
{ | |||
char text_buf[40]; | |||
char text_buf2[30]; | |||
if(LookupSpecial("_cap",text_buf2) != NULL) | |||
if(LookupSpecial(tr, "_cap", text_buf2) != NULL) | |||
{ | |||
sprintf(text_buf,"%s%s%s",tone_punct_on,text_buf2,tone_punct_off); | |||
j = strlen(text_buf); | |||
@@ -2200,7 +2200,7 @@ f_input = f_in; // for GetC etc | |||
} | |||
else | |||
if(iswalpha(c1)) | |||
clause_lower_count++; | |||
tr->clause_lower_count++; | |||
if(option_phoneme_input) | |||
{ | |||
@@ -2256,7 +2256,7 @@ if(option_ssml) parag=1; | |||
// if a list of allowed punctuation has been set up, check whether the character is in it | |||
if((option_punctuation == 1) || (wcschr(option_punctlist,c1) != NULL)) | |||
{ | |||
if((terminator = AnnouncePunctuation(c1, c2, buf, ix)) >= 0) | |||
if((terminator = AnnouncePunctuation(tr, c1, c2, buf, ix)) >= 0) | |||
return(terminator); | |||
} | |||
} | |||
@@ -2299,7 +2299,7 @@ if(option_ssml) parag=1; | |||
if((nl_count==0) && (c1 == '.')) | |||
{ | |||
if(iswdigit(cprev) && (langopts.numbers & 0x10000)) | |||
if(iswdigit(cprev) && (tr->langopts.numbers & 0x10000)) | |||
{ | |||
// dot after a number indicates an ordinal number | |||
c2 = ' '; | |||
@@ -2377,6 +2377,8 @@ void InitText2(void) | |||
{//================= | |||
int param; | |||
ungot_char = 0; | |||
n_ssml_stack =1; | |||
n_param_stack = 1; | |||
ssml_stack[0].tag_type = 0; |
@@ -253,8 +253,8 @@ static void DoEmbedded2(int &embix) | |||
} | |||
void Translator::CalcLengths() | |||
{//=========================== | |||
void CalcLengths(Translator *tr) | |||
{//============================== | |||
int ix; | |||
int ix2; | |||
PHONEME_LIST *prev; | |||
@@ -318,14 +318,14 @@ void Translator::CalcLengths() | |||
if(prev->type == phSTOP) | |||
p->prepause = 60; | |||
if((langopts.word_gap & 0x10) && (p->newword)) | |||
if((tr->langopts.word_gap & 0x10) && (p->newword)) | |||
p->prepause = 60; | |||
if(p->ph->phflags & phLENGTHENSTOP) | |||
p->prepause += 30; | |||
if(p->synthflags & SFLAG_LENGTHEN) | |||
p->prepause += langopts.long_stop; | |||
p->prepause += tr->langopts.long_stop; | |||
break; | |||
case phVFRICATIVE: | |||
@@ -353,7 +353,7 @@ void Translator::CalcLengths() | |||
else | |||
p->length = 256; | |||
if((langopts.word_gap & 0x10) && (p->newword)) | |||
if((tr->langopts.word_gap & 0x10) && (p->newword)) | |||
p->prepause = 30; | |||
break; | |||
@@ -383,14 +383,14 @@ void Translator::CalcLengths() | |||
p->prepause = 0; | |||
} | |||
} | |||
if((langopts.word_gap & 0x10) && (p->newword) && (p->prepause < 20)) | |||
if((tr->langopts.word_gap & 0x10) && (p->newword) && (p->prepause < 20)) | |||
p->prepause = 20; | |||
break; | |||
case phLIQUID: | |||
case phNASAL: | |||
p->amp = stress_amps[1]; // unless changed later | |||
p->amp = tr->stress_amps[1]; // unless changed later | |||
p->length = 256; // TEMPORARY | |||
min_drop = 0; | |||
@@ -457,9 +457,9 @@ void Translator::CalcLengths() | |||
if(stress > 7) stress = 7; | |||
if(pre_sonorant) | |||
p->amp = stress_amps[stress]-1; | |||
p->amp = tr->stress_amps[stress]-1; | |||
else | |||
p->amp = stress_amps[stress]; | |||
p->amp = tr->stress_amps[stress]; | |||
if(emphasized) | |||
p->amp = 25; | |||
@@ -467,8 +467,8 @@ void Translator::CalcLengths() | |||
if(ix >= (n_phoneme_list-3)) | |||
{ | |||
// last phoneme of a clause, limit its amplitude | |||
if(p->amp > langopts.param[LOPT_MAXAMP_EOC]) | |||
p->amp = langopts.param[LOPT_MAXAMP_EOC]; | |||
if(p->amp > tr->langopts.param[LOPT_MAXAMP_EOC]) | |||
p->amp = tr->langopts.param[LOPT_MAXAMP_EOC]; | |||
} | |||
// is the last syllable of a word ? | |||
@@ -501,19 +501,19 @@ void Translator::CalcLengths() | |||
if(more_syllables==0) | |||
{ | |||
len = langopts.length_mods0[next2->ph->length_mod *10+ next->ph->length_mod]; | |||
len = tr->langopts.length_mods0[next2->ph->length_mod *10+ next->ph->length_mod]; | |||
if((next->newword) && (langopts.word_gap & 0x20)) | |||
if((next->newword) && (tr->langopts.word_gap & 0x20)) | |||
{ | |||
// consider as a pause + first phoneme of the next word | |||
length_mod = (len + langopts.length_mods0[next->ph->length_mod *10+ 1])/2; | |||
length_mod = (len + tr->langopts.length_mods0[next->ph->length_mod *10+ 1])/2; | |||
} | |||
else | |||
length_mod = len; | |||
} | |||
else | |||
{ | |||
length_mod = langopts.length_mods[next2->ph->length_mod *10+ next->ph->length_mod]; | |||
length_mod = tr->langopts.length_mods[next2->ph->length_mod *10+ next->ph->length_mod]; | |||
if((next->type == phNASAL) && (next2->type == phSTOP || next2->type == phVSTOP) && (next3->ph->phflags & phFORTIS)) | |||
length_mod -= 15; | |||
@@ -546,8 +546,8 @@ void Translator::CalcLengths() | |||
length_mod += 20; | |||
} | |||
if((len = stress_lengths[stress]) == 0) | |||
len = stress_lengths[6]; | |||
if((len = tr->stress_lengths[stress]) == 0) | |||
len = tr->stress_lengths[6]; | |||
length_mod = (length_mod * len)/128; | |||
@@ -564,7 +564,7 @@ void Translator::CalcLengths() | |||
{ | |||
// this is the last syllable in the clause, lengthen it - more for short vowels | |||
len = p->ph->std_length; | |||
if(langopts.stress_flags & 0x40000) | |||
if(tr->langopts.stress_flags & 0x40000) | |||
len=200; // don't lengthen short vowels more than long vowels at end-of-clause | |||
length_mod = length_mod * (256 + (280 - len)/3)/256; | |||
} |
@@ -618,7 +618,7 @@ void sync_espeak_Key(const char *key) | |||
int letter; | |||
int ix; | |||
ix = utf8_in(&letter,key,0); | |||
ix = utf8_in(&letter,key); | |||
if(key[ix] == 0) | |||
{ | |||
// a single character |
@@ -568,7 +568,6 @@ void set_say_options(int reg2, int reg3) | |||
option_linelength = 0; | |||
option_phonemes = 0; | |||
option_waveout = 0; | |||
option_harmonic1 = 8; | |||
option_multibyte = 0; // auto | |||
option_capitals = 0; | |||
option_punctuation = 0; | |||
@@ -868,7 +867,6 @@ void command_line(char *arg_string, int wait) | |||
option_phonemes = 0; | |||
option_waveout = 0; | |||
option_quiet = 0; | |||
option_harmonic1 = 8; | |||
option_multibyte = 0; // auto | |||
option_capitals = 0; | |||
option_punctuation = 0; | |||
@@ -1263,7 +1261,7 @@ void terminate_module(void) | |||
{ | |||
RiscosCloseSound(); | |||
RemoveCallback(); | |||
delete translator; | |||
DeleteTranslator(translator); | |||
FreePhData(); | |||
} /* end of terminate_module */ | |||
@@ -106,8 +106,8 @@ void unload_MBR() | |||
#endif // USE_MBROLA_LIB | |||
MBROLA_TAB *mbrola_tab = NULL; | |||
int mbrola_control = 0; | |||
static MBROLA_TAB *mbrola_tab = NULL; | |||
static int mbrola_control = 0; | |||
@@ -200,8 +200,8 @@ espeak_ERROR LoadMbrolaTable(const char *mbrola_voice, const char *phtrans, int | |||
} // end of LoadMbrolaTable | |||
int GetMbrName(PHONEME_LIST *plist, PHONEME_TAB *ph, PHONEME_TAB *ph_prev, PHONEME_TAB *ph_next, int *name2, int *split, int *control) | |||
{//============================================================================================================== | |||
static int GetMbrName(PHONEME_LIST *plist, PHONEME_TAB *ph, PHONEME_TAB *ph_prev, PHONEME_TAB *ph_next, int *name2, int *split, int *control) | |||
{//========================================================================================================================================== | |||
// Look up a phoneme in the mbrola phoneme name translation table | |||
// It may give none, 1, or 2 mbrola phonemes | |||
int mnem = ph->mnemonic; | |||
@@ -431,8 +431,8 @@ static void MbrolaEmbedded(int &embix, int sourceix) | |||
#ifdef PLATFORM_WINDOWS | |||
int MbrolaSynth(char *p_mbrola) | |||
{//============================ | |||
static int MbrolaSynth(char *p_mbrola) | |||
{//=================================== | |||
// p_mbrola is a string of mbrola pho lines - Windows | |||
int len; | |||
int finished; | |||
@@ -472,8 +472,8 @@ int MbrolaSynth(char *p_mbrola) | |||
} // end of SynthMbrola | |||
#else | |||
int MbrolaSynth(char *p_mbrola) | |||
{//============================ | |||
static int MbrolaSynth(char *p_mbrola) | |||
{//=================================== | |||
// p_mbrola is a string of mbrola pho lines - Linux | |||
// This is wrong | |||
@@ -737,9 +737,9 @@ void MbrolaTranslate(PHONEME_LIST *plist, int n_phonemes, FILE *f_mbrola) | |||
#ifdef TEST_MBROLA | |||
PHONEME_LIST mbrola_phlist; | |||
int mbrola_n_ph; | |||
int mbrola_phix; | |||
static PHONEME_LIST mbrola_phlist; | |||
static int mbrola_n_ph; | |||
static int mbrola_phix; | |||
int MbrolaFill(int fill_zeros) |
@@ -53,7 +53,7 @@ static unsigned char *phoneme_tab_data = NULL; | |||
int n_phoneme_tables; | |||
PHONEME_TAB_LIST phoneme_tab_list[N_PHONEME_TABS]; | |||
static int phoneme_tab_number = 0; | |||
int phoneme_tab_number = 0; | |||
int wavefile_ix; // a wavefile to play along with the synthesis | |||
int wavefile_amp; | |||
@@ -65,15 +65,9 @@ int vowel_transition[4]; | |||
int vowel_transition0; | |||
int vowel_transition1; | |||
void FormantTransitions(frameref_t *seq, int &n_frames, PHONEME_TAB *this_ph, PHONEME_TAB *other_ph, int which); | |||
int FormantTransition2(frameref_t *seq, int &n_frames, unsigned int data1, unsigned int data2, PHONEME_TAB *other_ph, int which); | |||
const char *PhonemeTabName(void) | |||
{//============================= | |||
return(phoneme_tab_list[phoneme_tab_number].name); | |||
} | |||
static int ReadPhFile(char **ptr, const char *fname) | |||
{//================================================= | |||
@@ -173,8 +167,23 @@ void FreePhData(void) | |||
} | |||
int LookupPh(const char *string) | |||
{//============================= | |||
int PhonemeCode(unsigned int mnem) | |||
{//=============================== | |||
int ix; | |||
for(ix=0; ix<n_phoneme_tab; ix++) | |||
{ | |||
if(phoneme_tab[ix] == NULL) | |||
continue; | |||
if(phoneme_tab[ix]->mnemonic == mnem) | |||
return(phoneme_tab[ix]->code); | |||
} | |||
return(0); | |||
} | |||
int LookupPhonemeString(const char *string) | |||
{//======================================== | |||
int ix; | |||
unsigned char c; | |||
unsigned int mnem; | |||
@@ -188,19 +197,11 @@ int LookupPh(const char *string) | |||
mnem |= (c << (ix*8)); | |||
} | |||
for(ix=0; ix<n_phoneme_tab; ix++) | |||
{ | |||
if(phoneme_tab[ix] == NULL) | |||
continue; | |||
if(phoneme_tab[ix]->mnemonic == mnem) | |||
return(ix); | |||
} | |||
return(0); | |||
return(PhonemeCode(mnem)); | |||
} | |||
static unsigned int LookupSound2(int index, unsigned int other_phcode, int control) | |||
{//================================================================================ | |||
// control=1 get formant transition data only | |||
@@ -456,7 +457,6 @@ frameref_t *LookupSpect(PHONEME_TAB *this_ph, PHONEME_TAB *prev_ph, PHONEME_TAB | |||
if(*match_level == 0) | |||
seq_len_adjust = FormantTransition2(frames,nf,vowel_transition0,vowel_transition1,prev_ph,which); | |||
} | |||
// FormantTransitions(frames,nf,this_ph,other_ph,which); | |||
} | |||
nf1 = nf - 1; |
@@ -90,6 +90,7 @@ const char *WordToString(unsigned int word) | |||
} | |||
void SynthesizeInit() | |||
{//================== | |||
last_pitch_cmd = 0; | |||
@@ -477,8 +478,8 @@ static void AdjustFormants(frame_t *fr, int target, int min, int max, int f1_adj | |||
} | |||
int VowelCloseness(frame_t *fr) | |||
{//============================ | |||
static int VowelCloseness(frame_t *fr) | |||
{//=================================== | |||
// return a value 0-3 depending on the vowel's f1 | |||
int f1; | |||
@@ -1108,10 +1109,6 @@ static void DoEmbedded(int &embix, int sourceix) | |||
} | |||
void SwitchDictionary() | |||
{//==================== | |||
} | |||
int Generate(PHONEME_LIST *phoneme_list, int *n_ph, int resume) | |||
{//============================================================ | |||
@@ -1599,12 +1596,12 @@ int SpeakNextClause(FILE *f_in, const void *text_in, int control) | |||
// read the next clause from the input text file, translate it, and generate | |||
// entries in the wavegen command queue | |||
p_text = translator->TranslateClause(f_text,p_text,&clause_tone,&voice_change); | |||
p_text = TranslateClause(translator, f_text, p_text, &clause_tone, &voice_change); | |||
translator->CalcPitches(clause_tone); | |||
translator->CalcLengths(); | |||
CalcPitches(translator, clause_tone); | |||
CalcLengths(translator); | |||
translator->GetTranslatedPhonemeString(translator->phon_out,sizeof(translator->phon_out)); | |||
GetTranslatedPhonemeString(translator->phon_out,sizeof(translator->phon_out)); | |||
if(option_phonemes > 0) | |||
{ | |||
fprintf(f_trans,"%s\n",translator->phon_out); |
@@ -326,7 +326,7 @@ espeak_ERROR SetVoiceByProperties(espeak_VOICE *voice_selector); | |||
espeak_ERROR LoadMbrolaTable(const char *mbrola_voice, const char *phtrans, int srate); | |||
void SetParameter(int parameter, int value, int relative); | |||
void MbrolaTranslate(PHONEME_LIST *plist, int n_phonemes, FILE *f_mbrola); | |||
int MbrolaSynth(char *p_mbrola); | |||
//int MbrolaSynth(char *p_mbrola); | |||
int DoSample(PHONEME_TAB *ph1, PHONEME_TAB *ph2, int which, int length_mod, int amp); | |||
int DoSpect(PHONEME_TAB *this_ph, PHONEME_TAB *prev_ph, PHONEME_TAB *next_ph, | |||
int which, PHONEME_LIST *plist, int modulation); |
@@ -19,120 +19,3 @@ | |||
#include "StdAfx.h" | |||
#include <stdio.h> | |||
#include <ctype.h> | |||
#include <wctype.h> | |||
#include <stdlib.h> | |||
#include <string.h> | |||
#include <locale.h> | |||
#include "speak_lib.h" | |||
#include "speech.h" | |||
#include "phoneme.h" | |||
#include "synthesize.h" | |||
#include "translate.h" | |||
#include "tr_languages.h" | |||
Translator_English::Translator_English() : Translator() | |||
{//=================================== | |||
// static int stress_lengths2[8] = {182,140, 220,220, 220,240, 248,270}; | |||
static const short stress_lengths2[8] = {182,140, 220,220, 0,0, 248,275}; | |||
memcpy(stress_lengths,stress_lengths2,sizeof(stress_lengths)); | |||
langopts.stress_rule = 0; | |||
langopts.numbers = 0x841 + NUM_ROMAN; | |||
langopts.param[LOPT_COMBINE_WORDS] = 2; // allow "mc" to cmbine with the following word | |||
} | |||
static unsigned char initials_bitmap[86] = { | |||
0x00, 0x00, 0x00, 0x00, 0x22, 0x08, 0x00, 0x88, // 0 | |||
0x20, 0x24, 0x20, 0x80, 0x10, 0x00, 0x00, 0x00, | |||
0x00, 0x28, 0x08, 0x00, 0x88, 0x22, 0x04, 0x00, // 16 | |||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | |||
0x00, 0x88, 0x22, 0x04, 0x00, 0x02, 0x00, 0x04, // 32 | |||
0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | |||
0x00, 0x28, 0x8a, 0x03, 0x00, 0x00, 0x40, 0x00, // 48 | |||
0x02, 0x00, 0x41, 0xca, 0x9b, 0x06, 0x20, 0x80, | |||
0x91, 0x00, 0x00, 0x00, 0x00, 0x20, 0x08, 0x00, // 64 | |||
0x08, 0x20, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, | |||
0x00, 0x00, 0x22, 0x00, 0x01, 0x00, }; | |||
int Translator_English::Unpronouncable(char *word) | |||
{//=============================================== | |||
/* Determines whether a word in 'unpronouncable', i.e. whether it should | |||
be spoken as individual letters. | |||
This function is language specific. | |||
*/ | |||
int c; | |||
int vowel_posn=9; | |||
int index; | |||
int count; | |||
int ix; | |||
int apostrophe=0; | |||
// words which we pass through to the dictionary, even though they look unpronouncable | |||
static const char *exceptions[] = { | |||
"'s ", "st ","nd ","rd ","th ",NULL }; | |||
if((*word == ' ') || (*word == 0)) | |||
return(0); | |||
for(ix=0; exceptions[ix] != NULL; ix++) | |||
{ | |||
// Seemingly uncpronouncable words, but to be looked in the dictionary rules instead | |||
if(memcmp(word,exceptions[ix],3)==0) | |||
return(0); | |||
} | |||
index=0; | |||
count=0; | |||
for(;;) | |||
{ | |||
index += utf8_in(&c,&word[index],0); | |||
count++; | |||
if((c==0) || (c==' ')) | |||
break; | |||
if(IsVowel(c) || (c == 'y')) | |||
{ | |||
vowel_posn = count; | |||
break; | |||
} | |||
if(c == '\'') | |||
apostrophe = 1; | |||
else | |||
if(!IsAlpha(c)) | |||
return(0); // letter (not vowel) outside Latin character range or apostrophe, abort test | |||
} | |||
if((vowel_posn > 5) || ((word[0]!='s') && (vowel_posn > 4))) | |||
return(1); // no vowel, or no vowel in first four letters | |||
/* there is at least one vowel, is the initial letter combination valid ? */ | |||
if(vowel_posn < 3) | |||
return(0); /* vowel in first two letters, OK */ | |||
if(apostrophe) | |||
return(0); // first two letters not a-z, abort test | |||
index = (word[0]-'a') * 26 + (word[1]-'a'); | |||
if(initials_bitmap[index >> 3] & (1L << (index & 7))) | |||
return(0); | |||
else | |||
return(1); /****/ | |||
} /* end of Unpronounceable */ | |||
@@ -33,7 +33,6 @@ | |||
#include "phoneme.h" | |||
#include "synthesize.h" | |||
#include "translate.h" | |||
#include "tr_languages.h" | |||
@@ -53,6 +52,123 @@ | |||
#define OFFSET_MALAYALAM 0xd00 | |||
#define OFFSET_KOREAN 0x1100 | |||
static void Translator_Russian(Translator *tr); | |||
static void SetLetterVowel(Translator *tr, int c) | |||
{//============================================== | |||
tr->letter_bits[c] = (tr->letter_bits[c] & 0x40) | 0x81; // keep value for group 6 (front vowels e,i,y) | |||
} | |||
static void ResetLetterBits(Translator *tr, int groups) | |||
{//==================================================== | |||
// Clear all the specified groups | |||
unsigned int ix; | |||
unsigned int mask; | |||
mask = ~groups; | |||
for(ix=0; ix<sizeof(tr->letter_bits); ix++) | |||
{ | |||
tr->letter_bits[ix] &= mask; | |||
} | |||
} | |||
static void SetLetterBits(Translator *tr, int group, const char *string) | |||
{//===================================================================== | |||
int bits; | |||
unsigned char c; | |||
bits = (1L << group); | |||
while((c = *string++) != 0) | |||
tr->letter_bits[c] |= bits; | |||
} | |||
static void SetLetterBitsRange(Translator *tr, int group, int first, int last) | |||
{//=========================================================================== | |||
int bits; | |||
int ix; | |||
bits = (1L << group); | |||
for(ix=first; ix<=last; ix++) | |||
{ | |||
tr->letter_bits[ix] |= bits; | |||
} | |||
} | |||
static Translator* NewTranslator(void) | |||
{//=================================== | |||
Translator *tr; | |||
int ix; | |||
static const unsigned char stress_amps2[] = {17,17, 20,20, 20,22, 22,20 }; | |||
static const short stress_lengths2[8] = {182,140, 220,220, 220,240, 260,280}; | |||
static const wchar_t empty_wstring[1] = {0}; | |||
static const wchar_t punct_in_word[2] = {'\'', 0}; // allow hyphen within words | |||
tr = (Translator *)Alloc(sizeof(Translator)); | |||
if(tr == NULL) | |||
return(NULL); | |||
tr->charset_a0 = charsets[1]; // ISO-8859-1, this is for when the input is not utf8 | |||
dictionary_name[0] = 0; | |||
tr->dict_condition=0; | |||
tr->data_dictrules = NULL; // language_1 translation rules file | |||
tr->data_dictlist = NULL; // language_2 dictionary lookup file | |||
tr->transpose_offset = 0; | |||
// only need lower case | |||
tr->letter_bits_offset = 0; | |||
memset(tr->letter_bits,0,sizeof(tr->letter_bits)); | |||
memset(tr->letter_groups,0,sizeof(tr->letter_groups)); | |||
// 0-5 sets of characters matched by A B C E F G in pronunciation rules | |||
// these may be set differently for different languages | |||
SetLetterBits(tr,0,"aeiou"); // A vowels, except y | |||
SetLetterBits(tr,1,"bcdfgjklmnpqstvxz"); // B hard consonants, excluding h,r,w | |||
SetLetterBits(tr,2,"bcdfghjklmnpqrstvwxz"); // C all consonants | |||
SetLetterBits(tr,3,"hlmnr"); // H 'soft' consonants | |||
SetLetterBits(tr,4,"cfhkpqstx"); // F voiceless consonants | |||
SetLetterBits(tr,5,"bdgjlmnrvwyz"); // G voiced | |||
SetLetterBits(tr,6,"eiy"); // Letter group Y, front vowels | |||
SetLetterBits(tr,7,"aeiouy"); // vowels, including y | |||
tr->char_plus_apostrophe = empty_wstring; | |||
tr->punct_within_word = punct_in_word; | |||
for(ix=0; ix<8; ix++) | |||
{ | |||
tr->stress_amps[ix] = stress_amps2[ix]; | |||
tr->stress_amps_r[ix] = stress_amps2[ix] - 1; | |||
tr->stress_lengths[ix] = stress_lengths2[ix]; | |||
} | |||
memset(&(tr->langopts),0,sizeof(tr->langopts)); | |||
tr->langopts.stress_rule = 2; | |||
tr->langopts.unstressed_wd1 = 1; | |||
tr->langopts.unstressed_wd2 = 3; | |||
tr->langopts.param[LOPT_SONORANT_MIN] = 95; | |||
tr->langopts.param[LOPT_MAXAMP_EOC] = 19; | |||
tr->langopts.param[LOPT_UNPRONOUNCABLE] = 's'; // don't count this character at start of word | |||
tr->langopts.max_initial_consonants = 3; | |||
tr->langopts.replace_chars = NULL; | |||
SetLengthMods(tr,201); | |||
// tr->langopts.length_mods = length_mods_en; | |||
// tr->langopts.length_mods0 = length_mods_en0; | |||
tr->langopts.long_stop = 100; | |||
tr->langopts.max_roman = 49; | |||
tr->langopts.thousands_sep = ','; | |||
tr->langopts.decimal_sep = '.'; | |||
memcpy(tr->punct_to_tone, punctuation_to_tone, sizeof(tr->punct_to_tone)); | |||
return(tr); | |||
} | |||
static const unsigned int replace_cyrillic_latin[] = | |||
{0x430,'a', | |||
@@ -130,13 +246,13 @@ Translator *SelectTranslator(const char *name) | |||
while(*name != 0) | |||
name2 = (name2 << 8) + *name++; | |||
tr = NewTranslator(); | |||
switch(name2) | |||
{ | |||
case L('a','f'): | |||
{ | |||
static const short stress_lengths_af[8] = {170,140, 220,220, 0, 0, 250,270}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_af,NULL); | |||
tr->langopts.stress_rule = 0; | |||
@@ -155,7 +271,6 @@ Translator *SelectTranslator(const char *name) | |||
static const short stress_lengths_bn[8] = {180, 180, 210, 210, 0, 0, 230, 240}; | |||
static const unsigned char stress_amps_bn[8] = {18,18, 18,18, 20,20, 22,22 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_bn,stress_amps_bn); | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -175,7 +290,6 @@ Translator *SelectTranslator(const char *name) | |||
static const short stress_lengths_cy[8] = {170,220, 180,180, 0, 0, 250,270}; | |||
static const unsigned char stress_amps_cy[8] = {17,15, 18,18, 0,0, 22,20 }; // 'diminished' is used to mark a quieter, final unstressed syllable | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_cy,stress_amps_cy); | |||
tr->charset_a0 = charsets[14]; // ISO-8859-14 | |||
@@ -199,7 +313,6 @@ Translator *SelectTranslator(const char *name) | |||
case L('d','a'): // Danish | |||
{ | |||
static const short stress_lengths_da[8] = {160,140, 200,200, 0,0, 220,210}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_da,NULL); | |||
tr->langopts.stress_rule = 0; | |||
@@ -212,7 +325,6 @@ Translator *SelectTranslator(const char *name) | |||
case L('d','e'): | |||
{ | |||
static const short stress_lengths_de[8] = {150,130, 190,190, 0, 0, 260,275}; | |||
tr = new Translator(); | |||
tr->langopts.stress_rule = 0; | |||
tr->langopts.word_gap = 0x8; // don't use linking phonemes | |||
tr->langopts.vowel_pause = 0x30; | |||
@@ -225,7 +337,14 @@ Translator *SelectTranslator(const char *name) | |||
break; | |||
case L('e','n'): | |||
tr = new Translator_English(); | |||
{ | |||
static const short stress_lengths_en[8] = {182,140, 220,220, 0,0, 248,275}; | |||
SetupTranslator(tr,stress_lengths_en,NULL); | |||
tr->langopts.stress_rule = 0; | |||
tr->langopts.numbers = 0x841 + NUM_ROMAN; | |||
tr->langopts.param[LOPT_COMBINE_WORDS] = 2; // allow "mc" to cmbine with the following word | |||
} | |||
break; | |||
case L('e','l'): // Greek | |||
@@ -241,7 +360,6 @@ Translator *SelectTranslator(const char *name) | |||
static const char el_consonants[]={0x32,0x33,0x34,0x36,0x38,0x3a,0x3b,0x3c,0x3d,0x3e,0x40,0x41,0x42,0x43,0x44,0x46,0x47,0x48,0}; | |||
static const wchar_t el_char_apostrophe[] = {0x3c3,0}; // σ | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_el,stress_amps_el); | |||
tr->charset_a0 = charsets[7]; // ISO-8859-7 | |||
@@ -278,7 +396,6 @@ Translator *SelectTranslator(const char *name) | |||
static const unsigned char stress_amps_eo[] = {16,14, 20,20, 20,22, 22,21 }; | |||
static const wchar_t eo_char_apostrophe[2] = {'l',0}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_eo,stress_amps_eo); | |||
tr->charset_a0 = charsets[3]; // ISO-8859-3 | |||
@@ -303,7 +420,6 @@ Translator *SelectTranslator(const char *name) | |||
static const unsigned char stress_amps_es[8] = {16,12, 18,18, 20,20, 20,20 }; // 'diminished' is used to mark a quieter, final unstressed syllable | |||
static const wchar_t ca_punct_within_word[] = {'\'',0xb7,0}; // ca: allow middle-dot within word | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_es,stress_amps_es); | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -332,7 +448,6 @@ Translator *SelectTranslator(const char *name) | |||
static const unsigned char stress_amps_fi[8] = {18,16, 22,22, 20,22, 22,22 }; | |||
static const short stress_lengths_fi[8] = {150,180, 200,200, 0,0, 210,250}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_fi,stress_amps_fi); | |||
tr->langopts.stress_rule = 0; | |||
@@ -353,7 +468,6 @@ Translator *SelectTranslator(const char *name) | |||
static const short stress_lengths_fr[8] = {190, 170, 190, 200, 0, 0, 235, 240}; | |||
static const unsigned char stress_amps_fr[8] = {18,16, 20,20, 20,22, 22,21 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_fr,stress_amps_fr); | |||
tr->langopts.stress_rule = 3; // stress on final syllable | |||
tr->langopts.stress_flags = 0x0024; // don't use secondary stress | |||
@@ -367,8 +481,6 @@ Translator *SelectTranslator(const char *name) | |||
#ifdef deleted | |||
case L('g','a'): // Irish Gaelic | |||
{ | |||
tr = new Translator(); | |||
tr->langopts.stress_rule = 1; | |||
} | |||
break; | |||
@@ -380,7 +492,6 @@ Translator *SelectTranslator(const char *name) | |||
static const short stress_lengths_hi[8] = {190, 190, 210, 210, 0, 0, 230, 250}; | |||
static const unsigned char stress_amps_hi[8] = {17,14, 20,19, 20,22, 22,21 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_hi,stress_amps_hi); | |||
tr->charset_a0 = charsets[19]; // ISCII | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -403,8 +514,6 @@ Translator *SelectTranslator(const char *name) | |||
static const short stress_lengths_hr[8] = {180,160, 200,200, 0,0, 220,230}; | |||
static const short stress_lengths_sr[8] = {160,150, 200,200, 0,0, 250,260}; | |||
tr = new Translator(); | |||
if(name2 == L('s','r')) | |||
SetupTranslator(tr,stress_lengths_sr,stress_amps_hr); | |||
else | |||
@@ -433,7 +542,6 @@ Translator *SelectTranslator(const char *name) | |||
static const unsigned char stress_amps_hu[8] = {17,17, 19,19, 20,22, 22,21 }; | |||
static const short stress_lengths_hu[8] = {185,195, 195,190, 0,0, 210,220}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_hu,stress_amps_hu); | |||
tr->charset_a0 = charsets[2]; // ISO-8859-2 | |||
@@ -458,7 +566,7 @@ SetLengthMods(tr,3); // all equal | |||
static const char hy_vowels[] = {0x31, 0x35, 0x37, 0x38, 0x3b, 0x48, 0x55, 0}; | |||
static const char hy_consonants[] = {0x32,0x33,0x34,0x36,0x39,0x3a,0x3c,0x3d,0x3e,0x3f, | |||
0x40,0x41,0x42,0x43,0x44,0x45,0x46,0x47,0x49,0x4a,0x4b,0x4c,0x4d,0x4e,0x4f,0x50,0x51,0x52,0x53,0x54,0x56,0}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_hy,NULL); | |||
tr->langopts.stress_rule = 3; // default stress on final syllable | |||
@@ -475,7 +583,7 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const short stress_lengths_id[8] = {160, 200, 180, 180, 0, 0, 220, 240}; | |||
static const unsigned char stress_amps_id[8] = {16,18, 18,18, 20,22, 22,21 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_id,stress_amps_id); | |||
tr->langopts.stress_rule = 2; | |||
tr->langopts.numbers = 0x1009 + NUM_ROMAN; | |||
@@ -489,7 +597,6 @@ SetLengthMods(tr,3); // all equal | |||
static const short stress_lengths_is[8] = {180,160, 200,200, 0,0, 240,250}; | |||
static const wchar_t is_lettergroup_B[] = {'c','f','h','k','p','t','x',0xfe,0}; // voiceless conants, including 'þ' ?? 's' | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_is,NULL); | |||
tr->langopts.stress_rule = 0; | |||
tr->langopts.stress_flags = 0x10; | |||
@@ -511,7 +618,6 @@ SetLengthMods(tr,3); // all equal | |||
static const short stress_lengths_it[8] = {150, 140, 170, 170, 0, 0, 300, 330}; | |||
static const unsigned char stress_amps_it[8] = {15,14, 19,19, 20,22, 22,20 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_it,stress_amps_it); | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -528,16 +634,11 @@ SetLengthMods(tr,3); // all equal | |||
} | |||
break; | |||
case L('j','a'): // TEST | |||
tr = new Translator(); | |||
tr->langopts.param[LOPT_UNPRONOUNCABLE] = 1; // disable check for unpronouncable words | |||
break; | |||
case L_jbo: // Lojban | |||
{ | |||
static const short stress_lengths_jbo[8] = {145,145, 170,160, 0,0, 330,350}; | |||
static const wchar_t jbo_punct_within_word[] = {'.',',','\'',0x2c8,0}; // allow period and comma within a word, also stress marker (from LOPT_SYLLABLE_CAPS) | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_jbo,NULL); | |||
tr->langopts.stress_rule = 2; | |||
tr->langopts.vowel_pause = 0x20c; // pause before a word which starts with a vowel, or after a word which ends in a consonant | |||
@@ -552,7 +653,6 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const char ko_ivowels[] = {0x63,0x64,0x67,0x68,0x6d,0x72,0x74,0x75,0}; // y and i vowels | |||
static const char ko_voiced[] = {0x02,0x05,0x06,0xab,0xaf,0xb7,0xbc,0}; // voiced consonants, l,m,n,N | |||
tr = new Translator(); | |||
tr->letter_bits_offset = OFFSET_KOREAN; | |||
memset(tr->letter_bits,0,sizeof(tr->letter_bits)); | |||
@@ -571,7 +671,6 @@ SetLengthMods(tr,3); // all equal | |||
static const unsigned char stress_amps_ku[8] = {18,18, 20,20, 20,22, 22,21 }; | |||
static const short stress_lengths_ku[8] = {180,180, 190,180, 0,0, 230,240}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_ku,stress_amps_ku); | |||
tr->charset_a0 = charsets[9]; // ISO-8859-9 - Latin5 | |||
@@ -584,7 +683,6 @@ SetLengthMods(tr,3); // all equal | |||
case L('l','a'): //Latin | |||
{ | |||
tr = new Translator(); | |||
tr->charset_a0 = charsets[4]; // ISO-8859-4, includes a,e,i,o,u-macron | |||
tr->langopts.stress_rule = 2; | |||
tr->langopts.stress_flags = 0x20; | |||
@@ -600,7 +698,7 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const unsigned char stress_amps_lv[8] = {17,13, 20,20, 20,22, 22,21 }; | |||
static const short stress_lengths_lv[8] = {180,130, 210,210, 0,0, 210,210}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_lv,stress_amps_lv); | |||
tr->langopts.stress_rule = 0; | |||
@@ -618,7 +716,6 @@ SetLengthMods(tr,3); // all equal | |||
static const unsigned char stress_amps_mk[8] = {17,17, 20,20, 20,22, 22,21 }; | |||
static const short stress_lengths_mk[8] = {180,160, 200,200, 0,0, 220,230}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_mk,stress_amps_mk); | |||
tr->charset_a0 = charsets[5]; // ISO-8859-5 | |||
tr->letter_groups[0] = vowels_cyrillic; | |||
@@ -633,7 +730,6 @@ SetLengthMods(tr,3); // all equal | |||
case L('n','l'): // Dutch | |||
{ | |||
static const short stress_lengths_nl[8] = {160,135, 210,210, 0, 0, 260,280}; | |||
tr = new Translator(); | |||
tr->langopts.stress_rule = 0; | |||
tr->langopts.vowel_pause = 1; | |||
@@ -649,9 +745,8 @@ SetLengthMods(tr,3); // all equal | |||
case L('n','o'): // Norwegian | |||
{ | |||
static const short stress_lengths_no[8] = {160,140, 200,200, 0,0, 220,210}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_no,NULL); | |||
SetupTranslator(tr,stress_lengths_no,NULL); | |||
tr->langopts.stress_rule = 0; | |||
SetLetterVowel(tr,'y'); | |||
tr->langopts.numbers = 0x11849; | |||
@@ -662,9 +757,8 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const unsigned char stress_amps_om[] = {18,15, 20,20, 20,22, 22,22 }; | |||
static const short stress_lengths_om[8] = {200,200, 200,200, 0,0, 200,200}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_om,stress_amps_om); | |||
SetupTranslator(tr,stress_lengths_om,stress_amps_om); | |||
tr->langopts.stress_rule = 2; | |||
tr->langopts.stress_flags = 0x16 + 0x80000; | |||
} | |||
@@ -675,7 +769,6 @@ SetLengthMods(tr,3); // all equal | |||
static const short stress_lengths_pl[8] = {160, 190, 175, 175, 0, 0, 200, 210}; | |||
static const unsigned char stress_amps_pl[8] = {17,13, 19,19, 20,22, 22,21 }; // 'diminished' is used to mark a quieter, final unstressed syllable | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_pl,stress_amps_pl); | |||
tr->charset_a0 = charsets[2]; // ISO-8859-2 | |||
@@ -694,7 +787,7 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const short stress_lengths_pt[8] = {180, 125, 210, 210, 0, 0, 270, 295}; | |||
static const unsigned char stress_amps_pt[8] = {16,13, 19,19, 20,22, 22,21 }; // 'diminished' is used to mark a quieter, final unstressed syllable | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_pt,stress_amps_pt); | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -712,7 +805,6 @@ SetLengthMods(tr,3); // all equal | |||
static const short stress_lengths_ro[8] = {170, 170, 180, 180, 0, 0, 240, 260}; | |||
static const unsigned char stress_amps_ro[8] = {15,13, 18,18, 20,22, 22,21 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_ro,stress_amps_ro); | |||
tr->langopts.stress_rule = 2; | |||
@@ -725,12 +817,11 @@ SetLengthMods(tr,3); // all equal | |||
break; | |||
case L('r','u'): // Russian | |||
tr = new Translator_Russian(); | |||
Translator_Russian(tr); | |||
break; | |||
case L('r','w'): // Kiryarwanda | |||
{ | |||
tr = new Translator(); | |||
tr->langopts.stress_rule = 2; | |||
tr->langopts.stress_flags = 0x16; | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -745,7 +836,6 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const char *sk_voiced = "bdgjlmnrvwzaeiouy"; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_sk,stress_amps_sk); | |||
tr->charset_a0 = charsets[2]; // ISO-8859-2 | |||
@@ -777,7 +867,7 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const short stress_lengths_sq[8] = {150, 150, 180, 180, 0, 0, 300, 300}; | |||
static const unsigned char stress_amps_sq[8] = {16,12, 16,16, 20,20, 21,19 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_sq,stress_amps_sq); | |||
tr->langopts.stress_rule = 2; | |||
@@ -792,12 +882,10 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const unsigned char stress_amps_sv[] = {16,16, 20,20, 20,22, 22,21 }; | |||
static const short stress_lengths_sv[8] = {160,135, 220,220, 0,0, 250,280}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_sv,stress_amps_sv); | |||
tr->langopts.stress_rule = 0; | |||
SetLetterVowel(tr,'y'); | |||
// SetLetterBits(tr,6,"eiyäö"); // soft vowels NOTE accented letters don't work in SetLetterBits | |||
tr->langopts.numbers = 0x1909; | |||
tr->langopts.accents = 1; | |||
} | |||
@@ -807,8 +895,7 @@ SetLengthMods(tr,3); // all equal | |||
{ | |||
static const short stress_lengths_sw[8] = {160, 170, 200, 200, 0, 0, 320, 340}; | |||
static const unsigned char stress_amps_sw[] = {16,12, 19,19, 20,22, 22,21 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_sw,stress_amps_sw); | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -829,7 +916,6 @@ SetLengthMods(tr,3); // all equal | |||
static const short stress_lengths_ta[8] = {200, 200, 210, 210, 0, 0, 230, 230}; | |||
static const unsigned char stress_amps_ta[8] = {18,18, 18,18, 20,20, 22,22 }; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_ta,stress_amps_ta); | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -864,7 +950,6 @@ SetLengthMods(tr,3); // all equal | |||
static const short stress_lengths_th[8] = {230,150, 230,230, 230,0, 230,250}; | |||
static const unsigned char stress_amps_th[] = {22,16, 22,22, 22,22, 22,22 }; | |||
tr = new Translator; | |||
SetupTranslator(tr,stress_lengths_th,stress_amps_th); | |||
tr->langopts.stress_rule = 0; // stress on final syllable of a "word" | |||
@@ -882,7 +967,6 @@ SetLengthMods(tr,3); // all equal | |||
static const unsigned char stress_amps_tr[8] = {18,18, 20,20, 20,22, 22,21 }; | |||
static const short stress_lengths_tr[8] = {190,190, 190,190, 0,0, 250,270}; | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_tr,stress_amps_tr); | |||
tr->charset_a0 = charsets[9]; // ISO-8859-9 - Latin5 | |||
@@ -912,7 +996,6 @@ SetLengthMods(tr,3); // all equal | |||
0x1b0, 0x1eeb, 0x1ee9, 0x1eed, 0x1eef, 0x1ef1, // ư | |||
0x79, 0x1ef3, 0xfd, 0x1ef7, 0x1ef9, 0x1e, 0 }; // y | |||
tr = new Translator(); | |||
SetupTranslator(tr,stress_lengths_vi,stress_amps_vi); | |||
tr->langopts.length_mods0 = tr->langopts.length_mods; // don't lengthen vowels in the last syllable | |||
@@ -933,7 +1016,6 @@ SetLengthMods(tr,3); // all equal | |||
static const short stress_lengths_zh[8] = {230,150, 230,230, 230,0, 240,250}; // 1=tone5. end-of-sentence, 6=tone 1&4, 7=tone 2&3 | |||
static const unsigned char stress_amps_zh[] = {22,16, 22,22, 22,22, 22,22 }; | |||
tr = new Translator; | |||
SetupTranslator(tr,stress_lengths_zh,stress_amps_zh); | |||
tr->langopts.stress_rule = 3; // stress on final syllable of a "word" | |||
@@ -953,7 +1035,6 @@ SetLengthMods(tr,3); // all equal | |||
break; | |||
default: | |||
tr = new Translator(); | |||
break; | |||
} | |||
@@ -978,8 +1059,8 @@ SetLengthMods(tr,3); // all equal | |||
Translator_Russian::Translator_Russian() : Translator() | |||
{//=================================== | |||
static void Translator_Russian(Translator *tr) | |||
{//=========================================== | |||
static const unsigned char stress_amps_ru[] = {16,16, 18,18, 20,24, 24,22 }; | |||
static const short stress_lengths_ru[8] = {150,140, 220,220, 0,0, 260,280}; | |||
@@ -993,40 +1074,38 @@ Translator_Russian::Translator_Russian() : Translator() | |||
static const char ru_voiced[] = {0x11,0x12,0x13,0x14,0x16,0x17,0}; // letter group G (voiced obstruents) | |||
static const char ru_ivowels[] = {0x2c,0x15,0x31,0x18,0x2e,0x2f,0}; // letter group Y (iotated vowels & soft-sign) | |||
SetupTranslator(this,stress_lengths_ru,stress_amps_ru); | |||
charset_a0 = charsets[18]; // KOI8-R | |||
transpose_offset = 0x42f; // convert cyrillic from unicode into range 0x01 to 0x22 | |||
transpose_min = 0x430; | |||
transpose_max = 0x451; | |||
letter_bits_offset = OFFSET_CYRILLIC; | |||
memset(letter_bits,0,sizeof(letter_bits)); | |||
SetLetterBits(this,0,ru_vowels); | |||
SetLetterBits(this,1,ru_soft); | |||
SetLetterBits(this,2,ru_consonants); | |||
SetLetterBits(this,3,ru_hard); | |||
SetLetterBits(this,4,ru_nothard); | |||
SetLetterBits(this,5,ru_voiced); | |||
SetLetterBits(this,6,ru_ivowels); | |||
SetLetterBits(this,7,ru_vowels); | |||
langopts.param[LOPT_UNPRONOUNCABLE] = 0x432; // [v] don't count this character at start of word | |||
langopts.param[LOPT_REGRESSIVE_VOICING] = 1; | |||
langopts.param[LOPT_REDUCE] = 2; | |||
langopts.stress_rule = 5; | |||
langopts.stress_flags = 0x0020; // waas 0x1010 | |||
langopts.numbers = 0x0409; | |||
langopts.numbers2 = 0xc2; // variant numbers before thousands | |||
langopts.phoneme_change = 1; | |||
langopts.testing = 2; | |||
SetupTranslator(tr,stress_lengths_ru,stress_amps_ru); | |||
tr->charset_a0 = charsets[18]; // KOI8-R | |||
tr->transpose_offset = 0x42f; // convert cyrillic from unicode into range 0x01 to 0x22 | |||
tr->transpose_min = 0x430; | |||
tr->transpose_max = 0x451; | |||
tr->letter_bits_offset = OFFSET_CYRILLIC; | |||
memset(tr->letter_bits,0,sizeof(tr->letter_bits)); | |||
SetLetterBits(tr,0,ru_vowels); | |||
SetLetterBits(tr,1,ru_soft); | |||
SetLetterBits(tr,2,ru_consonants); | |||
SetLetterBits(tr,3,ru_hard); | |||
SetLetterBits(tr,4,ru_nothard); | |||
SetLetterBits(tr,5,ru_voiced); | |||
SetLetterBits(tr,6,ru_ivowels); | |||
SetLetterBits(tr,7,ru_vowels); | |||
tr->langopts.param[LOPT_UNPRONOUNCABLE] = 0x432; // [v] don't count this character at start of word | |||
tr->langopts.param[LOPT_REGRESSIVE_VOICING] = 1; | |||
tr->langopts.param[LOPT_REDUCE] = 2; | |||
tr->langopts.stress_rule = 5; | |||
tr->langopts.stress_flags = 0x0020; // waas 0x1010 | |||
tr->langopts.numbers = 0x0409; | |||
tr->langopts.numbers2 = 0xc2; // variant numbers before thousands | |||
tr->langopts.phoneme_change = 1; | |||
tr->langopts.testing = 2; | |||
} // end of Translator_Russian | |||
#define PH(c1,c2) (c2<<8)+c1 // combine two characters into an integer for phoneme name | |||
#define PY(c1,c2,c3) (c3<<16)+(c2<<8)+c1 | |||
/* | |||
typedef struct { | |||
@@ -1043,8 +1122,8 @@ typedef struct { | |||
#define RUSSIAN2 | |||
#ifdef RUSSIAN2 | |||
int Translator_Russian::ChangePhonemes(PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch) | |||
{//============================================================================================================== | |||
int ChangePhonemes_ru(Translator *tr, PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch) | |||
{//============================================================================================================= | |||
// Called for each phoneme in the phoneme list, to allow a language to make changes | |||
// ph The current phoneme | |||
@@ -1136,8 +1215,8 @@ PH('V','#'),PH('I','3'),PH('I','2'),PH('E','3')}; | |||
#else | |||
int Translator_Russian::ChangePhonemes(PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch) | |||
{//=============================================================================================================== | |||
int ChangePhonemes_ru(Translator *tr, PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch) | |||
{//============================================================================================================= | |||
// Called for each phoneme in the phoneme list, to allow a language to make changes | |||
// flags: bit 0=1 last phoneme in a word | |||
// bit 1=1 this is the highest stressed vowel in the current word |
@@ -18,28 +18,3 @@ | |||
***************************************************************************/ | |||
class Translator_English: public Translator | |||
{//======================================= | |||
public: | |||
Translator_English(); | |||
int Unpronouncable(char *word); | |||
}; // end of class Translator_English | |||
class Translator_Russian: public Translator | |||
{//======================================= | |||
public: | |||
Translator_Russian(); | |||
private: | |||
int ChangePhonemes(PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch); | |||
}; // end of class Translator_Russian | |||
@@ -397,18 +397,9 @@ typedef struct { | |||
#define NUM_SEP_SPACE 0x1000 // allow space as thousands separator (in addition to langopts.thousands_sep) | |||
#define NUM_DEC_IT 0x2000 // (LANG=it) speak post-decimal-point digits as a combined number not as single digits | |||
class Translator | |||
struct Translator | |||
{//============= | |||
public: | |||
Translator(); | |||
virtual ~Translator(); | |||
void *TranslateClause(FILE *f_text, const void *vp_input, int *tone, char **voice_change); | |||
int TranslateWord(char *word, int next_pause, WORD_TAB *wtab); | |||
int LoadDictionary(const char *name, int no_error); | |||
virtual void CalcLengths(); | |||
virtual void CalcPitches(int clause_tone); | |||
void GetTranslatedPhonemeString(char *phon_out, int n_phon_out); | |||
LANGUAGE_OPTIONS langopts; | |||
int translator_name; | |||
int transpose_offset; | |||
@@ -416,7 +407,6 @@ public: | |||
int transpose_min; | |||
char phon_out[300]; | |||
char word_phonemes[N_WORD_PHONEMES]; // a word translated into phoneme codes | |||
char phonemes_repeat[20]; | |||
int phonemes_repeat_count; | |||
@@ -441,56 +431,8 @@ public: | |||
char *data_dictrules; // language_1 translation rules file | |||
char *data_dictlist; // language_2 dictionary lookup file | |||
char *dict_hashtab[N_HASH_DICT]; // hash table to index dictionary lookup file | |||
char *letterGroups[N_LETTER_GROUPS]; | |||
private: | |||
int TranslateWord2(char *word, WORD_TAB *wtab, int pre_pause, int next_pause); | |||
int TranslateLetter(char *letter, char *phonemes, int control, int word_length); | |||
void SetSpellingStress(char *phonemes, int control, int n_chars); | |||
void WriteMnemonic(int *ix, int mnem); | |||
void MakePhonemeList(int post_pause, int new_sentence); | |||
int SubstitutePhonemes(PHONEME_LIST2 *plist_out); | |||
int ReadClause(FILE *f_in, char *buf, short *charix, int n_buf, int *tone_type); | |||
int AnnouncePunctuation(int c1, int c2, char *buf, int ix); | |||
const char *LookupDict2(const char *word, const char *word2, char *phonetic, unsigned int *flags, int end_flags, WORD_TAB *wtab); | |||
const char *LookupSpecial(const char *string, char *text_out); | |||
const char *LookupCharName(int c); | |||
void LookupLetter(unsigned int letter, int next_byte, char *ph_buf); | |||
int LookupLetter2(unsigned int letter, char *ph_buf); | |||
void LookupAccentedLetter(unsigned int letter, char *ph_buf); | |||
int LookupNum2(int value, int control, char *ph_out); | |||
int LookupNum3(int value, char *ph_out, int suppress_null, int thousandplex, int prev_thousands); | |||
int LookupThousands(int value, int thousandplex, char *ph_out); | |||
int TranslateNumber_1(char *word1, char *ph_out, unsigned int *flags, int wflags); | |||
int TranslateRoman(char *word, char *ph_out); | |||
int TranslateChar(char *ptr, int prev_in, unsigned int c, unsigned int next_in, int *insert); | |||
void InitGroups(void); | |||
void AppendPhonemes(char *string, int size, const char *ph); | |||
char *DecodeRule(const char *group, char *rule); | |||
void MatchRule(char *word[], const char *group, char *rule, MatchRecord *match_out, int end_flags, int dict_flags); | |||
int TranslateRules(char *p, char *phonemes, int size, char *end_phonemes, int end_flags, unsigned int *dict_flags); | |||
void ApplySpecialAttribute(char *phonemes, int dict_flags); | |||
int IsLetter(int letter, int group); | |||
int IsLetterGroup(char *word, int group, int pre); | |||
void CalcPitches_Tone(int clause_tone); | |||
protected: | |||
virtual int Unpronouncable(char *word); | |||
virtual void SetWordStress(char *output, unsigned int dictionary_flags, int tonic, int prev_stress); | |||
virtual int RemoveEnding(char *word, int end_type, char *word_copy); | |||
virtual int TranslateNumber(char *word1, char *ph_out, unsigned int *flags, int wflags); | |||
virtual int ChangePhonemes(PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch); | |||
int IsVowel(int letter); | |||
int LookupDictList(char **wordptr, char *ph_out, unsigned int *flags, int end_flags, WORD_TAB *wtab); | |||
int Lookup(const char *word, char *ph_out); | |||
// groups1 and groups2 are indexes into data_dictrules, set up by InitGroups() | |||
// the two-letter rules for each letter must be consecutive in the language_rules source | |||
@@ -501,22 +443,13 @@ protected: | |||
unsigned char groups2_count[256]; // number of 2 letter groups for this initial letter | |||
unsigned char groups2_start[256]; // index into groups2 | |||
char *letterGroups[N_LETTER_GROUPS]; | |||
int n_ph_list2; | |||
PHONEME_LIST2 ph_list2[N_PHONEME_LIST]; // first stage of text->phonemes | |||
int expect_verb; | |||
int expect_past; // expect past tense | |||
int expect_verb_s; | |||
int expect_noun; | |||
int word_flags; // word is all upper case | |||
int prev_last_stress; | |||
int prepause_timeout; | |||
int end_stressed_vowel; // word ends with stressed vowel | |||
int prev_dict_flags; // dictionary flags from previous word | |||
char *clause_end; | |||
int word_vowel_count; // number of vowels so far | |||
@@ -525,10 +458,12 @@ protected: | |||
int clause_upper_count; // number of upper case letters in the clause | |||
int clause_lower_count; // number of lower case letters in the clause | |||
int prepause_timeout; | |||
int end_stressed_vowel; // word ends with stressed vowel | |||
int prev_dict_flags; // dictionary flags from previous word | |||
}; // end of class Translator | |||
extern int option_tone1; | |||
extern int option_tone2; | |||
#define OPTION_EMPHASIZE_ALLCAPS 0x100 | |||
#define OPTION_EMPHASIZE_PENULTIMATE 0x200 | |||
@@ -538,7 +473,6 @@ extern int option_quiet; | |||
extern int option_phonemes; | |||
extern int option_phoneme_events; | |||
extern int option_linelength; // treat lines shorter than this as end-of-clause | |||
extern int option_harmonic1; | |||
extern int option_multibyte; | |||
extern int option_capitals; | |||
extern int option_punctuation; | |||
@@ -569,9 +503,6 @@ extern char skip_marker[N_MARKER_LENGTH]; | |||
#define N_PUNCTLIST 60 | |||
extern wchar_t option_punctlist[N_PUNCTLIST]; // which punctuation characters to announce | |||
extern unsigned char punctuation_to_tone[INTONATION_TYPES][PUNCT_INTONATIONS]; | |||
extern const unsigned short punct_chars[]; // punctuation chars fo end-of-clause | |||
extern int speech_parameters[]; | |||
extern Translator *translator; | |||
extern Translator *translator2; | |||
@@ -580,31 +511,23 @@ extern char dictionary_name[40]; | |||
extern char ctrl_embedded; // to allow an alternative CTRL for embedded commands | |||
extern char *p_textinput; | |||
extern wchar_t *p_wchar_input; | |||
extern int ungot_char; | |||
extern int dictionary_skipwords; | |||
extern int (* uri_callback)(int, const char *, const char *); | |||
extern int (* phoneme_callback)(const char *); | |||
extern void SetLengthMods(Translator *tr, int value); | |||
Translator *SelectTranslator(const char *name); | |||
int SetTranslator2(const char *name); | |||
int CompileDictionary(const char *dsource, const char *dict_name, FILE *log, char *err_name,int flags); | |||
void LoadConfig(void); | |||
int PhonemeCode(unsigned int mnem); | |||
void ChangeWordStress(Translator *tr, char *word, int new_stress); | |||
int TransposeAlphabet(char *text, int offset, int min, int max); | |||
int utf8_in(int *c, const char *buf, int backwards); | |||
int utf8_in(int *c, const char *buf); | |||
int utf8_in2(int *c, const char *buf, int backwards); | |||
int utf8_out(unsigned int c, char *buf); | |||
int utf8_nbytes(const char *buf); | |||
int lookupwchar(const unsigned short *list,int c); | |||
int Eof(void); | |||
char *strchr_w(const char *s, int c); | |||
int IsBracket(int c); | |||
void ResetLetterBits(Translator *tr, int groups); | |||
void SetLetterBits(Translator *tr, int group, const char *string); | |||
void SetLetterBitsRange(Translator *tr, int group, int first, int last); | |||
void SetLetterVowel(Translator *tr, int c); | |||
void InitNamedata(void); | |||
void InitText(int flags); | |||
void InitText2(void); | |||
@@ -612,6 +535,41 @@ int IsDigit(unsigned int c); | |||
int IsAlpha(unsigned int c); | |||
int isspace2(unsigned int c); | |||
int towlower2(unsigned int c); | |||
void GetTranslatedPhonemeString(char *phon_out, int n_phon_out); | |||
Translator *SelectTranslator(const char *name); | |||
int SetTranslator2(const char *name); | |||
void DeleteTranslator(Translator *tr); | |||
int Lookup(Translator *tr, const char *word, char *ph_out); | |||
int TranslateNumber(Translator *tr, char *word1, char *ph_out, unsigned int *flags, int wflags); | |||
int TranslateRoman(Translator *tr, char *word, char *ph_out); | |||
void ChangeWordStress(Translator *tr, char *word, int new_stress); | |||
void SetSpellingStress(Translator *tr, char *phonemes, int control, int n_chars); | |||
int TranslateLetter(Translator *tr, char *letter, char *phonemes, int control, int word_length); | |||
void LookupLetter(Translator *tr, unsigned int letter, int next_byte, char *ph_buf); | |||
void LookupAccentedLetter(Translator *tr, unsigned int letter, char *ph_buf); | |||
int LoadDictionary(Translator *tr, const char *name, int no_error); | |||
int LookupDictList(Translator *tr, char **wordptr, char *ph_out, unsigned int *flags, int end_flags, WORD_TAB *wtab); | |||
void MakePhonemeList(Translator *tr, int post_pause, int new_sentence); | |||
int ChangePhonemes_ru(Translator *tr, PHONEME_LIST2 *phlist, int n_ph, int index, PHONEME_TAB *ph, CHANGEPH *ch); | |||
void ApplySpecialAttribute(Translator *tr, char *phonemes, int dict_flags); | |||
void AppendPhonemes(Translator *tr, char *string, int size, const char *ph); | |||
void CalcLengths(Translator *tr); | |||
void CalcPitches(Translator *tr, int clause_tone); | |||
int RemoveEnding(Translator *tr, char *word, int end_type, char *word_copy); | |||
int Unpronouncable(Translator *tr, char *word); | |||
void SetWordStress(Translator *tr, char *output, unsigned int dictionary_flags, int tonic, int prev_stress); | |||
int TranslateRules(Translator *tr, char *p, char *phonemes, int size, char *end_phonemes, int end_flags, unsigned int *dict_flags); | |||
int TranslateWord(Translator *tr, char *word1, int next_pause, WORD_TAB *wtab); | |||
void *TranslateClause(Translator *tr, FILE *f_text, const void *vp_input, int *tone, char **voice_change); | |||
int ReadClause(Translator *tr, FILE *f_in, char *buf, short *charix, int n_buf, int *tone_type); | |||
void SetVoiceStack(espeak_VOICE *v); | |||
extern FILE *f_trans; // for logging |
@@ -423,11 +423,11 @@ void TranslDlg::OnCommand(wxCommandEvent& event) | |||
InitText(0); | |||
while((vp != NULL) && (n_ph_list < N_PH_LIST)) | |||
{ | |||
vp = translator->TranslateClause(NULL,vp,&clause_tone,NULL); | |||
translator->CalcPitches(clause_tone); | |||
translator->CalcLengths(); | |||
vp = TranslateClause(translator,NULL,vp,&clause_tone,NULL); | |||
CalcPitches(translator,clause_tone); | |||
CalcLengths(translator); | |||
translator->GetTranslatedPhonemeString(translator->phon_out,sizeof(translator->phon_out)); | |||
GetTranslatedPhonemeString(translator->phon_out,sizeof(translator->phon_out)); | |||
if(clause_count++ > 0) | |||
strcat(phon_out," ||"); | |||
strcat(phon_out,translator->phon_out); |
@@ -164,11 +164,12 @@ const char variants_male[N_VOICE_VARIANTS] = {1,2,3,4,5,0}; | |||
const char variants_female[N_VOICE_VARIANTS] = {11,12,13,14,0}; | |||
const char *variant_lists[3] = {variants_either, variants_male, variants_female}; | |||
voice_t voicedata; | |||
static voice_t voicedata; | |||
voice_t *voice = &voicedata; | |||
char *fgets_strip(char *buf, int size, FILE *f_in) | |||
{//=============================================== | |||
static char *fgets_strip(char *buf, int size, FILE *f_in) | |||
{//====================================================== | |||
// strip trailing spaces, and truncate lines at // comment | |||
int len; | |||
char *p; | |||
@@ -187,8 +188,8 @@ char *fgets_strip(char *buf, int size, FILE *f_in) | |||
} | |||
void SetToneAdjust(voice_t *voice, int *tone_pts) | |||
{//============================================== | |||
static void SetToneAdjust(voice_t *voice, int *tone_pts) | |||
{//===================================================== | |||
int ix; | |||
int pt; | |||
int y; | |||
@@ -420,7 +421,6 @@ SetToneAdjust(voice,tone_points); | |||
if(tone_only == 0) | |||
{ | |||
n_replace_phonemes = 0; | |||
option_tone1 = 0; | |||
option_quiet = 0; | |||
LoadMbrolaTable(NULL,NULL,0); | |||
} | |||
@@ -470,11 +470,11 @@ static void PhonemeReplacement(int type, char *p) | |||
if((n < 2) || (n_replace_phonemes >= N_REPLACE_PHONEMES)) | |||
return; | |||
if((phon = LookupPh(phon_string1)) == 0) | |||
if((phon = LookupPhonemeString(phon_string1)) == 0) | |||
return; // not recognised | |||
replace_phonemes[n_replace_phonemes].old_ph = phon; | |||
replace_phonemes[n_replace_phonemes].new_ph = LookupPh(phon_string2); | |||
replace_phonemes[n_replace_phonemes].new_ph = LookupPhonemeString(phon_string2); | |||
replace_phonemes[n_replace_phonemes++].type = flags; | |||
} // end of PhonemeReplacement | |||
@@ -581,7 +581,7 @@ voice_t *LoadVoice(const char *vname, int control) | |||
if(!tone_only && (translator != NULL)) | |||
{ | |||
delete translator; | |||
DeleteTranslator(translator); | |||
translator = NULL; | |||
} | |||
@@ -672,7 +672,7 @@ voice_t *LoadVoice(const char *vname, int control) | |||
SelectPhonemeTableName(phonemes_name); | |||
if(new_translator != NULL) | |||
delete new_translator; | |||
DeleteTranslator(new_translator); | |||
new_translator = SelectTranslator(translator_name); | |||
langopts = &new_translator->langopts; | |||
@@ -704,7 +704,7 @@ voice_t *LoadVoice(const char *vname, int control) | |||
sscanf(p,"%s",translator_name); | |||
if(new_translator != NULL) | |||
delete new_translator; | |||
DeleteTranslator(new_translator); | |||
new_translator = SelectTranslator(translator_name); | |||
langopts = &new_translator->langopts; | |||
@@ -919,7 +919,7 @@ voice_t *LoadVoice(const char *vname, int control) | |||
fprintf(stderr,"Unknown phoneme table: '%s'\n",phonemes_name); | |||
} | |||
voice->phoneme_tab_ix = ix; | |||
error = new_translator->LoadDictionary(new_dictionary, control & 4); | |||
error = LoadDictionary(new_translator, new_dictionary, control & 4); | |||
if(dictionary_name[0]==0) | |||
return(NULL); // no dictionary loaded | |||
@@ -962,8 +962,8 @@ voice_t *LoadVoice(const char *vname, int control) | |||
} // end of LoadVoice | |||
char *ExtractVoiceVariantName(char *vname, int variant_num) | |||
{//======================================================== | |||
static char *ExtractVoiceVariantName(char *vname, int variant_num) | |||
{//=============================================================== | |||
// Remove any voice variant suffix (name or number) from a voice name | |||
// Returns the voice variant name | |||
@@ -1453,8 +1453,8 @@ char const *SelectVoice(espeak_VOICE *voice_select, int *found) | |||
void GetVoices(const char *path) | |||
{//============================= | |||
static void GetVoices(const char *path) | |||
{//==================================== | |||
FILE *f_voice; | |||
espeak_VOICE *voice_data; | |||
int ftype; |
@@ -43,7 +43,6 @@ extern wxString path_phsource; | |||
extern char *spects_data; | |||
extern int n_phoneme_tables; | |||
extern PHONEME_TAB_LIST phoneme_tab_list[N_PHONEME_TABS]; | |||
// size of the vowelchart png |
@@ -59,7 +59,7 @@ voice_t *wvoice; | |||
FILE *f_log = NULL; | |||
int option_waveout = 0; | |||
int option_harmonic1 = 10; // 10 | |||
static int option_harmonic1 = 10; // 10 | |||
int option_log_frames = 0; | |||
static int flutter_amp = 64; | |||
@@ -86,7 +86,7 @@ static int echo_amp = 0; | |||
static short echo_buf[N_ECHO_BUF]; | |||
static int voicing; | |||
RESONATOR rbreath[N_PEAKS]; | |||
static RESONATOR rbreath[N_PEAKS]; | |||
static int harm_sqrt_n = 0; | |||
@@ -535,7 +535,7 @@ static int WaveCallback(const void *inputBuffer, void *outputBuffer, | |||
/* This is a fixed version of Pa_OpenDefaultStream() for use if the version in portaudio V19 | |||
is broken */ | |||
PaError Pa_OpenDefaultStream2( PaStream** stream, | |||
static PaError Pa_OpenDefaultStream2( PaStream** stream, | |||
int inputChannelCount, | |||
int outputChannelCount, | |||
PaSampleFormat sampleFormat, | |||
@@ -1073,8 +1073,8 @@ void InitBreath(void) | |||
void SetBreath() | |||
{//============= | |||
static void SetBreath() | |||
{//==================== | |||
#ifndef PLATFORM_RISCOS | |||
int pk; | |||
@@ -1094,8 +1094,8 @@ void SetBreath() | |||
} // end of SetBreath | |||
int ApplyBreath(void) | |||
{//================== | |||
static int ApplyBreath(void) | |||
{//========================= | |||
int value = 0; | |||
#ifndef PLATFORM_RISCOS | |||
int noise; |