mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-26 00:58:12 +00:00
assrender: Synchronize text and video by taking the running time instead of the stream time
This commit is contained in:
parent
8feed39593
commit
51da9d976f
1 changed files with 51 additions and 21 deletions
|
@ -175,6 +175,7 @@ gst_assrender_init (Gstassrender * render, GstassrenderClass * gclass)
|
||||||
render->embeddedfonts = TRUE;
|
render->embeddedfonts = TRUE;
|
||||||
|
|
||||||
gst_segment_init (&render->video_segment, GST_FORMAT_TIME);
|
gst_segment_init (&render->video_segment, GST_FORMAT_TIME);
|
||||||
|
gst_segment_init (&render->subtitle_segment, GST_FORMAT_TIME);
|
||||||
|
|
||||||
render->ass_library = ass_library_init ();
|
render->ass_library = ass_library_init ();
|
||||||
ass_set_fonts_dir (render->ass_library, "./");
|
ass_set_fonts_dir (render->ass_library, "./");
|
||||||
|
@ -264,6 +265,8 @@ gst_assrender_change_state (GstElement * element, GstStateChange transition)
|
||||||
switch (transition) {
|
switch (transition) {
|
||||||
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
||||||
render->subtitle_flushing = FALSE;
|
render->subtitle_flushing = FALSE;
|
||||||
|
gst_segment_init (&render->video_segment, GST_FORMAT_TIME);
|
||||||
|
gst_segment_init (&render->subtitle_segment, GST_FORMAT_TIME);
|
||||||
break;
|
break;
|
||||||
case GST_STATE_CHANGE_NULL_TO_READY:
|
case GST_STATE_CHANGE_NULL_TO_READY:
|
||||||
case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
|
case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
|
||||||
|
@ -451,17 +454,22 @@ gst_assrender_setcaps_text (GstPad * pad, GstCaps * caps)
|
||||||
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_assrender_process_text (Gstassrender * render, GstBuffer * buffer)
|
gst_assrender_process_text (Gstassrender * render, GstBuffer * buffer,
|
||||||
|
GstClockTime running_time, GstClockTime duration)
|
||||||
{
|
{
|
||||||
char *data = (gchar *) GST_BUFFER_DATA (buffer);
|
char *data = (gchar *) GST_BUFFER_DATA (buffer);
|
||||||
guint size = GST_BUFFER_SIZE (buffer);
|
guint size = GST_BUFFER_SIZE (buffer);
|
||||||
double pts_start, pts_end;
|
double pts_start, pts_end;
|
||||||
|
|
||||||
pts_start = GST_BUFFER_TIMESTAMP (buffer);
|
pts_start = running_time;
|
||||||
pts_start /= GST_MSECOND;
|
pts_start /= GST_MSECOND;
|
||||||
pts_end = GST_BUFFER_DURATION (buffer);
|
pts_end = duration;
|
||||||
pts_end /= GST_MSECOND;
|
pts_end /= GST_MSECOND;
|
||||||
|
|
||||||
|
GST_DEBUG_OBJECT (render,
|
||||||
|
"Processing subtitles with running time %" GST_TIME_FORMAT
|
||||||
|
" and duration %" GST_TIME_FORMAT, GST_TIME_ARGS (running_time),
|
||||||
|
GST_TIME_ARGS (duration));
|
||||||
ass_process_chunk (render->ass_track, data, size, pts_start, pts_end);
|
ass_process_chunk (render->ass_track, data, size, pts_start, pts_end);
|
||||||
gst_buffer_unref (buffer);
|
gst_buffer_unref (buffer);
|
||||||
}
|
}
|
||||||
|
@ -473,8 +481,6 @@ gst_assrender_chain_video (GstPad * pad, GstBuffer * buffer)
|
||||||
GstFlowReturn ret = GST_FLOW_OK;
|
GstFlowReturn ret = GST_FLOW_OK;
|
||||||
gboolean in_seg = FALSE;
|
gboolean in_seg = FALSE;
|
||||||
gint64 start, stop, clip_start = 0, clip_stop = 0;
|
gint64 start, stop, clip_start = 0, clip_stop = 0;
|
||||||
double timestamp;
|
|
||||||
double step;
|
|
||||||
ASS_Image *ass_image;
|
ASS_Image *ass_image;
|
||||||
|
|
||||||
render = GST_ASSRENDER (GST_PAD_PARENT (pad));
|
render = GST_ASSRENDER (GST_PAD_PARENT (pad));
|
||||||
|
@ -520,14 +526,25 @@ gst_assrender_chain_video (GstPad * pad, GstBuffer * buffer)
|
||||||
|
|
||||||
g_mutex_lock (render->subtitle_mutex);
|
g_mutex_lock (render->subtitle_mutex);
|
||||||
if (render->subtitle_pending) {
|
if (render->subtitle_pending) {
|
||||||
if (GST_BUFFER_TIMESTAMP (render->subtitle_pending) <=
|
GstClockTime sub_running_time, vid_running_time;
|
||||||
GST_BUFFER_TIMESTAMP (buffer) + GST_BUFFER_DURATION (buffer)) {
|
GstClockTime sub_duration, vid_duration;
|
||||||
gst_assrender_process_text (render, render->subtitle_pending);
|
|
||||||
|
sub_running_time =
|
||||||
|
gst_segment_to_running_time (&render->subtitle_segment, GST_FORMAT_TIME,
|
||||||
|
GST_BUFFER_TIMESTAMP (render->subtitle_pending));
|
||||||
|
vid_running_time =
|
||||||
|
gst_segment_to_running_time (&render->video_segment, GST_FORMAT_TIME,
|
||||||
|
GST_BUFFER_TIMESTAMP (buffer));
|
||||||
|
|
||||||
|
sub_duration = GST_BUFFER_DURATION (render->subtitle_pending);
|
||||||
|
vid_duration = GST_BUFFER_DURATION (buffer);
|
||||||
|
|
||||||
|
if (sub_running_time <= vid_running_time + vid_duration) {
|
||||||
|
gst_assrender_process_text (render, render->subtitle_pending,
|
||||||
|
sub_running_time, sub_duration);
|
||||||
render->subtitle_pending = NULL;
|
render->subtitle_pending = NULL;
|
||||||
g_cond_signal (render->subtitle_cond);
|
g_cond_signal (render->subtitle_cond);
|
||||||
} else if (GST_BUFFER_TIMESTAMP (render->subtitle_pending) +
|
} else if (sub_running_time + sub_duration <= vid_running_time) {
|
||||||
GST_BUFFER_DURATION (render->subtitle_pending) <=
|
|
||||||
GST_BUFFER_TIMESTAMP (buffer)) {
|
|
||||||
gst_buffer_unref (render->subtitle_pending);
|
gst_buffer_unref (render->subtitle_pending);
|
||||||
render->subtitle_pending = NULL;
|
render->subtitle_pending = NULL;
|
||||||
g_cond_signal (render->subtitle_cond);
|
g_cond_signal (render->subtitle_cond);
|
||||||
|
@ -538,19 +555,24 @@ gst_assrender_chain_video (GstPad * pad, GstBuffer * buffer)
|
||||||
/* now start rendering subtitles, if all conditions are met */
|
/* now start rendering subtitles, if all conditions are met */
|
||||||
if (render->renderer_init_ok && render->track_init_ok && render->enable) {
|
if (render->renderer_init_ok && render->track_init_ok && render->enable) {
|
||||||
int counter;
|
int counter;
|
||||||
|
GstClockTime running_time;
|
||||||
|
double timestamp;
|
||||||
|
double step;
|
||||||
|
|
||||||
timestamp = GST_BUFFER_TIMESTAMP (buffer);
|
running_time =
|
||||||
GST_DEBUG_OBJECT (render, "rendering frame for timestamp %" GST_TIME_FORMAT,
|
gst_segment_to_running_time (&render->video_segment, GST_FORMAT_TIME,
|
||||||
GST_TIME_ARGS (timestamp));
|
GST_BUFFER_TIMESTAMP (buffer));
|
||||||
|
GST_DEBUG_OBJECT (render,
|
||||||
|
"rendering frame for running time %" GST_TIME_FORMAT,
|
||||||
|
GST_TIME_ARGS (running_time));
|
||||||
/* libass needs timestamps in ms */
|
/* libass needs timestamps in ms */
|
||||||
timestamp = timestamp / GST_MSECOND;
|
timestamp = running_time / GST_MSECOND;
|
||||||
|
|
||||||
/* only for testing right now. could possibly be used for optimizations? */
|
/* only for testing right now. could possibly be used for optimizations? */
|
||||||
step = ass_step_sub (render->ass_track, timestamp, 1);
|
step = ass_step_sub (render->ass_track, timestamp, 1);
|
||||||
GST_DEBUG_OBJECT (render, "Current timestamp: %" GST_TIME_FORMAT
|
GST_DEBUG_OBJECT (render, "Current running time: %" GST_TIME_FORMAT
|
||||||
" // Next event: %" GST_TIME_FORMAT,
|
" // Next event: %" GST_TIME_FORMAT,
|
||||||
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)),
|
GST_TIME_ARGS (running_time), GST_TIME_ARGS (step * GST_MSECOND));
|
||||||
GST_TIME_ARGS (step * GST_MSECOND));
|
|
||||||
|
|
||||||
/* not sure what the last parameter to this call is for (detect_change) */
|
/* not sure what the last parameter to this call is for (detect_change) */
|
||||||
ass_image = ass_render_frame (render->ass_renderer, render->ass_track,
|
ass_image = ass_render_frame (render->ass_renderer, render->ass_track,
|
||||||
|
@ -612,6 +634,7 @@ gst_assrender_chain_text (GstPad * pad, GstBuffer * buffer)
|
||||||
GstFlowReturn ret = GST_FLOW_OK;
|
GstFlowReturn ret = GST_FLOW_OK;
|
||||||
Gstassrender *render;
|
Gstassrender *render;
|
||||||
GstClockTime timestamp, duration;
|
GstClockTime timestamp, duration;
|
||||||
|
GstClockTime sub_running_time, vid_running_time;
|
||||||
|
|
||||||
render = GST_ASSRENDER (GST_PAD_PARENT (pad));
|
render = GST_ASSRENDER (GST_PAD_PARENT (pad));
|
||||||
|
|
||||||
|
@ -629,7 +652,14 @@ gst_assrender_chain_text (GstPad * pad, GstBuffer * buffer)
|
||||||
return GST_FLOW_OK;
|
return GST_FLOW_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (timestamp > render->video_segment.last_stop) {
|
sub_running_time =
|
||||||
|
gst_segment_to_running_time (&render->subtitle_segment, GST_FORMAT_TIME,
|
||||||
|
timestamp);
|
||||||
|
vid_running_time =
|
||||||
|
gst_segment_to_running_time (&render->video_segment, GST_FORMAT_TIME,
|
||||||
|
render->video_segment.last_stop);
|
||||||
|
|
||||||
|
if (sub_running_time > vid_running_time) {
|
||||||
g_assert (render->subtitle_pending == NULL);
|
g_assert (render->subtitle_pending == NULL);
|
||||||
g_mutex_lock (render->subtitle_mutex);
|
g_mutex_lock (render->subtitle_mutex);
|
||||||
if (G_UNLIKELY (render->subtitle_flushing)) {
|
if (G_UNLIKELY (render->subtitle_flushing)) {
|
||||||
|
@ -642,12 +672,12 @@ gst_assrender_chain_text (GstPad * pad, GstBuffer * buffer)
|
||||||
render->subtitle_pending = buffer;
|
render->subtitle_pending = buffer;
|
||||||
g_cond_wait (render->subtitle_cond, render->subtitle_mutex);
|
g_cond_wait (render->subtitle_cond, render->subtitle_mutex);
|
||||||
g_mutex_unlock (render->subtitle_mutex);
|
g_mutex_unlock (render->subtitle_mutex);
|
||||||
} else if (timestamp + duration < render->video_segment.last_stop) {
|
} else if (sub_running_time + duration < vid_running_time) {
|
||||||
GST_DEBUG_OBJECT (render, "Too late text buffer, dropping");
|
GST_DEBUG_OBJECT (render, "Too late text buffer, dropping");
|
||||||
gst_buffer_unref (buffer);
|
gst_buffer_unref (buffer);
|
||||||
ret = GST_FLOW_OK;
|
ret = GST_FLOW_OK;
|
||||||
} else {
|
} else {
|
||||||
gst_assrender_process_text (render, buffer);
|
gst_assrender_process_text (render, buffer, sub_running_time, duration);
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (render,
|
GST_DEBUG_OBJECT (render,
|
||||||
|
|
Loading…
Reference in a new issue