mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-02 13:38:48 +00:00
mpegtsmux: add SPN/PTS indexing capabilities
Based on patch by Andreas Frisch <fraxinas@opendreambox.org> Fixes https://bugzilla.gnome.org/show_bug.cgi?id=644890
This commit is contained in:
parent
05dcabfe16
commit
bef06e8c8d
2 changed files with 97 additions and 4 deletions
|
@ -174,6 +174,12 @@ static GstStateChangeReturn mpegtsmux_change_state (GstElement * element,
|
|||
static void mpegtsdemux_set_header_on_caps (MpegTsMux * mux);
|
||||
static gboolean mpegtsmux_src_event (GstPad * pad, GstEvent * event);
|
||||
|
||||
static void mpegtsmux_set_index (GstElement * element, GstIndex * index);
|
||||
static GstIndex *mpegtsmux_get_index (GstElement * element);
|
||||
|
||||
static GstFormat pts_format;
|
||||
static GstFormat spn_format;
|
||||
|
||||
GST_BOILERPLATE (MpegTsMux, mpegtsmux, GstElement, GST_TYPE_ELEMENT);
|
||||
|
||||
static void
|
||||
|
@ -191,6 +197,10 @@ mpegtsmux_base_init (gpointer g_class)
|
|||
"MPEG Transport Stream Muxer", "Codec/Muxer",
|
||||
"Multiplexes media streams into an MPEG Transport Stream",
|
||||
"Fluendo <contact@fluendo.com>");
|
||||
|
||||
pts_format =
|
||||
gst_format_register ("PTS", "MPEG System Presentation Time Stamp");
|
||||
spn_format = gst_format_register ("SPN", "Source Packet Number");
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -207,6 +217,9 @@ mpegtsmux_class_init (MpegTsMuxClass * klass)
|
|||
gstelement_class->release_pad = mpegtsmux_release_pad;
|
||||
gstelement_class->change_state = mpegtsmux_change_state;
|
||||
|
||||
gstelement_class->set_index = GST_DEBUG_FUNCPTR (mpegtsmux_set_index);
|
||||
gstelement_class->get_index = GST_DEBUG_FUNCPTR (mpegtsmux_get_index);
|
||||
|
||||
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_PROG_MAP,
|
||||
g_param_spec_boxed ("prog-map", "Program map",
|
||||
"A GstStructure specifies the mapping from elementary streams to programs",
|
||||
|
@ -276,6 +289,7 @@ mpegtsmux_pad_reset (MpegTsPadData * pad_data)
|
|||
pad_data->cur_ts = GST_CLOCK_TIME_NONE;
|
||||
pad_data->prog_id = -1;
|
||||
pad_data->eos = FALSE;
|
||||
pad_data->element_index_writer_id = -1;
|
||||
|
||||
if (pad_data->free_func)
|
||||
pad_data->free_func (pad_data->prepare_data);
|
||||
|
@ -311,6 +325,13 @@ mpegtsmux_reset (MpegTsMux * mux, gboolean alloc)
|
|||
mux->force_key_unit_event = NULL;
|
||||
mux->pending_key_unit_ts = GST_CLOCK_TIME_NONE;
|
||||
|
||||
mux->spn_count = 0;
|
||||
|
||||
if (mux->element_index) {
|
||||
gst_object_unref (mux->element_index);
|
||||
mux->element_index = NULL;
|
||||
}
|
||||
|
||||
gst_adapter_clear (mux->adapter);
|
||||
gst_adapter_clear (mux->out_adapter);
|
||||
|
||||
|
@ -450,6 +471,36 @@ gst_mpegtsmux_get_property (GObject * object, guint prop_id,
|
|||
}
|
||||
}
|
||||
|
||||
static void
|
||||
mpegtsmux_set_index (GstElement * element, GstIndex * index)
|
||||
{
|
||||
MpegTsMux *mux = GST_MPEG_TSMUX (element);
|
||||
|
||||
GST_OBJECT_LOCK (mux);
|
||||
if (mux->element_index)
|
||||
gst_object_unref (mux->element_index);
|
||||
mux->element_index = index ? gst_object_ref (index) : NULL;
|
||||
GST_OBJECT_UNLOCK (mux);
|
||||
|
||||
GST_DEBUG_OBJECT (mux, "Set index %" GST_PTR_FORMAT, mux->element_index);
|
||||
}
|
||||
|
||||
static GstIndex *
|
||||
mpegtsmux_get_index (GstElement * element)
|
||||
{
|
||||
GstIndex *result = NULL;
|
||||
MpegTsMux *mux = GST_MPEG_TSMUX (element);
|
||||
|
||||
GST_OBJECT_LOCK (mux);
|
||||
if (mux->element_index)
|
||||
result = gst_object_ref (mux->element_index);
|
||||
GST_OBJECT_UNLOCK (mux);
|
||||
|
||||
GST_DEBUG_OBJECT (mux, "Returning index %" GST_PTR_FORMAT, result);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
static void
|
||||
release_buffer_cb (guint8 * data, void *user_data)
|
||||
{
|
||||
|
@ -578,6 +629,29 @@ mpegtsmux_create_stream (MpegTsMux * mux, MpegTsPadData * ts_data)
|
|||
ret = GST_FLOW_OK;
|
||||
}
|
||||
|
||||
GST_OBJECT_LOCK (mux);
|
||||
if (mux->element_index) {
|
||||
gboolean parsed = FALSE;
|
||||
|
||||
if (ts_data->stream->is_video_stream) {
|
||||
if (gst_structure_get_boolean (s, "parsed", &parsed) && parsed) {
|
||||
if (ts_data->element_index_writer_id == -1) {
|
||||
gst_index_get_writer_id (mux->element_index, GST_OBJECT (mux),
|
||||
&ts_data->element_index_writer_id);
|
||||
GST_DEBUG_OBJECT (mux, "created GstIndex writer_id = %d for stream",
|
||||
ts_data->element_index_writer_id);
|
||||
gst_index_add_format (mux->element_index,
|
||||
ts_data->element_index_writer_id, pts_format);
|
||||
gst_index_add_format (mux->element_index,
|
||||
ts_data->element_index_writer_id, spn_format);
|
||||
}
|
||||
} else {
|
||||
GST_WARNING_OBJECT (pad, "no indexing for (unparsed) stream !");
|
||||
}
|
||||
}
|
||||
}
|
||||
GST_OBJECT_UNLOCK (mux);
|
||||
|
||||
gst_caps_unref (caps);
|
||||
return ret;
|
||||
|
||||
|
@ -994,10 +1068,6 @@ mpegtsmux_collected (GstCollectPads2 * pads, MpegTsMux * mux)
|
|||
tsmux_program_set_pcr_stream (prog, best->stream);
|
||||
}
|
||||
|
||||
if (best->stream->is_video_stream)
|
||||
delta = GST_BUFFER_FLAG_IS_SET (buf, GST_BUFFER_FLAG_DELTA_UNIT);
|
||||
GST_DEBUG_OBJECT (mux, "delta: %d", delta);
|
||||
|
||||
GST_DEBUG_OBJECT (COLLECT_DATA_PAD (best),
|
||||
"Chose stream for output (PID: 0x%04x)", best->pid);
|
||||
|
||||
|
@ -1007,6 +1077,19 @@ mpegtsmux_collected (GstCollectPads2 * pads, MpegTsMux * mux)
|
|||
G_GINT64_FORMAT, GST_TIME_ARGS (best->cur_ts), pts);
|
||||
}
|
||||
|
||||
if (best->stream->is_video_stream) {
|
||||
delta = GST_BUFFER_FLAG_IS_SET (buf, GST_BUFFER_FLAG_DELTA_UNIT);
|
||||
GST_OBJECT_LOCK (mux);
|
||||
if (mux->element_index && !delta && best->element_index_writer_id != -1) {
|
||||
gst_index_add_association (mux->element_index,
|
||||
best->element_index_writer_id,
|
||||
GST_ASSOCIATION_FLAG_KEY_UNIT, spn_format, mux->spn_count,
|
||||
pts_format, pts, NULL);
|
||||
}
|
||||
GST_OBJECT_UNLOCK (mux);
|
||||
}
|
||||
GST_DEBUG_OBJECT (mux, "delta: %d", delta);
|
||||
|
||||
tsmux_stream_add_data (best->stream, GST_BUFFER_DATA (buf),
|
||||
GST_BUFFER_SIZE (buf), buf, pts, -1, !delta);
|
||||
|
||||
|
@ -1375,6 +1458,9 @@ new_packet_cb (GstBuffer * buf, void *user_data, gint64 new_pcr)
|
|||
MpegTsMux *mux = (MpegTsMux *) user_data;
|
||||
gint offset = 0;
|
||||
|
||||
GST_LOG_OBJECT (mux, "handling packet %d", mux->spn_count);
|
||||
mux->spn_count++;
|
||||
|
||||
offset = GST_BUFFER_DATA (buf) - GST_BUFFER_MALLOCDATA (buf);
|
||||
GST_BUFFER_TIMESTAMP (buf) = mux->last_ts;
|
||||
/* do common init (flags and streamheaders) */
|
||||
|
|
|
@ -165,6 +165,10 @@ struct MpegTsMux {
|
|||
GstBuffer *out_buffer;
|
||||
gint out_offset;
|
||||
gint last_size;
|
||||
|
||||
/* SPN/PTS index handling */
|
||||
GstIndex *element_index;
|
||||
gint spn_count;
|
||||
};
|
||||
|
||||
struct MpegTsMuxClass {
|
||||
|
@ -187,6 +191,9 @@ struct MpegTsPadData {
|
|||
/* most recent valid TS for this stream */
|
||||
GstClockTime last_ts;
|
||||
|
||||
/* (optional) index writing */
|
||||
gint element_index_writer_id;
|
||||
|
||||
/* optional codec data available in the caps */
|
||||
GstBuffer *codec_data;
|
||||
|
||||
|
|
Loading…
Reference in a new issue