123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882 |
- /***************************************************************************
- * Copyright (C) 2007, Gilles Casse <[email protected]> *
- * eSpeak driver for PulseAudio *
- * based on the XMMS PulseAudio Plugin *
- * *
- * 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; either version 3 of the License, or *
- * (at your option) any later version. *
- * *
- * 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. *
- ***************************************************************************/
- // TBD:
- // * ARCH_BIG
- // * uint64? a_timing_info.read_index
- // * prebuf,... size?
- // * 0.9.6: pb pulse_free using tlength=8820 (max size never returned -> tlength=10000 ok, but higher drain).
- //
- #include "speech.h"
-
- #ifdef USE_ASYNC
- // This source file is only used for asynchronious modes
-
- #include <stdio.h>
- #include <string.h>
- #include <stdlib.h>
- #include <math.h>
- #include <assert.h>
- #include <sys/time.h>
- #include <time.h>
- #include <pulse/pulseaudio.h>
-
- #ifndef PLATFORM_WINDOWS
- #include <unistd.h>
- #endif
- #include "wave.h"
- #include "debug.h"
-
- //<Definitions
-
- enum {ONE_BILLION=1000000000};
-
- #ifdef USE_PULSEAUDIO
-
-
- static t_wave_callback* my_callback_is_output_enabled=NULL;
-
- #define SAMPLE_RATE 22050
- #define FRAMES_PER_BUFFER 512
- #define BUFFER_LENGTH (SAMPLE_RATE*2*sizeof(uint16_t))
- #define THRESHOLD (BUFFER_LENGTH/5)
-
- #define ESPEAK_FORMAT PA_SAMPLE_S16LE
- #define ESPEAK_CHANNEL 1
-
-
- static pa_context *context = NULL;
- static pa_stream *stream = NULL;
- static pa_threaded_mainloop *mainloop = NULL;
-
- static pa_cvolume volume;
- static int volume_valid = 0;
-
- static int do_trigger = 0;
- static uint64_t written = 0;
- static int time_offset_msec = 0;
- static int just_flushed = 0;
-
- static int connected = 0;
-
- #define CHECK_DEAD_GOTO(label, warn) do { \
- if (!mainloop || \
- !context || pa_context_get_state(context) != PA_CONTEXT_READY || \
- !stream || pa_stream_get_state(stream) != PA_STREAM_READY) { \
- if (warn) \
- printf("Connection died: %s\n", context ? pa_strerror(pa_context_errno(context)) : "NULL"); \
- goto label; \
- } \
- } while(0);
-
- #define CHECK_CONNECTED(retval) \
- do { \
- if (!connected) return retval; \
- } while (0);
-
- //>
-
-
- // static void display_timing_info(const pa_timing_info* the_time)
- // {
- // const struct timeval *tv=&(the_time->timestamp);
-
- // SHOW_TIME("ti>");
- // SHOW("ti> timestamp=%03d.%03dms\n",(int)(tv->tv_sec%1000), (int)(tv->tv_usec/1000));
- // SHOW("ti> synchronized_clocks=%d\n",the_time->synchronized_clocks);
- // SHOW("ti> sink_usec=%ld\n",the_time->sink_usec);
- // SHOW("ti> source_usec=%ld\n",the_time->source_usec);
- // SHOW("ti> transport=%ld\n",the_time->transport_usec);
- // SHOW("ti> playing=%d\n",the_time->playing);
- // SHOW("ti> write_index_corrupt=%d\n",the_time->write_index_corrupt);
- // SHOW("ti> write_index=0x%lx\n",the_time->write_index);
- // SHOW("ti> read_index_corrupt=%d\n",the_time->read_index_corrupt);
- // SHOW("ti> read_index=0x%lx\n",the_time->read_index);
- // }
-
-
- static void info_cb(struct pa_context *c, const struct pa_sink_input_info *i, int is_last, void *userdata) {
- ENTER(__FUNCTION__);
- assert(c);
-
- if (!i)
- return;
-
- volume = i->volume;
- volume_valid = 1;
- }
-
- static void subscribe_cb(struct pa_context *c, enum pa_subscription_event_type t, uint32_t index, void *userdata) {
- pa_operation *o;
- ENTER(__FUNCTION__);
-
- assert(c);
-
- if (!stream ||
- index != pa_stream_get_index(stream) ||
- (t != (PA_SUBSCRIPTION_EVENT_SINK_INPUT|PA_SUBSCRIPTION_EVENT_CHANGE) &&
- t != (PA_SUBSCRIPTION_EVENT_SINK_INPUT|PA_SUBSCRIPTION_EVENT_NEW)))
- return;
-
- if (!(o = pa_context_get_sink_input_info(c, index, info_cb, NULL))) {
- printf("pa_context_get_sink_input_info() failed: %s\n", pa_strerror(pa_context_errno(c)));
- return;
- }
-
- pa_operation_unref(o);
- }
-
- static void context_state_cb(pa_context *c, void *userdata) {
- ENTER(__FUNCTION__);
- assert(c);
-
- switch (pa_context_get_state(c)) {
- case PA_CONTEXT_READY:
- case PA_CONTEXT_TERMINATED:
- case PA_CONTEXT_FAILED:
- pa_threaded_mainloop_signal(mainloop, 0);
- break;
-
- case PA_CONTEXT_UNCONNECTED:
- case PA_CONTEXT_CONNECTING:
- case PA_CONTEXT_AUTHORIZING:
- case PA_CONTEXT_SETTING_NAME:
- break;
- }
- }
-
- static void stream_state_cb(pa_stream *s, void * userdata) {
- ENTER(__FUNCTION__);
- assert(s);
-
- switch (pa_stream_get_state(s)) {
-
- case PA_STREAM_READY:
- case PA_STREAM_FAILED:
- case PA_STREAM_TERMINATED:
- pa_threaded_mainloop_signal(mainloop, 0);
- break;
-
- case PA_STREAM_UNCONNECTED:
- case PA_STREAM_CREATING:
- break;
- }
- }
-
- static void stream_success_cb(pa_stream *s, int success, void *userdata) {
- ENTER(__FUNCTION__);
- assert(s);
-
- if (userdata)
- *(int*) userdata = success;
-
- pa_threaded_mainloop_signal(mainloop, 0);
- }
-
- static void context_success_cb(pa_context *c, int success, void *userdata) {
- ENTER(__FUNCTION__);
- assert(c);
-
- if (userdata)
- *(int*) userdata = success;
-
- pa_threaded_mainloop_signal(mainloop, 0);
- }
-
- static void stream_request_cb(pa_stream *s, size_t length, void *userdata) {
- ENTER(__FUNCTION__);
- assert(s);
-
- pa_threaded_mainloop_signal(mainloop, 0);
- }
-
- static void stream_latency_update_cb(pa_stream *s, void *userdata) {
- // ENTER(__FUNCTION__);
- assert(s);
-
- pa_threaded_mainloop_signal(mainloop, 0);
- }
-
- static int pulse_free(void) {
- ENTER(__FUNCTION__);
- size_t l = 0;
- pa_operation *o = NULL;
-
- CHECK_CONNECTED(0);
-
- SHOW("pulse_free: %s (call)\n", "pa_threaded_main_loop_lock");
- pa_threaded_mainloop_lock(mainloop);
- CHECK_DEAD_GOTO(fail, 1);
-
- if ((l = pa_stream_writable_size(stream)) == (size_t) -1) {
- SHOW("pa_stream_writable_size() failed: %s", pa_strerror(pa_context_errno(context)));
- l = 0;
- goto fail;
- }
-
- SHOW("pulse_free: %s (ret=%d)\n", "pa_stream_writable_size", l);
-
- /* If this function is called twice with no pulse_write() call in
- * between this means we should trigger the playback */
- if (do_trigger) {
- int success = 0;
-
- SHOW("pulse_free: %s (call)\n", "pa_stream_trigger");
- if (!(o = pa_stream_trigger(stream, stream_success_cb, &success))) {
- SHOW("pa_stream_trigger() failed: %s", pa_strerror(pa_context_errno(context)));
- goto fail;
- }
-
- SHOW("pulse_free: %s (call)\n", "pa_threaded_main_loop");
- while (pa_operation_get_state(o) != PA_OPERATION_DONE) {
- CHECK_DEAD_GOTO(fail, 1);
- pa_threaded_mainloop_wait(mainloop);
- }
- SHOW("pulse_free: %s (ret)\n", "pa_threaded_main_loop");
-
- if (!success)
- SHOW("pa_stream_trigger() failed: %s", pa_strerror(pa_context_errno(context)));
- }
-
- fail:
- SHOW("pulse_free: %s (call)\n", "pa_operation_unref");
- if (o)
- pa_operation_unref(o);
-
- SHOW("pulse_free: %s (call)\n", "pa_threaded_main_loop_unlock");
- pa_threaded_mainloop_unlock(mainloop);
-
- do_trigger = !!l;
- SHOW("pulse_free: %d (ret)\n", (int)l);
- return (int) l;
- }
-
- static int pulse_playing(const pa_timing_info *the_timing_info) {
- ENTER(__FUNCTION__);
- int r = 0;
- const pa_timing_info *i;
-
- assert(the_timing_info);
-
- CHECK_CONNECTED(0);
-
- pa_threaded_mainloop_lock(mainloop);
-
- for (;;) {
- CHECK_DEAD_GOTO(fail, 1);
-
- if ((i = pa_stream_get_timing_info(stream)))
- {
- break;
- }
- if (pa_context_errno(context) != PA_ERR_NODATA) {
- SHOW("pa_stream_get_timing_info() failed: %s", pa_strerror(pa_context_errno(context)));
- goto fail;
- }
-
- pa_threaded_mainloop_wait(mainloop);
- }
-
- r = i->playing;
- memcpy((void*)the_timing_info, (void*)i, sizeof(pa_timing_info));
-
- // display_timing_info(i);
-
- fail:
- pa_threaded_mainloop_unlock(mainloop);
-
- return r;
- }
-
-
- // static void pulse_flush(int time) {
- // ENTER(__FUNCTION__);
- // pa_operation *o = NULL;
- // int success = 0;
-
- // CHECK_CONNECTED();
-
- // pa_threaded_mainloop_lock(mainloop);
- // CHECK_DEAD_GOTO(fail, 1);
-
- // if (!(o = pa_stream_flush(stream, stream_success_cb, &success))) {
- // SHOW("pa_stream_flush() failed: %s", pa_strerror(pa_context_errno(context)));
- // goto fail;
- // }
-
- // while (pa_operation_get_state(o) != PA_OPERATION_DONE) {
- // CHECK_DEAD_GOTO(fail, 1);
- // pa_threaded_mainloop_wait(mainloop);
- // }
-
- // if (!success)
- // SHOW("pa_stream_flush() failed: %s", pa_strerror(pa_context_errno(context)));
-
- // written = (uint64_t) (((double) time * pa_bytes_per_second(pa_stream_get_sample_spec(stream))) / 1000);
- // just_flushed = 1;
- // time_offset_msec = time;
-
- // fail:
- // if (o)
- // pa_operation_unref(o);
-
- // pa_threaded_mainloop_unlock(mainloop);
- // }
-
-
- static void pulse_write(void* ptr, int length) {
- ENTER(__FUNCTION__);
-
-
- SHOW("pulse_write > length=%d\n", length);
-
- CHECK_CONNECTED();
-
- pa_threaded_mainloop_lock(mainloop);
- CHECK_DEAD_GOTO(fail, 1);
-
- if (pa_stream_write(stream, ptr, length, NULL, PA_SEEK_RELATIVE, (pa_seek_mode_t)0) < 0) {
- SHOW("pa_stream_write() failed: %s", pa_strerror(pa_context_errno(context)));
- goto fail;
- }
-
- do_trigger = 0;
- written += length;
-
- fail:
-
- pa_threaded_mainloop_unlock(mainloop);
- }
-
- static void drain(void) {
- pa_operation *o = NULL;
- int success = 0;
-
- ENTER(__FUNCTION__);
-
- CHECK_CONNECTED();
-
- pa_threaded_mainloop_lock(mainloop);
- CHECK_DEAD_GOTO(fail, 0);
-
- SHOW_TIME("pa_stream_drain (call)");
- if (!(o = pa_stream_drain(stream, stream_success_cb, &success))) {
- printf("pa_stream_drain() failed: %s\n", pa_strerror(pa_context_errno(context)));
- goto fail;
- }
-
- SHOW_TIME("pa_threaded_mainloop_wait (call)");
- while (pa_operation_get_state(o) != PA_OPERATION_DONE) {
- CHECK_DEAD_GOTO(fail, 1);
- pa_threaded_mainloop_wait(mainloop);
- }
- SHOW_TIME("pa_threaded_mainloop_wait (ret)");
-
- if (!success)
- printf("pa_stream_drain() failed: %s\n", pa_strerror(pa_context_errno(context)));
-
- fail:
- SHOW_TIME("pa_operation_unref (call)");
- if (o)
- pa_operation_unref(o);
-
- pa_threaded_mainloop_unlock(mainloop);
- SHOW_TIME("drain (ret)");
- }
-
-
- static void pulse_close(void) {
-
- ENTER(__FUNCTION__);
-
- drain();
-
- connected = 0;
-
- if (mainloop)
- pa_threaded_mainloop_stop(mainloop);
-
- if (stream) {
- SHOW_TIME("pa_stream_disconnect (call)");
- pa_stream_disconnect(stream);
- pa_stream_unref(stream);
- stream = NULL;
- }
-
- if (context) {
- SHOW_TIME("pa_context_disconnect (call)");
- pa_context_disconnect(context);
- pa_context_unref(context);
- context = NULL;
- }
-
- if (mainloop) {
- SHOW_TIME("pa_threaded_mainloop_free (call)");
- pa_threaded_mainloop_free(mainloop);
- mainloop = NULL;
- }
- SHOW_TIME("pulse_close (ret)");
-
- }
-
-
- static int pulse_open()
- {
- ENTER(__FUNCTION__);
- pa_sample_spec ss;
- pa_operation *o = NULL;
- int success;
-
- assert(!mainloop);
- assert(!context);
- assert(!stream);
- assert(!connected);
-
- ss.format = ESPEAK_FORMAT;
- ss.rate = SAMPLE_RATE;
- ss.channels = ESPEAK_CHANNEL;
-
- if (!pa_sample_spec_valid(&ss))
- return false;
-
- /* if (!volume_valid) { */
- pa_cvolume_reset(&volume, ss.channels);
- volume_valid = 1;
- /* } else if (volume.channels != ss.channels) */
- /* pa_cvolume_set(&volume, ss.channels, pa_cvolume_avg(&volume)); */
-
- SHOW_TIME("pa_threaded_mainloop_new (call)");
- if (!(mainloop = pa_threaded_mainloop_new())) {
- printf("Failed to allocate main loop\n");
- goto fail;
- }
-
- pa_threaded_mainloop_lock(mainloop);
-
- SHOW_TIME("pa_context_new (call)");
- if (!(context = pa_context_new(pa_threaded_mainloop_get_api(mainloop), "eSpeak"))) {
- printf("Failed to allocate context\n");
- goto unlock_and_fail;
- }
-
- pa_context_set_state_callback(context, context_state_cb, NULL);
- pa_context_set_subscribe_callback(context, subscribe_cb, NULL);
-
- SHOW_TIME("pa_context_connect (call)");
- if (pa_context_connect(context, NULL, (pa_context_flags_t)0, NULL) < 0) {
- printf("Failed to connect to server: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- SHOW_TIME("pa_threaded_mainloop_start (call)");
- if (pa_threaded_mainloop_start(mainloop) < 0) {
- printf("Failed to start main loop");
- goto unlock_and_fail;
- }
-
- /* Wait until the context is ready */
- SHOW_TIME("pa_threaded_mainloop_wait");
- pa_threaded_mainloop_wait(mainloop);
-
- if (pa_context_get_state(context) != PA_CONTEXT_READY) {
- printf("Failed to connect to server: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- SHOW_TIME("pa_stream_new");
- if (!(stream = pa_stream_new(context, "unknown", &ss, NULL))) {
- printf("Failed to create stream: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- pa_stream_set_state_callback(stream, stream_state_cb, NULL);
- pa_stream_set_write_callback(stream, stream_request_cb, NULL);
- pa_stream_set_latency_update_callback(stream, stream_latency_update_cb, NULL);
-
-
-
- pa_buffer_attr a_attr;
-
- // TBD
- a_attr.maxlength=132300;
- a_attr.tlength=8820; //88200
- // a_attr.tlength=10000; //88200
- a_attr.prebuf=2200;
- a_attr.minreq=880;
- a_attr.fragsize=882;
-
-
-
- SHOW_TIME("pa_connect_playback");
- if (pa_stream_connect_playback(stream, NULL, &a_attr, (pa_stream_flags_t)(PA_STREAM_INTERPOLATE_TIMING|PA_STREAM_AUTO_TIMING_UPDATE), &volume, NULL) < 0) {
- printf("Failed to connect stream: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- /* Wait until the stream is ready */
- SHOW_TIME("pa_threaded_mainloop_wait");
- pa_threaded_mainloop_wait(mainloop);
-
- if (pa_stream_get_state(stream) != PA_STREAM_READY) {
- printf("Failed to connect stream: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- /* Now subscribe to events */
- SHOW_TIME("pa_context_subscribe");
- if (!(o = pa_context_subscribe(context, PA_SUBSCRIPTION_MASK_SINK_INPUT, context_success_cb, &success))) {
- printf("pa_context_subscribe() failed: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- success = 0;
- SHOW_TIME("pa_threaded_mainloop_wait");
- while (pa_operation_get_state(o) != PA_OPERATION_DONE) {
- CHECK_DEAD_GOTO(fail, 1);
- pa_threaded_mainloop_wait(mainloop);
- }
-
- if (!success) {
- printf("pa_context_subscribe() failed: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- pa_operation_unref(o);
-
- /* Now request the initial stream info */
- if (!(o = pa_context_get_sink_input_info(context, pa_stream_get_index(stream), info_cb, NULL))) {
- printf("pa_context_get_sink_input_info() failed: %s", pa_strerror(pa_context_errno(context)));
- goto unlock_and_fail;
- }
-
- SHOW_TIME("pa_threaded_mainloop_wait 2");
- while (pa_operation_get_state(o) != PA_OPERATION_DONE) {
- CHECK_DEAD_GOTO(fail, 1);
- pa_threaded_mainloop_wait(mainloop);
- }
-
- /* if (!volume_valid) { */
- /* printf("pa_context_get_sink_input_info() failed: %s", pa_strerror(pa_context_errno(context))); */
- /* goto unlock_and_fail; */
- /* } */
-
- do_trigger = 0;
- written = 0;
- time_offset_msec = 0;
- just_flushed = 0;
- connected = 1;
- // volume_time_event = NULL;
-
- pa_threaded_mainloop_unlock(mainloop);
- SHOW_TIME("pulse_open (ret true)");
-
- return true;
-
- unlock_and_fail:
-
- if (o)
- pa_operation_unref(o);
-
- pa_threaded_mainloop_unlock(mainloop);
-
- fail:
-
- pulse_close();
-
- SHOW_TIME("pulse_open (ret false)");
-
- return false;
-
-
-
- }
-
- void wave_flush(void* theHandler)
- {
- ENTER("wave_flush");
-
- // if (my_stream_could_start)
- // {
- // // #define buf 1024
- // // static char a_buffer[buf*2];
- // // memset(a_buffer,0,buf*2);
- // // wave_write(theHandler, a_buffer, buf*2);
- // start_stream();
- // }
- }
-
-
-
- //<wave_set_callback_is_output_enabled
-
- void wave_set_callback_is_output_enabled(t_wave_callback* cb)
- {
- my_callback_is_output_enabled = cb;
- }
-
- //>
- //<wave_init
-
- void display_buffer()
- {
-
- const pa_buffer_attr *a_attr = pa_stream_get_buffer_attr (stream);
-
- if (a_attr)
- {
- SHOW("attr> maxlength=%ld\n",a_attr->maxlength);
- SHOW("attr> tlength=%ld\n",a_attr->tlength);
- SHOW("attr> prebuf=%ld\n",a_attr->prebuf);
- SHOW("attr> minreq=%ld\n",a_attr->minreq);
- SHOW("attr> fragsize=%ld\n",a_attr->fragsize);
- }
- }
-
- void wave_init()
- {
- ENTER("wave_init");
-
- stream = NULL;
-
- pulse_open();
-
- display_buffer();
-
- }
-
- //>
- //<wave_open
-
- void* wave_open(char* the_api)
- {
- ENTER("wave_open");
- return((void*)1);
- }
-
- //>
- //<wave_write
-
- size_t wave_write(void* theHandler, char* theMono16BitsWaveBuffer, size_t theSize)
- {
- ENTER("wave_write");
- size_t bytes_to_write = theSize;
- char* aBuffer=theMono16BitsWaveBuffer;
-
- assert(stream);
-
- size_t aTotalFreeMem=0;
-
- while (1)
- {
- if (my_callback_is_output_enabled
- && (0==my_callback_is_output_enabled()))
- {
- SHOW_TIME("wave_write > my_callback_is_output_enabled: no!");
- return 0;
- }
-
- aTotalFreeMem = pulse_free();
- if (aTotalFreeMem >= bytes_to_write)
- {
- SHOW("wave_write > aTotalFreeMem(%d) >= bytes_to_write(%d)\n", aTotalFreeMem, bytes_to_write);
- break;
- }
- SHOW("wave_write > wait: aTotalFreeMem(%d) < bytes_to_write(%d)\n", aTotalFreeMem, bytes_to_write);
-
- // 500: threshold for avoiding to many calls to pulse_write
- if (aTotalFreeMem>500)
- {
- pulse_write(aBuffer, aTotalFreeMem);
- bytes_to_write -= aTotalFreeMem;
- aBuffer += aTotalFreeMem;
- }
-
- usleep(10000);
- }
-
- pulse_write(aBuffer, bytes_to_write);
-
-
- SHOW_TIME("wave_write > LEAVE");
-
- return theSize;
- }
-
- //>
- //<wave_close
-
- int wave_close(void* theHandler)
- {
- SHOW_TIME("wave_close > ENTER");
-
- drain();
-
- return 0;
- }
-
- //>
- //<wave_is_busy
-
- int wave_is_busy(void* theHandler)
- {
- SHOW_TIME("wave_is_busy");
-
- pa_timing_info a_timing_info;
- int active = pulse_playing(&a_timing_info);
- SHOW("wave_is_busy: %d\n",active);
- return active;
- }
-
- //>
- //<wave_terminate
-
- void wave_terminate()
- {
- ENTER("wave_terminate");
-
- // Pa_Terminate();
-
- pulse_close();
-
- }
-
- //>
- //<wave_get_read_position, wave_get_write_position, wave_get_remaining_time
-
- uint32_t wave_get_read_position(void* theHandler)
- {
- pa_timing_info a_timing_info;
- pulse_playing(&a_timing_info);
- SHOW("wave_get_read_position > %lx\n", a_timing_info.read_index);
- return a_timing_info.read_index;
- }
-
- uint32_t wave_get_write_position(void* theHandler)
- {
- pa_timing_info a_timing_info;
- pulse_playing(&a_timing_info);
- SHOW("wave_get_read_position > %lx\n", a_timing_info.write_index);
- return a_timing_info.write_index;
- }
-
- int wave_get_remaining_time(uint32_t sample, uint32_t* time)
- {
- double a_time=0;
-
- if (!time || !stream)
- {
- SHOW("event get_remaining_time> %s\n","audio device not available");
- return -1;
- }
-
- pa_timing_info a_timing_info;
- pulse_playing(&a_timing_info);
-
- if (sample > a_timing_info.read_index)
- {
- // TBD: take in account time suplied by portaudio V18 API
- a_time = sample - a_timing_info.read_index;
- a_time = 0.5 + (a_time * 1000.0) / SAMPLE_RATE;
- }
- else
- {
- a_time = 0;
- }
-
- SHOW("wave_get_remaining_time > sample=%d, time=%d\n", sample, (uint32_t)a_time);
-
- *time = (uint32_t)a_time;
-
- return 0;
- }
-
- //>
- //<wave_test_get_write_buffer
-
- void *wave_test_get_write_buffer()
- {
- return NULL;
- }
-
-
- #else
- // notdef USE_PULSEAUDIO
-
-
- void wave_init() {}
- void* wave_open(char* the_api) {return (void *)1;}
- size_t wave_write(void* theHandler, char* theMono16BitsWaveBuffer, size_t theSize) {return theSize;}
- int wave_close(void* theHandler) {return 0;}
- int wave_is_busy(void* theHandler) {return 0;}
- void wave_terminate() {}
- uint32_t wave_get_read_position(void* theHandler) {return 0;}
- uint32_t wave_get_write_position(void* theHandler) {return 0;}
- void wave_flush(void* theHandler) {}
- typedef int (t_wave_callback)(void);
- void wave_set_callback_is_output_enabled(t_wave_callback* cb) {}
- extern void* wave_test_get_write_buffer() {return NULL;}
-
- int wave_get_remaining_time(uint32_t sample, uint32_t* time)
- {
- if (!time) return(-1);
- *time = (uint32_t)0;
- return 0;
- }
-
- #endif // of USE_PORTAUDIO
-
- //>
- //<clock_gettime2, add_time_in_ms
-
- void clock_gettime2(struct timespec *ts)
- {
- struct timeval tv;
-
- if (!ts)
- {
- return;
- }
-
- assert (gettimeofday(&tv, NULL) != -1);
- ts->tv_sec = tv.tv_sec;
- ts->tv_nsec = tv.tv_usec*1000;
- }
-
- void add_time_in_ms(struct timespec *ts, int time_in_ms)
- {
- if (!ts)
- {
- return;
- }
-
- uint64_t t_ns = (uint64_t)ts->tv_nsec + 1000000 * (uint64_t)time_in_ms;
- while(t_ns >= ONE_BILLION)
- {
- SHOW("event > add_time_in_ms ns: %d sec %Lu nsec \n", ts->tv_sec, t_ns);
- ts->tv_sec += 1;
- t_ns -= ONE_BILLION;
- }
- ts->tv_nsec = (long int)t_ns;
- }
-
-
- #endif // USE_ASYNC
-
- //>
|