#include #include #include #include #include #include #include #include #include #include #include #include #include #define SYNTH_STATUS_CLEAR 0 #define SYNTH_STATUS_ON (1 << 0) typedef struct _synth_sine { int flags; float phase; size_t frequency; } synth_sine; static void usage(int argc, char **argv, const char *message, ...) { va_list args; va_start(args, message); if (message) { vfprintf(stderr, message, args); fputc('\n', stderr); } va_end(args); fprintf(stderr, "usage: %s output.wav\n", argv[0]); exit(EX_USAGE); } static int16_t sine_sample(xas_synth *synth, synth_sine *sine) { static float tau = 2.0f * M_PI; int16_t ret; if (sine->flags & SYNTH_STATUS_ON) { ret = (int16_t)roundf((INT16_MAX >> 2) * sinf(sine->phase)); sine->phase += tau / (synth->format.sample_rate / sine->frequency); if (sine->phase > tau) { sine->phase -= tau; } } else { ret = 0; } return ret; } static void sine_cleanup(xas_synth *synth, synth_sine *sine) { return; } int main(int argc, char **argv) { synth_sine sine_channels[2] = { { SYNTH_STATUS_ON, 0.0f, 220 }, { SYNTH_STATUS_ON, 0.0f, 420 }, }; xas_mixer *mixer; xas_bank *bank; xas_vox *vox; xas_audio_stream *voice, *bank_stream, *synth_l, *synth_r, *output; xas_audio_format format = { .channels = XAS_AUDIO_STEREO, .sample_size = XAS_AUDIO_PCM_16_BIT, .sample_rate = 44100 }; size_t buffer_size = 735, duration_frames = 3600, i; if (argc != 2) { usage(argc, argv, "No output file provided"); } if ((vox = xas_vox_new("/usr/bin/text2wave", format, buffer_size)) == NULL) { goto error_vox_new; } if ((bank = xas_bank_new(format, 2646000, 4)) == NULL) { goto error_bank_new; } if ((output = xas_riff_new_file(argv[1], format, O_WRONLY | O_CREAT | O_TRUNC)) == NULL) { goto error_riff_new_file; } if ((voice = xas_vox_stream_new(vox)) == NULL) { goto error_vox_stream_new; } if ((bank_stream = xas_bank_stream_new(bank)) == NULL) { goto error_bank_stream_new; } if ((synth_l = xas_synth_new((xas_synth_callback_sample)sine_sample, (xas_synth_callback_cleanup)sine_cleanup, format, buffer_size, &sine_channels[0])) == NULL) { goto error_synth_new_l; } if ((synth_r = xas_synth_new((xas_synth_callback_sample)sine_sample, (xas_synth_callback_cleanup)sine_cleanup, format, buffer_size, &sine_channels[1])) == NULL) { goto error_synth_new_r; } if ((mixer = xas_mixer_new(format, buffer_size)) == NULL) { goto error_mixer_new; } if (xas_mixer_input_add(mixer, bank_stream, 1.0, 0.0) == NULL) { goto error_mixer_input_add; } if (xas_mixer_input_add(mixer, synth_l, 0.5, -1.0) == NULL) { goto error_mixer_input_add; } if (xas_mixer_input_add(mixer, synth_r, 0.5, 1.0) == NULL) { goto error_mixer_input_add; } /* * Time to fill the sample bank, meow! */ xas_vox_say(vox, "I want to eat your soul. You don't understand. I really want to eat your soul.\n"); xas_vox_generate(vox); if (xas_bank_record(bank, voice, 0, 247018) < 0) { goto error_bank_record; } for (i=0; i= 300 && !xas_bank_active(bank)) { xas_bank_play(bank, 0, 1.0); } if ((readlen = xas_audio_stream_read(mixer->output, &buf, buffer_size)) < 0) { goto error_audio_stream_read; } if (xas_audio_stream_write(output, buf, readlen) < 0) { goto error_audio_stream_write; } } xas_audio_stream_flush(output); xas_mixer_destroy(mixer); xas_audio_stream_destroy(synth_r); xas_audio_stream_destroy(synth_l); xas_audio_stream_destroy(bank_stream); xas_audio_stream_destroy(voice); xas_audio_stream_destroy(output); xas_bank_destroy(bank); xas_vox_destroy(vox); return EX_OK; error_audio_stream_write: error_audio_stream_read: error_bank_record: error_mixer_input_add: xas_mixer_destroy(mixer); error_mixer_new: xas_audio_stream_destroy(synth_r); error_synth_new_r: xas_audio_stream_destroy(synth_l); error_synth_new_l: xas_audio_stream_destroy(bank_stream); error_bank_stream_new: xas_audio_stream_destroy(voice); error_vox_stream_new: xas_audio_stream_destroy(output); error_riff_new_file: xas_bank_destroy(bank); error_bank_new: xas_vox_destroy(vox); error_vox_new: return EX_OSERR; }