/*
* Copyright 2008-2013 Various Authors
* Copyright 2010-2011 Philipp 'ph3-der-loewe' Schafft
* Copyright 2006 Johannes Weißl
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program 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
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, see .
*/
#include
#include "../op.h"
#include "../mixer.h"
#include "../xmalloc.h"
#include "../utils.h"
#include "../misc.h"
#include "../debug.h"
// we do not use native 2^16-1 here as they use signed ints with 16 bit
// so we use 2^(16-1)-1 here.
#define MIXER_BASE_VOLUME 32767
static struct roar_connection con;
static roar_vs_t *vss = NULL;
static int err;
static sample_format_t format;
/* configuration */
static char *host = NULL;
static char *role = NULL;
static inline void _err_to_errno(void)
{
roar_err_set(err);
roar_err_to_errno();
}
static int op_roar_dummy(void)
{
return 0;
}
#if DEBUG > 1
static ssize_t op_roar_debug_write(struct roar_vio_calls *vio, void *buf_, size_t count)
{
char *buf = (char *) buf_;
int len = count;
if (len > 0 && buf[len-1] == '\n')
len--;
if (len > 0)
d_print("%*s\n", len, buf);
return count;
}
static struct roar_vio_calls op_roar_debug_cbs = {
.write = op_roar_debug_write
};
#endif
static int op_roar_init(void)
{
#if DEBUG > 1
roar_debug_set_stderr_mode(ROAR_DEBUG_MODE_VIO);
roar_debug_set_stderr_vio(&op_roar_debug_cbs);
#else
roar_debug_set_stderr_mode(ROAR_DEBUG_MODE_SYSLOG);
#endif
return 0;
}
static int op_roar_exit(void)
{
free(host);
free(role);
return 0;
}
static int _set_role(void)
{
int roleid = ROAR_ROLE_UNKNOWN;
if (role == NULL)
return 0;
roleid = roar_str2role(role);
if (roleid == ROAR_ROLE_UNKNOWN) {
// TODO: warn if role is invalid.
return 0;
}
if (roar_vs_role(vss, roleid, &err) == -1) {
return -OP_ERROR_ERRNO;
}
return 0;
}
static int op_roar_open(sample_format_t sf, const channel_position_t *channel_map)
{
struct roar_audio_info info;
int ret;
memset(&info, 0, sizeof(info));
ROAR_DBG("op_roar_open(*) = ?");
format = sf;
info.rate = sf_get_rate(sf);
info.channels = sf_get_channels(sf);
info.bits = sf_get_bits(sf);
if (sf_get_bigendian(sf)) {
if (sf_get_signed(sf)) {
info.codec = ROAR_CODEC_PCM_S_BE;
} else {
info.codec = ROAR_CODEC_PCM_U_BE;
}
} else {
if (sf_get_signed(sf)) {
info.codec = ROAR_CODEC_PCM_S_LE;
} else {
info.codec = ROAR_CODEC_PCM_U_LE;
}
}
ROAR_DBG("op_roar_open(*) = ?");
if (roar_libroar_set_server(host) == -1) {
ROAR_DBG("op_roar_open(*) = ?");
roar_err_to_errno();
return -OP_ERROR_ERRNO;
}
if (roar_simple_connect2(&con, NULL, "C* Music Player (cmus)", ROAR_ENUM_FLAG_NONBLOCK, 0) == -1) {
ROAR_DBG("op_roar_open(*) = ?");
roar_err_to_errno();
return -OP_ERROR_ERRNO;
}
vss = roar_vs_new_from_con(&con, &err);
if (vss == NULL) {
ROAR_DBG("op_roar_open(*) = ?");
roar_disconnect(&con);
_err_to_errno();
return -OP_ERROR_ERRNO;
}
if (roar_vs_stream(vss, &info, ROAR_DIR_PLAY, &err) == -1) {
ROAR_DBG("op_roar_open(*) = ?");
roar_disconnect(&con);
_err_to_errno();
return -OP_ERROR_ERRNO;
}
ROAR_DBG("op_roar_open(*) = ?");
if (roar_vs_buffer(vss, 2048*8, &err) == -1) {
roar_vs_close(vss, ROAR_VS_TRUE, NULL);
roar_disconnect(&con);
_err_to_errno();
return -OP_ERROR_ERRNO;
}
ROAR_DBG("op_roar_open(*) = ?");
ret = _set_role();
if (ret != 0) {
roar_vs_close(vss, ROAR_VS_TRUE, NULL);
roar_disconnect(&con);
_err_to_errno();
return ret;
}
ROAR_DBG("op_roar_open(*) = ?");
if (roar_vs_blocking(vss, ROAR_VS_FALSE, &err) == -1) {
/* FIXME: handle this error */
}
ROAR_DBG("op_roar_open(*) = 0");
return 0;
}
static int op_roar_close(void)
{
roar_vs_close(vss, ROAR_VS_FALSE, &err);
roar_disconnect(&con);
return 0;
}
static int op_roar_drop(void)
{
if (roar_vs_reset_buffer(vss, ROAR_VS_TRUE, ROAR_VS_TRUE, &err) == -1) {
/* FIXME: handle this error
* FIXME: I'm deprecated */
}
return 0;
}
static int op_roar_write(const char *buffer, int count)
{
int ret;
int i;
ret = roar_vs_write(vss, buffer, count, &err);
for (i = 0; i < 8; i++)
roar_vs_iterate(vss, ROAR_VS_NOWAIT, NULL);
return ret;
}
static int op_roar_buffer_space(void)
{
ssize_t ret;
int i;
int fs = sf_get_frame_size(format);
for (i = 0; i < 8; i++)
roar_vs_iterate(vss, ROAR_VS_NOWAIT, NULL);
ret = roar_vs_get_avail_write(vss, &err);
ret -= ret % fs;
return ret;
}
static int op_roar_pause(void) {
if (roar_vs_pause(vss, ROAR_VS_TRUE, &err) == -1) {
_err_to_errno();
return -OP_ERROR_ERRNO;
}
return 0;
}
static int op_roar_unpause(void) {
if (roar_vs_pause(vss, ROAR_VS_FALSE, &err) == -1) {
_err_to_errno();
return -OP_ERROR_ERRNO;
}
return 0;
}
static int op_roar_mixer_open(int *volume_max)
{
*volume_max = MIXER_BASE_VOLUME;
return 0;
}
static int op_roar_mixer_set_volume(int l, int r)
{
float lf, rf;
if (vss == NULL)
return -OP_ERROR_NOT_OPEN;
lf = (float)l / (float)MIXER_BASE_VOLUME;
rf = (float)r / (float)MIXER_BASE_VOLUME;
if (roar_vs_volume_stereo(vss, lf, rf, &err) == -1) {
_err_to_errno();
return -OP_ERROR_ERRNO;
}
return 0;
}
static int op_roar_mixer_get_volume(int *l, int *r)
{
float lf, rf;
if (vss == NULL)
return -OP_ERROR_NOT_OPEN;
if (roar_vs_volume_get(vss, &lf, &rf, &err) == -1) {
_err_to_errno();
return -OP_ERROR_ERRNO;
}
lf *= (float)MIXER_BASE_VOLUME;
rf *= (float)MIXER_BASE_VOLUME;
*l = lf;
*r = rf;
return 0;
}
static int op_roar_set_server(const char *val)
{
free(host);
host = xstrdup(val);
return 0;
}
static int op_roar_get_server(char **val)
{
if (host != NULL)
*val = xstrdup(host);
return 0;
}
static int op_roar_set_role(const char *val)
{
free(host);
host = xstrdup(val);
return 0;
}
static int op_roar_get_role(char **val)
{
if (role != NULL)
*val = xstrdup(role);
return 0;
}
const struct output_plugin_ops op_pcm_ops = {
.init = op_roar_init,
.exit = op_roar_exit,
.open = op_roar_open,
.close = op_roar_close,
.drop = op_roar_drop,
.write = op_roar_write,
.buffer_space = op_roar_buffer_space,
.pause = op_roar_pause,
.unpause = op_roar_unpause,
};
const struct output_plugin_opt op_pcm_options[] = {
OPT(op_roar, server),
OPT(op_roar, role),
{ NULL },
};
const struct mixer_plugin_ops op_mixer_ops = {
.init = op_roar_dummy,
.exit = op_roar_dummy,
.open = op_roar_mixer_open,
.close = op_roar_dummy,
.get_fds = NULL,
.set_volume = op_roar_mixer_set_volume,
.get_volume = op_roar_mixer_get_volume,
};
const struct mixer_plugin_opt op_mixer_options[] = {
{ NULL },
};
const int op_priority = -1;
const unsigned op_abi_version = OP_ABI_VERSION;