mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 18:35:35 +00:00
fd803921c9
This is too early and will confuse the event order. The other code that sets the caps is at the right position and does it properly already.
382 lines
11 KiB
C
382 lines
11 KiB
C
/* 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.
|
|
*/
|
|
|
|
/**
|
|
* SECTION:element-opusparse
|
|
* @see_also: opusenc, opusdec
|
|
*
|
|
* This element parses OPUS packets.
|
|
*
|
|
* <refsect2>
|
|
* <title>Example pipelines</title>
|
|
* |[
|
|
* gst-launch -v filesrc location=opusdata ! opusparse ! opusdec ! audioconvert ! audioresample ! alsasink
|
|
* ]| Decode and plays an unmuxed Opus file.
|
|
* </refsect2>
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <string.h>
|
|
#include <opus/opus.h>
|
|
#include "gstopusheader.h"
|
|
#include "gstopusparse.h"
|
|
|
|
GST_DEBUG_CATEGORY_STATIC (opusparse_debug);
|
|
#define GST_CAT_DEFAULT opusparse_debug
|
|
|
|
#define MAX_PAYLOAD_BYTES 1500
|
|
|
|
static GstStaticPadTemplate opus_parse_src_factory =
|
|
GST_STATIC_PAD_TEMPLATE ("src",
|
|
GST_PAD_SRC,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("audio/x-opus, framed = (boolean) true")
|
|
);
|
|
|
|
static GstStaticPadTemplate opus_parse_sink_factory =
|
|
GST_STATIC_PAD_TEMPLATE ("sink",
|
|
GST_PAD_SINK,
|
|
GST_PAD_ALWAYS,
|
|
GST_STATIC_CAPS ("audio/x-opus")
|
|
);
|
|
|
|
G_DEFINE_TYPE (GstOpusParse, gst_opus_parse, GST_TYPE_BASE_PARSE);
|
|
|
|
static gboolean gst_opus_parse_start (GstBaseParse * parse);
|
|
static gboolean gst_opus_parse_stop (GstBaseParse * parse);
|
|
static GstFlowReturn gst_opus_parse_handle_frame (GstBaseParse * base,
|
|
GstBaseParseFrame * frame, gint * skip);
|
|
static GstFlowReturn gst_opus_parse_parse_frame (GstBaseParse * base,
|
|
GstBaseParseFrame * frame);
|
|
|
|
static void
|
|
gst_opus_parse_class_init (GstOpusParseClass * klass)
|
|
{
|
|
GstBaseParseClass *bpclass;
|
|
GstElementClass *element_class;
|
|
|
|
bpclass = (GstBaseParseClass *) klass;
|
|
element_class = (GstElementClass *) klass;
|
|
|
|
bpclass->start = GST_DEBUG_FUNCPTR (gst_opus_parse_start);
|
|
bpclass->stop = GST_DEBUG_FUNCPTR (gst_opus_parse_stop);
|
|
bpclass->handle_frame = GST_DEBUG_FUNCPTR (gst_opus_parse_handle_frame);
|
|
|
|
gst_element_class_add_pad_template (element_class,
|
|
gst_static_pad_template_get (&opus_parse_src_factory));
|
|
gst_element_class_add_pad_template (element_class,
|
|
gst_static_pad_template_get (&opus_parse_sink_factory));
|
|
gst_element_class_set_static_metadata (element_class, "Opus audio parser",
|
|
"Codec/Parser/Audio",
|
|
"parses opus audio streams",
|
|
"Vincent Penquerc'h <vincent.penquerch@collabora.co.uk>");
|
|
|
|
GST_DEBUG_CATEGORY_INIT (opusparse_debug, "opusparse", 0,
|
|
"opus parsing element");
|
|
}
|
|
|
|
static void
|
|
gst_opus_parse_init (GstOpusParse * parse)
|
|
{
|
|
parse->header_sent = FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gst_opus_parse_start (GstBaseParse * base)
|
|
{
|
|
GstOpusParse *parse = GST_OPUS_PARSE (base);
|
|
|
|
parse->header_sent = FALSE;
|
|
parse->next_ts = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gst_opus_parse_stop (GstBaseParse * base)
|
|
{
|
|
GstOpusParse *parse = GST_OPUS_PARSE (base);
|
|
|
|
g_slist_foreach (parse->headers, (GFunc) gst_buffer_unref, NULL);
|
|
g_slist_free (parse->headers);
|
|
parse->headers = NULL;
|
|
|
|
parse->header_sent = FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_opus_parse_handle_frame (GstBaseParse * base,
|
|
GstBaseParseFrame * frame, gint * skip)
|
|
{
|
|
GstOpusParse *parse;
|
|
guint8 *data;
|
|
gsize size;
|
|
guint32 packet_size;
|
|
int ret = FALSE;
|
|
const unsigned char *frames[48];
|
|
unsigned char toc;
|
|
short frame_sizes[48];
|
|
int payload_offset;
|
|
int packet_offset = 0;
|
|
gboolean is_header, is_idheader, is_commentheader;
|
|
GstMapInfo map;
|
|
|
|
parse = GST_OPUS_PARSE (base);
|
|
|
|
*skip = -1;
|
|
|
|
gst_buffer_map (frame->buffer, &map, GST_MAP_READ);
|
|
data = map.data;
|
|
size = map.size;
|
|
GST_DEBUG_OBJECT (parse,
|
|
"Checking for frame, %" G_GSIZE_FORMAT " bytes in buffer", size);
|
|
|
|
/* check for headers */
|
|
is_idheader = gst_opus_header_is_id_header (frame->buffer);
|
|
is_commentheader = gst_opus_header_is_comment_header (frame->buffer);
|
|
is_header = is_idheader || is_commentheader;
|
|
|
|
if (!is_header) {
|
|
int nframes;
|
|
|
|
/* Next, check if there's an Opus packet there */
|
|
nframes =
|
|
opus_packet_parse (data, size, &toc, frames, frame_sizes,
|
|
&payload_offset);
|
|
|
|
if (nframes < 0) {
|
|
/* Then, check for the test vector framing */
|
|
GST_DEBUG_OBJECT (parse,
|
|
"No Opus packet found, trying test vector framing");
|
|
if (size < 4) {
|
|
GST_DEBUG_OBJECT (parse, "Too small");
|
|
goto beach;
|
|
}
|
|
packet_size = GST_READ_UINT32_BE (data);
|
|
GST_DEBUG_OBJECT (parse, "Packet size: %u bytes", packet_size);
|
|
if (packet_size > MAX_PAYLOAD_BYTES) {
|
|
GST_DEBUG_OBJECT (parse, "Too large");
|
|
goto beach;
|
|
}
|
|
if (packet_size > size - 4) {
|
|
GST_DEBUG_OBJECT (parse, "Truncated");
|
|
goto beach;
|
|
}
|
|
nframes =
|
|
opus_packet_parse (data + 8, packet_size, &toc, frames, frame_sizes,
|
|
&payload_offset);
|
|
if (nframes < 0) {
|
|
GST_DEBUG_OBJECT (parse, "No test vector framing either");
|
|
goto beach;
|
|
}
|
|
|
|
packet_offset = 8;
|
|
data += packet_offset;
|
|
|
|
/* for ad hoc framing, heed the framing, so we eat any padding */
|
|
payload_offset = packet_size;
|
|
}
|
|
}
|
|
|
|
if (is_header) {
|
|
*skip = 0;
|
|
} else {
|
|
*skip = packet_offset;
|
|
size = payload_offset;
|
|
}
|
|
|
|
GST_DEBUG_OBJECT (parse,
|
|
"Got Opus packet at offset %d, %" G_GSIZE_FORMAT " bytes", *skip, size);
|
|
ret = TRUE;
|
|
|
|
beach:
|
|
gst_buffer_unmap (frame->buffer, &map);
|
|
|
|
/* convert old style result to new one */
|
|
if (!ret) {
|
|
if (*skip < 0)
|
|
*skip = 1;
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
/* always skip first if needed */
|
|
if (*skip > 0)
|
|
return GST_FLOW_OK;
|
|
|
|
/* normalize again */
|
|
if (*skip < 0)
|
|
*skip = 0;
|
|
|
|
/* not enough */
|
|
if (size > map.size)
|
|
return GST_FLOW_OK;
|
|
|
|
/* FIXME some day ... should not mess with buffer itself */
|
|
if (!parse->header_sent) {
|
|
gst_buffer_replace (&frame->buffer,
|
|
gst_buffer_copy_region (frame->buffer, GST_BUFFER_COPY_ALL, 0, size));
|
|
gst_buffer_unref (frame->buffer);
|
|
}
|
|
|
|
ret = gst_opus_parse_parse_frame (base, frame);
|
|
|
|
if (ret == GST_BASE_PARSE_FLOW_DROPPED) {
|
|
frame->flags |= GST_BASE_PARSE_FRAME_FLAG_DROP;
|
|
ret = GST_FLOW_OK;
|
|
}
|
|
if (ret == GST_FLOW_OK)
|
|
ret = gst_base_parse_finish_frame (base, frame, size);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* Adapted copy of the one in gstoggstream.c... */
|
|
static guint64
|
|
packet_duration_opus (const guint8 * data, size_t len)
|
|
{
|
|
static const guint64 durations[32] = {
|
|
10000, 20000, 40000, 60000, /* Silk NB */
|
|
10000, 20000, 40000, 60000, /* Silk MB */
|
|
10000, 20000, 40000, 60000, /* Silk WB */
|
|
10000, 20000, /* Hybrid SWB */
|
|
10000, 20000, /* Hybrid FB */
|
|
2500, 5000, 10000, 20000, /* CELT NB */
|
|
2500, 5000, 10000, 20000, /* CELT NB */
|
|
2500, 5000, 10000, 20000, /* CELT NB */
|
|
2500, 5000, 10000, 20000, /* CELT NB */
|
|
};
|
|
|
|
gint64 duration;
|
|
gint64 frame_duration;
|
|
gint nframes;
|
|
guint8 toc;
|
|
|
|
if (len < 1)
|
|
return 0;
|
|
|
|
toc = data[0];
|
|
|
|
frame_duration = durations[toc >> 3] * 1000;
|
|
switch (toc & 3) {
|
|
case 0:
|
|
nframes = 1;
|
|
break;
|
|
case 1:
|
|
nframes = 2;
|
|
break;
|
|
case 2:
|
|
nframes = 2;
|
|
break;
|
|
case 3:
|
|
if (len < 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 > 120 * GST_MSECOND) {
|
|
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 / 1000000.f, nframes, duration / 1000000.f);
|
|
return duration;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_opus_parse_parse_frame (GstBaseParse * base, GstBaseParseFrame * frame)
|
|
{
|
|
guint64 duration;
|
|
GstOpusParse *parse;
|
|
gboolean is_idheader, is_commentheader;
|
|
GstMapInfo map;
|
|
|
|
parse = GST_OPUS_PARSE (base);
|
|
|
|
is_idheader = gst_opus_header_is_id_header (frame->buffer);
|
|
is_commentheader = gst_opus_header_is_comment_header (frame->buffer);
|
|
|
|
if (!parse->header_sent) {
|
|
GstCaps *caps;
|
|
guint8 channels;
|
|
|
|
/* Opus streams can decode to 1 or 2 channels, so use the header
|
|
value if we have one, or 2 otherwise */
|
|
if (is_idheader) {
|
|
gst_buffer_replace (&parse->id_header, frame->buffer);
|
|
GST_DEBUG_OBJECT (parse, "Found ID header, keeping");
|
|
return GST_BASE_PARSE_FLOW_DROPPED;
|
|
} else if (is_commentheader) {
|
|
gst_buffer_replace (&parse->comment_header, frame->buffer);
|
|
GST_DEBUG_OBJECT (parse, "Found comment header, keeping");
|
|
return GST_BASE_PARSE_FLOW_DROPPED;
|
|
}
|
|
|
|
g_slist_foreach (parse->headers, (GFunc) gst_buffer_unref, NULL);
|
|
g_slist_free (parse->headers);
|
|
parse->headers = NULL;
|
|
|
|
if (parse->id_header && parse->comment_header) {
|
|
gst_opus_header_create_caps_from_headers (&caps, &parse->headers,
|
|
parse->id_header, parse->comment_header);
|
|
} else {
|
|
guint8 channel_mapping_family, channel_mapping[256];
|
|
GST_INFO_OBJECT (parse,
|
|
"No headers, blindly setting up canonical stereo");
|
|
channels = 2;
|
|
channel_mapping_family = 0;
|
|
channel_mapping[0] = 0;
|
|
channel_mapping[1] = 1;
|
|
gst_opus_header_create_caps (&caps, &parse->headers, channels, 1, 0,
|
|
channel_mapping_family, channel_mapping, NULL);
|
|
}
|
|
|
|
gst_buffer_replace (&parse->id_header, NULL);
|
|
gst_buffer_replace (&parse->comment_header, NULL);
|
|
|
|
gst_pad_set_caps (GST_BASE_PARSE_SRC_PAD (parse), caps);
|
|
gst_caps_unref (caps);
|
|
parse->header_sent = TRUE;
|
|
}
|
|
|
|
GST_BUFFER_TIMESTAMP (frame->buffer) = parse->next_ts;
|
|
|
|
gst_buffer_map (frame->buffer, &map, GST_MAP_READ);
|
|
duration = packet_duration_opus (map.data, map.size);
|
|
gst_buffer_unmap (frame->buffer, &map);
|
|
parse->next_ts += duration;
|
|
|
|
GST_BUFFER_DURATION (frame->buffer) = duration;
|
|
GST_BUFFER_OFFSET_END (frame->buffer) =
|
|
gst_util_uint64_scale (parse->next_ts, 48000, GST_SECOND);
|
|
GST_BUFFER_OFFSET (frame->buffer) = parse->next_ts;
|
|
|
|
return GST_FLOW_OK;
|
|
}
|