mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 18:05:37 +00:00
cccombiner: Add new closed caption combiner element
This is the inverse of ccextractor: it takes a video and closed caption stream and places the closed caption metas on the video output buffers.
This commit is contained in:
parent
7ba9493ff1
commit
eb92865ea1
5 changed files with 578 additions and 4 deletions
|
@ -19,6 +19,8 @@ zvbi_headers = \
|
|||
libgstclosedcaption_la_SOURCES = \
|
||||
$(zvbi_sources) \
|
||||
$(zvbi_headers) \
|
||||
gstcccombiner.c \
|
||||
gstcccombiner.h \
|
||||
gstccextractor.c \
|
||||
gstccextractor.h \
|
||||
gstcea708decoder.c \
|
||||
|
@ -45,5 +47,6 @@ libgstclosedcaption_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)
|
|||
|
||||
noinst_HEADERS = \
|
||||
gstccextractor.h \
|
||||
gstcccombiner.h \
|
||||
gstline21dec.h \
|
||||
$(zvbi_headers)
|
||||
|
|
503
ext/closedcaption/gstcccombiner.c
Normal file
503
ext/closedcaption/gstcccombiner.c
Normal file
|
@ -0,0 +1,503 @@
|
|||
/*
|
||||
* GStreamer
|
||||
* Copyright (C) 2018 Sebastian Dröge <sebastian@centricular.com>
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
|
||||
#ifdef HAVE_CONFIG_H
|
||||
# include <config.h>
|
||||
#endif
|
||||
|
||||
#include <gst/gst.h>
|
||||
#include <gst/base/base.h>
|
||||
#include <gst/video/video.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "gstcccombiner.h"
|
||||
|
||||
GST_DEBUG_CATEGORY_STATIC (gst_cc_combiner_debug);
|
||||
#define GST_CAT_DEFAULT gst_cc_combiner_debug
|
||||
|
||||
static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink",
|
||||
GST_PAD_SINK,
|
||||
GST_PAD_ALWAYS,
|
||||
GST_STATIC_CAPS_ANY);
|
||||
|
||||
static GstStaticPadTemplate srctemplate = GST_STATIC_PAD_TEMPLATE ("src",
|
||||
GST_PAD_SRC,
|
||||
GST_PAD_ALWAYS,
|
||||
GST_STATIC_CAPS_ANY);
|
||||
|
||||
static GstStaticPadTemplate captiontemplate =
|
||||
GST_STATIC_PAD_TEMPLATE ("caption",
|
||||
GST_PAD_SINK,
|
||||
GST_PAD_REQUEST,
|
||||
GST_STATIC_CAPS
|
||||
("closedcaption/x-cea-608,format={ (string) raw, (string) cc_data}; "
|
||||
"closedcaption/x-cea-708,format={ (string) cc_data, (string) cdp }"));
|
||||
|
||||
G_DEFINE_TYPE (GstCCCombiner, gst_cc_combiner, GST_TYPE_AGGREGATOR);
|
||||
#define parent_class gst_cc_combiner_parent_class
|
||||
|
||||
typedef struct
|
||||
{
|
||||
GstVideoCaptionType caption_type;
|
||||
GstBuffer *buffer;
|
||||
} CaptionData;
|
||||
|
||||
static void
|
||||
caption_data_clear (CaptionData * data)
|
||||
{
|
||||
gst_buffer_unref (data->buffer);
|
||||
}
|
||||
|
||||
static void
|
||||
gst_cc_combiner_finalize (GObject * object)
|
||||
{
|
||||
GstCCCombiner *self = GST_CCCOMBINER (object);
|
||||
|
||||
g_array_unref (self->current_frame_captions);
|
||||
self->current_frame_captions = NULL;
|
||||
|
||||
G_OBJECT_CLASS (parent_class)->finalize (object);
|
||||
}
|
||||
|
||||
#define GST_FLOW_NEED_DATA GST_FLOW_CUSTOM_SUCCESS
|
||||
|
||||
static GstFlowReturn
|
||||
gst_cc_combiner_collect_captions (GstCCCombiner * self)
|
||||
{
|
||||
GstAggregatorPad *caption_pad;
|
||||
GstBuffer *video_buf;
|
||||
|
||||
g_assert (self->current_video_buffer != NULL);
|
||||
|
||||
caption_pad =
|
||||
GST_AGGREGATOR_PAD_CAST (gst_element_get_static_pad (GST_ELEMENT_CAST
|
||||
(self), "caption"));
|
||||
|
||||
/* No caption pad, forward buffer directly */
|
||||
if (!caption_pad) {
|
||||
GST_LOG_OBJECT (self, "No caption pad, passing through video");
|
||||
video_buf = self->current_video_buffer;
|
||||
self->current_video_buffer = NULL;
|
||||
return gst_aggregator_finish_buffer (GST_AGGREGATOR_CAST (self), video_buf);
|
||||
}
|
||||
|
||||
GST_LOG_OBJECT (self, "Trying to collect captions for queued video buffer");
|
||||
do {
|
||||
GstBuffer *caption_buf;
|
||||
GstClockTime caption_time;
|
||||
CaptionData caption_data;
|
||||
|
||||
caption_buf = gst_aggregator_pad_peek_buffer (caption_pad);
|
||||
if (!caption_buf) {
|
||||
if (gst_aggregator_pad_is_eos (caption_pad)) {
|
||||
GST_DEBUG_OBJECT (self, "Caption pad is EOS, we're done");
|
||||
break;
|
||||
} else {
|
||||
GST_DEBUG_OBJECT (self, "Need more caption data");
|
||||
gst_object_unref (caption_pad);
|
||||
return GST_FLOW_NEED_DATA;
|
||||
}
|
||||
}
|
||||
|
||||
caption_time = GST_BUFFER_PTS (caption_buf);
|
||||
if (!GST_CLOCK_TIME_IS_VALID (caption_time)) {
|
||||
GST_ERROR_OBJECT (self, "Caption buffer without PTS");
|
||||
|
||||
gst_buffer_unref (caption_buf);
|
||||
gst_object_unref (caption_pad);
|
||||
|
||||
return GST_FLOW_ERROR;
|
||||
}
|
||||
|
||||
caption_time =
|
||||
gst_segment_to_running_time (&caption_pad->segment, GST_FORMAT_TIME,
|
||||
caption_time);
|
||||
|
||||
if (!GST_CLOCK_TIME_IS_VALID (caption_time)) {
|
||||
GST_DEBUG_OBJECT (self, "Caption buffer outside segment, dropping");
|
||||
|
||||
gst_aggregator_pad_drop_buffer (caption_pad);
|
||||
gst_buffer_unref (caption_buf);
|
||||
|
||||
continue;
|
||||
}
|
||||
|
||||
/* Collected all caption buffers for this video buffer */
|
||||
if (caption_time >= self->current_video_running_time_end) {
|
||||
gst_buffer_unref (caption_buf);
|
||||
break;
|
||||
} else if (caption_time < self->current_video_running_time) {
|
||||
GST_DEBUG_OBJECT (self,
|
||||
"Caption buffer before current video frame, dropping");
|
||||
|
||||
gst_aggregator_pad_drop_buffer (caption_pad);
|
||||
gst_buffer_unref (caption_buf);
|
||||
continue;
|
||||
}
|
||||
|
||||
/* This caption buffer has to be collected */
|
||||
GST_LOG_OBJECT (self,
|
||||
"Collecting caption buffer %p %" GST_TIME_FORMAT " for video buffer %p",
|
||||
caption_buf, GST_TIME_ARGS (caption_time), self->current_video_buffer);
|
||||
caption_data.caption_type = self->current_caption_type;
|
||||
caption_data.buffer = caption_buf;
|
||||
g_array_append_val (self->current_frame_captions, caption_data);
|
||||
gst_aggregator_pad_drop_buffer (caption_pad);
|
||||
} while (TRUE);
|
||||
|
||||
if (self->current_frame_captions->len > 0) {
|
||||
guint i;
|
||||
|
||||
GST_LOG_OBJECT (self, "Attaching %u captions to buffer %p",
|
||||
self->current_frame_captions->len, self->current_video_buffer);
|
||||
video_buf = gst_buffer_make_writable (self->current_video_buffer);
|
||||
self->current_video_buffer = NULL;
|
||||
|
||||
for (i = 0; i < self->current_frame_captions->len; i++) {
|
||||
CaptionData *caption_data =
|
||||
&g_array_index (self->current_frame_captions, CaptionData, i);
|
||||
GstMapInfo map;
|
||||
|
||||
gst_buffer_map (caption_data->buffer, &map, GST_MAP_READ);
|
||||
gst_buffer_add_video_caption_meta (video_buf, caption_data->caption_type,
|
||||
map.data, map.size);
|
||||
gst_buffer_unmap (caption_data->buffer, &map);
|
||||
}
|
||||
|
||||
g_array_set_size (self->current_frame_captions, 0);
|
||||
} else {
|
||||
GST_LOG_OBJECT (self, "No captions for buffer %p",
|
||||
self->current_video_buffer);
|
||||
video_buf = self->current_video_buffer;
|
||||
self->current_video_buffer = NULL;
|
||||
}
|
||||
|
||||
gst_object_unref (caption_pad);
|
||||
|
||||
return gst_aggregator_finish_buffer (GST_AGGREGATOR_CAST (self), video_buf);
|
||||
}
|
||||
|
||||
static GstFlowReturn
|
||||
gst_cc_combiner_aggregate (GstAggregator * aggregator, gboolean timeout)
|
||||
{
|
||||
GstCCCombiner *self = GST_CCCOMBINER (aggregator);
|
||||
GstFlowReturn flow_ret = GST_FLOW_OK;
|
||||
|
||||
/* If we have no current video buffer, queue one. If we have one but
|
||||
* its end running time is not known yet, try to determine it from the
|
||||
* next video buffer */
|
||||
if (!self->current_video_buffer
|
||||
|| !GST_CLOCK_TIME_IS_VALID (self->current_video_running_time_end)) {
|
||||
GstAggregatorPad *video_pad;
|
||||
GstClockTime video_start;
|
||||
GstBuffer *video_buf;
|
||||
|
||||
video_pad =
|
||||
GST_AGGREGATOR_PAD_CAST (gst_element_get_static_pad (GST_ELEMENT_CAST
|
||||
(aggregator), "sink"));
|
||||
video_buf = gst_aggregator_pad_peek_buffer (video_pad);
|
||||
if (!video_buf) {
|
||||
if (gst_aggregator_pad_is_eos (video_pad)) {
|
||||
GST_DEBUG_OBJECT (aggregator, "Video pad is EOS, we're done");
|
||||
|
||||
/* Assume that this buffer ends where it started +50ms (25fps) and handle it */
|
||||
if (self->current_video_buffer) {
|
||||
self->current_video_running_time_end =
|
||||
self->current_video_running_time + 50 * GST_MSECOND;
|
||||
flow_ret = gst_cc_combiner_collect_captions (self);
|
||||
}
|
||||
|
||||
/* If we collected all captions for the remaining video frame we're
|
||||
* done, otherwise get called another time and go directly into the
|
||||
* outer branch for finishing the current video frame */
|
||||
if (flow_ret == GST_FLOW_NEED_DATA)
|
||||
flow_ret = GST_FLOW_OK;
|
||||
else
|
||||
flow_ret = GST_FLOW_EOS;
|
||||
} else {
|
||||
GST_LOG_OBJECT (aggregator, "Video pad has no data, waiting");
|
||||
flow_ret = GST_FLOW_OK;
|
||||
}
|
||||
|
||||
gst_object_unref (video_pad);
|
||||
return flow_ret;
|
||||
}
|
||||
|
||||
video_start = GST_BUFFER_PTS (video_buf);
|
||||
if (!GST_CLOCK_TIME_IS_VALID (video_start)) {
|
||||
gst_buffer_unref (video_buf);
|
||||
gst_object_unref (video_pad);
|
||||
|
||||
GST_ERROR_OBJECT (aggregator, "Video buffer without PTS");
|
||||
|
||||
return GST_FLOW_ERROR;
|
||||
}
|
||||
|
||||
video_start =
|
||||
gst_segment_to_running_time (&video_pad->segment, GST_FORMAT_TIME,
|
||||
video_start);
|
||||
if (!GST_CLOCK_TIME_IS_VALID (video_start)) {
|
||||
GST_DEBUG_OBJECT (aggregator, "Buffer outside segment, dropping");
|
||||
gst_aggregator_pad_drop_buffer (video_pad);
|
||||
gst_buffer_unref (video_buf);
|
||||
gst_object_unref (video_pad);
|
||||
return GST_FLOW_OK;
|
||||
}
|
||||
|
||||
if (self->current_video_buffer) {
|
||||
/* If we already have a video buffer just update the current end running
|
||||
* time accordingly. That's what was missing and why we got here */
|
||||
self->current_video_running_time_end = video_start;
|
||||
gst_buffer_unref (video_buf);
|
||||
GST_LOG_OBJECT (self,
|
||||
"Determined end timestamp for video buffer: %p %" GST_TIME_FORMAT
|
||||
" - %" GST_TIME_FORMAT, self->current_video_buffer,
|
||||
GST_TIME_ARGS (self->current_video_running_time),
|
||||
GST_TIME_ARGS (self->current_video_running_time_end));
|
||||
} else {
|
||||
/* Otherwise we had no buffer queued currently. Let's do that now
|
||||
* so that we can collect captions for it */
|
||||
gst_buffer_replace (&self->current_video_buffer, video_buf);
|
||||
self->current_video_running_time = video_start;
|
||||
gst_aggregator_pad_drop_buffer (video_pad);
|
||||
gst_buffer_unref (video_buf);
|
||||
|
||||
if (GST_BUFFER_DURATION_IS_VALID (video_buf)) {
|
||||
GstClockTime end_time =
|
||||
GST_BUFFER_PTS (video_buf) + GST_BUFFER_DURATION (video_buf);
|
||||
if (video_pad->segment.stop != -1 && end_time > video_pad->segment.stop)
|
||||
end_time = video_pad->segment.stop;
|
||||
self->current_video_running_time_end =
|
||||
gst_segment_to_running_time (&video_pad->segment, GST_FORMAT_TIME,
|
||||
end_time);
|
||||
} else if (self->video_fps_n != 0 && self->video_fps_d != 0) {
|
||||
GstClockTime end_time =
|
||||
GST_BUFFER_PTS (video_buf) + gst_util_uint64_scale_int (GST_SECOND,
|
||||
self->video_fps_d, self->video_fps_n);
|
||||
if (video_pad->segment.stop != -1 && end_time > video_pad->segment.stop)
|
||||
end_time = video_pad->segment.stop;
|
||||
self->current_video_running_time_end =
|
||||
gst_segment_to_running_time (&video_pad->segment, GST_FORMAT_TIME,
|
||||
end_time);
|
||||
} else {
|
||||
self->current_video_running_time_end = GST_CLOCK_TIME_NONE;
|
||||
}
|
||||
|
||||
GST_LOG_OBJECT (self,
|
||||
"Queued new video buffer: %p %" GST_TIME_FORMAT " - %"
|
||||
GST_TIME_FORMAT, self->current_video_buffer,
|
||||
GST_TIME_ARGS (self->current_video_running_time),
|
||||
GST_TIME_ARGS (self->current_video_running_time_end));
|
||||
}
|
||||
|
||||
gst_object_unref (video_pad);
|
||||
}
|
||||
|
||||
/* At this point we have a video buffer queued and can start collecting
|
||||
* caption buffers for it */
|
||||
g_assert (self->current_video_buffer != NULL);
|
||||
g_assert (GST_CLOCK_TIME_IS_VALID (self->current_video_running_time));
|
||||
g_assert (GST_CLOCK_TIME_IS_VALID (self->current_video_running_time_end));
|
||||
|
||||
flow_ret = gst_cc_combiner_collect_captions (self);
|
||||
|
||||
/* Only if we collected all captions we replace the current video buffer
|
||||
* with NULL and continue with the next one on the next call */
|
||||
if (flow_ret == GST_FLOW_NEED_DATA) {
|
||||
flow_ret = GST_FLOW_OK;
|
||||
} else {
|
||||
gst_buffer_replace (&self->current_video_buffer, NULL);
|
||||
self->current_video_running_time = self->current_video_running_time_end =
|
||||
GST_CLOCK_TIME_NONE;
|
||||
}
|
||||
|
||||
return flow_ret;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_cc_combiner_sink_event (GstAggregator * aggregator,
|
||||
GstAggregatorPad * agg_pad, GstEvent * event)
|
||||
{
|
||||
GstCCCombiner *self = GST_CCCOMBINER (aggregator);
|
||||
|
||||
switch (GST_EVENT_TYPE (event)) {
|
||||
case GST_EVENT_CAPS:{
|
||||
GstCaps *caps;
|
||||
GstStructure *s;
|
||||
|
||||
gst_event_parse_caps (event, &caps);
|
||||
s = gst_caps_get_structure (caps, 0);
|
||||
|
||||
if (strcmp (GST_OBJECT_NAME (agg_pad), "caption") == 0) {
|
||||
const gchar *format;
|
||||
|
||||
format = gst_structure_get_string (s, "format");
|
||||
if (gst_structure_has_name (s, "closedcaption/x-cea-608")) {
|
||||
if (strcmp (format, "raw") == 0) {
|
||||
self->current_caption_type = GST_VIDEO_CAPTION_TYPE_CEA608_RAW;
|
||||
} else if (strcmp (format, "cc_data") == 0) {
|
||||
self->current_caption_type =
|
||||
GST_VIDEO_CAPTION_TYPE_CEA608_IN_CEA708_RAW;
|
||||
} else {
|
||||
g_assert_not_reached ();
|
||||
}
|
||||
} else if (gst_structure_has_name (s, "closedcaption/x-cea-708")) {
|
||||
if (strcmp (format, "cc_data") == 0) {
|
||||
self->current_caption_type = GST_VIDEO_CAPTION_TYPE_CEA708_RAW;
|
||||
} else if (strcmp (format, "cdp") == 0) {
|
||||
self->current_caption_type = GST_VIDEO_CAPTION_TYPE_CEA708_CDP;
|
||||
} else {
|
||||
g_assert_not_reached ();
|
||||
}
|
||||
} else {
|
||||
g_assert_not_reached ();
|
||||
}
|
||||
} else {
|
||||
if (!gst_structure_get_fraction (s, "framerate", &self->video_fps_n,
|
||||
&self->video_fps_d))
|
||||
self->video_fps_n = self->video_fps_d = 0;
|
||||
|
||||
gst_aggregator_set_src_caps (aggregator, caps);
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return GST_AGGREGATOR_CLASS (parent_class)->sink_event (aggregator, agg_pad,
|
||||
event);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
gst_cc_combiner_stop (GstAggregator * aggregator)
|
||||
{
|
||||
GstCCCombiner *self = GST_CCCOMBINER (aggregator);
|
||||
|
||||
self->video_fps_n = self->video_fps_d = 0;
|
||||
self->current_video_running_time = self->current_video_running_time_end =
|
||||
GST_CLOCK_TIME_NONE;
|
||||
gst_buffer_replace (&self->current_video_buffer, NULL);
|
||||
|
||||
g_array_set_size (self->current_frame_captions, 0);
|
||||
self->current_caption_type = GST_VIDEO_CAPTION_TYPE_UNKNOWN;
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static GstFlowReturn
|
||||
gst_cc_combiner_flush (GstAggregator * aggregator)
|
||||
{
|
||||
GstCCCombiner *self = GST_CCCOMBINER (aggregator);
|
||||
|
||||
self->current_video_running_time = self->current_video_running_time_end =
|
||||
GST_CLOCK_TIME_NONE;
|
||||
gst_buffer_replace (&self->current_video_buffer, NULL);
|
||||
|
||||
g_array_set_size (self->current_frame_captions, 0);
|
||||
|
||||
return GST_FLOW_OK;
|
||||
}
|
||||
|
||||
static GstAggregatorPad *
|
||||
gst_cc_combiner_create_new_pad (GstAggregator * aggregator,
|
||||
GstPadTemplate * templ, const gchar * req_name, const GstCaps * caps)
|
||||
{
|
||||
GstCCCombiner *self = GST_CCCOMBINER (aggregator);
|
||||
GstAggregatorPad *agg_pad;
|
||||
|
||||
if (templ->direction != GST_PAD_SINK)
|
||||
return NULL;
|
||||
|
||||
if (templ->presence != GST_PAD_REQUEST)
|
||||
return NULL;
|
||||
|
||||
if (strcmp (templ->name_template, "caption") != 0)
|
||||
return NULL;
|
||||
|
||||
GST_OBJECT_LOCK (self);
|
||||
agg_pad = g_object_new (GST_TYPE_AGGREGATOR_PAD,
|
||||
"name", "caption", "direction", GST_PAD_SINK, "template", templ, NULL);
|
||||
self->current_caption_type = GST_VIDEO_CAPTION_TYPE_UNKNOWN;
|
||||
GST_OBJECT_UNLOCK (self);
|
||||
|
||||
return agg_pad;
|
||||
}
|
||||
|
||||
static void
|
||||
gst_cc_combiner_class_init (GstCCCombinerClass * klass)
|
||||
{
|
||||
GObjectClass *gobject_class;
|
||||
GstElementClass *gstelement_class;
|
||||
GstAggregatorClass *aggregator_class;
|
||||
|
||||
gobject_class = (GObjectClass *) klass;
|
||||
gstelement_class = (GstElementClass *) klass;
|
||||
aggregator_class = (GstAggregatorClass *) klass;
|
||||
|
||||
gobject_class->finalize = gst_cc_combiner_finalize;
|
||||
|
||||
gst_element_class_set_static_metadata (gstelement_class,
|
||||
"Closed Caption Combiner",
|
||||
"Filter",
|
||||
"Combines GstVideoCaptionMeta with video input stream",
|
||||
"Sebastian Dröge <sebastian@centricular.com>");
|
||||
|
||||
gst_element_class_add_static_pad_template_with_gtype (gstelement_class,
|
||||
&sinktemplate, GST_TYPE_AGGREGATOR_PAD);
|
||||
gst_element_class_add_static_pad_template_with_gtype (gstelement_class,
|
||||
&srctemplate, GST_TYPE_AGGREGATOR_PAD);
|
||||
gst_element_class_add_static_pad_template_with_gtype (gstelement_class,
|
||||
&captiontemplate, GST_TYPE_AGGREGATOR_PAD);
|
||||
|
||||
aggregator_class->aggregate = gst_cc_combiner_aggregate;
|
||||
aggregator_class->stop = gst_cc_combiner_stop;
|
||||
aggregator_class->flush = gst_cc_combiner_flush;
|
||||
aggregator_class->create_new_pad = gst_cc_combiner_create_new_pad;
|
||||
aggregator_class->sink_event = gst_cc_combiner_sink_event;
|
||||
|
||||
GST_DEBUG_CATEGORY_INIT (gst_cc_combiner_debug, "cccombiner",
|
||||
0, "Closed Caption combiner");
|
||||
}
|
||||
|
||||
static void
|
||||
gst_cc_combiner_init (GstCCCombiner * self)
|
||||
{
|
||||
GstPadTemplate *templ;
|
||||
GstAggregatorPad *agg_pad;
|
||||
|
||||
templ = gst_static_pad_template_get (&sinktemplate);
|
||||
agg_pad = g_object_new (GST_TYPE_AGGREGATOR_PAD,
|
||||
"name", "sink", "direction", GST_PAD_SINK, "template", templ, NULL);
|
||||
gst_object_unref (templ);
|
||||
gst_element_add_pad (GST_ELEMENT_CAST (self), GST_PAD_CAST (agg_pad));
|
||||
|
||||
self->current_frame_captions =
|
||||
g_array_new (FALSE, FALSE, sizeof (CaptionData));
|
||||
g_array_set_clear_func (self->current_frame_captions,
|
||||
(GDestroyNotify) caption_data_clear);
|
||||
|
||||
self->current_video_running_time = self->current_video_running_time_end =
|
||||
GST_CLOCK_TIME_NONE;
|
||||
|
||||
self->current_caption_type = GST_VIDEO_CAPTION_TYPE_UNKNOWN;
|
||||
}
|
64
ext/closedcaption/gstcccombiner.h
Normal file
64
ext/closedcaption/gstcccombiner.h
Normal file
|
@ -0,0 +1,64 @@
|
|||
/*
|
||||
* GStreamer
|
||||
* Copyright (C) 2018 Sebastian Dröge <sebastian@centricular.com>
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
#ifndef __GST_CCCOMBINER_H__
|
||||
#define __GST_CCCOMBINER_H__
|
||||
|
||||
#include <gst/gst.h>
|
||||
#include <gst/base/base.h>
|
||||
#include <gst/video/video.h>
|
||||
|
||||
G_BEGIN_DECLS
|
||||
#define GST_TYPE_CCCOMBINER \
|
||||
(gst_cc_combiner_get_type())
|
||||
#define GST_CCCOMBINER(obj) \
|
||||
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_CCCOMBINER,GstCCCombiner))
|
||||
#define GST_CCCOMBINER_CLASS(klass) \
|
||||
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_CCCOMBINER,GstCCCombinerClass))
|
||||
#define GST_IS_CCCOMBINER(obj) \
|
||||
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_CCCOMBINER))
|
||||
#define GST_IS_CCCOMBINER_CLASS(klass) \
|
||||
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_CCCOMBINER))
|
||||
|
||||
typedef struct _GstCCCombiner GstCCCombiner;
|
||||
typedef struct _GstCCCombinerClass GstCCCombinerClass;
|
||||
|
||||
struct _GstCCCombiner
|
||||
{
|
||||
GstAggregator parent;
|
||||
|
||||
gint video_fps_n, video_fps_d;
|
||||
GstClockTime current_video_running_time;
|
||||
GstClockTime current_video_running_time_end;
|
||||
GstBuffer *current_video_buffer;
|
||||
|
||||
GArray *current_frame_captions;
|
||||
GstVideoCaptionType current_caption_type;
|
||||
};
|
||||
|
||||
struct _GstCCCombinerClass
|
||||
{
|
||||
GstAggregatorClass parent_class;
|
||||
};
|
||||
|
||||
GType gst_cc_combiner_get_type (void);
|
||||
|
||||
G_END_DECLS
|
||||
#endif /* __GST_CCCOMBINER_H__ */
|
|
@ -25,6 +25,7 @@
|
|||
|
||||
#include <gst/gst.h>
|
||||
|
||||
#include "gstcccombiner.h"
|
||||
#include "gstccextractor.h"
|
||||
#include "gstline21dec.h"
|
||||
#include "gstceaccoverlay.h"
|
||||
|
@ -34,13 +35,16 @@ closedcaption_init (GstPlugin * plugin)
|
|||
{
|
||||
gboolean ret;
|
||||
|
||||
ret = gst_element_register (plugin, "ccextractor", GST_RANK_NONE,
|
||||
ret = gst_element_register (plugin, "cccombiner", GST_RANK_NONE,
|
||||
GST_TYPE_CCCOMBINER);
|
||||
|
||||
ret &= gst_element_register (plugin, "ccextractor", GST_RANK_NONE,
|
||||
GST_TYPE_CCEXTRACTOR);
|
||||
|
||||
ret &= gst_element_register (plugin, "line21decoder", GST_RANK_NONE,
|
||||
GST_TYPE_LINE21DECODER);
|
||||
|
||||
ret = gst_element_register (plugin, "cc708overlay", GST_RANK_PRIMARY,
|
||||
ret &= gst_element_register (plugin, "cc708overlay", GST_RANK_PRIMARY,
|
||||
GST_TYPE_CEA_CC_OVERLAY);
|
||||
|
||||
return ret;
|
||||
|
|
|
@ -10,12 +10,12 @@ zvbi_sources = [
|
|||
|
||||
if pangocairo_dep.found()
|
||||
gstclosedcaption = library('gstclosedcaption',
|
||||
'gstccextractor.c', 'gstclosedcaption.c', 'gstline21dec.c',
|
||||
'gstcccombiner.c', 'gstccextractor.c', 'gstclosedcaption.c', 'gstline21dec.c',
|
||||
'gstcea708decoder.c', 'gstceaccoverlay.c', zvbi_sources,
|
||||
c_args : gst_plugins_bad_args,
|
||||
link_args : noseh_link_args,
|
||||
include_directories : [configinc],
|
||||
dependencies : [gstvideo_dep, pangocairo_dep],
|
||||
dependencies : [gstvideo_dep, gstbase_dep, gst_dep, pangocairo_dep],
|
||||
install : true,
|
||||
install_dir : plugins_install_dir,
|
||||
)
|
||||
|
|
Loading…
Reference in a new issue