mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 18:35:35 +00:00
2020 lines
58 KiB
C
2020 lines
58 KiB
C
|
/* GStreamer
|
||
|
*
|
||
|
* SPDX-License-Identifier: LGPL-2.1
|
||
|
*
|
||
|
* Copyright (C) 2013 Google Inc. All rights reserved.
|
||
|
* Copyright (C) 2013 Orange
|
||
|
* Copyright (C) 2013-2020 Apple Inc. All rights reserved.
|
||
|
* Copyright (C) 2014 Sebastian Dröge <sebastian@centricular.com>
|
||
|
* Copyright (C) 2015, 2016, 2017 Igalia, S.L
|
||
|
* Copyright (C) 2015, 2016, 2017 Metrological Group B.V.
|
||
|
* Copyright (C) 2022, 2023 Collabora Ltd.
|
||
|
*
|
||
|
* 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:gstsourcebuffer
|
||
|
* @title: GstSourceBuffer
|
||
|
* @short_description: Source Buffer
|
||
|
* @include: mse/mse.h
|
||
|
* @symbols:
|
||
|
* - GstSourceBuffer
|
||
|
*
|
||
|
* The Source Buffer is the primary means of data flow between an application
|
||
|
* and the Media Source API. It represents a single timeline of media,
|
||
|
* containing some combination of audio, video, and text tracks.
|
||
|
* An application is responsible for feeding raw data into the Source Buffer
|
||
|
* using gst_source_buffer_append_buffer() and the Source Buffer will
|
||
|
* asynchronously process the data into tracks of time-coded multimedia samples.
|
||
|
*
|
||
|
* The application as well as the associated playback component can then select
|
||
|
* to play media from any subset of tracks across all Source Buffers of a Media
|
||
|
* Source.
|
||
|
*
|
||
|
* A few control points are also provided to customize the behavior.
|
||
|
*
|
||
|
* - #GstSourceBuffer:append-mode controls how timestamps of processed samples are
|
||
|
* interpreted. They are either inserted in the timeline directly where the
|
||
|
* decoded media states they should, or inserted directly after the previously
|
||
|
* encountered sample.
|
||
|
*
|
||
|
* - #GstSourceBuffer:append-window-start / #GstSourceBuffer:append-window-end
|
||
|
* control the planned time window where media from appended data can be added
|
||
|
* to the current timeline. Any samples outside that range may be ignored.
|
||
|
*
|
||
|
* - #GstSourceBuffer:timestamp-offset is added to the start time of any sample
|
||
|
* processed.
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
|
||
|
#ifdef HAVE_CONFIG_H
|
||
|
#include "config.h"
|
||
|
#endif
|
||
|
|
||
|
#include <gst/mse/mse-enumtypes.h>
|
||
|
#include "gstsourcebuffer.h"
|
||
|
#include "gstsourcebuffer-private.h"
|
||
|
|
||
|
#include "gstmselogging-private.h"
|
||
|
#include "gstmsemediatype-private.h"
|
||
|
#include "gstmseeventqueue-private.h"
|
||
|
|
||
|
#include "gstappendpipeline-private.h"
|
||
|
#include "gstmediasource.h"
|
||
|
#include "gstmediasource-private.h"
|
||
|
#include "gstmediasourcetrack-private.h"
|
||
|
#include "gstmediasourcetrackbuffer-private.h"
|
||
|
#include "gstsourcebufferlist-private.h"
|
||
|
#include "gstmsesrc.h"
|
||
|
#include "gstmsesrc-private.h"
|
||
|
|
||
|
#define g_array_new_ranges() \
|
||
|
(g_array_new (TRUE, FALSE, sizeof (GstMediaSourceRange)))
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
GstSourceBufferCallbacks callbacks;
|
||
|
gpointer user_data;
|
||
|
} Callbacks;
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
typedef struct
|
||
|
{
|
||
|
GstSourceBuffer *parent;
|
||
|
|
||
|
GstMediaSourceTrack *track;
|
||
|
GstMediaSourceTrackBuffer *buffer;
|
||
|
|
||
|
GstTask *task;
|
||
|
GRecMutex lock;
|
||
|
|
||
|
gboolean cancelled;
|
||
|
} TrackFeedTask;
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
gsize n_samples;
|
||
|
GstSample *current_sample;
|
||
|
GstClockTime current_dts;
|
||
|
} TrackFeedAccumulator;
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
const GstClockTime time;
|
||
|
gboolean buffered;
|
||
|
} IsBufferedAccumulator;
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
const GstClockTime start;
|
||
|
const GstClockTime end;
|
||
|
gboolean start_buffered;
|
||
|
gboolean end_buffered;
|
||
|
} IsRangeBufferedAccumulator;
|
||
|
|
||
|
struct _GstSourceBuffer
|
||
|
{
|
||
|
GstObject parent_instance;
|
||
|
|
||
|
GstSourceBufferAppendMode append_mode;
|
||
|
GstClockTime append_window_start;
|
||
|
GstClockTime append_window_end;
|
||
|
gchar *content_type;
|
||
|
gboolean generate_timestamps;
|
||
|
GstClockTime timestamp_offset;
|
||
|
gboolean updating;
|
||
|
gboolean errored;
|
||
|
gsize size_limit;
|
||
|
gsize size;
|
||
|
GstBuffer *pending_data;
|
||
|
GstTask *append_to_buffer_task;
|
||
|
GRecMutex append_to_buffer_lock;
|
||
|
GstClockTime seek_time;
|
||
|
GstAppendPipeline *append_pipeline;
|
||
|
GstMseEventQueue *event_queue;
|
||
|
|
||
|
gboolean processed_init_segment;
|
||
|
|
||
|
GHashTable *track_buffers;
|
||
|
GHashTable *track_feeds;
|
||
|
|
||
|
Callbacks callbacks;
|
||
|
};
|
||
|
|
||
|
G_DEFINE_TYPE (GstSourceBuffer, gst_source_buffer, GST_TYPE_OBJECT);
|
||
|
|
||
|
enum
|
||
|
{
|
||
|
PROP_0,
|
||
|
|
||
|
PROP_APPEND_MODE,
|
||
|
PROP_APPEND_WINDOW_START,
|
||
|
PROP_APPEND_WINDOW_END,
|
||
|
PROP_BUFFERED,
|
||
|
PROP_CONTENT_TYPE,
|
||
|
PROP_TIMESTAMP_OFFSET,
|
||
|
PROP_UDPATING,
|
||
|
|
||
|
N_PROPS,
|
||
|
};
|
||
|
|
||
|
typedef enum
|
||
|
{
|
||
|
ON_UPDATE_START,
|
||
|
ON_UPDATE,
|
||
|
ON_UPDATE_END,
|
||
|
ON_ERROR,
|
||
|
ON_ABORT,
|
||
|
|
||
|
N_SIGNALS,
|
||
|
} SourceBufferEvent;
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
GstDataQueueItem item;
|
||
|
SourceBufferEvent event;
|
||
|
} SourceBufferEventItem;
|
||
|
|
||
|
#define DEFAULT_BUFFER_SIZE 1 << 24
|
||
|
#define DEFAULT_APPEND_MODE GST_SOURCE_BUFFER_APPEND_MODE_SEGMENTS
|
||
|
|
||
|
static GParamSpec *properties[N_PROPS];
|
||
|
static guint signals[N_SIGNALS];
|
||
|
|
||
|
static void call_received_init_segment (GstSourceBuffer * self);
|
||
|
static void call_duration_changed (GstSourceBuffer * self);
|
||
|
static void call_active_state_changed (GstSourceBuffer * self);
|
||
|
|
||
|
static inline gboolean is_removed (GstSourceBuffer * self);
|
||
|
static void reset_parser_state (GstSourceBuffer * self);
|
||
|
static void append_error (GstSourceBuffer * self);
|
||
|
|
||
|
static void seek_track_buffer (GstMediaSourceTrack * track,
|
||
|
GstMediaSourceTrackBuffer * buffer, GstSourceBuffer * self);
|
||
|
static void dispatch_event (SourceBufferEventItem * item, GstSourceBuffer *
|
||
|
self);
|
||
|
static void schedule_event (GstSourceBuffer * self, SourceBufferEvent event);
|
||
|
static void append_to_buffer_task (GstSourceBuffer * self);
|
||
|
static void track_feed_task (TrackFeedTask * feed);
|
||
|
static void clear_track_feed (TrackFeedTask * feed);
|
||
|
static void stop_track_feed (TrackFeedTask * feed);
|
||
|
static void start_track_feed (TrackFeedTask * feed);
|
||
|
static void reset_track_feed (TrackFeedTask * feed);
|
||
|
static TrackFeedTask *get_track_feed (GstSourceBuffer * self,
|
||
|
GstMediaSourceTrack * track);
|
||
|
static GstMediaSourceTrackBuffer *get_track_buffer (GstSourceBuffer * self,
|
||
|
GstMediaSourceTrack * track);
|
||
|
static void add_track_feed (GstMediaSourceTrack * track,
|
||
|
GstMediaSourceTrackBuffer * track_buffer, GstSourceBuffer * self);
|
||
|
static void add_track_buffer (GstMediaSourceTrack * track, GstSourceBuffer *
|
||
|
self);
|
||
|
static void update_msesrc_ready_state (GstSourceBuffer * self);
|
||
|
|
||
|
static void on_duration_changed (GstAppendPipeline * pipeline,
|
||
|
gpointer user_data);
|
||
|
static void on_eos (GstAppendPipeline * pipeline, GstMediaSourceTrack * track,
|
||
|
gpointer user_data);
|
||
|
static void on_error (GstAppendPipeline * pipeline, gpointer user_data);
|
||
|
static void on_new_sample (GstAppendPipeline * pipeline,
|
||
|
GstMediaSourceTrack * track, GstSample * sample, gpointer user_data);
|
||
|
static void on_received_init_segment (GstAppendPipeline * pipeline,
|
||
|
gpointer user_data);
|
||
|
|
||
|
static inline GstMediaSource *
|
||
|
get_media_source (GstSourceBuffer * self)
|
||
|
{
|
||
|
return GST_MEDIA_SOURCE (gst_object_get_parent (GST_OBJECT (self)));
|
||
|
}
|
||
|
|
||
|
static GstMseSrc *
|
||
|
get_msesrc (GstSourceBuffer * self)
|
||
|
{
|
||
|
GstMediaSource *media_source = get_media_source (self);
|
||
|
if (media_source == NULL) {
|
||
|
return NULL;
|
||
|
}
|
||
|
return gst_media_source_get_source_element (media_source);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
clear_pending_data (GstSourceBuffer * self)
|
||
|
{
|
||
|
gst_clear_buffer (&self->pending_data);
|
||
|
}
|
||
|
|
||
|
static GstBuffer *
|
||
|
take_pending_data (GstSourceBuffer * self)
|
||
|
{
|
||
|
return g_steal_pointer (&self->pending_data);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
set_pending_data (GstSourceBuffer * self, GstBuffer * buffer)
|
||
|
{
|
||
|
clear_pending_data (self);
|
||
|
self->pending_data = buffer;
|
||
|
}
|
||
|
|
||
|
GstSourceBuffer *
|
||
|
gst_source_buffer_new (const gchar * content_type, GstObject * parent,
|
||
|
GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_MEDIA_SOURCE (parent), NULL);
|
||
|
g_return_val_if_fail (content_type != NULL, NULL);
|
||
|
|
||
|
gst_mse_init_logging ();
|
||
|
|
||
|
GstMediaSourceMediaType media_type = GST_MEDIA_SOURCE_MEDIA_TYPE_INIT;
|
||
|
gst_media_source_media_type_parse (&media_type, content_type);
|
||
|
|
||
|
gboolean generate_timestamps = gst_media_source_media_type_generates_timestamp
|
||
|
(&media_type);
|
||
|
gst_media_source_media_type_reset (&media_type);
|
||
|
|
||
|
GstSourceBufferAppendMode append_mode = generate_timestamps
|
||
|
? GST_SOURCE_BUFFER_APPEND_MODE_SEQUENCE
|
||
|
: GST_SOURCE_BUFFER_APPEND_MODE_SEGMENTS;
|
||
|
|
||
|
GstSourceBuffer *self = g_object_new (GST_TYPE_SOURCE_BUFFER,
|
||
|
"parent", parent, NULL);
|
||
|
|
||
|
self->generate_timestamps = generate_timestamps;
|
||
|
self->append_mode = append_mode;
|
||
|
self->content_type = g_strdup (content_type);
|
||
|
|
||
|
GstAppendPipelineCallbacks callbacks = {
|
||
|
.duration_changed = on_duration_changed,
|
||
|
.eos = on_eos,
|
||
|
.error = on_error,
|
||
|
.new_sample = on_new_sample,
|
||
|
.received_init_segment = on_received_init_segment,
|
||
|
};
|
||
|
GError *append_pipeline_error = NULL;
|
||
|
self->append_pipeline =
|
||
|
gst_append_pipeline_new (&callbacks, self, &append_pipeline_error);
|
||
|
if (append_pipeline_error) {
|
||
|
g_propagate_prefixed_error (error, append_pipeline_error,
|
||
|
"failed to create source buffer");
|
||
|
goto error;
|
||
|
}
|
||
|
|
||
|
return gst_object_ref_sink (self);
|
||
|
error:
|
||
|
gst_clear_object (&self);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
GstSourceBuffer *
|
||
|
gst_source_buffer_new_with_callbacks (const gchar * content_type,
|
||
|
GstObject * parent, GstSourceBufferCallbacks * callbacks,
|
||
|
gpointer user_data, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (callbacks, NULL);
|
||
|
|
||
|
GError *source_buffer_error = NULL;
|
||
|
GstSourceBuffer *self =
|
||
|
gst_source_buffer_new (content_type, parent, &source_buffer_error);
|
||
|
if (source_buffer_error) {
|
||
|
g_propagate_error (error, source_buffer_error);
|
||
|
gst_clear_object (&self);
|
||
|
return NULL;
|
||
|
}
|
||
|
self->callbacks.callbacks = *callbacks;
|
||
|
self->callbacks.user_data = user_data;
|
||
|
|
||
|
return self;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_source_buffer_dispose (GObject * object)
|
||
|
{
|
||
|
GstSourceBuffer *self = (GstSourceBuffer *) object;
|
||
|
|
||
|
if (self->append_to_buffer_task) {
|
||
|
gst_task_join (self->append_to_buffer_task);
|
||
|
}
|
||
|
gst_clear_object (&self->append_to_buffer_task);
|
||
|
|
||
|
gst_clear_object (&self->append_pipeline);
|
||
|
|
||
|
g_hash_table_remove_all (self->track_feeds);
|
||
|
|
||
|
if (!is_removed (self)) {
|
||
|
GstMediaSource *parent = get_media_source (self);
|
||
|
gst_media_source_remove_source_buffer (parent, self, NULL);
|
||
|
gst_object_unref (parent);
|
||
|
}
|
||
|
|
||
|
gst_clear_object (&self->event_queue);
|
||
|
|
||
|
G_OBJECT_CLASS (gst_source_buffer_parent_class)->dispose (object);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_source_buffer_finalize (GObject * object)
|
||
|
{
|
||
|
GstSourceBuffer *self = (GstSourceBuffer *) object;
|
||
|
|
||
|
g_clear_pointer (&self->content_type, g_free);
|
||
|
g_rec_mutex_clear (&self->append_to_buffer_lock);
|
||
|
|
||
|
g_hash_table_unref (self->track_buffers);
|
||
|
g_hash_table_unref (self->track_feeds);
|
||
|
|
||
|
G_OBJECT_CLASS (gst_source_buffer_parent_class)->finalize (object);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_source_buffer_get_property (GObject * object, guint prop_id, GValue * value,
|
||
|
GParamSpec * pspec)
|
||
|
{
|
||
|
GstSourceBuffer *self = GST_SOURCE_BUFFER (object);
|
||
|
|
||
|
switch (prop_id) {
|
||
|
case PROP_APPEND_MODE:
|
||
|
g_value_set_enum (value, gst_source_buffer_get_append_mode (self));
|
||
|
break;
|
||
|
case PROP_APPEND_WINDOW_START:
|
||
|
g_value_set_uint64 (value,
|
||
|
gst_source_buffer_get_append_window_start (self));
|
||
|
break;
|
||
|
case PROP_APPEND_WINDOW_END:
|
||
|
g_value_set_uint64 (value,
|
||
|
gst_source_buffer_get_append_window_end (self));
|
||
|
break;
|
||
|
case PROP_BUFFERED:
|
||
|
g_value_take_boxed (value, gst_source_buffer_get_buffered (self, NULL));
|
||
|
break;
|
||
|
case PROP_CONTENT_TYPE:
|
||
|
g_value_take_string (value, gst_source_buffer_get_content_type (self));
|
||
|
break;
|
||
|
case PROP_TIMESTAMP_OFFSET:
|
||
|
g_value_set_int64 (value, gst_source_buffer_get_timestamp_offset (self));
|
||
|
break;
|
||
|
case PROP_UDPATING:
|
||
|
g_value_set_boolean (value, gst_source_buffer_get_updating (self));
|
||
|
break;
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_source_buffer_set_property (GObject * object, guint prop_id,
|
||
|
const GValue * value, GParamSpec * pspec)
|
||
|
{
|
||
|
GstSourceBuffer *self = GST_SOURCE_BUFFER (object);
|
||
|
|
||
|
switch (prop_id) {
|
||
|
case PROP_APPEND_MODE:
|
||
|
gst_source_buffer_set_append_mode (self, g_value_get_enum (value), NULL);
|
||
|
break;
|
||
|
case PROP_CONTENT_TYPE:
|
||
|
gst_source_buffer_change_content_type (self,
|
||
|
g_value_get_string (value), NULL);
|
||
|
break;
|
||
|
case PROP_TIMESTAMP_OFFSET:
|
||
|
gst_source_buffer_set_timestamp_offset (self,
|
||
|
g_value_get_int64 (value), NULL);
|
||
|
break;
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_source_buffer_class_init (GstSourceBufferClass * klass)
|
||
|
{
|
||
|
GObjectClass *oclass = G_OBJECT_CLASS (klass);
|
||
|
|
||
|
oclass->dispose = GST_DEBUG_FUNCPTR (gst_source_buffer_dispose);
|
||
|
oclass->finalize = GST_DEBUG_FUNCPTR (gst_source_buffer_finalize);
|
||
|
oclass->get_property = GST_DEBUG_FUNCPTR (gst_source_buffer_get_property);
|
||
|
oclass->set_property = GST_DEBUG_FUNCPTR (gst_source_buffer_set_property);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:append-mode:
|
||
|
*
|
||
|
* Affects how timestamps of processed media segments are interpreted.
|
||
|
* In %GST_SOURCE_BUFFER_APPEND_MODE_SEGMENTS, the start timestamp of a
|
||
|
* processed media segment is used directly along with
|
||
|
* #GstSourceBuffer:timestamp-offset .
|
||
|
* In %GST_SOURCE_BUFFER_APPEND_MODE_SEQUENCE, the timestamp of a
|
||
|
* processed media segment is ignored and replaced with the end time of the
|
||
|
* most recently appended segment.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-mode)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
properties[PROP_APPEND_MODE] = g_param_spec_enum ("append-mode",
|
||
|
"Append Mode",
|
||
|
"Either Segments or Sequence",
|
||
|
GST_TYPE_SOURCE_BUFFER_APPEND_MODE, DEFAULT_APPEND_MODE,
|
||
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:append-window-start:
|
||
|
*
|
||
|
* Any segments processed which end before this value will be ignored by this
|
||
|
* Source Buffer.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-appendwindowstart)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
properties[PROP_APPEND_WINDOW_START] =
|
||
|
g_param_spec_uint64 ("append-window-start", "Append Window Start",
|
||
|
"The timestamp representing the start of the append window", 0,
|
||
|
GST_CLOCK_TIME_NONE, 0, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:append-window-end:
|
||
|
*
|
||
|
* Any segments processed which have a start time greater than this value will
|
||
|
* be ignored by this Source Buffer.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-appendwindowend)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
properties[PROP_APPEND_WINDOW_END] = g_param_spec_uint64 ("append-window-end",
|
||
|
"Append Window End",
|
||
|
"The timestamp representing the end of the append window",
|
||
|
0, GST_CLOCK_TIME_NONE, GST_CLOCK_TIME_NONE, G_PARAM_READABLE |
|
||
|
G_PARAM_STATIC_STRINGS);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:buffered:
|
||
|
*
|
||
|
* The set of Time Intervals that have been loaded into the current Source
|
||
|
* Buffer
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-buffered)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
properties[PROP_BUFFERED] = g_param_spec_boxed ("buffered",
|
||
|
"Buffered Time Intervals",
|
||
|
"The set of Time Intervals that have been loaded into"
|
||
|
" the current Source Buffer", G_TYPE_ARRAY, G_PARAM_READABLE |
|
||
|
G_PARAM_STATIC_STRINGS);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:content-type:
|
||
|
*
|
||
|
* The MIME content-type of the data stream
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
properties[PROP_CONTENT_TYPE] = g_param_spec_string ("content-type",
|
||
|
"Content Type",
|
||
|
"The MIME content-type of the data stream", NULL,
|
||
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:timestamp-offset:
|
||
|
*
|
||
|
* The next media segment appended to the current Source Buffer will have its
|
||
|
* start timestamp increased by this amount.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-timestampoffset)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
properties[PROP_TIMESTAMP_OFFSET] = g_param_spec_int64 ("timestamp-offset",
|
||
|
"Timestamp Offset",
|
||
|
"The next media segment appended to the current Source Buffer"
|
||
|
" will have its start timestamp increased by this amount",
|
||
|
0, G_MAXINT64, 0, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer:updating:
|
||
|
*
|
||
|
* Whether the current source buffer is still asynchronously processing
|
||
|
* previously issued commands.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-updating)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
properties[PROP_UDPATING] = g_param_spec_boolean ("updating",
|
||
|
"Updating",
|
||
|
"Whether the current Source Buffer is still"
|
||
|
" asynchronously processing previously issued commands",
|
||
|
FALSE, G_PARAM_READABLE);
|
||
|
|
||
|
g_object_class_install_properties (oclass, N_PROPS, properties);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer::on-update-start:
|
||
|
* @self: The #GstSourceBuffer that has just started updating
|
||
|
*
|
||
|
* Emitted when @self has begun to process data after a call to
|
||
|
* gst_source_buffer_append_buffer().
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-onupdatestart)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
signals[ON_UPDATE_START] = g_signal_new ("on-update-start",
|
||
|
G_TYPE_FROM_CLASS (klass),
|
||
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer::on-update:
|
||
|
* @self: The #GstSourceBuffer that has just updated
|
||
|
*
|
||
|
* Emitted when @self has successfully processed data after a call to
|
||
|
* gst_source_buffer_append_buffer().
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-onupdate)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
signals[ON_UPDATE] = g_signal_new ("on-update",
|
||
|
G_TYPE_FROM_CLASS (klass),
|
||
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer::on-update-end:
|
||
|
* @self: The #GstSourceBuffer that is no longer updating
|
||
|
*
|
||
|
* Emitted when @self is no longer in the updating state after a call to
|
||
|
* gst_source_buffer_append_buffer(). This can happen after a successful or
|
||
|
* unsuccessful append.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-onupdateend)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
signals[ON_UPDATE_END] = g_signal_new ("on-update-end",
|
||
|
G_TYPE_FROM_CLASS (klass),
|
||
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer::on-error:
|
||
|
* @self: The #GstSourceBuffer that has encountered an error
|
||
|
*
|
||
|
* Emitted when @self has encountered an error after a call to
|
||
|
* gst_source_buffer_append_buffer().
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-onerror)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
signals[ON_ERROR] = g_signal_new ("on-error",
|
||
|
G_TYPE_FROM_CLASS (klass),
|
||
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0);
|
||
|
|
||
|
/**
|
||
|
* GstSourceBuffer::on-abort:
|
||
|
* @self: The #GstSourceBuffer that has been aborted.
|
||
|
*
|
||
|
* Emitted when @self was aborted after a call to gst_source_buffer_abort().
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-onabort)
|
||
|
*
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
signals[ON_ABORT] = g_signal_new ("on-abort",
|
||
|
G_TYPE_FROM_CLASS (klass),
|
||
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 0);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_duration_changed (GstAppendPipeline * pipeline, gpointer user_data)
|
||
|
{
|
||
|
GstSourceBuffer *self = GST_SOURCE_BUFFER (user_data);
|
||
|
if (is_removed (self)) {
|
||
|
return;
|
||
|
}
|
||
|
call_duration_changed (self);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_eos (GstAppendPipeline * pipeline, GstMediaSourceTrack * track,
|
||
|
gpointer user_data)
|
||
|
{
|
||
|
GstSourceBuffer *self = GST_SOURCE_BUFFER (user_data);
|
||
|
if (GST_IS_MEDIA_SOURCE_TRACK (track)) {
|
||
|
GST_DEBUG_OBJECT (self, "got EOS event on %" GST_PTR_FORMAT, track);
|
||
|
GstMediaSourceTrackBuffer *buffer = get_track_buffer (self, track);
|
||
|
gst_media_source_track_buffer_eos (buffer);
|
||
|
}
|
||
|
update_msesrc_ready_state (self);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_error (GstAppendPipeline * pipeline, gpointer user_data)
|
||
|
{
|
||
|
GstSourceBuffer *self = GST_SOURCE_BUFFER (user_data);
|
||
|
append_error (self);
|
||
|
}
|
||
|
|
||
|
static GstMediaSourceTrackBuffer *
|
||
|
get_track_buffer (GstSourceBuffer * self, GstMediaSourceTrack * track)
|
||
|
{
|
||
|
g_return_val_if_fail (g_hash_table_contains (self->track_buffers, track),
|
||
|
NULL);
|
||
|
return g_hash_table_lookup (self->track_buffers, track);
|
||
|
}
|
||
|
|
||
|
static inline TrackFeedTask *
|
||
|
get_track_feed (GstSourceBuffer * self, GstMediaSourceTrack * track)
|
||
|
{
|
||
|
g_return_val_if_fail (g_hash_table_contains (self->track_feeds, track), NULL);
|
||
|
return g_hash_table_lookup (self->track_feeds, track);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
add_track_buffer (GstMediaSourceTrack * track, GstSourceBuffer * self)
|
||
|
{
|
||
|
const gchar *id = gst_media_source_track_get_id (track);
|
||
|
if (g_hash_table_contains (self->track_buffers, track)) {
|
||
|
GST_DEBUG_OBJECT (self, "already have a track buffer for track %s", id);
|
||
|
return;
|
||
|
}
|
||
|
GstMediaSourceTrackBuffer *buf = gst_media_source_track_buffer_new ();
|
||
|
g_hash_table_insert (self->track_buffers, track, buf);
|
||
|
GST_DEBUG_OBJECT (self, "added track buffer for track %s", id);
|
||
|
|
||
|
add_track_feed (track, buf, self);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
add_track_feed (GstMediaSourceTrack * track,
|
||
|
GstMediaSourceTrackBuffer * track_buffer, GstSourceBuffer * self)
|
||
|
{
|
||
|
TrackFeedTask *feed = g_new0 (TrackFeedTask, 1);
|
||
|
GstTask *task = gst_task_new ((GstTaskFunction) track_feed_task, feed, NULL);
|
||
|
g_rec_mutex_init (&feed->lock);
|
||
|
gst_task_set_lock (task, &feed->lock);
|
||
|
feed->task = task;
|
||
|
feed->buffer = track_buffer;
|
||
|
feed->track = gst_object_ref (track);
|
||
|
feed->parent = self;
|
||
|
feed->cancelled = FALSE;
|
||
|
g_hash_table_insert (self->track_feeds, track, feed);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
clear_track_feed (TrackFeedTask * feed)
|
||
|
{
|
||
|
gst_object_unref (feed->task);
|
||
|
g_rec_mutex_clear (&feed->lock);
|
||
|
gst_object_unref (feed->track);
|
||
|
g_free (feed);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
stop_track_feed (TrackFeedTask * feed)
|
||
|
{
|
||
|
g_return_if_fail (feed != NULL);
|
||
|
gst_media_source_track_flush (feed->track);
|
||
|
g_atomic_int_set (&feed->cancelled, TRUE);
|
||
|
gst_task_join (feed->task);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
start_track_feed (TrackFeedTask * feed)
|
||
|
{
|
||
|
g_return_if_fail (feed != NULL);
|
||
|
g_atomic_int_set (&feed->cancelled, FALSE);
|
||
|
gst_media_source_track_resume (feed->track);
|
||
|
gst_task_start (feed->task);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
reset_track_feed (TrackFeedTask * feed)
|
||
|
{
|
||
|
stop_track_feed (feed);
|
||
|
start_track_feed (feed);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
is_within_append_window (GstSourceBuffer * self, GstSample * sample)
|
||
|
{
|
||
|
GstBuffer *buffer = gst_sample_get_buffer (sample);
|
||
|
GstClockTime start = GST_BUFFER_PTS (buffer);
|
||
|
GstClockTime end = start + GST_BUFFER_DURATION (buffer);
|
||
|
|
||
|
if (start < self->append_window_start) {
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (!GST_CLOCK_TIME_IS_VALID (self->append_window_end)) {
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
return end <= self->append_window_end;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_new_sample (GstAppendPipeline * pipeline, GstMediaSourceTrack * track,
|
||
|
GstSample * sample, gpointer user_data)
|
||
|
{
|
||
|
GstSourceBuffer *self = GST_SOURCE_BUFFER (user_data);
|
||
|
|
||
|
g_return_if_fail (self->processed_init_segment);
|
||
|
|
||
|
GST_OBJECT_LOCK (self);
|
||
|
|
||
|
if (is_within_append_window (self, sample)) {
|
||
|
GstMediaSourceTrackBuffer *track_buffer = get_track_buffer (self, track);
|
||
|
GST_TRACE_OBJECT (self, "new sample on %s with %" GST_PTR_FORMAT,
|
||
|
gst_media_source_track_get_id (track), gst_sample_get_buffer (sample));
|
||
|
gst_media_source_track_buffer_add (track_buffer, sample);
|
||
|
TrackFeedTask *feed = get_track_feed (self, track);
|
||
|
start_track_feed (feed);
|
||
|
}
|
||
|
|
||
|
GST_OBJECT_UNLOCK (self);
|
||
|
|
||
|
update_msesrc_ready_state (self);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
call_received_init_segment (GstSourceBuffer * self)
|
||
|
{
|
||
|
GstSourceBufferCallbacks *callbacks = &self->callbacks.callbacks;
|
||
|
if (callbacks->received_init_segment) {
|
||
|
callbacks->received_init_segment (self, self->callbacks.user_data);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
call_duration_changed (GstSourceBuffer * self)
|
||
|
{
|
||
|
GstSourceBufferCallbacks *callbacks = &self->callbacks.callbacks;
|
||
|
if (callbacks->duration_changed) {
|
||
|
callbacks->duration_changed (self, self->callbacks.user_data);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
call_active_state_changed (GstSourceBuffer * self)
|
||
|
{
|
||
|
GstSourceBufferCallbacks *callbacks = &self->callbacks.callbacks;
|
||
|
if (callbacks->active_state_changed) {
|
||
|
callbacks->active_state_changed (self, self->callbacks.user_data);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
update_track_buffer_modes (GstSourceBuffer * self)
|
||
|
{
|
||
|
GHashTableIter iter;
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
gboolean enabled =
|
||
|
self->append_mode == GST_SOURCE_BUFFER_APPEND_MODE_SEQUENCE;
|
||
|
for (gpointer value; g_hash_table_iter_next (&iter, NULL, &value);) {
|
||
|
GstMediaSourceTrackBuffer *buffer = GST_MEDIA_SOURCE_TRACK_BUFFER (value);
|
||
|
gst_media_source_track_buffer_process_init_segment (buffer, enabled);
|
||
|
gst_media_source_track_buffer_set_group_start (buffer,
|
||
|
self->timestamp_offset);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
on_received_init_segment (GstAppendPipeline * pipeline, gpointer user_data)
|
||
|
{
|
||
|
GstSourceBuffer *self = GST_SOURCE_BUFFER (user_data);
|
||
|
GST_DEBUG_OBJECT (self, "got init segment, have duration %" GST_TIME_FORMAT,
|
||
|
GST_TIME_ARGS (gst_append_pipeline_get_duration (pipeline)));
|
||
|
|
||
|
GST_OBJECT_LOCK (self);
|
||
|
|
||
|
if (!self->processed_init_segment) {
|
||
|
GST_DEBUG_OBJECT (self, "processing first init segment");
|
||
|
|
||
|
GPtrArray *audio_tracks = gst_append_pipeline_get_audio_tracks (pipeline);
|
||
|
GPtrArray *text_tracks = gst_append_pipeline_get_text_tracks (pipeline);
|
||
|
GPtrArray *video_tracks = gst_append_pipeline_get_video_tracks (pipeline);
|
||
|
|
||
|
g_ptr_array_foreach (audio_tracks, (GFunc) add_track_buffer, self);
|
||
|
g_ptr_array_foreach (text_tracks, (GFunc) add_track_buffer, self);
|
||
|
g_ptr_array_foreach (video_tracks, (GFunc) add_track_buffer, self);
|
||
|
}
|
||
|
|
||
|
self->processed_init_segment = TRUE;
|
||
|
|
||
|
update_track_buffer_modes (self);
|
||
|
|
||
|
GST_OBJECT_UNLOCK (self);
|
||
|
|
||
|
call_received_init_segment (self);
|
||
|
call_active_state_changed (self);
|
||
|
}
|
||
|
|
||
|
static guint
|
||
|
track_buffer_hash (GstMediaSourceTrack * track)
|
||
|
{
|
||
|
return g_str_hash (gst_media_source_track_get_id (track));
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
track_buffer_equal (GstMediaSourceTrack * a, GstMediaSourceTrack * b)
|
||
|
{
|
||
|
return g_str_equal (gst_media_source_track_get_id (a),
|
||
|
gst_media_source_track_get_id (b));
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gst_source_buffer_init (GstSourceBuffer * self)
|
||
|
{
|
||
|
|
||
|
self->append_mode = DEFAULT_APPEND_MODE;
|
||
|
self->append_window_start = 0;
|
||
|
self->append_window_end = GST_CLOCK_TIME_NONE;
|
||
|
self->content_type = NULL;
|
||
|
self->timestamp_offset = 0;
|
||
|
self->updating = FALSE;
|
||
|
self->errored = FALSE;
|
||
|
self->size_limit = DEFAULT_BUFFER_SIZE;
|
||
|
self->size = 0;
|
||
|
self->pending_data = NULL;
|
||
|
self->processed_init_segment = FALSE;
|
||
|
self->event_queue =
|
||
|
gst_mse_event_queue_new ((GstMseEventQueueCallback) dispatch_event, self);
|
||
|
|
||
|
g_rec_mutex_init (&self->append_to_buffer_lock);
|
||
|
self->append_to_buffer_task = gst_task_new (
|
||
|
(GstTaskFunction) append_to_buffer_task, self, NULL);
|
||
|
gst_task_set_lock (self->append_to_buffer_task, &self->append_to_buffer_lock);
|
||
|
self->track_buffers = g_hash_table_new_full ((GHashFunc) track_buffer_hash,
|
||
|
(GEqualFunc) track_buffer_equal, NULL, gst_object_unref);
|
||
|
|
||
|
self->track_feeds = g_hash_table_new_full (g_direct_hash, g_direct_equal,
|
||
|
NULL, (GDestroyNotify) clear_track_feed);
|
||
|
self->seek_time = 0;
|
||
|
self->callbacks.callbacks.duration_changed = NULL;
|
||
|
self->callbacks.user_data = NULL;
|
||
|
}
|
||
|
|
||
|
static inline gboolean
|
||
|
is_removed (GstSourceBuffer * self)
|
||
|
{
|
||
|
GstObject *parent = gst_object_get_parent (GST_OBJECT (self));
|
||
|
if (parent == NULL) {
|
||
|
return TRUE;
|
||
|
}
|
||
|
gst_object_unref (parent);
|
||
|
|
||
|
GstMediaSource *source = get_media_source (self);
|
||
|
GstSourceBufferList *buffers = gst_media_source_get_source_buffers (source);
|
||
|
gboolean removed = !gst_source_buffer_list_contains (buffers, self);
|
||
|
|
||
|
gst_object_unref (source);
|
||
|
gst_object_unref (buffers);
|
||
|
|
||
|
return removed;
|
||
|
}
|
||
|
|
||
|
static inline gboolean
|
||
|
is_updating (GstSourceBuffer * self)
|
||
|
{
|
||
|
return g_atomic_int_get (&self->updating);
|
||
|
}
|
||
|
|
||
|
static inline void
|
||
|
set_updating (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_atomic_int_set (&self->updating, TRUE);
|
||
|
}
|
||
|
|
||
|
static inline void
|
||
|
clear_updating (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_atomic_int_set (&self->updating, FALSE);
|
||
|
}
|
||
|
|
||
|
static inline gboolean
|
||
|
is_errored (GstSourceBuffer * self)
|
||
|
{
|
||
|
return g_atomic_int_get (&self->errored);
|
||
|
}
|
||
|
|
||
|
static inline void
|
||
|
set_errored (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_atomic_int_set (&self->errored, TRUE);
|
||
|
}
|
||
|
|
||
|
static inline void
|
||
|
clear_errored (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_atomic_int_set (&self->errored, FALSE);
|
||
|
}
|
||
|
|
||
|
static inline gboolean
|
||
|
is_ended (GstSourceBuffer * self)
|
||
|
{
|
||
|
if (is_removed (self)) {
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
GstMediaSource *source = get_media_source (self);
|
||
|
gboolean ended = gst_media_source_get_ready_state (source) ==
|
||
|
GST_MEDIA_SOURCE_READY_STATE_ENDED;
|
||
|
|
||
|
gst_object_unref (source);
|
||
|
|
||
|
return ended;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
open_parent (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_if_fail (!is_removed (self));
|
||
|
GstMediaSource *source = get_media_source (self);
|
||
|
gst_media_source_open (source);
|
||
|
gst_object_unref (source);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_get_append_mode:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-mode)
|
||
|
*
|
||
|
* Returns: The current #GstSourceBufferAppendMode
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
GstSourceBufferAppendMode
|
||
|
gst_source_buffer_get_append_mode (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), DEFAULT_APPEND_MODE);
|
||
|
return self->append_mode;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_set_append_mode:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @mode: #GstSourceBufferAppendMode the desired Append Mode
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Changes the Append Mode of @self. This influences what timestamps will be
|
||
|
* assigned to media processed by this Source Buffer. In Segment mode, the
|
||
|
* timestamps in each segment determine the position of each sample after it
|
||
|
* is processed. In Sequence mode, the timestamp of each processed sample is
|
||
|
* generated based on the end of the most recently processed segment.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-mode)
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_set_append_mode (GstSourceBuffer * self,
|
||
|
GstSourceBufferAppendMode mode, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
|
||
|
if (is_removed (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"buffer is removed");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_updating (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"buffer is still updating");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (self->generate_timestamps && mode ==
|
||
|
GST_SOURCE_BUFFER_APPEND_MODE_SEGMENTS) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_TYPE,
|
||
|
"cannot change to segments mode while generate timestamps is active");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_ended (self)) {
|
||
|
open_parent (self);
|
||
|
}
|
||
|
|
||
|
self->append_mode = mode;
|
||
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_APPEND_MODE]);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_get_append_window_start:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
*
|
||
|
* Returns the current append window start time. Any segment processed that ends
|
||
|
* earlier than this value will be ignored.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-appendwindowstart)
|
||
|
*
|
||
|
* Returns: The current Append Window start time as a #GstClockTime
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
GstClockTime
|
||
|
gst_source_buffer_get_append_window_start (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), GST_CLOCK_TIME_NONE);
|
||
|
return self->append_window_start;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_set_append_window_start:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @start: the append window end
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Modifies the current append window start of @self. If successful, samples
|
||
|
* processed after setting this value that end before this point will be
|
||
|
* ignored.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-appendwindowstart)
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_set_append_window_start (GstSourceBuffer * self,
|
||
|
GstClockTime start, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
|
||
|
if (is_removed (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"append window start cannot be set on source buffer "
|
||
|
"with no media source");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_updating (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"append window start cannot be set on source buffer while updating");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (start < 0 || start <= self->append_window_end) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_TYPE,
|
||
|
"append window start must be between zero and append window end");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
self->append_window_start = start;
|
||
|
g_object_notify_by_pspec (G_OBJECT (self),
|
||
|
properties[PROP_APPEND_WINDOW_START]);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_get_append_window_end:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
*
|
||
|
* Returns the current append window end time. Any segment processed that starts
|
||
|
* after this value will be ignored.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-appendwindowend)
|
||
|
*
|
||
|
* Returns: The current Append Window end time as a #GstClockTime
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
GstClockTime
|
||
|
gst_source_buffer_get_append_window_end (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), GST_CLOCK_TIME_NONE);
|
||
|
return self->append_window_end;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_set_append_window_end:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @end: the append window end
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Modifies the current append window end of @self. If successful, samples
|
||
|
* processed after setting this value that start after this point will be
|
||
|
* ignored.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-appendwindowend)
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_set_append_window_end (GstSourceBuffer * self,
|
||
|
GstClockTime end, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
|
||
|
if (is_removed (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"append window end cannot be set on source buffer "
|
||
|
"with no media source");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_updating (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"append window end cannot be set on source buffer while updating");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (end <= self->append_window_start) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_TYPE,
|
||
|
"append window end must be after append window start");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
self->append_window_end = end;
|
||
|
g_object_notify_by_pspec (G_OBJECT (self),
|
||
|
properties[PROP_APPEND_WINDOW_END]);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
get_intersection (GstMediaSourceRange * a, GstMediaSourceRange * b,
|
||
|
GstMediaSourceRange * intersection)
|
||
|
{
|
||
|
g_return_val_if_fail (a != NULL, FALSE);
|
||
|
g_return_val_if_fail (b != NULL, FALSE);
|
||
|
g_return_val_if_fail (intersection != NULL, FALSE);
|
||
|
GstMediaSourceRange range = {
|
||
|
.start = MAX (a->start, b->start),
|
||
|
.end = MIN (a->end, b->end),
|
||
|
};
|
||
|
if (range.start >= range.end) {
|
||
|
return FALSE;
|
||
|
}
|
||
|
*intersection = range;
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
static GArray *
|
||
|
intersect_ranges (GstMediaSourceRange * a, GstMediaSourceRange * a_end,
|
||
|
GstMediaSourceRange * b, GstMediaSourceRange * b_end)
|
||
|
{
|
||
|
GArray *intersection = g_array_new_ranges ();
|
||
|
while (a < a_end && b < b_end) {
|
||
|
GstMediaSourceRange range;
|
||
|
if (!get_intersection (a, b, &range)) {
|
||
|
if (a->end < b->end) {
|
||
|
a++;
|
||
|
} else {
|
||
|
b++;
|
||
|
}
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if (a->end < b->end) {
|
||
|
a++;
|
||
|
} else {
|
||
|
b++;
|
||
|
}
|
||
|
|
||
|
g_array_append_val (intersection, range);
|
||
|
}
|
||
|
return intersection;
|
||
|
}
|
||
|
|
||
|
static inline gboolean
|
||
|
contributes_to_buffered (GstMediaSourceTrack * track)
|
||
|
{
|
||
|
switch (gst_media_source_track_get_track_type (track)) {
|
||
|
case GST_MEDIA_SOURCE_TRACK_TYPE_AUDIO:
|
||
|
case GST_MEDIA_SOURCE_TRACK_TYPE_VIDEO:
|
||
|
return TRUE;
|
||
|
default:
|
||
|
return FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_get_buffered:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Returns a sequence of #GstMediaSourceRange values representing which segments
|
||
|
* of @self are buffered in memory.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-buffered)
|
||
|
*
|
||
|
* Returns: (transfer full) (element-type GstMediaSourceRange): a #GArray of #GstMediaSourceRange values.
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
GArray *
|
||
|
gst_source_buffer_get_buffered (GstSourceBuffer * self, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), NULL);
|
||
|
GHashTableIter iter;
|
||
|
GArray *buffered = NULL;
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
for (gpointer key, value; g_hash_table_iter_next (&iter, &key, &value);) {
|
||
|
GstMediaSourceTrack *track = GST_MEDIA_SOURCE_TRACK (key);
|
||
|
GstMediaSourceTrackBuffer *buffer = GST_MEDIA_SOURCE_TRACK_BUFFER (value);
|
||
|
if (!contributes_to_buffered (track)) {
|
||
|
continue;
|
||
|
}
|
||
|
GArray *current_ranges = gst_media_source_track_buffer_get_ranges (buffer);
|
||
|
if (buffered == NULL) {
|
||
|
buffered = current_ranges;
|
||
|
continue;
|
||
|
}
|
||
|
GArray *intersection = intersect_ranges (
|
||
|
(GstMediaSourceRange *) buffered->data,
|
||
|
((GstMediaSourceRange *) buffered->data) + buffered->len,
|
||
|
(GstMediaSourceRange *) current_ranges->data,
|
||
|
((GstMediaSourceRange *) current_ranges->data) + current_ranges->len);
|
||
|
g_array_unref (buffered);
|
||
|
buffered = intersection;
|
||
|
}
|
||
|
if (buffered == NULL) {
|
||
|
return g_array_new_ranges ();
|
||
|
} else {
|
||
|
return buffered;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_get_content_type:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
*
|
||
|
* Returns the current content type of @self.
|
||
|
*
|
||
|
* Returns: (transfer full): a string representing the content type
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gchar *
|
||
|
gst_source_buffer_get_content_type (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), NULL);
|
||
|
|
||
|
GST_OBJECT_LOCK (self);
|
||
|
gchar *content_type = g_strdup (self->content_type);
|
||
|
GST_OBJECT_UNLOCK (self);
|
||
|
|
||
|
return content_type;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_change_content_type:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @type: (transfer none): the desired content type
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Attempts to change the content type of @self to @type. Any new data appended
|
||
|
* to the Source Buffer must be of the supplied @type afterward.
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_change_content_type (GstSourceBuffer * self,
|
||
|
const gchar * type, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
|
||
|
if (type == NULL || g_strcmp0 (type, "") == 0) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_TYPE,
|
||
|
"content type must not be empty");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_removed (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"content type cannot be set on source buffer with no media source");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_updating (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"content type cannot be set on source buffer that is updating");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_NOT_SUPPORTED,
|
||
|
"content type cannot be changed");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_remove:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @start: The beginning timestamp of data to remove
|
||
|
* @end: The end timestamp of data to remove
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Attempts to remove any parsed data between @start and @end from @self.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-remove)
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_remove (GstSourceBuffer * self, GstClockTime start,
|
||
|
GstClockTime end, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_get_timestamp_offset:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-timestampoffset)
|
||
|
*
|
||
|
* Returns: The current timestamp offset as a #GstClockTime
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
GstClockTime
|
||
|
gst_source_buffer_get_timestamp_offset (GstSourceBuffer * self)
|
||
|
{
|
||
|
return self->timestamp_offset;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_set_timestamp_offset:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @offset: The new timestamp offset
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Attempt to set the timestamp offset of @self. Any media processed after this
|
||
|
* value is set will have this value added to its start time.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-timestampoffset)
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_set_timestamp_offset (GstSourceBuffer * self, GstClockTime
|
||
|
offset, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
if (is_removed (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"source buffer is removed");
|
||
|
return FALSE;
|
||
|
}
|
||
|
if (is_updating (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"source buffer is still updating");
|
||
|
return FALSE;
|
||
|
}
|
||
|
if (is_ended (self)) {
|
||
|
GstMediaSource *parent = get_media_source (self);
|
||
|
gst_media_source_open (parent);
|
||
|
gst_clear_object (&parent);
|
||
|
}
|
||
|
GST_OBJECT_LOCK (self);
|
||
|
GHashTableIter iter;
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
for (gpointer value; g_hash_table_iter_next (&iter, NULL, &value);) {
|
||
|
GstMediaSourceTrackBuffer *buffer = value;
|
||
|
gst_media_source_track_buffer_set_group_start (buffer, offset);
|
||
|
}
|
||
|
self->timestamp_offset = offset;
|
||
|
GST_OBJECT_UNLOCK (self);
|
||
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_TIMESTAMP_OFFSET]);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_get_updating:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-updating)
|
||
|
*
|
||
|
* Returns: Whether @self is currently adding or removing media content.
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_get_updating (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
return is_updating (self);
|
||
|
}
|
||
|
|
||
|
static gsize
|
||
|
compute_total_size_unlocked (GstSourceBuffer * self)
|
||
|
{
|
||
|
GHashTableIter iter;
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
gsize total_size = 0;
|
||
|
for (gpointer value; g_hash_table_iter_next (&iter, NULL, &value);) {
|
||
|
GstMediaSourceTrackBuffer *buffer = value;
|
||
|
total_size += gst_media_source_track_buffer_get_storage_size (buffer);
|
||
|
}
|
||
|
return total_size;
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
will_overflow (GstSourceBuffer * self, gsize bytes)
|
||
|
{
|
||
|
GST_OBJECT_LOCK (self);
|
||
|
gsize total = compute_total_size_unlocked (self);
|
||
|
GST_OBJECT_UNLOCK (self);
|
||
|
return total + bytes > self->size_limit;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
evict_coded_frames (GstSourceBuffer * self, gsize space_required,
|
||
|
gsize size_limit, GstClockTime position, GstClockTime duration)
|
||
|
{
|
||
|
if (!will_overflow (self, space_required)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (!GST_CLOCK_TIME_IS_VALID (position)) {
|
||
|
GST_ERROR ("invalid position, cannot delete anything");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
GstClockTime min_distance_from_position = GST_SECOND * 5;
|
||
|
GstClockTime max_dts = position > min_distance_from_position ?
|
||
|
position - min_distance_from_position : 0;
|
||
|
|
||
|
GST_DEBUG_OBJECT (self, "position=%" GST_TIMEP_FORMAT
|
||
|
", attempting removal from 0 to %" GST_TIMEP_FORMAT, &position, &max_dts);
|
||
|
|
||
|
GST_OBJECT_LOCK (self);
|
||
|
GHashTableIter iter;
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
for (gpointer value; g_hash_table_iter_next (&iter, NULL, &value);) {
|
||
|
GstMediaSourceTrackBuffer *buffer = value;
|
||
|
gst_media_source_track_buffer_remove_range (buffer, 0, max_dts);
|
||
|
}
|
||
|
self->size = compute_total_size_unlocked (self);
|
||
|
GST_OBJECT_UNLOCK (self);
|
||
|
|
||
|
GST_DEBUG_OBJECT (self, "capacity=%" G_GSIZE_FORMAT "/%" G_GSIZE_FORMAT
|
||
|
"(%" G_GSIZE_FORMAT "%%)", self->size, self->size_limit,
|
||
|
self->size * 100 / size_limit);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
reset_parser_state (GstSourceBuffer * self)
|
||
|
{
|
||
|
clear_pending_data (self);
|
||
|
if (gst_append_pipeline_reset (self->append_pipeline)) {
|
||
|
clear_errored (self);
|
||
|
} else {
|
||
|
set_errored (self);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
append_error (GstSourceBuffer * self)
|
||
|
{
|
||
|
gst_task_stop (self->append_to_buffer_task);
|
||
|
reset_parser_state (self);
|
||
|
clear_updating (self);
|
||
|
|
||
|
if (is_removed (self)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
schedule_event (self, ON_ERROR);
|
||
|
schedule_event (self, ON_UPDATE_END);
|
||
|
|
||
|
GstMediaSource *source = get_media_source (self);
|
||
|
gst_media_source_end_of_stream (source, GST_MEDIA_SOURCE_EOS_ERROR_DECODE,
|
||
|
NULL);
|
||
|
gst_object_unref (source);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
append_successful (GstSourceBuffer * self, gboolean ended)
|
||
|
{
|
||
|
gst_task_stop (self->append_to_buffer_task);
|
||
|
clear_updating (self);
|
||
|
schedule_event (self, ON_UPDATE);
|
||
|
schedule_event (self, ON_UPDATE_END);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
encountered_bad_bytes (GstSourceBuffer * self)
|
||
|
{
|
||
|
return gst_append_pipeline_get_failed (self->append_pipeline);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
append_to_buffer_task (GstSourceBuffer * self)
|
||
|
{
|
||
|
if (is_removed (self)) {
|
||
|
append_successful (self, TRUE);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (encountered_bad_bytes (self)) {
|
||
|
append_error (self);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
GstBuffer *pending_data = take_pending_data (self);
|
||
|
|
||
|
if (!GST_IS_BUFFER (pending_data)) {
|
||
|
GST_LOG_OBJECT (self, "no pending data");
|
||
|
append_successful (self, is_ended (self));
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
GstFlowReturn result = gst_append_pipeline_append (self->append_pipeline,
|
||
|
pending_data);
|
||
|
|
||
|
if (result != GST_FLOW_OK) {
|
||
|
GST_ERROR_OBJECT (self, "failed to append: %s", gst_flow_get_name (result));
|
||
|
append_error (self);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
append_successful (self, is_ended (self));
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
track_feed_fold (const GValue * item, TrackFeedAccumulator * acc,
|
||
|
TrackFeedTask * feed)
|
||
|
{
|
||
|
GstSample *sample = gst_sample_ref (gst_value_get_sample (item));
|
||
|
GstClockTime dts = GST_BUFFER_DTS (gst_sample_get_buffer (sample));
|
||
|
acc->n_samples++;
|
||
|
acc->current_dts = dts;
|
||
|
gst_clear_sample (&acc->current_sample);
|
||
|
acc->current_sample = gst_sample_ref (sample);
|
||
|
if (gst_media_source_track_push (feed->track, sample)) {
|
||
|
return TRUE;
|
||
|
} else {
|
||
|
gst_sample_unref (sample);
|
||
|
return FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
track_feed_task (TrackFeedTask * feed)
|
||
|
{
|
||
|
GstSourceBuffer *self = feed->parent;
|
||
|
GstMediaSourceTrack *track = feed->track;
|
||
|
GstMediaSourceTrackBuffer *buffer = feed->buffer;
|
||
|
GstClockTime time = feed->parent->seek_time;
|
||
|
const gchar *track_id = gst_media_source_track_get_id (track);
|
||
|
|
||
|
GST_DEBUG_OBJECT (self, "%s: feed starting@%" GST_TIMEP_FORMAT, track_id,
|
||
|
&time);
|
||
|
|
||
|
TrackFeedAccumulator acc = {
|
||
|
.n_samples = 0,
|
||
|
.current_dts = time,
|
||
|
.current_sample = NULL,
|
||
|
};
|
||
|
while (TRUE) {
|
||
|
gboolean eos = gst_media_source_track_buffer_is_eos (buffer);
|
||
|
GstIterator *it = gst_media_source_track_buffer_iter_samples (buffer,
|
||
|
acc.current_dts, acc.current_sample);
|
||
|
while (TRUE) {
|
||
|
GstIteratorResult fold_result = gst_iterator_fold (it,
|
||
|
(GstIteratorFoldFunction) track_feed_fold, (GValue *) & acc, feed);
|
||
|
if (fold_result != GST_ITERATOR_RESYNC) {
|
||
|
break;
|
||
|
}
|
||
|
if (g_atomic_int_get (&feed->cancelled)) {
|
||
|
break;
|
||
|
}
|
||
|
gst_iterator_resync (it);
|
||
|
}
|
||
|
gst_iterator_free (it);
|
||
|
|
||
|
if (eos) {
|
||
|
GST_DEBUG_OBJECT (self, "%s: enqueued all %" G_GSIZE_FORMAT " samples",
|
||
|
track_id, acc.n_samples);
|
||
|
gst_media_source_track_push_eos (track);
|
||
|
GST_DEBUG_OBJECT (self, "%s: marked EOS", track_id);
|
||
|
gst_task_stop (feed->task);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (g_atomic_int_get (&feed->cancelled)) {
|
||
|
GST_DEBUG_OBJECT (self, "feed is cancelled, stopping task");
|
||
|
gst_task_stop (feed->task);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
GST_DEBUG_OBJECT (self, "%s: resume after %" G_GSIZE_FORMAT " samples",
|
||
|
track_id, acc.n_samples);
|
||
|
gint64 deadline = g_get_monotonic_time () + G_TIME_SPAN_SECOND;
|
||
|
gst_media_source_track_buffer_await_eos_until (buffer, deadline);
|
||
|
}
|
||
|
|
||
|
gst_clear_sample (&acc.current_sample);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
dispatch_event (SourceBufferEventItem * item, GstSourceBuffer * self)
|
||
|
{
|
||
|
g_signal_emit (self, signals[item->event], 0);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
schedule_event (GstSourceBuffer * self, SourceBufferEvent event)
|
||
|
{
|
||
|
g_return_if_fail (event < N_SIGNALS);
|
||
|
if (is_removed (self)) {
|
||
|
return;
|
||
|
}
|
||
|
SourceBufferEventItem item = {
|
||
|
.item = {.destroy = g_free,.visible = TRUE,.size = 1,.object = NULL},
|
||
|
.event = event,
|
||
|
};
|
||
|
gst_mse_event_queue_push (self->event_queue, g_memdup2 (&item,
|
||
|
sizeof (SourceBufferEventItem)));
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
schedule_append_to_buffer_task (GstSourceBuffer * self)
|
||
|
{
|
||
|
GstTask *task = self->append_to_buffer_task;
|
||
|
g_return_if_fail (GST_IS_TASK (task));
|
||
|
g_return_if_fail (gst_task_get_state (task) != GST_TASK_STARTED);
|
||
|
gst_task_start (task);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
update_msesrc_ready_state (GstSourceBuffer * self)
|
||
|
{
|
||
|
GstMseSrc *element = get_msesrc (self);
|
||
|
if (element == NULL) {
|
||
|
return;
|
||
|
}
|
||
|
gst_mse_src_update_ready_state (element);
|
||
|
gst_object_unref (element);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_append_buffer:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @buf: (transfer full):The media data to append
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Schedules the bytes inside @buf to be processed by @self. When it is possible
|
||
|
* to accept the supplied data, it will be processed asynchronously and fill in
|
||
|
* the track buffers for playback purposes.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-appendbuffer)
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_append_buffer (GstSourceBuffer * self, GstBuffer * buf,
|
||
|
GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
g_return_val_if_fail (GST_IS_BUFFER (buf), FALSE);
|
||
|
|
||
|
if (is_removed (self) || is_updating (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"source buffer is removed or still updating");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_errored (self)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"source buffer has encountered error");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if (is_ended (self)) {
|
||
|
open_parent (self);
|
||
|
}
|
||
|
|
||
|
GstMediaSource *source = get_media_source (self);
|
||
|
gsize buffer_size = gst_buffer_get_size (buf);
|
||
|
GstClockTime position = gst_media_source_get_position (source);
|
||
|
GstClockTime duration = gst_media_source_get_duration (source);
|
||
|
|
||
|
gst_object_unref (source);
|
||
|
|
||
|
evict_coded_frames (self, buffer_size, self->size_limit, position, duration);
|
||
|
|
||
|
if (will_overflow (self, buffer_size)) {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_QUOTA_EXCEEDED,
|
||
|
"buffer is full");
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
g_return_val_if_fail (self->pending_data == NULL, FALSE);
|
||
|
|
||
|
set_pending_data (self, buf);
|
||
|
set_updating (self);
|
||
|
|
||
|
schedule_event (self, ON_UPDATE_START);
|
||
|
schedule_append_to_buffer_task (self);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* gst_source_buffer_abort:
|
||
|
* @self: #GstSourceBuffer instance
|
||
|
* @error: (out) (optional) (nullable) (transfer full): the resulting error or `NULL`
|
||
|
*
|
||
|
* Attempts to end any processing of the currently pending data and reset the
|
||
|
* media parser.
|
||
|
*
|
||
|
* [Specification](https://www.w3.org/TR/media-source-2/#dom-sourcebuffer-abort)
|
||
|
*
|
||
|
* Returns: `TRUE` on success, `FALSE` otherwise
|
||
|
* Since: 1.24
|
||
|
*/
|
||
|
gboolean
|
||
|
gst_source_buffer_abort (GstSourceBuffer * self, GError ** error)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
if (gst_append_pipeline_eos (self->append_pipeline) == GST_FLOW_OK) {
|
||
|
return TRUE;
|
||
|
} else {
|
||
|
g_set_error (error,
|
||
|
GST_MEDIA_SOURCE_ERROR, GST_MEDIA_SOURCE_ERROR_INVALID_STATE,
|
||
|
"failed to abort source buffer");
|
||
|
return FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
gst_source_buffer_has_init_segment (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), FALSE);
|
||
|
return gst_append_pipeline_has_init_segment (self->append_pipeline);
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
is_buffered_fold (const GValue * item, IsBufferedAccumulator * acc,
|
||
|
GstSourceBuffer * self)
|
||
|
{
|
||
|
GstSample *sample = gst_value_get_sample (item);
|
||
|
GstBuffer *buffer = gst_sample_get_buffer (sample);
|
||
|
GstClockTime buffer_start = GST_BUFFER_DTS (buffer);
|
||
|
GstClockTime buffer_end = buffer_start + GST_BUFFER_DURATION (buffer);
|
||
|
if (acc->time < buffer_start) {
|
||
|
GST_TRACE_OBJECT (self, "position precedes buffer start, done");
|
||
|
acc->buffered = FALSE;
|
||
|
return FALSE;
|
||
|
}
|
||
|
if (acc->time >= buffer_start && acc->time < buffer_end) {
|
||
|
GST_TRACE_OBJECT (self, "position is within buffer, done");
|
||
|
acc->buffered = TRUE;
|
||
|
return FALSE;
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
gst_source_buffer_is_buffered (GstSourceBuffer * self, GstClockTime time)
|
||
|
{
|
||
|
GHashTableIter iter;
|
||
|
gboolean buffered = TRUE;
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
for (gpointer key, value;
|
||
|
buffered && g_hash_table_iter_next (&iter, &key, &value);) {
|
||
|
GstMediaSourceTrack *track = key;
|
||
|
if (!gst_media_source_track_get_active (track)) {
|
||
|
continue;
|
||
|
}
|
||
|
GstMediaSourceTrackBuffer *track_buffer = value;
|
||
|
IsBufferedAccumulator acc = {
|
||
|
.time = time,
|
||
|
.buffered = FALSE,
|
||
|
};
|
||
|
GstIterator *iter =
|
||
|
gst_media_source_track_buffer_iter_samples (track_buffer, time, NULL);
|
||
|
while (gst_iterator_fold (iter, (GstIteratorFoldFunction) is_buffered_fold,
|
||
|
(GValue *) & acc, self) == GST_ITERATOR_RESYNC) {
|
||
|
gst_iterator_resync (iter);
|
||
|
}
|
||
|
gst_iterator_free (iter);
|
||
|
buffered = acc.buffered;
|
||
|
}
|
||
|
return buffered;
|
||
|
}
|
||
|
|
||
|
static gboolean
|
||
|
is_range_buffered_fold (const GValue * item, IsRangeBufferedAccumulator * acc,
|
||
|
GstSourceBuffer * self)
|
||
|
{
|
||
|
GstSample *sample = gst_value_get_sample (item);
|
||
|
GstBuffer *buffer = gst_sample_get_buffer (sample);
|
||
|
GstClockTime buffer_start = GST_BUFFER_DTS (buffer);
|
||
|
GstClockTime buffer_end = buffer_start + GST_BUFFER_DURATION (buffer);
|
||
|
|
||
|
GstClockTime start = acc->start;
|
||
|
GstClockTime end = acc->end;
|
||
|
|
||
|
if (!acc->start_buffered) {
|
||
|
if (start < buffer_start) {
|
||
|
GST_TRACE_OBJECT (self, "start position precedes buffer start, done");
|
||
|
return FALSE;
|
||
|
}
|
||
|
if (start >= buffer_start && start < buffer_end) {
|
||
|
GST_TRACE_OBJECT (self, "start position is within buffer, checking end");
|
||
|
acc->start_buffered = TRUE;
|
||
|
return TRUE;
|
||
|
}
|
||
|
} else {
|
||
|
if (end < buffer_start) {
|
||
|
GST_TRACE_OBJECT (self, "end position precedes buffer start, done");
|
||
|
return FALSE;
|
||
|
}
|
||
|
if (end <= buffer_end) {
|
||
|
GST_TRACE_OBJECT (self, "end position is within buffer, done");
|
||
|
acc->end_buffered = TRUE;
|
||
|
return FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
gst_source_buffer_is_range_buffered (GstSourceBuffer * self, GstClockTime start,
|
||
|
GstClockTime end)
|
||
|
{
|
||
|
GHashTableIter iter;
|
||
|
gboolean buffered = TRUE;
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
for (gpointer key, value;
|
||
|
buffered && g_hash_table_iter_next (&iter, &key, &value);) {
|
||
|
GstMediaSourceTrack *track = key;
|
||
|
if (!gst_media_source_track_get_active (track)) {
|
||
|
continue;
|
||
|
}
|
||
|
GstMediaSourceTrackBuffer *track_buffer = value;
|
||
|
IsRangeBufferedAccumulator acc = {
|
||
|
.start = start,
|
||
|
.end = end,
|
||
|
.start_buffered = FALSE,
|
||
|
.end_buffered = FALSE,
|
||
|
};
|
||
|
GstIterator *iter =
|
||
|
gst_media_source_track_buffer_iter_samples (track_buffer, start, NULL);
|
||
|
while (gst_iterator_fold (iter,
|
||
|
(GstIteratorFoldFunction) is_range_buffered_fold, (GValue *) & acc,
|
||
|
self) == GST_ITERATOR_RESYNC) {
|
||
|
gst_iterator_resync (iter);
|
||
|
}
|
||
|
buffered = acc.end_buffered;
|
||
|
gst_iterator_free (iter);
|
||
|
}
|
||
|
return buffered;
|
||
|
}
|
||
|
|
||
|
GstClockTime
|
||
|
gst_source_buffer_get_duration (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), GST_CLOCK_TIME_NONE);
|
||
|
return gst_append_pipeline_get_duration (self->append_pipeline);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
gst_source_buffer_teardown (GstSourceBuffer * self)
|
||
|
{
|
||
|
reset_parser_state (self);
|
||
|
clear_updating (self);
|
||
|
}
|
||
|
|
||
|
GPtrArray *
|
||
|
gst_source_buffer_get_all_tracks (GstSourceBuffer * self)
|
||
|
{
|
||
|
g_return_val_if_fail (GST_IS_SOURCE_BUFFER (self), NULL);
|
||
|
|
||
|
GPtrArray *tracks = g_ptr_array_new ();
|
||
|
|
||
|
GPtrArray *audio_tracks = gst_append_pipeline_get_audio_tracks
|
||
|
(self->append_pipeline);
|
||
|
GPtrArray *text_tracks = gst_append_pipeline_get_text_tracks
|
||
|
(self->append_pipeline);
|
||
|
GPtrArray *video_tracks = gst_append_pipeline_get_video_tracks
|
||
|
(self->append_pipeline);
|
||
|
|
||
|
if (audio_tracks) {
|
||
|
g_ptr_array_extend (tracks, audio_tracks, NULL, NULL);
|
||
|
}
|
||
|
if (text_tracks) {
|
||
|
g_ptr_array_extend (tracks, text_tracks, NULL, NULL);
|
||
|
}
|
||
|
if (video_tracks) {
|
||
|
g_ptr_array_extend (tracks, video_tracks, NULL, NULL);
|
||
|
}
|
||
|
|
||
|
return tracks;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
seek_track_buffer (GstMediaSourceTrack * track,
|
||
|
GstMediaSourceTrackBuffer * buffer, GstSourceBuffer * self)
|
||
|
{
|
||
|
TrackFeedTask *feed = get_track_feed (self, track);
|
||
|
|
||
|
const gchar *track_id = gst_media_source_track_get_id (track);
|
||
|
GST_DEBUG_OBJECT (self, "%s: seeking", track_id);
|
||
|
reset_track_feed (feed);
|
||
|
GST_DEBUG_OBJECT (self, "%s: restarted track feed", track_id);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
gst_source_buffer_seek (GstSourceBuffer * self, GstClockTime time)
|
||
|
{
|
||
|
g_return_if_fail (GST_IS_SOURCE_BUFFER (self));
|
||
|
g_return_if_fail (GST_CLOCK_TIME_IS_VALID (time));
|
||
|
self->seek_time = time;
|
||
|
g_hash_table_foreach (self->track_buffers, (GHFunc) seek_track_buffer, self);
|
||
|
}
|
||
|
|
||
|
gboolean
|
||
|
gst_source_buffer_get_active (GstSourceBuffer * self)
|
||
|
{
|
||
|
gboolean active = FALSE;
|
||
|
GHashTableIter iter;
|
||
|
GST_OBJECT_LOCK (self);
|
||
|
g_hash_table_iter_init (&iter, self->track_buffers);
|
||
|
for (gpointer key; !active && g_hash_table_iter_next (&iter, &key, NULL);) {
|
||
|
GstMediaSourceTrack *track = GST_MEDIA_SOURCE_TRACK (key);
|
||
|
active |= gst_media_source_track_get_active (track);
|
||
|
}
|
||
|
GST_OBJECT_UNLOCK (self);
|
||
|
return active;
|
||
|
}
|