#include #include #include #include #if defined(BYTE_ORDER) && BYTE_ORDER == BIG_ENDIAN #define IS_BIG_ENDIAN 1 #else #define IS_BIG_ENDIAN 0 #endif #if IS_BIG_ENDIAN # define SWAP_USHORT(val) ((unsigned short) ( \ (unsigned short) ((unsigned short) (val) >> 8) | \ (unsigned short) ((unsigned short) (val) << 8))) # define SWAP_UINT(val) ((unsigned int) ( \ (((unsigned int) (val) & (unsigned int) 0x000000ffU) << 24) | \ (((unsigned int) (val) & (unsigned int) 0x0000ff00U) << 8) | \ (((unsigned int) (val) & (unsigned int) 0x00ff0000U) >> 8) | \ (((unsigned int) (val) & (unsigned int) 0xff000000U) >> 24))) #else # define SWAP_USHORT(val) (val) # define SWAP_UINT(val) (val) #endif #define N_PHONEME_TAB_NAME 32 typedef struct { unsigned int mnemonic; unsigned int phflags; unsigned short std_length; unsigned short spect; unsigned short before; unsigned short after; unsigned char code; unsigned char type; unsigned char start_type; unsigned char end_type; unsigned char length_mod; unsigned char reduce_to; unsigned char alternative_ph; unsigned char link_out; } PHONEME_TAB; typedef struct { short frflags; unsigned char length; unsigned char rms; short ffreq[9]; unsigned char fheight[9]; unsigned char fwidth[6]; unsigned char fright[6]; } frame_t; #define N_SEQ_FRAMES 25 typedef struct { short length; unsigned char n_frames; unsigned char flags; frame_t frame[N_SEQ_FRAMES]; } SPECT_SEQ; void swap_phondata (const char *infile, const char *outfile, const char *manifest); void swap_phonindex (const char *infile, const char *outfile); void swap_phontab (const char *infile, const char *outfile); void usage (const char *program_name); int main (int argc, char *argv[]) { const char *indir = "/usr/share/espeak-data"; const char *outdir = "."; const char *manifest = "phondata-manifest"; char *f1, *f2; if (argc > 4) usage (argv[0]); if (argc > 1) { if (strcmp (argv[1], "-h") == 0 || strcmp (argv[1], "--help") == 0) usage (argv[0]); indir = argv[1]; } if (argc > 2) outdir = argv[2]; if (argc > 3) manifest = argv[3]; f1 = (char *) malloc (strlen (indir) + 20); if (f1 == NULL) { fprintf (stderr, "Unable to allocate memory\n"); exit (1); } f2 = (char *) malloc (strlen (outdir) + 20); if (f2 == NULL) { fprintf (stderr, "Unable to allocate memory\n"); exit (1); } #if IS_BIG_ENDIAN printf ("Host seems to be big-endian ..\n"); #else printf ("Host seems to be little-endian ..\n"); #endif sprintf (f1, "%s/phontab", indir); sprintf (f2, "%s/temp_1", outdir); printf ("Processing phontab ..\n"); swap_phontab (f1, f2); sprintf (f1, "%s/phontab", outdir); rename (f2, f1); sprintf (f1, "%s/phonindex", indir); sprintf (f2, "%s/temp_1", outdir); printf ("Processing phonindex ..\n"); swap_phonindex (f1, f2); sprintf (f1, "%s/phonindex", outdir); rename (f2, f1); sprintf (f1, "%s/phondata", indir); sprintf (f2, "%s/temp_1", outdir); printf ("Processing phondata ..\n"); swap_phondata (f1, f2, manifest); sprintf (f1, "%s/phondata", outdir); rename (f2, f1); free (f1); free (f2); printf ("Done.\n"); return 0; } void swap_phondata (const char *infile, const char *outfile, const char *manifest) { FILE *in, *mfest, *out; char line[1024]; unsigned char buf_4[4]; in = fopen (infile, "rb"); if (in == NULL) { fprintf (stderr, "Unable to read from file %s\n", infile); exit (1); } mfest = fopen (manifest, "rb"); if (mfest == NULL) { fprintf (stderr, "Unable to read from file %s\n", manifest); exit (1); } out = fopen (outfile, "wb"); if (out == NULL) { fprintf (stderr, "Unable to open file %s for writing\n", outfile); exit (1); } fread (buf_4, 4, 1, in); fwrite (buf_4, 4, 1, out); while (fgets (line, 1024, mfest)) { if (line[0] == 'S') { SPECT_SEQ buf_spect; size_t ix; int n; fread (&buf_spect.length, 2, 1, in); fread (&buf_spect.n_frames, 1, 1, in); fseek (in, -3, SEEK_CUR); ix = (char *)(&buf_spect.frame[buf_spect.n_frames]) - (char *)(&buf_spect); ix = (ix+3) & 0xfffc; fread (&buf_spect, ix, 1, in); buf_spect.length = (short) SWAP_USHORT (buf_spect.length); for (n = 0; n < buf_spect.n_frames; n++) { int k; buf_spect.frame[n].frflags = (short) SWAP_USHORT (buf_spect.frame[n].frflags); for (k = 0; k < 9; k++) { buf_spect.frame[n].ffreq[k] = (short) SWAP_USHORT (buf_spect.frame[n].ffreq[k]); } } fwrite (&buf_spect, ix, 1, out); } else if (line[0] == 'W') { long pos; int length; char *wave_data; fread (buf_4, 4, 1, in); fwrite (buf_4, 4, 1, out); length = buf_4[1] * 256 + buf_4[0]; wave_data = (char *) malloc (length); if (wave_data == NULL) { fprintf (stderr, "Memory allocation error\n"); exit (1); } fread (wave_data, 1, length, in); fwrite (wave_data, 1, length, out); pos = ftell (in); while((pos & 3) != 0) { fgetc (in); pos++; } pos = ftell (out); while((pos & 3) != 0) { fputc (0, out); pos++; } free (wave_data); } else if (line[0] == 'E') { char env_buf[128]; fread (env_buf, 1, 128, in); fwrite (env_buf, 1, 128, out); } } fclose (in); fclose (out); fclose (mfest); } void swap_phonindex (const char *infile, const char *outfile) { FILE *in, *out; unsigned int val; in = fopen (infile, "rb"); if (in == NULL) { fprintf (stderr, "Unable to read from file %s\n", infile); exit (1); } out = fopen (outfile, "wb"); if (out == NULL) { fprintf (stderr, "Unable to open file %s for writing\n", outfile); exit (1); } while (! feof (in)) { size_t n; n = fread (&val, 4, 1, in); if (n != 1) break; val = SWAP_UINT (val); fwrite (&val, 4, 1, out); } fclose (in); fclose (out); } void swap_phontab (const char *infile, const char *outfile) { FILE *in, *out; char buf_4[4]; int i, n_phoneme_tables; in = fopen (infile, "rb"); if (in == NULL) { fprintf (stderr, "Unable to read from file %s\n", infile); exit (1); } out = fopen (outfile, "wb"); if (out == NULL) { fprintf (stderr, "Unable to open file %s for writing\n", outfile); exit (1); } fread (buf_4, 4, 1, in); fwrite (buf_4, 4, 1, out); n_phoneme_tables = buf_4[0]; for (i = 0; i < n_phoneme_tables; i++) { int n_phonemes, j; char tab_name[N_PHONEME_TAB_NAME]; fread (buf_4, 4, 1, in); fwrite (buf_4, 4, 1, out); n_phonemes = buf_4[0]; fread (tab_name, N_PHONEME_TAB_NAME, 1, in); fwrite (tab_name, N_PHONEME_TAB_NAME, 1, out); for (j = 0; j < n_phonemes; j++) { PHONEME_TAB table; fread (&table, sizeof (PHONEME_TAB), 1, in); table.mnemonic = SWAP_UINT (table.mnemonic); table.phflags = SWAP_UINT (table.phflags); table.std_length = SWAP_USHORT (table.std_length); table.spect = SWAP_USHORT (table.spect); table.before = SWAP_USHORT (table.before); table.after = SWAP_USHORT (table.after); fwrite (&table, sizeof (PHONEME_TAB), 1, out); } } fclose (in); fclose (out); } void usage (const char *program_name) { fprintf (stderr, "This program copies the phontab, phonindex and phondata files from a given\n" "directory, swapping values to big-endian form if necessary.\n\n" "Usage:\n" " %s [INPUT_DIR] [OUTPUT_DIR] [MANIFEST_FILE]\n\n" "By default, the MANIFEST_FILE used is a file called 'phondata-manifest' in\n" "the current directory. The default INPUT_DIR is /usr/share/espeak-data and\n" "OUTPUT_DIR is the current directory.\n", program_name); exit (1); }