|
rpm-build |
ca19af |
/* PulseAudio Output.
|
|
rpm-build |
ca19af |
*
|
|
rpm-build |
ca19af |
* Copyright (C) 2016 Reece H. Dunn
|
|
rpm-build |
ca19af |
*
|
|
rpm-build |
ca19af |
* This file is part of pcaudiolib.
|
|
rpm-build |
ca19af |
*
|
|
rpm-build |
ca19af |
* pcaudiolib is free software: you can redistribute it and/or modify
|
|
rpm-build |
ca19af |
* it under the terms of the GNU General Public License as published by
|
|
rpm-build |
ca19af |
* the Free Software Foundation, either version 3 of the License, or
|
|
rpm-build |
ca19af |
* (at your option) any later version.
|
|
rpm-build |
ca19af |
*
|
|
rpm-build |
ca19af |
* pcaudiolib is distributed in the hope that it will be useful,
|
|
rpm-build |
ca19af |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
rpm-build |
ca19af |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
rpm-build |
ca19af |
* GNU General Public License for more details.
|
|
rpm-build |
ca19af |
*
|
|
rpm-build |
ca19af |
* You should have received a copy of the GNU General Public License
|
|
rpm-build |
ca19af |
* along with pcaudiolib. If not, see <http://www.gnu.org/licenses/>.
|
|
rpm-build |
ca19af |
*/
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
#include "config.h"
|
|
rpm-build |
ca19af |
#include "audio_priv.h"
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
#ifdef HAVE_PULSE_SIMPLE_H
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
#include <pulse/error.h>
|
|
rpm-build |
ca19af |
#include <pulse/simple.h>
|
|
rpm-build |
ca19af |
#include <string.h>
|
|
rpm-build |
ca19af |
#include <stdbool.h>
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
struct pulseaudio_object
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
struct audio_object vtable;
|
|
rpm-build |
ca19af |
pa_sample_spec ss;
|
|
rpm-build |
ca19af |
pa_simple *s;
|
|
rpm-build |
ca19af |
char *device;
|
|
rpm-build |
ca19af |
char *application_name;
|
|
rpm-build |
ca19af |
char *description;
|
|
rpm-build |
ca19af |
};
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
#define to_pulseaudio_object(object) container_of(object, struct pulseaudio_object, vtable)
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int
|
|
rpm-build |
ca19af |
pulseaudio_object_open(struct audio_object *object,
|
|
rpm-build |
ca19af |
enum audio_object_format format,
|
|
rpm-build |
ca19af |
uint32_t rate,
|
|
rpm-build |
ca19af |
uint8_t channels)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
struct pulseaudio_object *self = to_pulseaudio_object(object);
|
|
rpm-build |
ca19af |
if (self->s)
|
|
rpm-build |
ca19af |
return PA_ERR_EXIST;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
self->ss.rate = rate;
|
|
rpm-build |
ca19af |
self->ss.channels = channels;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
switch (format)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_ALAW: self->ss.format = PA_SAMPLE_ALAW; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_ULAW: self->ss.format = PA_SAMPLE_ULAW; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_U8: self->ss.format = PA_SAMPLE_U8; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S16LE: self->ss.format = PA_SAMPLE_S16LE; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S16BE: self->ss.format = PA_SAMPLE_S16BE; break;
|
|
rpm-build |
ca19af |
#ifdef PA_SAMPLE_S24LE
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S24LE: self->ss.format = PA_SAMPLE_S24LE; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S24BE: self->ss.format = PA_SAMPLE_S24BE; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S24_32LE: self->ss.format = PA_SAMPLE_S24_32LE; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S24_32BE: self->ss.format = PA_SAMPLE_S24_32BE; break;
|
|
rpm-build |
ca19af |
#endif
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S32LE: self->ss.format = PA_SAMPLE_S32LE; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_S32BE: self->ss.format = PA_SAMPLE_S32BE; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_FLOAT32LE: self->ss.format = PA_SAMPLE_FLOAT32LE; break;
|
|
rpm-build |
ca19af |
case AUDIO_OBJECT_FORMAT_FLOAT32BE: self->ss.format = PA_SAMPLE_FLOAT32BE; break;
|
|
rpm-build |
ca19af |
default: return PA_ERR_INVALID; // Invalid argument.
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int error = 0;
|
|
rpm-build |
ca19af |
self->s = pa_simple_new(NULL,
|
|
rpm-build |
ca19af |
self->application_name,
|
|
rpm-build |
ca19af |
PA_STREAM_PLAYBACK,
|
|
rpm-build |
ca19af |
self->device,
|
|
rpm-build |
ca19af |
self->description,
|
|
rpm-build |
ca19af |
&self->ss,
|
|
rpm-build |
ca19af |
NULL,
|
|
rpm-build |
ca19af |
NULL,
|
|
rpm-build |
ca19af |
&error);
|
|
rpm-build |
ca19af |
return error;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
void
|
|
rpm-build |
ca19af |
pulseaudio_object_close(struct audio_object *object)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
struct pulseaudio_object *self = to_pulseaudio_object(object);
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
if (self->s) {
|
|
rpm-build |
ca19af |
pa_simple_free(self->s);
|
|
rpm-build |
ca19af |
self->s = NULL;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
void
|
|
rpm-build |
ca19af |
pulseaudio_object_destroy(struct audio_object *object)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
struct pulseaudio_object *self = to_pulseaudio_object(object);
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
free(self->device);
|
|
rpm-build |
ca19af |
free(self->application_name);
|
|
rpm-build |
ca19af |
free(self->description);
|
|
rpm-build |
ca19af |
free(self);
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int
|
|
rpm-build |
ca19af |
pulseaudio_object_drain(struct audio_object *object)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
struct pulseaudio_object *self = to_pulseaudio_object(object);
|
|
rpm-build |
ca19af |
if (!self->s)
|
|
rpm-build |
ca19af |
return 0;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int error = 0;
|
|
rpm-build |
ca19af |
pa_simple_drain(self->s, &error);
|
|
rpm-build |
ca19af |
return error;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int
|
|
rpm-build |
ca19af |
pulseaudio_object_flush(struct audio_object *object)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
struct pulseaudio_object *self = to_pulseaudio_object(object);
|
|
rpm-build |
ca19af |
if (!self->s)
|
|
rpm-build |
ca19af |
return 0;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int error = 0;
|
|
rpm-build |
ca19af |
pa_simple_flush(self->s, &error);
|
|
rpm-build |
ca19af |
return error;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int
|
|
rpm-build |
ca19af |
pulseaudio_object_write(struct audio_object *object,
|
|
rpm-build |
ca19af |
const void *data,
|
|
rpm-build |
ca19af |
size_t bytes)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
struct pulseaudio_object *self = to_pulseaudio_object(object);
|
|
rpm-build |
ca19af |
if (!self->s)
|
|
rpm-build |
ca19af |
return 0;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
int error = 0;
|
|
rpm-build |
ca19af |
pa_simple_write(self->s, data, bytes, &error);
|
|
rpm-build |
ca19af |
return error;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
const char *
|
|
rpm-build |
ca19af |
pulseaudio_object_strerror(struct audio_object *object,
|
|
rpm-build |
ca19af |
int error)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
return pa_strerror(error);
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
static bool
|
|
rpm-build |
ca19af |
pulseaudio_is_available(const char *device,
|
|
rpm-build |
ca19af |
const char *application_name,
|
|
rpm-build |
ca19af |
const char *description)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
pa_sample_spec ss;
|
|
rpm-build |
ca19af |
ss.format = PA_SAMPLE_S16LE;
|
|
rpm-build |
ca19af |
ss.rate = 44100;
|
|
rpm-build |
ca19af |
ss.channels = 1;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
pa_simple *s = pa_simple_new(NULL,
|
|
rpm-build |
ca19af |
application_name,
|
|
rpm-build |
ca19af |
PA_STREAM_PLAYBACK,
|
|
rpm-build |
ca19af |
device,
|
|
rpm-build |
ca19af |
description,
|
|
rpm-build |
ca19af |
&ss,
|
|
rpm-build |
ca19af |
NULL,
|
|
rpm-build |
ca19af |
NULL,
|
|
rpm-build |
ca19af |
NULL);
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
if (!s)
|
|
rpm-build |
ca19af |
return false;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
pa_simple_free(s);
|
|
rpm-build |
ca19af |
return true;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
struct audio_object *
|
|
rpm-build |
ca19af |
create_pulseaudio_object(const char *device,
|
|
rpm-build |
ca19af |
const char *application_name,
|
|
rpm-build |
ca19af |
const char *description)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
if (!pulseaudio_is_available(device, application_name, description))
|
|
rpm-build |
ca19af |
return NULL;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
struct pulseaudio_object *self = malloc(sizeof(struct pulseaudio_object));
|
|
rpm-build |
ca19af |
if (!self)
|
|
rpm-build |
ca19af |
return NULL;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
self->s = NULL;
|
|
rpm-build |
ca19af |
self->device = device ? strdup(device) : NULL;
|
|
rpm-build |
ca19af |
self->application_name = application_name ? strdup(application_name) : NULL;
|
|
rpm-build |
ca19af |
self->description = description ? strdup(description) : NULL;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
self->vtable.open = pulseaudio_object_open;
|
|
rpm-build |
ca19af |
self->vtable.close = pulseaudio_object_close;
|
|
rpm-build |
ca19af |
self->vtable.destroy = pulseaudio_object_destroy;
|
|
rpm-build |
ca19af |
self->vtable.write = pulseaudio_object_write;
|
|
rpm-build |
ca19af |
self->vtable.drain = pulseaudio_object_drain;
|
|
rpm-build |
ca19af |
self->vtable.flush = pulseaudio_object_flush;
|
|
rpm-build |
ca19af |
self->vtable.strerror = pulseaudio_object_strerror;
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
return &self->vtable;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
#else
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
struct audio_object *
|
|
rpm-build |
ca19af |
create_pulseaudio_object(const char *device,
|
|
rpm-build |
ca19af |
const char *application_name,
|
|
rpm-build |
ca19af |
const char *description)
|
|
rpm-build |
ca19af |
{
|
|
rpm-build |
ca19af |
return NULL;
|
|
rpm-build |
ca19af |
}
|
|
rpm-build |
ca19af |
|
|
rpm-build |
ca19af |
#endif
|