diff options
author | Max Kellermann <max@duempel.org> | 2009-07-03 01:06:17 +0200 |
---|---|---|
committer | Max Kellermann <max@duempel.org> | 2009-07-03 01:06:17 +0200 |
commit | e3c436f4113086b5bc478aa87c369741b2bfecd3 (patch) | |
tree | 33f4955edff10f85116abf0ddbc9cbf4c4c26b08 | |
parent | 48f3e13beccec63de5322e2cc0c241527efeabef (diff) | |
download | mpd-e3c436f4113086b5bc478aa87c369741b2bfecd3.tar.gz mpd-e3c436f4113086b5bc478aa87c369741b2bfecd3.tar.xz mpd-e3c436f4113086b5bc478aa87c369741b2bfecd3.zip |
filter: added "volume" plugin
The "volume" filter plugin will replace the current software volume
code. One "volume" filter may be attached to each output device.
This will allow the user to use hardware mixers for some devices, and
software mixers for other devices at the same time.
Currently, neither the filter API nor the "volume" plugin is
integrated into MPD.
-rw-r--r-- | Makefile.am | 1 | ||||
-rw-r--r-- | src/filter/volume_filter_plugin.c | 136 | ||||
-rw-r--r-- | src/filter_registry.c | 1 | ||||
-rw-r--r-- | src/filter_registry.h | 1 |
4 files changed, 139 insertions, 0 deletions
diff --git a/Makefile.am b/Makefile.am index 0a5a1d505..a558ae488 100644 --- a/Makefile.am +++ b/Makefile.am @@ -194,6 +194,7 @@ src_mpd_SOURCES = \ src/filter_plugin.c \ src/filter_registry.c \ src/filter/null_filter_plugin.c \ + src/filter/volume_filter_plugin.c \ src/update.c \ src/client.c \ src/listen.c \ diff --git a/src/filter/volume_filter_plugin.c b/src/filter/volume_filter_plugin.c new file mode 100644 index 000000000..0a5e13a49 --- /dev/null +++ b/src/filter/volume_filter_plugin.c @@ -0,0 +1,136 @@ +/* + * Copyright (C) 2003-2009 The Music Player Daemon Project + * http://www.musicpd.org + * + * 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, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + */ + +#include "filter_plugin.h" +#include "filter_internal.h" +#include "filter_registry.h" +#include "conf.h" +#include "pcm_buffer.h" +#include "pcm_volume.h" +#include "volume.h" +#include "audio_format.h" + +#include <assert.h> +#include <string.h> + +struct volume_filter { + struct filter filter; + + struct audio_format audio_format; + + struct pcm_buffer buffer; +}; + +static inline GQuark +volume_quark(void) +{ + return g_quark_from_static_string("pcm_volume"); +} + +static struct filter * +volume_filter_init(G_GNUC_UNUSED const struct config_param *param, + G_GNUC_UNUSED GError **error_r) +{ + struct volume_filter *filter = g_new(struct volume_filter, 1); + + filter_init(&filter->filter, &volume_filter_plugin); + return &filter->filter; +} + +static void +volume_filter_finish(struct filter *filter) +{ + g_free(filter); +} + +static const struct audio_format * +volume_filter_open(struct filter *_filter, + const struct audio_format *audio_format, + GError **error_r) +{ + struct volume_filter *filter = (struct volume_filter *)_filter; + + if (audio_format->bits != 8 && audio_format->bits != 16 && + audio_format->bits != 24) { + g_set_error(error_r, volume_quark(), 0, + "Unsupported audio format"); + return false; + } + + filter->audio_format = *audio_format; + pcm_buffer_init(&filter->buffer); + + return &filter->audio_format; +} + +static void +volume_filter_close(struct filter *_filter) +{ + struct volume_filter *filter = (struct volume_filter *)_filter; + + pcm_buffer_deinit(&filter->buffer); +} + +static const void * +volume_filter_filter(struct filter *_filter, const void *src, size_t src_size, + size_t *dest_size_r, GError **error_r) +{ + struct volume_filter *filter = (struct volume_filter *)_filter; + int volume; + bool success; + void *dest; + + volume = volume_level_get(); /* XXX don't use volume_level_get() */ + if (volume < 0 || volume >= PCM_VOLUME_1) { + /* optimized special case: 100% volume = no-op */ + *dest_size_r = src_size; + return src; + } + + dest = pcm_buffer_get(&filter->buffer, src_size); + *dest_size_r = src_size; + + if (volume == 0) { + /* optimized special case: 0% volume = memset(0) */ + /* XXX is this valid for all sample formats? What + about floating point? */ + memset(dest, 0, src_size); + return dest; + } + + memcpy(dest, src, src_size); + + success = pcm_volume(dest, src_size, &filter->audio_format, volume); + if (!success) { + g_set_error(error_r, volume_quark(), 0, + "pcm_volume() has failed"); + return NULL; + } + + return dest; +} + +const struct filter_plugin volume_filter_plugin = { + .name = "volume", + .init = volume_filter_init, + .finish = volume_filter_finish, + .open = volume_filter_open, + .close = volume_filter_close, + .filter = volume_filter_filter, +}; diff --git a/src/filter_registry.c b/src/filter_registry.c index 90ba6fe1c..db47ab7df 100644 --- a/src/filter_registry.c +++ b/src/filter_registry.c @@ -25,6 +25,7 @@ const struct filter_plugin *const filter_plugins[] = { &null_filter_plugin, + &volume_filter_plugin, NULL, }; diff --git a/src/filter_registry.h b/src/filter_registry.h index e5988f96e..af873316e 100644 --- a/src/filter_registry.h +++ b/src/filter_registry.h @@ -27,6 +27,7 @@ #define MPD_FILTER_REGISTRY_H extern const struct filter_plugin null_filter_plugin; +extern const struct filter_plugin volume_filter_plugin; const struct filter_plugin * filter_plugin_by_name(const char *name); |