pipewire/src/examples/audio-src.c

139 lines
3.5 KiB
C
Raw Normal View History

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
#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)
{
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++) {
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;
for (c = 0; c < DEFAULT_CHANNELS; c++)
2018-03-14 11:24:23 +01:00
*dst++ = val;
}
}
static void on_process(void *userdata)
2018-03-14 11:24:23 +01:00
{
struct data *data = userdata;
struct pw_buffer *b;
2018-03-14 11:24:23 +01:00
struct spa_buffer *buf;
uint8_t *p;
if ((b = pw_stream_dequeue_buffer(data->stream)) == NULL)
2018-03-14 11:24:23 +01:00
return;
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;
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,
.process = on_process,
2018-03-14 11:24:23 +01:00
};
int main(int argc, char *argv[])
{
struct data data = { 0, };
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);
data.stream = pw_stream_new_simple(
pw_main_loop_get_loop(data.loop),
"audio-src",
pw_properties_new(
PW_NODE_PROP_MEDIA, "Audio",
PW_NODE_PROP_CATEGORY, "Playback",
PW_NODE_PROP_ROLE, "Music",
NULL),
&stream_events,
&data);
2018-03-14 11:24:23 +01:00
data.remote = pw_stream_get_remote(data.stream);
2018-03-14 11:24:23 +01:00
params[0] = spa_pod_builder_object(&b,
SPA_PARAM_EnumFormat, SPA_ID_OBJECT_Format,
"I", SPA_MEDIA_TYPE_audio,
"I", SPA_MEDIA_SUBTYPE_raw,
":", SPA_FORMAT_AUDIO_format, "I", SPA_AUDIO_FORMAT_F32,
":", SPA_FORMAT_AUDIO_layout, "I", SPA_AUDIO_LAYOUT_INTERLEAVED,
":", SPA_FORMAT_AUDIO_channels, "i", DEFAULT_CHANNELS,
":", SPA_FORMAT_AUDIO_rate, "i", DEFAULT_RATE);
pw_stream_connect(data.stream,
PW_DIRECTION_OUTPUT,
2018-04-05 15:37:34 +02:00
argc > 1 ? argv[1] : NULL,
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);
pw_stream_destroy(data.stream);
2018-03-14 11:24:23 +01:00
pw_main_loop_destroy(data.loop);
return 0;
}