2018-03-14 11:24:23 +01:00
|
|
|
/* PipeWire
|
|
|
|
|
* Copyright (C) 2018 Wim Taymans <wim.taymans@gmail.com>
|
|
|
|
|
*
|
|
|
|
|
* 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; if not, write to the
|
|
|
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <sys/mman.h>
|
|
|
|
|
|
|
|
|
|
#include <spa/param/audio/format-utils.h>
|
|
|
|
|
#include <spa/param/props.h>
|
|
|
|
|
|
|
|
|
|
#include <pipewire/pipewire.h>
|
|
|
|
|
|
|
|
|
|
#define M_PI_M2 ( M_PI + M_PI )
|
|
|
|
|
|
|
|
|
|
#define DEFAULT_RATE 44100
|
2018-04-12 10:12:40 +02:00
|
|
|
#define DEFAULT_CHANNELS 2
|
2018-06-22 17:30:19 +02:00
|
|
|
#define DEFAULT_VOLUME 0.7
|
2018-03-14 11:24:23 +01:00
|
|
|
|
|
|
|
|
struct data {
|
|
|
|
|
struct pw_main_loop *loop;
|
|
|
|
|
|
|
|
|
|
struct pw_core *core;
|
|
|
|
|
struct pw_remote *remote;
|
|
|
|
|
|
|
|
|
|
struct pw_stream *stream;
|
|
|
|
|
|
|
|
|
|
double accumulator;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void fill_f32(struct data *d, void *dest, int avail)
|
|
|
|
|
{
|
2018-03-22 18:07:57 +01:00
|
|
|
float *dst = dest, val;
|
|
|
|
|
int n_samples = avail / (sizeof(float) * DEFAULT_CHANNELS);
|
2018-03-14 11:24:23 +01:00
|
|
|
int i, c;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < n_samples; i++) {
|
2018-03-22 18:07:57 +01:00
|
|
|
d->accumulator += M_PI_M2 * 440 / DEFAULT_RATE;
|
2018-03-14 11:24:23 +01:00
|
|
|
if (d->accumulator >= M_PI_M2)
|
|
|
|
|
d->accumulator -= M_PI_M2;
|
|
|
|
|
|
2018-06-22 17:30:19 +02:00
|
|
|
val = sin(d->accumulator) * DEFAULT_VOLUME;
|
2018-03-22 18:07:57 +01:00
|
|
|
for (c = 0; c < DEFAULT_CHANNELS; c++)
|
2018-03-14 11:24:23 +01:00
|
|
|
*dst++ = val;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-22 16:40:27 +01:00
|
|
|
static void on_process(void *userdata)
|
2018-03-14 11:24:23 +01:00
|
|
|
{
|
|
|
|
|
struct data *data = userdata;
|
2018-03-22 16:40:27 +01:00
|
|
|
struct pw_buffer *b;
|
2018-03-14 11:24:23 +01:00
|
|
|
struct spa_buffer *buf;
|
|
|
|
|
uint8_t *p;
|
|
|
|
|
|
2018-03-22 16:40:27 +01:00
|
|
|
if ((b = pw_stream_dequeue_buffer(data->stream)) == NULL)
|
2018-03-14 11:24:23 +01:00
|
|
|
return;
|
|
|
|
|
|
2018-03-22 16:40:27 +01:00
|
|
|
buf = b->buffer;
|
2018-03-14 11:24:23 +01:00
|
|
|
if ((p = buf->datas[0].data) == NULL)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
fill_f32(data, p, buf->datas[0].maxsize);
|
|
|
|
|
|
|
|
|
|
buf->datas[0].chunk->size = buf->datas[0].maxsize;
|
|
|
|
|
|
2018-03-22 16:40:27 +01:00
|
|
|
pw_stream_queue_buffer(data->stream, b);
|
2018-03-14 11:24:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const struct pw_stream_events stream_events = {
|
|
|
|
|
PW_VERSION_STREAM_EVENTS,
|
2018-03-22 16:40:27 +01:00
|
|
|
.process = on_process,
|
2018-03-14 11:24:23 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
|
{
|
|
|
|
|
struct data data = { 0, };
|
2018-03-22 18:07:57 +01:00
|
|
|
const struct spa_pod *params[1];
|
|
|
|
|
uint8_t buffer[1024];
|
|
|
|
|
struct spa_pod_builder b = SPA_POD_BUILDER_INIT(buffer, sizeof(buffer));
|
2018-03-14 11:24:23 +01:00
|
|
|
|
|
|
|
|
pw_init(&argc, &argv);
|
|
|
|
|
|
|
|
|
|
data.loop = pw_main_loop_new(NULL);
|
|
|
|
|
|
2018-03-22 18:07:57 +01:00
|
|
|
data.stream = pw_stream_new_simple(
|
|
|
|
|
pw_main_loop_get_loop(data.loop),
|
|
|
|
|
"audio-src",
|
2018-04-12 09:52:15 +02:00
|
|
|
pw_properties_new(
|
|
|
|
|
PW_NODE_PROP_MEDIA, "Audio",
|
|
|
|
|
PW_NODE_PROP_CATEGORY, "Playback",
|
|
|
|
|
PW_NODE_PROP_ROLE, "Music",
|
|
|
|
|
NULL),
|
2018-03-22 18:07:57 +01:00
|
|
|
&stream_events,
|
|
|
|
|
&data);
|
2018-03-14 11:24:23 +01:00
|
|
|
|
2018-03-22 18:07:57 +01:00
|
|
|
data.remote = pw_stream_get_remote(data.stream);
|
2018-03-14 11:24:23 +01:00
|
|
|
|
2018-03-22 18:07:57 +01:00
|
|
|
params[0] = spa_pod_builder_object(&b,
|
2018-08-27 15:03:11 +02:00
|
|
|
SPA_TYPE_OBJECT_Format, SPA_PARAM_EnumFormat,
|
2018-08-23 17:47:57 +02:00
|
|
|
"I", SPA_MEDIA_TYPE_audio,
|
|
|
|
|
"I", SPA_MEDIA_SUBTYPE_raw,
|
|
|
|
|
":", SPA_FORMAT_AUDIO_format, "I", SPA_AUDIO_FORMAT_F32,
|
2018-08-25 12:08:29 +02:00
|
|
|
":", SPA_FORMAT_AUDIO_layout, "I", SPA_AUDIO_LAYOUT_INTERLEAVED,
|
2018-08-23 17:47:57 +02:00
|
|
|
":", SPA_FORMAT_AUDIO_channels, "i", DEFAULT_CHANNELS,
|
|
|
|
|
":", SPA_FORMAT_AUDIO_rate, "i", DEFAULT_RATE);
|
2018-03-22 18:07:57 +01:00
|
|
|
|
|
|
|
|
pw_stream_connect(data.stream,
|
|
|
|
|
PW_DIRECTION_OUTPUT,
|
2018-04-05 15:37:34 +02:00
|
|
|
argc > 1 ? argv[1] : NULL,
|
2018-03-22 18:07:57 +01:00
|
|
|
PW_STREAM_FLAG_AUTOCONNECT |
|
|
|
|
|
PW_STREAM_FLAG_MAP_BUFFERS |
|
|
|
|
|
PW_STREAM_FLAG_RT_PROCESS,
|
|
|
|
|
params, 1);
|
2018-03-14 11:24:23 +01:00
|
|
|
|
|
|
|
|
pw_main_loop_run(data.loop);
|
|
|
|
|
|
2018-03-22 18:07:57 +01:00
|
|
|
pw_stream_destroy(data.stream);
|
2018-03-14 11:24:23 +01:00
|
|
|
pw_main_loop_destroy(data.loop);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|