--- Makefile 2012-11-07 09:46:17.000000000 +0100 +++ Makefile 2013-01-04 07:15:35.293069953 +0100 @@ -74,6 +74,7 @@ mikmod-objs := mikmod.lo modplug-objs := modplug.lo mpc-objs := mpc.lo vorbis-objs := vorbis.lo +opus-objs := opus.lo wavpack-objs := wavpack.lo wav-objs := wav.lo mp4-objs := mp4.lo @@ -88,6 +89,7 @@ ip-$(CONFIG_MIKMOD) += mikmod.so ip-$(CONFIG_MODPLUG) += modplug.so ip-$(CONFIG_MPC) += mpc.so ip-$(CONFIG_VORBIS) += vorbis.so +ip-$(CONFIG_OPUS) += opus.so ip-$(CONFIG_WAVPACK) += wavpack.so ip-$(CONFIG_WAV) += wav.so ip-$(CONFIG_MP4) += mp4.so @@ -102,6 +104,7 @@ $(mikmod-objs): CFLAGS += $(MIKMOD_CFLA $(modplug-objs): CFLAGS += $(MODPLUG_CFLAGS) $(mpc-objs): CFLAGS += $(MPC_CFLAGS) $(vorbis-objs): CFLAGS += $(VORBIS_CFLAGS) +$(opus-objs): CFLAGS += $(OPUS_CFLAGS) $(wavpack-objs): CFLAGS += $(WAVPACK_CFLAGS) $(mp4-objs): CFLAGS += $(MP4_CFLAGS) $(aac-objs): CFLAGS += $(AAC_CFLAGS) @@ -129,6 +132,9 @@ mpc.so: $(mpc-objs) $(libcmus-y) vorbis.so: $(vorbis-objs) $(libcmus-y) $(call cmd,ld_dl,$(VORBIS_LIBS)) +opus.so: $(opus-objs) $(libcmus-y) + $(call cmd,ld_dl,$(OPUS_LIBS)) + wavpack.so: $(wavpack-objs) $(libcmus-y) $(call cmd,ld_dl,$(WAVPACK_LIBS)) --- configure 2012-11-07 09:46:17.000000000 +0100 +++ configure 2013-01-04 07:15:35.293069953 +0100 @@ -228,6 +228,12 @@ check_vorbis() fi } +check_opus() +{ + pkg_config OPUS "opusfile" + return $? +} + check_wavpack() { pkg_config WAVPACK "wavpack" "" "-lwavpack" @@ -425,6 +431,7 @@ Optional Features: y/n CONFIG_MPC libmpcdec (Musepack .mpc, .mpp, .mp+) [auto] CONFIG_VORBIS Ogg/Vorbis (.ogg, application/ogg, audio/x-ogg) [auto] CONFIG_TREMOR Use Tremor as Ogg/Vorbis input plugin [n] + CONFIG_OPUS Opus (.opus) [auto] CONFIG_WAV WAV [y] CONFIG_WAVPACK WavPack (.wv, audio/x-wavpack) [auto] CONFIG_MP4 MPEG-4 AAC (.mp4, .m4a, .m4b) [auto] @@ -485,6 +492,7 @@ check check_mikmod CONFIG_MIKMOD check check_modplug CONFIG_MODPLUG check check_mpc CONFIG_MPC check check_vorbis CONFIG_VORBIS +check check_opus CONFIG_OPUS check check_wavpack CONFIG_WAVPACK check check_mp4 CONFIG_MP4 check check_aac CONFIG_AAC @@ -534,7 +542,7 @@ config_header config/cue.h CONFIG_CUE CFLAGS="${CFLAGS} -DHAVE_CONFIG" makefile_vars bindir datadir libdir mandir exampledir -makefile_vars CONFIG_CDIO CONFIG_FLAC CONFIG_MAD CONFIG_MIKMOD CONFIG_MODPLUG CONFIG_MPC CONFIG_VORBIS CONFIG_WAVPACK CONFIG_WAV CONFIG_MP4 CONFIG_AAC CONFIG_FFMPEG CONFIG_CUE +makefile_vars CONFIG_CDIO CONFIG_FLAC CONFIG_MAD CONFIG_MIKMOD CONFIG_MODPLUG CONFIG_MPC CONFIG_VORBIS CONFIG_OPUS CONFIG_WAVPACK CONFIG_WAV CONFIG_MP4 CONFIG_AAC CONFIG_FFMPEG CONFIG_CUE makefile_vars CONFIG_ROAR CONFIG_PULSE CONFIG_ALSA CONFIG_AO CONFIG_ARTS CONFIG_OSS CONFIG_SUN CONFIG_WAVEOUT generate_config_mk --- opus.c 1970-01-01 01:00:00.000000000 +0100 +++ opus.c 2013-01-04 07:15:35.493069945 +0100 @@ -0,0 +1,335 @@ +/* + * Copyright 2008-2012 Various Authors + * Copyright 2004-2005 Timo Hirvonen + * + * 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 <http://www.gnu.org/licenses/>. + */ + +#include "ip.h" +#include "xmalloc.h" +#include "read_wrapper.h" +#include "debug.h" +#include "comment.h" + +#include <opusfile.h> + +#include <errno.h> +#include <string.h> +#include <sys/types.h> +#include <unistd.h> + +#define SAMPLING_RATE 48000 +#define CHANNELS 2 + +struct opus_private { + OggOpusFile *of; + int current_link; +}; + +static int read_func(void *datasource, unsigned char *ptr, int size) +{ + struct input_plugin_data *ip_data = datasource; + return read_wrapper(ip_data, ptr, size); +} + +static int seek_func(void *datasource, opus_int64 offset, int whence) +{ + struct input_plugin_data *ip_data = datasource; + return lseek(ip_data->fd, offset, whence); +} + +static int close_func(void *datasource) +{ + struct input_plugin_data *ip_data; + int rc; + + ip_data = datasource; + rc = close(ip_data->fd); + ip_data->fd = -1; + return rc; +} + +static opus_int64 tell_func(void *datasource) +{ + struct input_plugin_data *ip_data = datasource; + return lseek(ip_data->fd, 0, SEEK_CUR); +} + +static OpusFileCallbacks callbacks = { + .read = read_func, + .seek = seek_func, + .tell = tell_func, + .close = close_func +}; + +static int opus_open(struct input_plugin_data *ip_data) +{ + struct opus_private *priv; + int rc; + void *source; + + priv = xnew(struct opus_private, 1); + priv->current_link = -1; + priv->of = NULL; + + source = op_fdopen(&callbacks, ip_data->fd, "r"); + if (source == NULL) { + free(priv); + return -IP_ERROR_INTERNAL; + } + + priv->of = op_open_callbacks(source, &callbacks, NULL, 0, &rc); + if (rc != 0) { + d_print("op_open_callbacks failed: %d:%s\n", rc, strerror(rc)); + free(priv); + /* ogg is a container format, so it is likely to contain + * something else if it isn't opus */ + return -IP_ERROR_UNSUPPORTED_FILE_TYPE; + } + ip_data->private = priv; + + ip_data->sf = sf_rate(SAMPLING_RATE) + | sf_channels(CHANNELS) + | sf_bits(16) + | sf_signed(1); +#ifdef WORDS_BIGENDIAN + ip_data->sf |= sf_bigendian(1); +#endif + return 0; +} + +static int opus_close(struct input_plugin_data *ip_data) +{ + struct opus_private *priv = ip_data->private; + /* this closes ip_data->fd! */ + op_free(priv->of); + ip_data->fd = -1; + free(priv); + ip_data->private = NULL; + return 0; +} + +/* + * -n + * indicates error + * 0 + * indicates EOF + * n + * indicates actual number of bytes read + */ +static int opus_read(struct input_plugin_data *ip_data, char *buffer, int count) +{ + struct opus_private *priv; + int samples, current_link, rc; + + priv = ip_data->private; + + /* samples = number of samples read per channel */ + samples = op_read_stereo(priv->of, (opus_int16*)buffer, + count / sizeof(opus_int16)); + if (samples < 0) { + switch (samples) { + case OP_HOLE: + errno = EAGAIN; + rc = -1; + break; + case OP_EREAD: + errno = EINVAL; + rc = -1; + break; + case OP_EFAULT: + errno = EINVAL; + rc = -1; + break; + case OP_EIMPL: + rc = -IP_ERROR_FUNCTION_NOT_SUPPORTED; + break; + case OP_EINVAL: + errno = EINVAL; + rc = -1; + break; + case OP_ENOTFORMAT: + rc = -IP_ERROR_FILE_FORMAT; + break; + case OP_EBADHEADER: + rc = -IP_ERROR_FILE_FORMAT; + break; + case OP_EVERSION: + rc = -IP_ERROR_FILE_FORMAT; + break; + case OP_EBADPACKET: + errno = EINVAL; + rc = -1; + break; + case OP_EBADLINK: + errno = EINVAL; + rc = -1; + break; + case OP_EBADTIMESTAMP: + rc = -IP_ERROR_FILE_FORMAT; + break; + default: + d_print("error: %d\n", samples); + rc = -IP_ERROR_FILE_FORMAT; + } + } else if (samples == 0) { + /* EOF or buffer too small */ + rc = 0; + } else { + current_link = op_current_link(priv->of); + if (current_link < 0) { + d_print("error: %d\n", current_link); + rc = -1; + } else { + if (ip_data->remote && current_link != priv->current_link) { + ip_data->metadata_changed = 1; + priv->current_link = current_link; + } + + /* bytes = samples * channels * sample_size */ + rc = samples * CHANNELS * sizeof(opus_int16); + } + } + + return rc; +} + +static int opus_seek(struct input_plugin_data *ip_data, double offset) +{ + struct opus_private *priv; + int rc; + + priv = ip_data->private; + + rc = op_pcm_seek(priv->of, offset * SAMPLING_RATE); + switch (rc) { + case OP_ENOSEEK: + return -IP_ERROR_FUNCTION_NOT_SUPPORTED; + case OP_EINVAL: + return -IP_ERROR_INTERNAL; + case OP_EREAD: + return -IP_ERROR_INTERNAL; + case OP_EFAULT: + return -IP_ERROR_INTERNAL; + case OP_EBADLINK: + return -IP_ERROR_INTERNAL; + } + return 0; +} + +static int opus_read_comments(struct input_plugin_data *ip_data, + struct keyval **comments) +{ + GROWING_KEYVALS(c); + struct opus_private *priv; + const OpusTags *ot; + int i; + + priv = ip_data->private; + + ot = op_tags(priv->of, -1); + if (ot == NULL) { + d_print("ot == NULL\n"); + *comments = keyvals_new(0); + return 0; + } + + for (i = 0; i < ot->comments; i++) { + const char *str = ot->user_comments[i]; + const char *eq = strchr(str, '='); + char *key; + + if (!eq) { + d_print("invalid comment: '%s' ('=' expected)\n", str); + continue; + } + + key = xstrndup(str, eq - str); + comments_add_const(&c, key, eq + 1); + free(key); + } + keyvals_terminate(&c); + *comments = c.keyvals; + return 0; +} + +static int opus_duration(struct input_plugin_data *ip_data) +{ + struct opus_private *priv; + ogg_int64_t samples; + + priv = ip_data->private; + + samples = op_pcm_total(priv->of, -1); + if (samples < 0) + return -IP_ERROR_FUNCTION_NOT_SUPPORTED; + + return samples / SAMPLING_RATE; +} + +static long opus_bitrate(struct input_plugin_data *ip_data) +{ + struct opus_private *priv; + opus_int32 bitrate; + + priv = ip_data->private; + + bitrate = op_bitrate(priv->of, -1); + if (bitrate < 0) + return -IP_ERROR_FUNCTION_NOT_SUPPORTED; + else + return bitrate; +} + +static long opus_current_bitrate(struct input_plugin_data *ip_data) +{ + struct opus_private *priv; + opus_int32 bitrate; + + priv = ip_data->private; + + bitrate = op_bitrate_instant(priv->of); + if (bitrate < 0) + return -IP_ERROR_FUNCTION_NOT_SUPPORTED; + else + return bitrate; +} + +static char *opus_codec(struct input_plugin_data *ip_data) +{ + return xstrdup("opus"); +} + +static char *opus_codec_profile(struct input_plugin_data *ip_data) +{ + return NULL; +} + +const struct input_plugin_ops ip_ops = { + .open = opus_open, + .close = opus_close, + .read = opus_read, + .seek = opus_seek, + .read_comments = opus_read_comments, + .duration = opus_duration, + .bitrate = opus_bitrate, + .bitrate_current = opus_current_bitrate, + .codec = opus_codec, + .codec_profile = opus_codec_profile +}; + +const int ip_priority = 50; +const char * const ip_extensions[] = { "opus", NULL }; +const char * const ip_mime_types[] = { NULL }; +const char * const ip_options[] = { NULL };