| /* GStreamer |
| * Copyright (C) 2004 Wim Taymans <wim@fluendo.com> |
| * Copyright (C) 2006 Tim-Philipp Müller <tim centricular net> |
| * Copyright (C) 2008 Sebastian Dröge <sebastian.droege@collabora.co.uk> |
| * Copyright (C) 2011-2012 Vincent Penquerc'h <vincent.penquerch@collabora.co.uk> |
| * |
| * 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. |
| */ |
| |
| /* |
| * Based on the speexdec element. |
| */ |
| |
| /** |
| * SECTION:element-opusdec |
| * @title: opusdec |
| * @see_also: opusenc, oggdemux |
| * |
| * This element decodes a OPUS stream to raw integer audio. |
| * |
| * ## Example pipelines |
| * |[ |
| * gst-launch-1.0 -v filesrc location=opus.ogg ! oggdemux ! opusdec ! audioconvert ! audioresample ! alsasink |
| * ]| |
| * Decode an Ogg/Opus file. To create an Ogg/Opus file refer to the documentation of opusenc. |
| * |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include "config.h" |
| #endif |
| |
| #include <math.h> |
| #include <string.h> |
| #include <stdio.h> |
| #include "gstopusheader.h" |
| #include "gstopuscommon.h" |
| #include "gstopusdec.h" |
| #include <gst/pbutils/pbutils.h> |
| |
| GST_DEBUG_CATEGORY_STATIC (opusdec_debug); |
| #define GST_CAT_DEFAULT opusdec_debug |
| |
| static GstStaticPadTemplate opus_dec_src_factory = |
| GST_STATIC_PAD_TEMPLATE ("src", |
| GST_PAD_SRC, |
| GST_PAD_ALWAYS, |
| GST_STATIC_CAPS ("audio/x-raw, " |
| "format = (string) " GST_AUDIO_NE (S16) ", " |
| "layout = (string) interleaved, " |
| "rate = (int) { 48000, 24000, 16000, 12000, 8000 }, " |
| "channels = (int) [ 1, 8 ] ") |
| ); |
| |
| static GstStaticPadTemplate opus_dec_sink_factory = |
| GST_STATIC_PAD_TEMPLATE ("sink", |
| GST_PAD_SINK, |
| GST_PAD_ALWAYS, |
| GST_STATIC_CAPS ("audio/x-opus, " |
| "channel-mapping-family = (int) 0; " |
| "audio/x-opus, " |
| "channel-mapping-family = (int) [1, 255], " |
| "channels = (int) [1, 255], " |
| "stream-count = (int) [1, 255], " "coupled-count = (int) [0, 255]") |
| ); |
| |
| G_DEFINE_TYPE (GstOpusDec, gst_opus_dec, GST_TYPE_AUDIO_DECODER); |
| |
| #define DB_TO_LINEAR(x) pow (10., (x) / 20.) |
| |
| #define DEFAULT_USE_INBAND_FEC FALSE |
| #define DEFAULT_APPLY_GAIN TRUE |
| |
| enum |
| { |
| PROP_0, |
| PROP_USE_INBAND_FEC, |
| PROP_APPLY_GAIN |
| }; |
| |
| |
| static GstFlowReturn gst_opus_dec_parse_header (GstOpusDec * dec, |
| GstBuffer * buf); |
| static gboolean gst_opus_dec_start (GstAudioDecoder * dec); |
| static gboolean gst_opus_dec_stop (GstAudioDecoder * dec); |
| static GstFlowReturn gst_opus_dec_handle_frame (GstAudioDecoder * dec, |
| GstBuffer * buffer); |
| static gboolean gst_opus_dec_set_format (GstAudioDecoder * bdec, |
| GstCaps * caps); |
| static void gst_opus_dec_get_property (GObject * object, guint prop_id, |
| GValue * value, GParamSpec * pspec); |
| static void gst_opus_dec_set_property (GObject * object, guint prop_id, |
| const GValue * value, GParamSpec * pspec); |
| static GstCaps *gst_opus_dec_getcaps (GstAudioDecoder * dec, GstCaps * filter); |
| |
| |
| static void |
| gst_opus_dec_class_init (GstOpusDecClass * klass) |
| { |
| GObjectClass *gobject_class; |
| GstAudioDecoderClass *adclass; |
| GstElementClass *element_class; |
| |
| gobject_class = (GObjectClass *) klass; |
| adclass = (GstAudioDecoderClass *) klass; |
| element_class = (GstElementClass *) klass; |
| |
| gobject_class->set_property = gst_opus_dec_set_property; |
| gobject_class->get_property = gst_opus_dec_get_property; |
| |
| adclass->start = GST_DEBUG_FUNCPTR (gst_opus_dec_start); |
| adclass->stop = GST_DEBUG_FUNCPTR (gst_opus_dec_stop); |
| adclass->handle_frame = GST_DEBUG_FUNCPTR (gst_opus_dec_handle_frame); |
| adclass->set_format = GST_DEBUG_FUNCPTR (gst_opus_dec_set_format); |
| adclass->getcaps = GST_DEBUG_FUNCPTR (gst_opus_dec_getcaps); |
| |
| gst_element_class_add_static_pad_template (element_class, |
| &opus_dec_src_factory); |
| gst_element_class_add_static_pad_template (element_class, |
| &opus_dec_sink_factory); |
| gst_element_class_set_static_metadata (element_class, "Opus audio decoder", |
| "Codec/Decoder/Audio", "decode opus streams to audio", |
| "Vincent Penquerc'h <vincent.penquerch@collabora.co.uk>"); |
| g_object_class_install_property (gobject_class, PROP_USE_INBAND_FEC, |
| g_param_spec_boolean ("use-inband-fec", "Use in-band FEC", |
| "Use forward error correction if available (needs PLC enabled)", |
| DEFAULT_USE_INBAND_FEC, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| g_object_class_install_property (gobject_class, PROP_APPLY_GAIN, |
| g_param_spec_boolean ("apply-gain", "Apply gain", |
| "Apply gain if any is specified in the header", DEFAULT_APPLY_GAIN, |
| G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); |
| |
| GST_DEBUG_CATEGORY_INIT (opusdec_debug, "opusdec", 0, |
| "opus decoding element"); |
| } |
| |
| static void |
| gst_opus_dec_reset (GstOpusDec * dec) |
| { |
| dec->packetno = 0; |
| if (dec->state) { |
| opus_multistream_decoder_destroy (dec->state); |
| dec->state = NULL; |
| } |
| |
| gst_buffer_replace (&dec->streamheader, NULL); |
| gst_buffer_replace (&dec->vorbiscomment, NULL); |
| gst_buffer_replace (&dec->last_buffer, NULL); |
| dec->primed = FALSE; |
| |
| dec->pre_skip = 0; |
| dec->r128_gain = 0; |
| dec->sample_rate = 0; |
| dec->n_channels = 0; |
| dec->leftover_plc_duration = 0; |
| dec->last_known_buffer_duration = GST_CLOCK_TIME_NONE; |
| } |
| |
| static void |
| gst_opus_dec_init (GstOpusDec * dec) |
| { |
| dec->use_inband_fec = FALSE; |
| dec->apply_gain = DEFAULT_APPLY_GAIN; |
| |
| gst_audio_decoder_set_needs_format (GST_AUDIO_DECODER (dec), TRUE); |
| gst_audio_decoder_set_use_default_pad_acceptcaps (GST_AUDIO_DECODER_CAST |
| (dec), TRUE); |
| GST_PAD_SET_ACCEPT_TEMPLATE (GST_AUDIO_DECODER_SINK_PAD (dec)); |
| |
| gst_opus_dec_reset (dec); |
| } |
| |
| static gboolean |
| gst_opus_dec_start (GstAudioDecoder * dec) |
| { |
| GstOpusDec *odec = GST_OPUS_DEC (dec); |
| |
| gst_opus_dec_reset (odec); |
| |
| /* we know about concealment */ |
| gst_audio_decoder_set_plc_aware (dec, TRUE); |
| |
| if (odec->use_inband_fec) { |
| /* opusdec outputs samples directly from an input buffer, except if |
| * FEC is on, in which case it buffers one buffer in case one buffer |
| * goes missing. |
| */ |
| gst_audio_decoder_set_latency (dec, 120 * GST_MSECOND, 120 * GST_MSECOND); |
| } |
| |
| return TRUE; |
| } |
| |
| static gboolean |
| gst_opus_dec_stop (GstAudioDecoder * dec) |
| { |
| GstOpusDec *odec = GST_OPUS_DEC (dec); |
| |
| gst_opus_dec_reset (odec); |
| |
| return TRUE; |
| } |
| |
| static double |
| gst_opus_dec_get_r128_gain (gint16 r128_gain) |
| { |
| return r128_gain / (double) (1 << 8); |
| } |
| |
| static double |
| gst_opus_dec_get_r128_volume (gint16 r128_gain) |
| { |
| return DB_TO_LINEAR (gst_opus_dec_get_r128_gain (r128_gain)); |
| } |
| |
| static gboolean |
| gst_opus_dec_negotiate (GstOpusDec * dec, const GstAudioChannelPosition * pos) |
| { |
| GstCaps *caps = gst_pad_get_allowed_caps (GST_AUDIO_DECODER_SRC_PAD (dec)); |
| GstStructure *s; |
| GstAudioInfo info; |
| |
| if (caps) { |
| gint rate = dec->sample_rate, channels = dec->n_channels; |
| GstCaps *constraint, *inter; |
| |
| constraint = gst_caps_from_string ("audio/x-raw"); |
| if (dec->n_channels <= 2) { /* including 0 */ |
| gst_caps_set_simple (constraint, "channels", GST_TYPE_INT_RANGE, 1, 2, |
| NULL); |
| } else { |
| gst_caps_set_simple (constraint, "channels", G_TYPE_INT, dec->n_channels, |
| NULL); |
| } |
| |
| inter = gst_caps_intersect (caps, constraint); |
| gst_caps_unref (constraint); |
| |
| if (gst_caps_is_empty (inter)) { |
| GST_DEBUG_OBJECT (dec, "Empty intersection, failed to negotiate"); |
| gst_caps_unref (inter); |
| gst_caps_unref (caps); |
| return FALSE; |
| } |
| |
| inter = gst_caps_truncate (inter); |
| s = gst_caps_get_structure (inter, 0); |
| rate = dec->sample_rate > 0 ? dec->sample_rate : 48000; |
| gst_structure_fixate_field_nearest_int (s, "rate", dec->sample_rate); |
| gst_structure_get_int (s, "rate", &rate); |
| channels = dec->n_channels > 0 ? dec->n_channels : 2; |
| gst_structure_fixate_field_nearest_int (s, "channels", dec->n_channels); |
| gst_structure_get_int (s, "channels", &channels); |
| |
| gst_caps_unref (inter); |
| |
| dec->sample_rate = rate; |
| dec->n_channels = channels; |
| gst_caps_unref (caps); |
| } |
| |
| if (dec->n_channels == 0) { |
| GST_DEBUG_OBJECT (dec, "Using a default of 2 channels"); |
| dec->n_channels = 2; |
| pos = NULL; |
| } |
| |
| if (dec->sample_rate == 0) { |
| GST_DEBUG_OBJECT (dec, "Using a default of 48kHz sample rate"); |
| dec->sample_rate = 48000; |
| } |
| |
| GST_INFO_OBJECT (dec, "Negotiated %d channels, %d Hz", dec->n_channels, |
| dec->sample_rate); |
| |
| /* pass valid order to audio info */ |
| if (pos) { |
| memcpy (dec->opus_pos, pos, sizeof (pos[0]) * dec->n_channels); |
| gst_audio_channel_positions_to_valid_order (dec->opus_pos, dec->n_channels); |
| } |
| |
| /* set up source format */ |
| gst_audio_info_init (&info); |
| gst_audio_info_set_format (&info, GST_AUDIO_FORMAT_S16, |
| dec->sample_rate, dec->n_channels, pos ? dec->opus_pos : NULL); |
| gst_audio_decoder_set_output_format (GST_AUDIO_DECODER (dec), &info); |
| |
| /* but we still need the opus order for later reordering */ |
| if (pos) { |
| memcpy (dec->opus_pos, pos, sizeof (pos[0]) * dec->n_channels); |
| } else { |
| dec->opus_pos[0] = GST_AUDIO_CHANNEL_POSITION_INVALID; |
| } |
| |
| dec->info = info; |
| |
| return TRUE; |
| } |
| |
| static GstFlowReturn |
| gst_opus_dec_parse_header (GstOpusDec * dec, GstBuffer * buf) |
| { |
| GstAudioChannelPosition pos[64]; |
| const GstAudioChannelPosition *posn = NULL; |
| |
| if (!gst_opus_header_is_id_header (buf)) { |
| GST_ELEMENT_ERROR (dec, STREAM, FORMAT, (NULL), |
| ("Header is not an Opus ID header")); |
| return GST_FLOW_ERROR; |
| } |
| |
| if (!gst_codec_utils_opus_parse_header (buf, |
| &dec->sample_rate, |
| &dec->n_channels, |
| &dec->channel_mapping_family, |
| &dec->n_streams, |
| &dec->n_stereo_streams, |
| dec->channel_mapping, &dec->pre_skip, &dec->r128_gain)) { |
| GST_ELEMENT_ERROR (dec, STREAM, DECODE, (NULL), |
| ("Failed to parse Opus ID header")); |
| return GST_FLOW_ERROR; |
| } |
| dec->r128_gain_volume = gst_opus_dec_get_r128_volume (dec->r128_gain); |
| |
| GST_INFO_OBJECT (dec, |
| "Found pre-skip of %u samples, R128 gain %d (volume %f)", |
| dec->pre_skip, dec->r128_gain, dec->r128_gain_volume); |
| |
| if (dec->channel_mapping_family == 1) { |
| GST_INFO_OBJECT (dec, "Channel mapping family 1, Vorbis mapping"); |
| switch (dec->n_channels) { |
| case 1: |
| case 2: |
| /* nothing */ |
| break; |
| case 3: |
| case 4: |
| case 5: |
| case 6: |
| case 7: |
| case 8: |
| posn = gst_opus_channel_positions[dec->n_channels - 1]; |
| break; |
| default:{ |
| gint i; |
| |
| GST_ELEMENT_WARNING (GST_ELEMENT (dec), STREAM, DECODE, |
| (NULL), ("Using NONE channel layout for more than 8 channels")); |
| |
| for (i = 0; i < dec->n_channels; i++) |
| pos[i] = GST_AUDIO_CHANNEL_POSITION_NONE; |
| |
| posn = pos; |
| } |
| } |
| } else { |
| GST_INFO_OBJECT (dec, "Channel mapping family %d", |
| dec->channel_mapping_family); |
| } |
| |
| if (!gst_opus_dec_negotiate (dec, posn)) |
| return GST_FLOW_NOT_NEGOTIATED; |
| |
| return GST_FLOW_OK; |
| } |
| |
| |
| static GstFlowReturn |
| gst_opus_dec_parse_comments (GstOpusDec * dec, GstBuffer * buf) |
| { |
| return GST_FLOW_OK; |
| } |
| |
| /* adapted from ext/ogg/gstoggstream.c */ |
| static gint64 |
| packet_duration_opus (const unsigned char *data, size_t bytes) |
| { |
| static const guint64 durations[32] = { |
| 480, 960, 1920, 2880, /* Silk NB */ |
| 480, 960, 1920, 2880, /* Silk MB */ |
| 480, 960, 1920, 2880, /* Silk WB */ |
| 480, 960, /* Hybrid SWB */ |
| 480, 960, /* Hybrid FB */ |
| 120, 240, 480, 960, /* CELT NB */ |
| 120, 240, 480, 960, /* CELT NB */ |
| 120, 240, 480, 960, /* CELT NB */ |
| 120, 240, 480, 960, /* CELT NB */ |
| }; |
| |
| gint64 duration; |
| gint64 frame_duration; |
| gint nframes = 0; |
| guint8 toc; |
| |
| if (bytes < 1) |
| return 0; |
| |
| /* headers */ |
| if (bytes >= 8 && !memcmp (data, "Opus", 4)) |
| return 0; |
| |
| toc = data[0]; |
| |
| frame_duration = durations[toc >> 3]; |
| switch (toc & 3) { |
| case 0: |
| nframes = 1; |
| break; |
| case 1: |
| nframes = 2; |
| break; |
| case 2: |
| nframes = 2; |
| break; |
| case 3: |
| if (bytes < 2) { |
| GST_WARNING ("Code 3 Opus packet has less than 2 bytes"); |
| return 0; |
| } |
| nframes = data[1] & 63; |
| break; |
| } |
| |
| duration = nframes * frame_duration; |
| if (duration > 5760) { |
| GST_WARNING ("Opus packet duration > 120 ms, invalid"); |
| return 0; |
| } |
| GST_LOG ("Opus packet: frame size %.1f ms, %d frames, duration %.1f ms", |
| frame_duration / 48.f, nframes, duration / 48.f); |
| return duration / 48.f * 1000000; |
| } |
| |
| static GstFlowReturn |
| opus_dec_chain_parse_data (GstOpusDec * dec, GstBuffer * buffer) |
| { |
| GstFlowReturn res = GST_FLOW_OK; |
| gsize size; |
| guint8 *data; |
| GstBuffer *outbuf, *bufd; |
| gint16 *out_data; |
| int n, err; |
| int samples; |
| unsigned int packet_size; |
| GstBuffer *buf; |
| GstMapInfo map, omap; |
| GstAudioClippingMeta *cmeta = NULL; |
| |
| if (dec->state == NULL) { |
| /* If we did not get any headers, default to 2 channels */ |
| if (dec->n_channels == 0) { |
| GST_INFO_OBJECT (dec, "No header, assuming single stream"); |
| dec->n_channels = 2; |
| dec->sample_rate = 48000; |
| /* default stereo mapping */ |
| dec->channel_mapping_family = 0; |
| dec->channel_mapping[0] = 0; |
| dec->channel_mapping[1] = 1; |
| dec->n_streams = 1; |
| dec->n_stereo_streams = 1; |
| |
| if (!gst_opus_dec_negotiate (dec, NULL)) |
| return GST_FLOW_NOT_NEGOTIATED; |
| } |
| |
| if (dec->n_channels == 2 && dec->n_streams == 1 |
| && dec->n_stereo_streams == 0) { |
| /* if we are automatically decoding 2 channels, but only have |
| a single encoded one, direct both channels to it */ |
| dec->channel_mapping[1] = 0; |
| } |
| |
| GST_DEBUG_OBJECT (dec, "Creating decoder with %d channels, %d Hz", |
| dec->n_channels, dec->sample_rate); |
| #ifndef GST_DISABLE_GST_DEBUG |
| gst_opus_common_log_channel_mapping_table (GST_ELEMENT (dec), opusdec_debug, |
| "Mapping table", dec->n_channels, dec->channel_mapping); |
| #endif |
| |
| GST_DEBUG_OBJECT (dec, "%d streams, %d stereo", dec->n_streams, |
| dec->n_stereo_streams); |
| dec->state = |
| opus_multistream_decoder_create (dec->sample_rate, dec->n_channels, |
| dec->n_streams, dec->n_stereo_streams, dec->channel_mapping, &err); |
| if (!dec->state || err != OPUS_OK) |
| goto creation_failed; |
| } |
| |
| if (buffer) { |
| GST_DEBUG_OBJECT (dec, "Received buffer of size %" G_GSIZE_FORMAT, |
| gst_buffer_get_size (buffer)); |
| } else { |
| GST_DEBUG_OBJECT (dec, "Received missing buffer"); |
| } |
| |
| /* if using in-band FEC, we introdude one extra frame's delay as we need |
| to potentially wait for next buffer to decode a missing buffer */ |
| if (dec->use_inband_fec && !dec->primed) { |
| GST_DEBUG_OBJECT (dec, "First buffer received in FEC mode, early out"); |
| gst_buffer_replace (&dec->last_buffer, buffer); |
| dec->primed = TRUE; |
| goto done; |
| } |
| |
| /* That's the buffer we'll be sending to the opus decoder. */ |
| buf = (dec->use_inband_fec |
| && gst_buffer_get_size (dec->last_buffer) > |
| 0) ? dec->last_buffer : buffer; |
| |
| /* That's the buffer we get duration from */ |
| bufd = dec->use_inband_fec ? dec->last_buffer : buffer; |
| |
| if (buf && gst_buffer_get_size (buf) > 0) { |
| gst_buffer_map (buf, &map, GST_MAP_READ); |
| data = map.data; |
| size = map.size; |
| GST_DEBUG_OBJECT (dec, "Using buffer of size %" G_GSIZE_FORMAT, size); |
| } else { |
| /* concealment data, pass NULL as the bits parameters */ |
| GST_DEBUG_OBJECT (dec, "Using NULL buffer"); |
| data = NULL; |
| size = 0; |
| } |
| |
| if (gst_buffer_get_size (bufd) == 0) { |
| GstClockTime const opus_plc_alignment = 2500 * GST_USECOND; |
| GstClockTime aligned_missing_duration; |
| GstClockTime missing_duration = GST_BUFFER_DURATION (bufd); |
| |
| if (!GST_CLOCK_TIME_IS_VALID (missing_duration) || missing_duration == 0) { |
| if (GST_CLOCK_TIME_IS_VALID (dec->last_known_buffer_duration)) { |
| missing_duration = dec->last_known_buffer_duration; |
| GST_WARNING_OBJECT (dec, |
| "Missing duration, using last duration %" GST_TIME_FORMAT, |
| GST_TIME_ARGS (missing_duration)); |
| } else { |
| GST_WARNING_OBJECT (dec, |
| "Missing buffer, but unknown duration, and no previously known duration, assuming 20 ms"); |
| missing_duration = 20 * GST_MSECOND; |
| } |
| } |
| |
| GST_DEBUG_OBJECT (dec, |
| "missing buffer, doing PLC duration %" GST_TIME_FORMAT |
| " plus leftover %" GST_TIME_FORMAT, GST_TIME_ARGS (missing_duration), |
| GST_TIME_ARGS (dec->leftover_plc_duration)); |
| |
| /* add the leftover PLC duration to that of the buffer */ |
| missing_duration += dec->leftover_plc_duration; |
| |
| /* align the combined buffer and leftover PLC duration to multiples |
| * of 2.5ms, rounding to nearest, and store excess duration for later */ |
| aligned_missing_duration = |
| ((missing_duration + |
| opus_plc_alignment / 2) / opus_plc_alignment) * opus_plc_alignment; |
| dec->leftover_plc_duration = missing_duration - aligned_missing_duration; |
| |
| /* Opus' PLC cannot operate with less than 2.5ms; skip PLC |
| * and accumulate the missing duration in the leftover_plc_duration |
| * for the next PLC attempt */ |
| if (aligned_missing_duration < opus_plc_alignment) { |
| GST_DEBUG_OBJECT (dec, |
| "current duration %" GST_TIME_FORMAT |
| " of missing data not enough for PLC (minimum needed: %" |
| GST_TIME_FORMAT ") - skipping", GST_TIME_ARGS (missing_duration), |
| GST_TIME_ARGS (opus_plc_alignment)); |
| goto done; |
| } |
| |
| /* convert the duration (in nanoseconds) to sample count */ |
| samples = |
| gst_util_uint64_scale_int (aligned_missing_duration, dec->sample_rate, |
| GST_SECOND); |
| |
| GST_DEBUG_OBJECT (dec, |
| "calculated PLC frame length: %" GST_TIME_FORMAT |
| " num frame samples: %d new leftover: %" GST_TIME_FORMAT, |
| GST_TIME_ARGS (aligned_missing_duration), samples, |
| GST_TIME_ARGS (dec->leftover_plc_duration)); |
| } else { |
| /* use maximum size (120 ms) as the number of returned samples is |
| not constant over the stream. */ |
| samples = 120 * dec->sample_rate / 1000; |
| } |
| packet_size = samples * dec->n_channels * 2; |
| |
| outbuf = |
| gst_audio_decoder_allocate_output_buffer (GST_AUDIO_DECODER (dec), |
| packet_size); |
| if (!outbuf) { |
| goto buffer_failed; |
| } |
| |
| if (size > 0) |
| dec->last_known_buffer_duration = packet_duration_opus (data, size); |
| |
| gst_buffer_map (outbuf, &omap, GST_MAP_WRITE); |
| out_data = (gint16 *) omap.data; |
| |
| do { |
| if (dec->use_inband_fec) { |
| if (gst_buffer_get_size (dec->last_buffer) > 0) { |
| /* normal delayed decode */ |
| GST_LOG_OBJECT (dec, "FEC enabled, decoding last delayed buffer"); |
| n = opus_multistream_decode (dec->state, data, size, out_data, samples, |
| 0); |
| } else { |
| /* FEC reconstruction decode */ |
| GST_LOG_OBJECT (dec, "FEC enabled, reconstructing last buffer"); |
| n = opus_multistream_decode (dec->state, data, size, out_data, samples, |
| 1); |
| } |
| } else { |
| /* normal decode */ |
| GST_LOG_OBJECT (dec, "FEC disabled, decoding buffer"); |
| n = opus_multistream_decode (dec->state, data, size, out_data, samples, |
| 0); |
| } |
| if (n == OPUS_BUFFER_TOO_SMALL) { |
| /* if too small, add 2.5 milliseconds and try again, up to the |
| * Opus max size of 120 milliseconds */ |
| if (samples >= 120 * dec->sample_rate / 1000) |
| break; |
| samples += 25 * dec->sample_rate / 10000; |
| packet_size = samples * dec->n_channels * 2; |
| gst_buffer_unmap (outbuf, &omap); |
| gst_buffer_unref (outbuf); |
| outbuf = |
| gst_audio_decoder_allocate_output_buffer (GST_AUDIO_DECODER (dec), |
| packet_size); |
| if (!outbuf) { |
| goto buffer_failed; |
| } |
| gst_buffer_map (outbuf, &omap, GST_MAP_WRITE); |
| out_data = (gint16 *) omap.data; |
| } |
| } while (n == OPUS_BUFFER_TOO_SMALL); |
| gst_buffer_unmap (outbuf, &omap); |
| if (data != NULL) |
| gst_buffer_unmap (buf, &map); |
| |
| if (n < 0) { |
| GstFlowReturn ret = GST_FLOW_ERROR; |
| |
| gst_buffer_unref (outbuf); |
| GST_AUDIO_DECODER_ERROR (dec, 1, STREAM, DECODE, (NULL), |
| ("Decoding error (%d): %s", n, opus_strerror (n)), ret); |
| return ret; |
| } |
| GST_DEBUG_OBJECT (dec, "decoded %d samples", n); |
| gst_buffer_set_size (outbuf, n * 2 * dec->n_channels); |
| GST_BUFFER_DURATION (outbuf) = samples * GST_SECOND / dec->sample_rate; |
| samples = n; |
| |
| cmeta = gst_buffer_get_audio_clipping_meta (buf); |
| |
| g_assert (!cmeta || cmeta->format == GST_FORMAT_DEFAULT); |
| |
| /* Skip any samples that need skipping */ |
| if (cmeta && cmeta->start) { |
| guint pre_skip = cmeta->start; |
| guint scaled_pre_skip = pre_skip * dec->sample_rate / 48000; |
| guint skip = scaled_pre_skip > n ? n : scaled_pre_skip; |
| guint scaled_skip = skip * 48000 / dec->sample_rate; |
| |
| gst_buffer_resize (outbuf, skip * 2 * dec->n_channels, -1); |
| |
| GST_INFO_OBJECT (dec, |
| "Skipping %u samples at the beginning (%u at 48000 Hz)", |
| skip, scaled_skip); |
| } |
| |
| if (cmeta && cmeta->end) { |
| guint post_skip = cmeta->end; |
| guint scaled_post_skip = post_skip * dec->sample_rate / 48000; |
| guint skip = scaled_post_skip > n ? n : scaled_post_skip; |
| guint scaled_skip = skip * 48000 / dec->sample_rate; |
| guint outsize = gst_buffer_get_size (outbuf); |
| guint skip_bytes = skip * 2 * dec->n_channels; |
| |
| if (outsize > skip_bytes) |
| outsize -= skip_bytes; |
| else |
| outsize = 0; |
| |
| gst_buffer_resize (outbuf, 0, outsize); |
| |
| GST_INFO_OBJECT (dec, |
| "Skipping %u samples at the end (%u at 48000 Hz)", skip, scaled_skip); |
| } |
| |
| if (gst_buffer_get_size (outbuf) == 0) { |
| gst_buffer_unref (outbuf); |
| outbuf = NULL; |
| } else if (dec->opus_pos[0] != GST_AUDIO_CHANNEL_POSITION_INVALID) { |
| gst_audio_buffer_reorder_channels (outbuf, GST_AUDIO_FORMAT_S16, |
| dec->n_channels, dec->opus_pos, dec->info.position); |
| } |
| |
| /* Apply gain */ |
| /* Would be better off leaving this to a volume element, as this is |
| a naive conversion that does too many int/float conversions. |
| However, we don't have control over the pipeline... |
| So make it optional if the user program wants to use a volume, |
| but do it by default so the correct volume goes out by default */ |
| if (dec->apply_gain && outbuf && dec->r128_gain) { |
| gsize rsize; |
| unsigned int i, nsamples; |
| double volume = dec->r128_gain_volume; |
| gint16 *samples; |
| |
| gst_buffer_map (outbuf, &omap, GST_MAP_READWRITE); |
| samples = (gint16 *) omap.data; |
| rsize = omap.size; |
| GST_DEBUG_OBJECT (dec, "Applying gain: volume %f", volume); |
| nsamples = rsize / 2; |
| for (i = 0; i < nsamples; ++i) { |
| int sample = (int) (samples[i] * volume + 0.5); |
| samples[i] = sample < -32768 ? -32768 : sample > 32767 ? 32767 : sample; |
| } |
| gst_buffer_unmap (outbuf, &omap); |
| } |
| |
| if (dec->use_inband_fec) { |
| gst_buffer_replace (&dec->last_buffer, buffer); |
| } |
| |
| res = gst_audio_decoder_finish_frame (GST_AUDIO_DECODER (dec), outbuf, 1); |
| |
| if (res != GST_FLOW_OK) |
| GST_DEBUG_OBJECT (dec, "flow: %s", gst_flow_get_name (res)); |
| |
| done: |
| return res; |
| |
| creation_failed: |
| GST_ELEMENT_ERROR (dec, LIBRARY, INIT, ("Failed to create Opus decoder"), |
| ("Failed to create Opus decoder (%d): %s", err, opus_strerror (err))); |
| return GST_FLOW_ERROR; |
| |
| buffer_failed: |
| GST_ELEMENT_ERROR (dec, STREAM, DECODE, (NULL), |
| ("Failed to create %u byte buffer", packet_size)); |
| return GST_FLOW_ERROR; |
| } |
| |
| static gboolean |
| gst_opus_dec_set_format (GstAudioDecoder * bdec, GstCaps * caps) |
| { |
| GstOpusDec *dec = GST_OPUS_DEC (bdec); |
| gboolean ret = TRUE; |
| GstStructure *s; |
| const GValue *streamheader; |
| GstCaps *old_caps; |
| |
| GST_DEBUG_OBJECT (dec, "set_format: %" GST_PTR_FORMAT, caps); |
| |
| if ((old_caps = gst_pad_get_current_caps (GST_AUDIO_DECODER_SINK_PAD (bdec)))) { |
| if (gst_caps_is_equal (caps, old_caps)) { |
| gst_caps_unref (old_caps); |
| GST_DEBUG_OBJECT (dec, "caps didn't change"); |
| goto done; |
| } |
| |
| GST_DEBUG_OBJECT (dec, "caps have changed, resetting decoder"); |
| gst_opus_dec_reset (dec); |
| gst_caps_unref (old_caps); |
| } |
| |
| s = gst_caps_get_structure (caps, 0); |
| if ((streamheader = gst_structure_get_value (s, "streamheader")) && |
| G_VALUE_HOLDS (streamheader, GST_TYPE_ARRAY) && |
| gst_value_array_get_size (streamheader) >= 2) { |
| const GValue *header, *vorbiscomment; |
| GstBuffer *buf; |
| GstFlowReturn res = GST_FLOW_OK; |
| |
| header = gst_value_array_get_value (streamheader, 0); |
| if (header && G_VALUE_HOLDS (header, GST_TYPE_BUFFER)) { |
| buf = gst_value_get_buffer (header); |
| res = gst_opus_dec_parse_header (dec, buf); |
| if (res != GST_FLOW_OK) { |
| ret = FALSE; |
| goto done; |
| } |
| gst_buffer_replace (&dec->streamheader, buf); |
| } |
| |
| vorbiscomment = gst_value_array_get_value (streamheader, 1); |
| if (vorbiscomment && G_VALUE_HOLDS (vorbiscomment, GST_TYPE_BUFFER)) { |
| buf = gst_value_get_buffer (vorbiscomment); |
| res = gst_opus_dec_parse_comments (dec, buf); |
| if (res != GST_FLOW_OK) { |
| ret = FALSE; |
| goto done; |
| } |
| gst_buffer_replace (&dec->vorbiscomment, buf); |
| } |
| } else { |
| const GstAudioChannelPosition *posn = NULL; |
| |
| if (!gst_codec_utils_opus_parse_caps (caps, &dec->sample_rate, |
| &dec->n_channels, &dec->channel_mapping_family, &dec->n_streams, |
| &dec->n_stereo_streams, dec->channel_mapping)) { |
| ret = FALSE; |
| goto done; |
| } |
| |
| if (dec->channel_mapping_family == 1 && dec->n_channels <= 8) |
| posn = gst_opus_channel_positions[dec->n_channels - 1]; |
| |
| if (!gst_opus_dec_negotiate (dec, posn)) |
| return FALSE; |
| } |
| |
| done: |
| return ret; |
| } |
| |
| static gboolean |
| memcmp_buffers (GstBuffer * buf1, GstBuffer * buf2) |
| { |
| gsize size1, size2; |
| gboolean res; |
| GstMapInfo map; |
| |
| size1 = gst_buffer_get_size (buf1); |
| size2 = gst_buffer_get_size (buf2); |
| |
| if (size1 != size2) |
| return FALSE; |
| |
| gst_buffer_map (buf1, &map, GST_MAP_READ); |
| res = gst_buffer_memcmp (buf2, 0, map.data, map.size) == 0; |
| gst_buffer_unmap (buf1, &map); |
| |
| return res; |
| } |
| |
| static GstFlowReturn |
| gst_opus_dec_handle_frame (GstAudioDecoder * adec, GstBuffer * buf) |
| { |
| GstFlowReturn res; |
| GstOpusDec *dec; |
| |
| /* no fancy draining */ |
| if (G_UNLIKELY (!buf)) |
| return GST_FLOW_OK; |
| |
| dec = GST_OPUS_DEC (adec); |
| GST_LOG_OBJECT (dec, |
| "Got buffer ts %" GST_TIME_FORMAT ", duration %" GST_TIME_FORMAT, |
| GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)), |
| GST_TIME_ARGS (GST_BUFFER_DURATION (buf))); |
| |
| /* If we have the streamheader and vorbiscomment from the caps already |
| * ignore them here */ |
| if (dec->streamheader && dec->vorbiscomment) { |
| if (memcmp_buffers (dec->streamheader, buf)) { |
| GST_DEBUG_OBJECT (dec, "found streamheader"); |
| gst_audio_decoder_finish_frame (adec, NULL, 1); |
| res = GST_FLOW_OK; |
| } else if (memcmp_buffers (dec->vorbiscomment, buf)) { |
| GST_DEBUG_OBJECT (dec, "found vorbiscomments"); |
| gst_audio_decoder_finish_frame (adec, NULL, 1); |
| res = GST_FLOW_OK; |
| } else { |
| res = opus_dec_chain_parse_data (dec, buf); |
| } |
| } else { |
| /* Otherwise fall back to packet counting and assume that the |
| * first two packets might be the headers, checking magic. */ |
| switch (dec->packetno) { |
| case 0: |
| if (gst_opus_header_is_header (buf, "OpusHead", 8)) { |
| GST_DEBUG_OBJECT (dec, "found streamheader"); |
| res = gst_opus_dec_parse_header (dec, buf); |
| gst_audio_decoder_finish_frame (adec, NULL, 1); |
| } else { |
| res = opus_dec_chain_parse_data (dec, buf); |
| } |
| break; |
| case 1: |
| if (gst_opus_header_is_header (buf, "OpusTags", 8)) { |
| GST_DEBUG_OBJECT (dec, "counted vorbiscomments"); |
| res = gst_opus_dec_parse_comments (dec, buf); |
| gst_audio_decoder_finish_frame (adec, NULL, 1); |
| } else { |
| res = opus_dec_chain_parse_data (dec, buf); |
| } |
| break; |
| default: |
| { |
| res = opus_dec_chain_parse_data (dec, buf); |
| break; |
| } |
| } |
| } |
| |
| dec->packetno++; |
| |
| return res; |
| } |
| |
| static void |
| gst_opus_dec_get_property (GObject * object, guint prop_id, GValue * value, |
| GParamSpec * pspec) |
| { |
| GstOpusDec *dec = GST_OPUS_DEC (object); |
| |
| switch (prop_id) { |
| case PROP_USE_INBAND_FEC: |
| g_value_set_boolean (value, dec->use_inband_fec); |
| break; |
| case PROP_APPLY_GAIN: |
| g_value_set_boolean (value, dec->apply_gain); |
| break; |
| default: |
| G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); |
| break; |
| } |
| } |
| |
| static void |
| gst_opus_dec_set_property (GObject * object, guint prop_id, |
| const GValue * value, GParamSpec * pspec) |
| { |
| GstOpusDec *dec = GST_OPUS_DEC (object); |
| |
| switch (prop_id) { |
| case PROP_USE_INBAND_FEC: |
| dec->use_inband_fec = g_value_get_boolean (value); |
| break; |
| case PROP_APPLY_GAIN: |
| dec->apply_gain = g_value_get_boolean (value); |
| break; |
| default: |
| G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); |
| break; |
| } |
| } |
| |
| /* caps must be writable */ |
| static void |
| gst_opus_dec_caps_extend_channels_options (GstCaps * caps) |
| { |
| unsigned n; |
| int channels; |
| |
| for (n = 0; n < gst_caps_get_size (caps); ++n) { |
| GstStructure *s = gst_caps_get_structure (caps, n); |
| if (gst_structure_get_int (s, "channels", &channels)) { |
| if (channels == 1 || channels == 2) { |
| GValue v = { 0 }; |
| g_value_init (&v, GST_TYPE_INT_RANGE); |
| gst_value_set_int_range (&v, 1, 2); |
| gst_structure_set_value (s, "channels", &v); |
| g_value_unset (&v); |
| } |
| } |
| } |
| } |
| |
| static void |
| gst_opus_dec_value_list_append_int (GValue * list, gint i) |
| { |
| GValue v = { 0 }; |
| |
| g_value_init (&v, G_TYPE_INT); |
| g_value_set_int (&v, i); |
| gst_value_list_append_value (list, &v); |
| g_value_unset (&v); |
| } |
| |
| static void |
| gst_opus_dec_caps_extend_rate_options (GstCaps * caps) |
| { |
| unsigned n; |
| GValue v = { 0 }; |
| |
| g_value_init (&v, GST_TYPE_LIST); |
| gst_opus_dec_value_list_append_int (&v, 48000); |
| gst_opus_dec_value_list_append_int (&v, 24000); |
| gst_opus_dec_value_list_append_int (&v, 16000); |
| gst_opus_dec_value_list_append_int (&v, 12000); |
| gst_opus_dec_value_list_append_int (&v, 8000); |
| |
| for (n = 0; n < gst_caps_get_size (caps); ++n) { |
| GstStructure *s = gst_caps_get_structure (caps, n); |
| |
| gst_structure_set_value (s, "rate", &v); |
| } |
| g_value_unset (&v); |
| } |
| |
| GstCaps * |
| gst_opus_dec_getcaps (GstAudioDecoder * dec, GstCaps * filter) |
| { |
| GstCaps *caps, *proxy_filter = NULL, *ret; |
| |
| if (filter) { |
| proxy_filter = gst_caps_copy (filter); |
| gst_opus_dec_caps_extend_channels_options (proxy_filter); |
| gst_opus_dec_caps_extend_rate_options (proxy_filter); |
| } |
| caps = gst_audio_decoder_proxy_getcaps (dec, NULL, proxy_filter); |
| if (proxy_filter) |
| gst_caps_unref (proxy_filter); |
| if (caps) { |
| caps = gst_caps_make_writable (caps); |
| gst_opus_dec_caps_extend_channels_options (caps); |
| gst_opus_dec_caps_extend_rate_options (caps); |
| } |
| |
| if (filter) { |
| ret = gst_caps_intersect (caps, filter); |
| gst_caps_unref (caps); |
| } else { |
| ret = caps; |
| } |
| return ret; |
| } |