mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-21 22:58:16 +00:00
1000 lines
29 KiB
C
1000 lines
29 KiB
C
/* GStreamer RealAudio demuxer
|
|
* Copyright (C) 2006 Tim-Philipp Müller <tim centricular net>
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* SECTION:element-rademux
|
|
* @title: rademux
|
|
*
|
|
* Demuxes/parses a RealAudio (.ra) file or stream into compressed audio.
|
|
*
|
|
* ## Example launch line
|
|
* |[
|
|
* gst-launch-1.0 filesrc location=interview.ra ! rademux ! avdec_real_288 ! audioconvert ! audioresample ! autoaudiosink
|
|
* ]| Read a RealAudio file and decode it and output it to the soundcard using
|
|
* the ALSA element. The .ra file is assumed to contain RealAudio version 2.
|
|
* |[
|
|
* gst-launch-1.0 souphttpsrc location=http://www.example.org/interview.ra ! rademux ! ac3parse ! a52dec ! audioconvert ! audioresample ! autoaudiosink
|
|
* ]| Stream RealAudio data containing AC3 (dnet) compressed audio and decode it
|
|
* and output it to the soundcard.
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "rademux.h"
|
|
#include "rmdemux.h"
|
|
#include "rmutils.h"
|
|
|
|
#include <string.h>
|
|
|
|
static GstStaticPadTemplate sink_template = GST_STATIC_PAD_TEMPLATE ("sink",
|
|
GST_PAD_SINK,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("application/x-pn-realaudio")
|
|
);
|
|
|
|
static GstStaticPadTemplate src_template = GST_STATIC_PAD_TEMPLATE ("src",
|
|
GST_PAD_SRC,
|
|
GST_PAD_SOMETIMES,
|
|
GST_STATIC_CAPS_ANY);
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (real_audio_demux_debug);
|
|
#define GST_CAT_DEFAULT real_audio_demux_debug
|
|
|
|
#define gst_real_audio_demux_parent_class parent_class
|
|
G_DEFINE_TYPE (GstRealAudioDemux, gst_real_audio_demux, GST_TYPE_ELEMENT);
|
|
GST_ELEMENT_REGISTER_DEFINE (rademux, "rademux",
|
|
GST_RANK_SECONDARY, GST_TYPE_REAL_AUDIO_DEMUX);
|
|
|
|
static GstStateChangeReturn gst_real_audio_demux_change_state (GstElement * e,
|
|
GstStateChange transition);
|
|
static GstFlowReturn gst_real_audio_demux_chain (GstPad * pad,
|
|
GstObject * parent, GstBuffer * buf);
|
|
static gboolean gst_real_audio_demux_sink_event (GstPad * pad,
|
|
GstObject * parent, GstEvent * ev);
|
|
static gboolean gst_real_audio_demux_src_event (GstPad * pad,
|
|
GstObject * parent, GstEvent * ev);
|
|
static gboolean gst_real_audio_demux_src_query (GstPad * pad,
|
|
GstObject * parent, GstQuery * query);
|
|
static void gst_real_audio_demux_loop (GstRealAudioDemux * demux);
|
|
static gboolean gst_real_audio_demux_sink_activate (GstPad * sinkpad,
|
|
GstObject * parent);
|
|
static gboolean gst_real_audio_demux_sink_activate_mode (GstPad * sinkpad,
|
|
GstObject * parent, GstPadMode mode, gboolean active);
|
|
|
|
static void
|
|
gst_real_audio_demux_finalize (GObject * obj)
|
|
{
|
|
GstRealAudioDemux *demux = GST_REAL_AUDIO_DEMUX (obj);
|
|
|
|
g_object_unref (demux->adapter);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (obj);
|
|
}
|
|
|
|
static void
|
|
gst_real_audio_demux_class_init (GstRealAudioDemuxClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = (GObjectClass *) klass;
|
|
GstElementClass *gstelement_class = (GstElementClass *) klass;
|
|
|
|
gobject_class->finalize = gst_real_audio_demux_finalize;
|
|
|
|
gst_element_class_add_static_pad_template (gstelement_class, &sink_template);
|
|
gst_element_class_add_static_pad_template (gstelement_class, &src_template);
|
|
|
|
gst_element_class_set_static_metadata (gstelement_class, "RealAudio Demuxer",
|
|
"Codec/Demuxer",
|
|
"Demultiplex a RealAudio file",
|
|
"Tim-Philipp Müller <tim centricular net>");
|
|
|
|
gstelement_class->change_state =
|
|
GST_DEBUG_FUNCPTR (gst_real_audio_demux_change_state);
|
|
|
|
GST_DEBUG_CATEGORY_INIT (real_audio_demux_debug, "rademux",
|
|
0, "Demuxer for RealAudio streams");
|
|
}
|
|
|
|
static void
|
|
gst_real_audio_demux_reset (GstRealAudioDemux * demux)
|
|
{
|
|
gst_adapter_clear (demux->adapter);
|
|
|
|
if (demux->srcpad) {
|
|
GST_DEBUG_OBJECT (demux, "Removing source pad");
|
|
gst_element_remove_pad (GST_ELEMENT (demux), demux->srcpad);
|
|
demux->srcpad = NULL;
|
|
}
|
|
|
|
if (demux->pending_tags) {
|
|
gst_tag_list_unref (demux->pending_tags);
|
|
demux->pending_tags = NULL;
|
|
}
|
|
|
|
demux->state = REAL_AUDIO_DEMUX_STATE_MARKER;
|
|
demux->ra_version = 0;
|
|
demux->data_offset = 0;
|
|
demux->packet_size = 0;
|
|
|
|
demux->sample_rate = 0;
|
|
demux->sample_width = 0;
|
|
demux->channels = 0;
|
|
demux->fourcc = 0;
|
|
|
|
demux->need_newsegment = TRUE;
|
|
|
|
demux->segment_running = FALSE;
|
|
|
|
demux->byterate_num = 0;
|
|
demux->byterate_denom = 0;
|
|
|
|
demux->duration = 0;
|
|
demux->upstream_size = 0;
|
|
|
|
demux->offset = 0;
|
|
|
|
demux->have_group_id = FALSE;
|
|
demux->group_id = G_MAXUINT;
|
|
|
|
gst_adapter_clear (demux->adapter);
|
|
}
|
|
|
|
static void
|
|
gst_real_audio_demux_init (GstRealAudioDemux * demux)
|
|
{
|
|
demux->sinkpad = gst_pad_new_from_static_template (&sink_template, "sink");
|
|
|
|
gst_pad_set_chain_function (demux->sinkpad,
|
|
GST_DEBUG_FUNCPTR (gst_real_audio_demux_chain));
|
|
gst_pad_set_event_function (demux->sinkpad,
|
|
GST_DEBUG_FUNCPTR (gst_real_audio_demux_sink_event));
|
|
gst_pad_set_activate_function (demux->sinkpad,
|
|
GST_DEBUG_FUNCPTR (gst_real_audio_demux_sink_activate));
|
|
gst_pad_set_activatemode_function (demux->sinkpad,
|
|
GST_DEBUG_FUNCPTR (gst_real_audio_demux_sink_activate_mode));
|
|
|
|
gst_element_add_pad (GST_ELEMENT (demux), demux->sinkpad);
|
|
|
|
demux->adapter = gst_adapter_new ();
|
|
gst_real_audio_demux_reset (demux);
|
|
}
|
|
|
|
static gboolean
|
|
gst_real_audio_demux_sink_activate (GstPad * sinkpad, GstObject * parent)
|
|
{
|
|
GstQuery *query;
|
|
gboolean pull_mode;
|
|
|
|
query = gst_query_new_scheduling ();
|
|
|
|
if (!gst_pad_peer_query (sinkpad, query)) {
|
|
gst_query_unref (query);
|
|
goto activate_push;
|
|
}
|
|
|
|
pull_mode = gst_query_has_scheduling_mode_with_flags (query,
|
|
GST_PAD_MODE_PULL, GST_SCHEDULING_FLAG_SEEKABLE);
|
|
gst_query_unref (query);
|
|
|
|
if (!pull_mode)
|
|
goto activate_push;
|
|
|
|
GST_DEBUG_OBJECT (sinkpad, "activating pull");
|
|
return gst_pad_activate_mode (sinkpad, GST_PAD_MODE_PULL, TRUE);
|
|
|
|
activate_push:
|
|
{
|
|
GST_DEBUG_OBJECT (sinkpad, "activating push");
|
|
return gst_pad_activate_mode (sinkpad, GST_PAD_MODE_PUSH, TRUE);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_real_audio_demux_sink_activate_mode (GstPad * sinkpad, GstObject * parent,
|
|
GstPadMode mode, gboolean active)
|
|
{
|
|
gboolean res;
|
|
GstRealAudioDemux *demux;
|
|
|
|
demux = GST_REAL_AUDIO_DEMUX (parent);
|
|
|
|
switch (mode) {
|
|
case GST_PAD_MODE_PUSH:
|
|
demux->seekable = FALSE;
|
|
res = TRUE;
|
|
break;
|
|
case GST_PAD_MODE_PULL:
|
|
if (active) {
|
|
demux->seekable = TRUE;
|
|
|
|
res = gst_pad_start_task (sinkpad,
|
|
(GstTaskFunction) gst_real_audio_demux_loop, demux, NULL);
|
|
} else {
|
|
demux->seekable = FALSE;
|
|
res = gst_pad_stop_task (sinkpad);
|
|
}
|
|
break;
|
|
default:
|
|
res = FALSE;
|
|
break;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_real_audio_demux_parse_marker (GstRealAudioDemux * demux)
|
|
{
|
|
guint8 data[6];
|
|
|
|
if (gst_adapter_available (demux->adapter) < 6) {
|
|
GST_LOG_OBJECT (demux, "need at least 6 bytes, waiting for more data");
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
gst_adapter_copy (demux->adapter, data, 0, 6);
|
|
if (memcmp (data, ".ra\375", 4) != 0)
|
|
goto wrong_format;
|
|
|
|
demux->ra_version = GST_READ_UINT16_BE (data + 4);
|
|
GST_DEBUG_OBJECT (demux, "ra_version = %u", demux->ra_version);
|
|
if (demux->ra_version != 4 && demux->ra_version != 3)
|
|
goto unsupported_ra_version;
|
|
|
|
gst_adapter_flush (demux->adapter, 6);
|
|
demux->state = REAL_AUDIO_DEMUX_STATE_HEADER;
|
|
return GST_FLOW_OK;
|
|
|
|
/* ERRORS */
|
|
wrong_format:
|
|
{
|
|
GST_ELEMENT_ERROR (GST_ELEMENT (demux), STREAM, WRONG_TYPE, (NULL), (NULL));
|
|
return GST_FLOW_ERROR;
|
|
}
|
|
|
|
unsupported_ra_version:
|
|
{
|
|
GST_ELEMENT_ERROR (GST_ELEMENT (demux), STREAM, DECODE,
|
|
("Cannot decode this RealAudio file, please file a bug"),
|
|
("ra_version = %u", demux->ra_version));
|
|
return GST_FLOW_ERROR;
|
|
}
|
|
}
|
|
|
|
static GstClockTime
|
|
gst_real_demux_get_timestamp_from_offset (GstRealAudioDemux * demux,
|
|
guint64 offset)
|
|
{
|
|
if (offset >= demux->data_offset && demux->byterate_num > 0 &&
|
|
demux->byterate_denom > 0) {
|
|
return gst_util_uint64_scale (offset - demux->data_offset,
|
|
demux->byterate_denom * GST_SECOND, demux->byterate_num);
|
|
} else if (offset == demux->data_offset) {
|
|
return (GstClockTime) 0;
|
|
} else {
|
|
return GST_CLOCK_TIME_NONE;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_real_audio_demux_get_data_offset_from_header (GstRealAudioDemux * demux)
|
|
{
|
|
guint8 data[16];
|
|
|
|
gst_adapter_copy (demux->adapter, data, 0, 16);
|
|
|
|
switch (demux->ra_version) {
|
|
case 3:
|
|
demux->data_offset = GST_READ_UINT16_BE (data) + 8;
|
|
break;
|
|
case 4:
|
|
demux->data_offset = GST_READ_UINT32_BE (data + 12) + 16;
|
|
break;
|
|
default:
|
|
demux->data_offset = 0;
|
|
g_return_val_if_reached (FALSE);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_real_audio_demux_parse_header (GstRealAudioDemux * demux)
|
|
{
|
|
const guint8 *data;
|
|
gchar *codec_name = NULL;
|
|
GstCaps *caps = NULL;
|
|
GstEvent *event;
|
|
gchar *stream_id;
|
|
guint avail;
|
|
|
|
g_assert (demux->ra_version == 4 || demux->ra_version == 3);
|
|
|
|
avail = gst_adapter_available (demux->adapter);
|
|
if (avail < 16)
|
|
return GST_FLOW_OK;
|
|
|
|
if (!gst_real_audio_demux_get_data_offset_from_header (demux))
|
|
return GST_FLOW_ERROR; /* shouldn't happen */
|
|
|
|
GST_DEBUG_OBJECT (demux, "data_offset = %u", demux->data_offset);
|
|
|
|
if (avail + 6 < demux->data_offset) {
|
|
GST_DEBUG_OBJECT (demux, "Need %u bytes, but only %u available now",
|
|
demux->data_offset - 6, avail);
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
data = gst_adapter_map (demux->adapter, demux->data_offset - 6);
|
|
g_assert (data);
|
|
|
|
switch (demux->ra_version) {
|
|
case 3:
|
|
demux->fourcc = GST_RM_AUD_14_4;
|
|
demux->packet_size = 20;
|
|
demux->sample_rate = 8000;
|
|
demux->channels = 1;
|
|
demux->sample_width = 16;
|
|
demux->flavour = 1;
|
|
demux->leaf_size = 0;
|
|
demux->height = 0;
|
|
break;
|
|
case 4:
|
|
demux->flavour = GST_READ_UINT16_BE (data + 16);
|
|
/* demux->frame_size = GST_READ_UINT32_BE (data + 36); */
|
|
demux->leaf_size = GST_READ_UINT16_BE (data + 38);
|
|
demux->height = GST_READ_UINT16_BE (data + 34);
|
|
demux->packet_size = GST_READ_UINT32_BE (data + 18);
|
|
demux->sample_rate = GST_READ_UINT16_BE (data + 42);
|
|
demux->sample_width = GST_READ_UINT16_BE (data + 46);
|
|
demux->channels = GST_READ_UINT16_BE (data + 48);
|
|
demux->fourcc = GST_READ_UINT32_LE (data + 56);
|
|
demux->pending_tags = gst_rm_utils_read_tags (data + 63,
|
|
demux->data_offset - 63, gst_rm_utils_read_string8);
|
|
if (demux->pending_tags)
|
|
gst_tag_list_set_scope (demux->pending_tags, GST_TAG_SCOPE_GLOBAL);
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
#if 0
|
|
case 5:
|
|
demux->flavour = GST_READ_UINT16_BE (data + 16);
|
|
/* demux->frame_size = GST_READ_UINT32_BE (data + 36); */
|
|
demux->leaf_size = GST_READ_UINT16_BE (data + 38);
|
|
demux->height = GST_READ_UINT16_BE (data + 34);
|
|
|
|
demux->sample_rate = GST_READ_UINT16_BE (data + 48);
|
|
demux->sample_width = GST_READ_UINT16_BE (data + 52);
|
|
demux->n_channels = GST_READ_UINT16_BE (data + 54);
|
|
demux->fourcc = RMDEMUX_FOURCC_GET (data + 60);
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
GST_INFO_OBJECT (demux, "packet_size = %u", demux->packet_size);
|
|
GST_INFO_OBJECT (demux, "sample_rate = %u", demux->sample_rate);
|
|
GST_INFO_OBJECT (demux, "sample_width = %u", demux->sample_width);
|
|
GST_INFO_OBJECT (demux, "channels = %u", demux->channels);
|
|
GST_INFO_OBJECT (demux, "fourcc = '%" GST_FOURCC_FORMAT "' (%08X)",
|
|
GST_FOURCC_ARGS (demux->fourcc), demux->fourcc);
|
|
|
|
switch (demux->fourcc) {
|
|
case GST_RM_AUD_14_4:
|
|
caps = gst_caps_new_simple ("audio/x-pn-realaudio", "raversion",
|
|
G_TYPE_INT, 1, NULL);
|
|
demux->byterate_num = 1000;
|
|
demux->byterate_denom = 1;
|
|
break;
|
|
|
|
case GST_RM_AUD_28_8:
|
|
/* FIXME: needs descrambling */
|
|
caps = gst_caps_new_simple ("audio/x-pn-realaudio", "raversion",
|
|
G_TYPE_INT, 2, NULL);
|
|
break;
|
|
|
|
case GST_RM_AUD_DNET:
|
|
caps = gst_caps_new_simple ("audio/x-ac3", "rate", G_TYPE_INT,
|
|
demux->sample_rate, NULL);
|
|
if (demux->packet_size == 0 || demux->sample_rate == 0)
|
|
goto broken_file;
|
|
demux->byterate_num = demux->packet_size * demux->sample_rate;
|
|
demux->byterate_denom = 1536;
|
|
break;
|
|
|
|
/* Sipro/ACELP.NET Voice Codec (MIME unknown) */
|
|
case GST_RM_AUD_SIPR:
|
|
caps = gst_caps_new_empty_simple ("audio/x-sipro");
|
|
break;
|
|
|
|
default:
|
|
GST_WARNING_OBJECT (demux, "unknown fourcc %08X", demux->fourcc);
|
|
break;
|
|
}
|
|
|
|
if (caps == NULL)
|
|
goto unknown_fourcc;
|
|
|
|
gst_caps_set_simple (caps,
|
|
"flavor", G_TYPE_INT, demux->flavour,
|
|
"rate", G_TYPE_INT, demux->sample_rate,
|
|
"channels", G_TYPE_INT, demux->channels,
|
|
"width", G_TYPE_INT, demux->sample_width,
|
|
"leaf_size", G_TYPE_INT, demux->leaf_size,
|
|
"packet_size", G_TYPE_INT, demux->packet_size,
|
|
"height", G_TYPE_INT, demux->height, NULL);
|
|
|
|
GST_INFO_OBJECT (demux, "Adding source pad, caps %" GST_PTR_FORMAT, caps);
|
|
demux->srcpad = gst_pad_new_from_static_template (&src_template, "src");
|
|
gst_pad_set_event_function (demux->srcpad,
|
|
GST_DEBUG_FUNCPTR (gst_real_audio_demux_src_event));
|
|
gst_pad_set_query_function (demux->srcpad,
|
|
GST_DEBUG_FUNCPTR (gst_real_audio_demux_src_query));
|
|
gst_pad_set_active (demux->srcpad, TRUE);
|
|
gst_pad_use_fixed_caps (demux->srcpad);
|
|
|
|
stream_id =
|
|
gst_pad_create_stream_id (demux->srcpad, GST_ELEMENT_CAST (demux), NULL);
|
|
|
|
event = gst_pad_get_sticky_event (demux->sinkpad, GST_EVENT_STREAM_START, 0);
|
|
if (event) {
|
|
if (gst_event_parse_group_id (event, &demux->group_id))
|
|
demux->have_group_id = TRUE;
|
|
else
|
|
demux->have_group_id = FALSE;
|
|
gst_event_unref (event);
|
|
} else if (!demux->have_group_id) {
|
|
demux->have_group_id = TRUE;
|
|
demux->group_id = gst_util_group_id_next ();
|
|
}
|
|
|
|
event = gst_event_new_stream_start (stream_id);
|
|
if (demux->have_group_id)
|
|
gst_event_set_group_id (event, demux->group_id);
|
|
|
|
gst_pad_push_event (demux->srcpad, event);
|
|
g_free (stream_id);
|
|
|
|
gst_pad_set_caps (demux->srcpad, caps);
|
|
codec_name = gst_pb_utils_get_codec_description (caps);
|
|
gst_caps_unref (caps);
|
|
|
|
gst_element_add_pad (GST_ELEMENT (demux), demux->srcpad);
|
|
|
|
if (demux->byterate_num > 0 && demux->byterate_denom > 0) {
|
|
GstFormat bformat = GST_FORMAT_BYTES;
|
|
gint64 size_bytes = 0;
|
|
|
|
GST_INFO_OBJECT (demux, "byte rate = %u/%u = %u bytes/sec",
|
|
demux->byterate_num, demux->byterate_denom,
|
|
demux->byterate_num / demux->byterate_denom);
|
|
|
|
if (gst_pad_peer_query_duration (demux->sinkpad, bformat, &size_bytes)) {
|
|
demux->duration =
|
|
gst_real_demux_get_timestamp_from_offset (demux, size_bytes);
|
|
demux->upstream_size = size_bytes;
|
|
GST_INFO_OBJECT (demux, "upstream_size = %" G_GUINT64_FORMAT,
|
|
demux->upstream_size);
|
|
GST_INFO_OBJECT (demux, "duration = %" GST_TIME_FORMAT,
|
|
GST_TIME_ARGS (demux->duration));
|
|
}
|
|
}
|
|
|
|
demux->need_newsegment = TRUE;
|
|
|
|
if (codec_name) {
|
|
if (demux->pending_tags == NULL) {
|
|
demux->pending_tags = gst_tag_list_new_empty ();
|
|
gst_tag_list_set_scope (demux->pending_tags, GST_TAG_SCOPE_GLOBAL);
|
|
}
|
|
|
|
gst_tag_list_add (demux->pending_tags, GST_TAG_MERGE_REPLACE,
|
|
GST_TAG_AUDIO_CODEC, codec_name, NULL);
|
|
g_free (codec_name);
|
|
}
|
|
|
|
gst_adapter_unmap (demux->adapter);
|
|
gst_adapter_flush (demux->adapter, demux->data_offset - 6);
|
|
|
|
demux->state = REAL_AUDIO_DEMUX_STATE_DATA;
|
|
demux->need_newsegment = TRUE;
|
|
|
|
return GST_FLOW_OK;
|
|
|
|
/* ERRORS */
|
|
unknown_fourcc:
|
|
{
|
|
GST_ELEMENT_ERROR (GST_ELEMENT (demux), STREAM, DECODE, (NULL),
|
|
("Unknown fourcc '0x%" G_GINT32_MODIFIER "x'", demux->fourcc));
|
|
return GST_FLOW_ERROR;
|
|
}
|
|
broken_file:
|
|
{
|
|
GST_ELEMENT_ERROR (GST_ELEMENT (demux), STREAM, DECODE, (NULL),
|
|
("Broken file - invalid sample_rate or other header value"));
|
|
return GST_FLOW_ERROR;
|
|
}
|
|
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_real_audio_demux_parse_data (GstRealAudioDemux * demux)
|
|
{
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
guint avail, unit_size;
|
|
|
|
avail = gst_adapter_available (demux->adapter);
|
|
|
|
if (demux->packet_size > 0)
|
|
unit_size = demux->packet_size;
|
|
else
|
|
unit_size = avail & 0xfffffff0; /* round down to next multiple of 16 */
|
|
|
|
GST_LOG_OBJECT (demux, "available = %u, unit_size = %u", avail, unit_size);
|
|
|
|
while (ret == GST_FLOW_OK && unit_size > 0 && avail >= unit_size) {
|
|
GstClockTime ts;
|
|
GstBuffer *buf;
|
|
|
|
buf = gst_adapter_take_buffer (demux->adapter, unit_size);
|
|
avail -= unit_size;
|
|
|
|
if (demux->need_newsegment) {
|
|
gst_pad_push_event (demux->srcpad,
|
|
gst_event_new_segment (&demux->segment));
|
|
demux->need_newsegment = FALSE;
|
|
}
|
|
|
|
if (demux->pending_tags) {
|
|
gst_pad_push_event (demux->srcpad,
|
|
gst_event_new_tag (demux->pending_tags));
|
|
demux->pending_tags = NULL;
|
|
}
|
|
|
|
if (demux->fourcc == GST_RM_AUD_DNET) {
|
|
buf = gst_rm_utils_descramble_dnet_buffer (buf);
|
|
}
|
|
|
|
ts = gst_real_demux_get_timestamp_from_offset (demux, demux->offset);
|
|
GST_BUFFER_TIMESTAMP (buf) = ts;
|
|
|
|
demux->segment.position = ts;
|
|
|
|
ret = gst_pad_push (demux->srcpad, buf);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_real_audio_demux_handle_buffer (GstRealAudioDemux * demux, GstBuffer * buf)
|
|
{
|
|
GstFlowReturn ret;
|
|
|
|
gst_adapter_push (demux->adapter, buf);
|
|
buf = NULL;
|
|
|
|
switch (demux->state) {
|
|
case REAL_AUDIO_DEMUX_STATE_MARKER:{
|
|
ret = gst_real_audio_demux_parse_marker (demux);
|
|
if (ret != GST_FLOW_OK || demux->state != REAL_AUDIO_DEMUX_STATE_HEADER)
|
|
break;
|
|
/* otherwise fall through */
|
|
}
|
|
case REAL_AUDIO_DEMUX_STATE_HEADER:{
|
|
ret = gst_real_audio_demux_parse_header (demux);
|
|
if (ret != GST_FLOW_OK || demux->state != REAL_AUDIO_DEMUX_STATE_DATA)
|
|
break;
|
|
/* otherwise fall through */
|
|
}
|
|
case REAL_AUDIO_DEMUX_STATE_DATA:{
|
|
ret = gst_real_audio_demux_parse_data (demux);
|
|
break;
|
|
}
|
|
default:
|
|
g_return_val_if_reached (GST_FLOW_ERROR);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_real_audio_demux_chain (GstPad * pad, GstObject * parent, GstBuffer * buf)
|
|
{
|
|
GstRealAudioDemux *demux;
|
|
|
|
demux = GST_REAL_AUDIO_DEMUX (parent);
|
|
|
|
return gst_real_audio_demux_handle_buffer (demux, buf);
|
|
}
|
|
|
|
static void
|
|
gst_real_audio_demux_loop (GstRealAudioDemux * demux)
|
|
{
|
|
GstFlowReturn ret;
|
|
GstBuffer *buf;
|
|
guint bytes_needed;
|
|
|
|
/* check how much data we need */
|
|
switch (demux->state) {
|
|
case REAL_AUDIO_DEMUX_STATE_MARKER:
|
|
bytes_needed = 6 + 16; /* 16 are beginning of header */
|
|
break;
|
|
case REAL_AUDIO_DEMUX_STATE_HEADER:
|
|
if (!gst_real_audio_demux_get_data_offset_from_header (demux))
|
|
goto parse_header_error;
|
|
bytes_needed = demux->data_offset - (6 + 16);
|
|
break;
|
|
case REAL_AUDIO_DEMUX_STATE_DATA:
|
|
if (demux->packet_size > 0) {
|
|
/* TODO: should probably take into account width/height as well? */
|
|
bytes_needed = demux->packet_size;
|
|
} else {
|
|
bytes_needed = 1024;
|
|
}
|
|
break;
|
|
default:
|
|
g_return_if_reached ();
|
|
}
|
|
|
|
/* now get the data */
|
|
GST_LOG_OBJECT (demux, "getting data: %5u bytes @ %8" G_GINT64_MODIFIER "u",
|
|
bytes_needed, demux->offset);
|
|
|
|
if (demux->upstream_size > 0 && demux->offset >= demux->upstream_size)
|
|
goto eos;
|
|
|
|
buf = NULL;
|
|
ret = gst_pad_pull_range (demux->sinkpad, demux->offset, bytes_needed, &buf);
|
|
|
|
if (ret != GST_FLOW_OK)
|
|
goto pull_range_error;
|
|
|
|
if (gst_buffer_get_size (buf) != bytes_needed)
|
|
goto pull_range_short_read;
|
|
|
|
ret = gst_real_audio_demux_handle_buffer (demux, buf);
|
|
if (ret != GST_FLOW_OK)
|
|
goto handle_flow_error;
|
|
|
|
/* TODO: increase this in chain function too (for timestamps)? */
|
|
demux->offset += bytes_needed;
|
|
|
|
/* check for the end of the segment */
|
|
if (demux->segment.stop != -1 && demux->segment.position != -1 &&
|
|
demux->segment.position > demux->segment.stop) {
|
|
GST_DEBUG_OBJECT (demux, "reached end of segment");
|
|
goto eos;
|
|
}
|
|
|
|
return;
|
|
|
|
/* ERRORS */
|
|
parse_header_error:
|
|
{
|
|
GST_ELEMENT_ERROR (demux, STREAM, DECODE, (NULL), (NULL));
|
|
goto pause_task;
|
|
}
|
|
handle_flow_error:
|
|
{
|
|
GST_WARNING_OBJECT (demux, "handle_buf flow: %s", gst_flow_get_name (ret));
|
|
goto pause_task;
|
|
}
|
|
pull_range_error:
|
|
{
|
|
GST_WARNING_OBJECT (demux, "pull range flow: %s", gst_flow_get_name (ret));
|
|
goto pause_task;
|
|
}
|
|
pull_range_short_read:
|
|
{
|
|
GST_WARNING_OBJECT (demux, "pull range short read: wanted %u bytes, but "
|
|
"got only %" G_GSIZE_FORMAT " bytes", bytes_needed,
|
|
gst_buffer_get_size (buf));
|
|
gst_buffer_unref (buf);
|
|
goto eos;
|
|
}
|
|
eos:
|
|
{
|
|
if (demux->state != REAL_AUDIO_DEMUX_STATE_DATA) {
|
|
GST_WARNING_OBJECT (demux, "reached EOS before finished parsing header");
|
|
goto parse_header_error;
|
|
}
|
|
GST_INFO_OBJECT (demux, "EOS");
|
|
if ((demux->segment.flags & GST_SEEK_FLAG_SEGMENT) != 0) {
|
|
gint64 stop;
|
|
|
|
/* for segment playback we need to post when (in stream time)
|
|
* we stopped, this is either stop (when set) or the duration. */
|
|
if ((stop = demux->segment.stop) == -1)
|
|
stop = demux->segment.duration;
|
|
|
|
GST_DEBUG_OBJECT (demux, "sending segment done, at end of segment");
|
|
gst_element_post_message (GST_ELEMENT (demux),
|
|
gst_message_new_segment_done (GST_OBJECT (demux), GST_FORMAT_TIME,
|
|
stop));
|
|
gst_pad_push_event (demux->srcpad,
|
|
gst_event_new_segment_done (GST_FORMAT_TIME, stop));
|
|
} else {
|
|
/* normal playback, send EOS event downstream */
|
|
GST_DEBUG_OBJECT (demux, "sending EOS event, at end of stream");
|
|
gst_pad_push_event (demux->srcpad, gst_event_new_eos ());
|
|
}
|
|
goto pause_task;
|
|
}
|
|
pause_task:
|
|
{
|
|
demux->segment_running = FALSE;
|
|
gst_pad_pause_task (demux->sinkpad);
|
|
GST_DEBUG_OBJECT (demux, "pausing task");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_real_audio_demux_sink_event (GstPad * pad, GstObject * parent,
|
|
GstEvent * event)
|
|
{
|
|
GstRealAudioDemux *demux;
|
|
gboolean ret;
|
|
|
|
demux = GST_REAL_AUDIO_DEMUX (parent);
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_SEGMENT:{
|
|
/* FIXME */
|
|
gst_event_unref (event);
|
|
demux->need_newsegment = TRUE;
|
|
ret = TRUE;
|
|
break;
|
|
}
|
|
default:
|
|
ret = gst_pad_event_default (pad, parent, event);
|
|
break;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
gst_real_audio_demux_handle_seek (GstRealAudioDemux * demux, GstEvent * event)
|
|
{
|
|
GstFormat format;
|
|
GstSeekFlags flags;
|
|
GstSeekType cur_type, stop_type;
|
|
gboolean flush, update;
|
|
gdouble rate;
|
|
guint64 seek_pos;
|
|
gint64 cur, stop;
|
|
|
|
if (!demux->seekable)
|
|
goto not_seekable;
|
|
|
|
if (demux->byterate_num == 0 || demux->byterate_denom == 0)
|
|
goto no_bitrate;
|
|
|
|
gst_event_parse_seek (event, &rate, &format, &flags,
|
|
&cur_type, &cur, &stop_type, &stop);
|
|
|
|
if (format != GST_FORMAT_TIME)
|
|
goto only_time_format_supported;
|
|
|
|
if (rate <= 0.0)
|
|
goto cannot_do_backwards_playback;
|
|
|
|
flush = ((flags & GST_SEEK_FLAG_FLUSH) != 0);
|
|
|
|
GST_DEBUG_OBJECT (demux, "flush=%d, rate=%g", flush, rate);
|
|
|
|
/* unlock streaming thread and make streaming stop */
|
|
if (flush) {
|
|
gst_pad_push_event (demux->sinkpad, gst_event_new_flush_start ());
|
|
gst_pad_push_event (demux->srcpad, gst_event_new_flush_start ());
|
|
} else {
|
|
gst_pad_pause_task (demux->sinkpad);
|
|
}
|
|
|
|
GST_PAD_STREAM_LOCK (demux->sinkpad);
|
|
|
|
gst_segment_do_seek (&demux->segment, rate, format, flags,
|
|
cur_type, cur, stop_type, stop, &update);
|
|
|
|
GST_DEBUG_OBJECT (demux, "segment: %" GST_SEGMENT_FORMAT, &demux->segment);
|
|
|
|
seek_pos = gst_util_uint64_scale (demux->segment.start,
|
|
demux->byterate_num, demux->byterate_denom * GST_SECOND);
|
|
if (demux->packet_size > 0) {
|
|
seek_pos -= seek_pos % demux->packet_size;
|
|
}
|
|
seek_pos += demux->data_offset;
|
|
|
|
GST_DEBUG_OBJECT (demux, "seek_pos = %" G_GUINT64_FORMAT, seek_pos);
|
|
|
|
/* stop flushing */
|
|
gst_pad_push_event (demux->sinkpad, gst_event_new_flush_stop (TRUE));
|
|
gst_pad_push_event (demux->srcpad, gst_event_new_flush_stop (TRUE));
|
|
|
|
demux->offset = seek_pos;
|
|
demux->need_newsegment = TRUE;
|
|
|
|
/* notify start of new segment */
|
|
if (demux->segment.flags & GST_SEEK_FLAG_SEGMENT) {
|
|
gst_element_post_message (GST_ELEMENT (demux),
|
|
gst_message_new_segment_start (GST_OBJECT (demux),
|
|
GST_FORMAT_TIME, demux->segment.position));
|
|
}
|
|
|
|
demux->segment_running = TRUE;
|
|
/* restart our task since it might have been stopped when we did the flush */
|
|
gst_pad_start_task (demux->sinkpad,
|
|
(GstTaskFunction) gst_real_audio_demux_loop, demux, NULL);
|
|
|
|
/* streaming can continue now */
|
|
GST_PAD_STREAM_UNLOCK (demux->sinkpad);
|
|
|
|
return TRUE;
|
|
|
|
/* ERRORS */
|
|
not_seekable:
|
|
{
|
|
GST_DEBUG_OBJECT (demux, "seek failed: cannot seek in streaming mode");
|
|
return FALSE;
|
|
}
|
|
no_bitrate:
|
|
{
|
|
GST_DEBUG_OBJECT (demux, "seek failed: bitrate unknown");
|
|
return FALSE;
|
|
}
|
|
only_time_format_supported:
|
|
{
|
|
GST_DEBUG_OBJECT (demux, "can only seek in TIME format");
|
|
return FALSE;
|
|
}
|
|
cannot_do_backwards_playback:
|
|
{
|
|
GST_DEBUG_OBJECT (demux, "can only seek with positive rate, not %lf", rate);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_real_audio_demux_src_event (GstPad * pad, GstObject * parent,
|
|
GstEvent * event)
|
|
{
|
|
GstRealAudioDemux *demux;
|
|
gboolean ret = FALSE;
|
|
|
|
demux = GST_REAL_AUDIO_DEMUX (parent);
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_QOS:
|
|
gst_event_unref (event);
|
|
break;
|
|
case GST_EVENT_SEEK:
|
|
ret = gst_real_audio_demux_handle_seek (demux, event);
|
|
gst_event_unref (event);
|
|
break;
|
|
default:
|
|
ret = gst_pad_event_default (pad, parent, event);
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
gst_real_audio_demux_src_query (GstPad * pad, GstObject * parent,
|
|
GstQuery * query)
|
|
{
|
|
GstRealAudioDemux *demux;
|
|
gboolean ret = FALSE;
|
|
|
|
demux = GST_REAL_AUDIO_DEMUX (parent);
|
|
|
|
switch (GST_QUERY_TYPE (query)) {
|
|
case GST_QUERY_DURATION:{
|
|
GstFormat format;
|
|
|
|
gst_query_parse_duration (query, &format, NULL);
|
|
if (format == GST_FORMAT_TIME && demux->duration > 0) {
|
|
gst_query_set_duration (query, GST_FORMAT_TIME, demux->duration);
|
|
ret = TRUE;
|
|
} else if (format == GST_FORMAT_BYTES && demux->upstream_size > 0) {
|
|
gst_query_set_duration (query, GST_FORMAT_BYTES,
|
|
demux->upstream_size - demux->data_offset);
|
|
ret = TRUE;
|
|
}
|
|
break;
|
|
}
|
|
case GST_QUERY_SEEKING:{
|
|
GstFormat format;
|
|
gboolean seekable;
|
|
|
|
gst_query_parse_seeking (query, &format, NULL, NULL, NULL);
|
|
seekable = (format == GST_FORMAT_TIME && demux->seekable);
|
|
gst_query_set_seeking (query, format, seekable, 0,
|
|
(format == GST_FORMAT_TIME) ? demux->duration : -1);
|
|
ret = TRUE;
|
|
break;
|
|
}
|
|
case GST_QUERY_SEGMENT:
|
|
{
|
|
GstFormat format;
|
|
gint64 start, stop;
|
|
|
|
format = demux->segment.format;
|
|
|
|
start =
|
|
gst_segment_to_stream_time (&demux->segment, format,
|
|
demux->segment.start);
|
|
if ((stop = demux->segment.stop) == -1)
|
|
stop = demux->segment.duration;
|
|
else
|
|
stop = gst_segment_to_stream_time (&demux->segment, format, stop);
|
|
|
|
gst_query_set_segment (query, demux->segment.rate, format, start, stop);
|
|
ret = TRUE;
|
|
break;
|
|
}
|
|
default:
|
|
ret = gst_pad_query_default (pad, parent, query);
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static GstStateChangeReturn
|
|
gst_real_audio_demux_change_state (GstElement * element,
|
|
GstStateChange transition)
|
|
{
|
|
GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS;
|
|
GstRealAudioDemux *demux = GST_REAL_AUDIO_DEMUX (element);
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_NULL_TO_READY:
|
|
break;
|
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
|
demux->state = REAL_AUDIO_DEMUX_STATE_MARKER;
|
|
demux->segment_running = FALSE;
|
|
gst_segment_init (&demux->segment, GST_FORMAT_TIME);
|
|
gst_adapter_clear (demux->adapter);
|
|
break;
|
|
case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
|
|
break;
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:{
|
|
gst_real_audio_demux_reset (demux);
|
|
gst_segment_init (&demux->segment, GST_FORMAT_UNDEFINED);
|
|
break;
|
|
}
|
|
case GST_STATE_CHANGE_READY_TO_NULL:
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|