summaryrefslogtreecommitdiffstats
path: root/examples/artscmt.c
diff options
context:
space:
mode:
Diffstat (limited to 'examples/artscmt.c')
-rw-r--r--examples/artscmt.c143
1 files changed, 143 insertions, 0 deletions
diff --git a/examples/artscmt.c b/examples/artscmt.c
new file mode 100644
index 0000000..bc9e98d
--- /dev/null
+++ b/examples/artscmt.c
@@ -0,0 +1,143 @@
+ /*
+
+ Copyright (C) 2003 Stefan Westerfeld
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License as published by the Free Software Foundation; either
+ version 2 of the License, or (at your option) any later version.
+
+ This library 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
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ Boston, MA 02111-1307, USA.
+
+ */
+
+/*
+ * This is a threaded example using the aRts C API.
+ *
+ * Compile programs using the aRts C API with
+ *
+ * cc -o artscmt artscmt.c `artsc-config --cflags` `artsc-config --libs`
+ *
+ * If you are using a makefile, it could look like this:
+ *
+ * CFLAGS=`artsc-config --cflags`
+ * LDFLAGS=`artsc-config --libs`
+ *
+ * artscmt: artscmt.c
+ */
+#include <artsc.h>
+#include <pthread.h>
+#include <assert.h>
+#include <math.h>
+#include <unistd.h>
+#include <stdio.h>
+
+pthread_mutex_t arts_mutex; /* pthread mutex */
+
+struct Writer
+{
+ pthread_t thread;
+ arts_stream_t stream;
+ float freq;
+};
+
+#define BUFFER_SIZE 1024
+
+static void* writer(void *arg)
+{
+ struct Writer *self = arg;
+ int pos = 0;
+
+ while(pos < 44100*10)
+ {
+ char buffer[BUFFER_SIZE], *to = buffer;
+ int i, written;
+
+ for(i=0;i<BUFFER_SIZE/4;i++)
+ {
+ /* generate two sin waves */
+ float fpos = ((float)pos)/44100.0;
+ long sample = (long)(sin(fpos*6.28*self->freq)*15000.0);
+ pos++;
+
+ /* put the samples in the packet */
+ *to++ = sample & 0xff;
+ *to++ = (sample >> 8) & 0xff;
+ *to++ = sample & 0xff;
+ *to++ = (sample >> 8) & 0xff;
+ }
+
+ written = 0;
+ do
+ {
+ int space;
+
+ /*
+ * Since there is more than one thread, it is important not to keep the lock
+ * for a long time. We definitely don't want arts_write to block, while we
+ * keep the lock, to ensure that other threads can do something as well. So
+ * we check the available buffer space before writing to avoid blocking.
+ */
+ pthread_mutex_lock(&arts_mutex);
+ space = arts_stream_get(self->stream, ARTS_P_BUFFER_SPACE);
+ if (space >= BUFFER_SIZE)
+ {
+ written = arts_write(self->stream, buffer, BUFFER_SIZE);
+ assert(written == BUFFER_SIZE); /* should handle errors here */
+ }
+ pthread_mutex_unlock(&arts_mutex);
+
+ /*
+ * If the buffer is full, wait some time to get free space again. The amout of
+ * time to wait needs to correspond to the buffer size we use for refilling.
+ */
+ if (!written)
+ usleep(10000); /* 10ms */
+ } while(!written);
+ }
+ return 0;
+}
+
+int main()
+{
+ struct Writer writer1, writer2;
+ int error;
+
+ error = arts_init();
+ if(error < 0)
+ {
+ fprintf(stderr, "error initializing aRts driver: %s\n", arts_error_text(error));
+ return 1;
+ }
+
+ pthread_mutex_init(&arts_mutex, 0);
+
+ writer1.stream = arts_play_stream(44100, 16, 2, "artscmt1");
+ writer1.freq = 440;
+ pthread_create(&writer1.thread, NULL, writer, &writer1);
+
+ writer2.stream = arts_play_stream(44100, 16, 2, "artscmt2");
+ writer2.freq = 880;
+ pthread_create(&writer2.thread, NULL, writer, &writer2);
+
+ pthread_join(writer1.thread, NULL);
+ pthread_join(writer2.thread, NULL);
+
+ arts_close_stream(writer1.stream);
+ arts_close_stream(writer2.stream);
+
+ pthread_mutex_destroy(&arts_mutex);
+
+ arts_free();
+
+ return 0;
+}