summaryrefslogtreecommitdiff
path: root/aoview/aoview_flite.c
blob: 2673824dbdcfaf0780babfd20093fa4a0ca775a6 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
/*
 * Copyright © 2009 Keith Packard <keithp@keithp.com>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; version 2 of the License.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
 */

#include <stdio.h>
#include <flite/flite.h>
#include "aoview.h"
#include <alsa/asoundlib.h>

cst_voice *register_cmu_us_kal();
static cst_voice *voice;

static FILE *pipe_write;
static GThread *aoview_flite_thread;

static snd_pcm_t	*alsa_handle;

gpointer
aoview_flite_task(gpointer data)
{
	FILE		*input = data;
	char		line[1024];
	cst_wave	*wave;
	int		rate;
	int		channels;
	int		err;

	err = snd_pcm_open(&alsa_handle, "default",
			   SND_PCM_STREAM_PLAYBACK, 0);
	if (err >= 0)
	{
		if (err < 0) {
			snd_pcm_close(alsa_handle);
			alsa_handle = 0;
		}
	}
	rate = 0;
	channels = 0;
	while (fgets(line, sizeof (line) - 1, input) != NULL) {
		if (!alsa_handle)
			continue;
		wave = flite_text_to_wave(line, voice);
		if (wave->sample_rate != rate ||
		    wave->num_channels != channels)
		{
			rate = wave->sample_rate;
			channels = wave->num_channels;
			snd_pcm_set_params(alsa_handle,
					   SND_PCM_FORMAT_S16,
					   SND_PCM_ACCESS_RW_INTERLEAVED,
					   channels,
					   rate,
					   1,
					   100000);
		}
		snd_pcm_prepare(alsa_handle);
		err = snd_pcm_writei(alsa_handle,
				     wave->samples,
				     wave->num_samples);
		if (err < 0)
			fprintf(stderr, "alsa write error %s\n",
				strerror(-err));
		snd_pcm_drain(alsa_handle);
		delete_wave(wave);
	}
	snd_pcm_close(alsa_handle);
	alsa_handle = 0;
	return NULL;
}

void
aoview_flite_stop(void)
{
	int status;
	if (pipe_write) {
		fclose(pipe_write);
		pipe_write = NULL;
	}
	if (aoview_flite_thread) {
		g_thread_join(aoview_flite_thread);
		aoview_flite_thread = NULL;
	}
}

FILE *
aoview_flite_start(void)
{
	static once;
	int	p[2];
	GError	*error;
	FILE	*pipe_read;

	if (!once) {
		flite_init();
		voice = register_cmu_us_kal();
		if (!voice) {
			perror("register voice");
			exit(1);
		}
	}
	aoview_flite_stop();
	pipe(p);
	pipe_read = fdopen(p[0], "r");
	pipe_write = fdopen(p[1], "w");
	g_thread_create(aoview_flite_task, pipe_read, TRUE, &error);
	return pipe_write;
}