2004-07-07 00:22:46 +00:00
|
|
|
#include <signal.h>
|
2004-06-23 23:17:30 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
#include "polyp.h"
|
2004-07-07 00:22:46 +00:00
|
|
|
#include "polyp-error.h"
|
2004-06-23 23:17:30 +00:00
|
|
|
#include "mainloop.h"
|
2004-07-07 00:22:46 +00:00
|
|
|
#include "mainloop-signal.h"
|
2004-06-23 23:17:30 +00:00
|
|
|
|
|
|
|
|
static struct pa_context *context = NULL;
|
|
|
|
|
static struct pa_stream *stream = NULL;
|
|
|
|
|
static struct pa_mainloop_api *mainloop_api = NULL;
|
|
|
|
|
|
|
|
|
|
static void *buffer = NULL;
|
|
|
|
|
static size_t buffer_length = 0, buffer_index = 0;
|
|
|
|
|
|
|
|
|
|
static void* stdin_source = NULL;
|
|
|
|
|
|
2004-07-07 00:22:46 +00:00
|
|
|
static void quit(int ret) {
|
|
|
|
|
assert(mainloop_api);
|
|
|
|
|
mainloop_api->quit(mainloop_api, ret);
|
|
|
|
|
}
|
|
|
|
|
|
2004-06-23 23:17:30 +00:00
|
|
|
static void context_die_callback(struct pa_context *c, void *userdata) {
|
|
|
|
|
assert(c);
|
|
|
|
|
fprintf(stderr, "Connection to server shut down, exiting.\n");
|
2004-07-07 00:22:46 +00:00
|
|
|
quit(1);
|
2004-06-23 23:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void stream_die_callback(struct pa_stream *s, void *userdata) {
|
|
|
|
|
assert(s);
|
|
|
|
|
fprintf(stderr, "Stream deleted, exiting.\n");
|
2004-07-07 00:22:46 +00:00
|
|
|
quit(1);
|
2004-06-23 23:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
2004-06-27 17:50:02 +00:00
|
|
|
static void do_write(size_t length) {
|
2004-06-23 23:17:30 +00:00
|
|
|
size_t l;
|
2004-07-07 00:22:46 +00:00
|
|
|
assert(length);
|
|
|
|
|
|
|
|
|
|
if (!buffer || !buffer_length)
|
|
|
|
|
return;
|
2004-06-23 23:17:30 +00:00
|
|
|
|
|
|
|
|
l = length;
|
|
|
|
|
if (l > buffer_length)
|
|
|
|
|
l = buffer_length;
|
|
|
|
|
|
2004-06-27 17:50:02 +00:00
|
|
|
pa_stream_write(stream, buffer+buffer_index, l);
|
2004-06-23 23:17:30 +00:00
|
|
|
buffer_length -= l;
|
|
|
|
|
buffer_index += l;
|
|
|
|
|
|
|
|
|
|
if (!buffer_length) {
|
|
|
|
|
free(buffer);
|
|
|
|
|
buffer = NULL;
|
|
|
|
|
buffer_index = buffer_length = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2004-06-27 17:50:02 +00:00
|
|
|
static void stream_write_callback(struct pa_stream *s, size_t length, void *userdata) {
|
|
|
|
|
assert(s && length);
|
2004-07-07 00:22:46 +00:00
|
|
|
|
|
|
|
|
if (stdin_source)
|
|
|
|
|
mainloop_api->enable_io(mainloop_api, stdin_source, PA_MAINLOOP_API_IO_EVENT_INPUT);
|
2004-06-27 17:50:02 +00:00
|
|
|
|
|
|
|
|
if (!buffer)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
do_write(length);
|
|
|
|
|
}
|
|
|
|
|
|
2004-07-06 00:08:44 +00:00
|
|
|
static void stream_complete_callback(struct pa_stream*s, int success, void *userdata) {
|
|
|
|
|
assert(s);
|
2004-06-23 23:17:30 +00:00
|
|
|
|
2004-07-06 00:08:44 +00:00
|
|
|
if (!success) {
|
2004-07-07 00:22:46 +00:00
|
|
|
fprintf(stderr, "Stream creation failed: %s\n", pa_strerror(pa_context_errno(pa_stream_get_context(s))));
|
|
|
|
|
quit(1);
|
2004-06-27 17:50:02 +00:00
|
|
|
return;
|
2004-06-23 23:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
2004-07-07 00:22:46 +00:00
|
|
|
fprintf(stderr, "Stream created.\n");
|
2004-06-23 23:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void context_complete_callback(struct pa_context *c, int success, void *userdata) {
|
|
|
|
|
static const struct pa_sample_spec ss = {
|
2004-07-03 23:35:12 +00:00
|
|
|
.format = PA_SAMPLE_S16LE,
|
2004-06-23 23:17:30 +00:00
|
|
|
.rate = 44100,
|
2004-07-03 23:35:12 +00:00
|
|
|
.channels = 2
|
2004-06-23 23:17:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
assert(c && !stream);
|
|
|
|
|
|
|
|
|
|
if (!success) {
|
2004-07-07 00:22:46 +00:00
|
|
|
fprintf(stderr, "Connection failed: %s\n", pa_strerror(pa_context_errno(c)));
|
2004-06-23 23:17:30 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
2004-07-07 00:22:46 +00:00
|
|
|
|
|
|
|
|
fprintf(stderr, "Connection established.\n");
|
2004-06-23 23:17:30 +00:00
|
|
|
|
2004-07-06 00:08:44 +00:00
|
|
|
if (!(stream = pa_stream_new(c, PA_STREAM_PLAYBACK, NULL, "pacat", &ss, NULL, stream_complete_callback, NULL))) {
|
2004-07-07 00:22:46 +00:00
|
|
|
fprintf(stderr, "pa_stream_new() failed: %s\n", pa_strerror(pa_context_errno(c)));
|
2004-06-23 23:17:30 +00:00
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
|
2004-07-07 00:22:46 +00:00
|
|
|
pa_stream_set_die_callback(stream, stream_die_callback, NULL);
|
|
|
|
|
pa_stream_set_write_callback(stream, stream_write_callback, NULL);
|
|
|
|
|
|
2004-06-23 23:17:30 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
fail:
|
2004-07-07 00:22:46 +00:00
|
|
|
quit(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void context_drain_complete(struct pa_context*c, void *userdata) {
|
|
|
|
|
quit(0);
|
2004-06-23 23:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void stdin_callback(struct pa_mainloop_api*a, void *id, int fd, enum pa_mainloop_api_io_events events, void *userdata) {
|
2004-06-27 17:50:02 +00:00
|
|
|
size_t l, w = 0;
|
2004-06-23 23:17:30 +00:00
|
|
|
ssize_t r;
|
2004-07-07 00:22:46 +00:00
|
|
|
assert(a == mainloop_api && id && fd == STDIN_FILENO && events == PA_MAINLOOP_API_IO_EVENT_INPUT && stdin_source == id);
|
2004-06-23 23:17:30 +00:00
|
|
|
|
|
|
|
|
if (buffer) {
|
|
|
|
|
mainloop_api->enable_io(mainloop_api, stdin_source, PA_MAINLOOP_API_IO_EVENT_NULL);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2004-07-07 00:22:46 +00:00
|
|
|
if (!stream || !pa_stream_is_ready(stream) || !(l = w = pa_stream_writable_size(stream)))
|
2004-06-23 23:17:30 +00:00
|
|
|
l = 4096;
|
2004-07-07 00:22:46 +00:00
|
|
|
|
2004-06-23 23:17:30 +00:00
|
|
|
buffer = malloc(l);
|
|
|
|
|
assert(buffer);
|
|
|
|
|
if ((r = read(fd, buffer, l)) <= 0) {
|
2004-07-07 00:22:46 +00:00
|
|
|
if (r == 0) {
|
|
|
|
|
fprintf(stderr, "Got EOF.\n");
|
|
|
|
|
if (pa_context_drain(context, context_drain_complete, NULL) < 0)
|
|
|
|
|
quit(0);
|
|
|
|
|
else
|
|
|
|
|
fprintf(stderr, "Draining connection to server.\n");
|
|
|
|
|
} else {
|
2004-06-23 23:17:30 +00:00
|
|
|
fprintf(stderr, "read() failed: %s\n", strerror(errno));
|
2004-07-07 00:22:46 +00:00
|
|
|
quit(1);
|
2004-06-23 23:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
2004-07-07 00:22:46 +00:00
|
|
|
mainloop_api->cancel_io(mainloop_api, stdin_source);
|
|
|
|
|
stdin_source = NULL;
|
2004-06-23 23:17:30 +00:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
buffer_length = r;
|
|
|
|
|
buffer_index = 0;
|
2004-06-27 17:50:02 +00:00
|
|
|
|
|
|
|
|
if (w)
|
|
|
|
|
do_write(w);
|
2004-06-23 23:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
2004-07-07 00:22:46 +00:00
|
|
|
|
|
|
|
|
static void exit_signal_callback(void *id, int sig, void *userdata) {
|
|
|
|
|
fprintf(stderr, "Got SIGINT, exiting.\n");
|
|
|
|
|
quit(0);
|
|
|
|
|
}
|
|
|
|
|
|
2004-06-23 23:17:30 +00:00
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
|
struct pa_mainloop* m;
|
2004-07-07 00:22:46 +00:00
|
|
|
int ret = 1, r;
|
2004-06-23 23:17:30 +00:00
|
|
|
|
|
|
|
|
if (!(m = pa_mainloop_new())) {
|
|
|
|
|
fprintf(stderr, "pa_mainloop_new() failed.\n");
|
|
|
|
|
goto quit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mainloop_api = pa_mainloop_get_api(m);
|
|
|
|
|
|
2004-07-07 00:22:46 +00:00
|
|
|
r = pa_signal_init(mainloop_api);
|
|
|
|
|
assert(r == 0);
|
|
|
|
|
pa_signal_register(SIGINT, exit_signal_callback, NULL);
|
|
|
|
|
signal(SIGPIPE, SIG_IGN);
|
|
|
|
|
|
2004-06-23 23:17:30 +00:00
|
|
|
if (!(stdin_source = mainloop_api->source_io(mainloop_api, STDIN_FILENO, PA_MAINLOOP_API_IO_EVENT_INPUT, stdin_callback, NULL))) {
|
|
|
|
|
fprintf(stderr, "source_io() failed.\n");
|
|
|
|
|
goto quit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(context = pa_context_new(mainloop_api, argv[0]))) {
|
|
|
|
|
fprintf(stderr, "pa_context_new() failed.\n");
|
|
|
|
|
goto quit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pa_context_connect(context, NULL, context_complete_callback, NULL) < 0) {
|
|
|
|
|
fprintf(stderr, "pa_context_connext() failed.\n");
|
|
|
|
|
goto quit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pa_context_set_die_callback(context, context_die_callback, NULL);
|
|
|
|
|
|
|
|
|
|
if (pa_mainloop_run(m, &ret) < 0) {
|
|
|
|
|
fprintf(stderr, "pa_mainloop_run() failed.\n");
|
|
|
|
|
goto quit;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
quit:
|
|
|
|
|
if (stream)
|
|
|
|
|
pa_stream_free(stream);
|
|
|
|
|
if (context)
|
|
|
|
|
pa_context_free(context);
|
|
|
|
|
if (m)
|
|
|
|
|
pa_mainloop_free(m);
|
|
|
|
|
if (buffer)
|
|
|
|
|
free(buffer);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|