clip: secure adding clip to layer

Add more checks when adding a clip to a layer, or moving a clip to a new
layer. Also, mark the "layer" property as explicit-notify.
This commit is contained in:
Henry Wilkes 2020-04-06 12:42:03 +01:00
parent 46e6a074bb
commit e57c345f72
2 changed files with 75 additions and 29 deletions

View file

@ -106,6 +106,7 @@ struct _GESClipPrivate
GList *copied_track_elements; GList *copied_track_elements;
GESLayer *copied_layer; GESLayer *copied_layer;
GESTimeline *copied_timeline;
gboolean prevent_priority_offset_update; gboolean prevent_priority_offset_update;
gboolean prevent_resort; gboolean prevent_resort;
@ -1225,6 +1226,7 @@ _deep_copy (GESTimelineElement * element, GESTimelineElement * copy)
} }
ccopy->priv->copied_layer = g_object_ref (self->priv->layer); ccopy->priv->copied_layer = g_object_ref (self->priv->layer);
ccopy->priv->copied_timeline = self->priv->layer->timeline;
} }
static GESTimelineElement * static GESTimelineElement *
@ -1233,6 +1235,7 @@ _paste (GESTimelineElement * element, GESTimelineElement * ref,
{ {
GList *tmp; GList *tmp;
GESClip *self = GES_CLIP (element); GESClip *self = GES_CLIP (element);
GESLayer *layer = self->priv->copied_layer;
GESClip *nclip = GES_CLIP (ges_timeline_element_copy (element, FALSE)); GESClip *nclip = GES_CLIP (ges_timeline_element_copy (element, FALSE));
ges_timeline_element_set_start (GES_TIMELINE_ELEMENT (nclip), paste_position); ges_timeline_element_set_start (GES_TIMELINE_ELEMENT (nclip), paste_position);
@ -1241,8 +1244,18 @@ _paste (GESTimelineElement * element, GESTimelineElement * ref,
for (tmp = self->priv->copied_track_elements; tmp; tmp = tmp->next) for (tmp = self->priv->copied_track_elements; tmp; tmp = tmp->next)
ges_clip_copy_track_element_into (nclip, tmp->data, GST_CLOCK_TIME_NONE); ges_clip_copy_track_element_into (nclip, tmp->data, GST_CLOCK_TIME_NONE);
if (self->priv->copied_layer) { if (layer) {
if (!ges_layer_add_clip (self->priv->copied_layer, nclip)) { if (layer->timeline != self->priv->copied_timeline) {
GST_WARNING_OBJECT (self, "Cannot be pasted into the layer %"
GST_PTR_FORMAT " because its timeline has changed", layer);
gst_object_ref_sink (nclip);
gst_object_unref (nclip);
return NULL;
}
/* adding the clip to the layer will add it to the tracks, but not
* necessarily the same ones depending on select-tracks-for-object */
if (!ges_layer_add_clip (layer, nclip)) {
GST_INFO ("%" GES_FORMAT " could not be pasted to %" GST_TIME_FORMAT, GST_INFO ("%" GES_FORMAT " could not be pasted to %" GST_TIME_FORMAT,
GES_ARGS (element), GST_TIME_ARGS (paste_position)); GES_ARGS (element), GST_TIME_ARGS (paste_position));
@ -1366,7 +1379,7 @@ ges_clip_class_init (GESClipClass * klass)
*/ */
properties[PROP_LAYER] = g_param_spec_object ("layer", "Layer", properties[PROP_LAYER] = g_param_spec_object ("layer", "Layer",
"The GESLayer where this clip is being used.", "The GESLayer where this clip is being used.",
GES_TYPE_LAYER, G_PARAM_READABLE); GES_TYPE_LAYER, G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY);
g_object_class_install_property (object_class, PROP_LAYER, g_object_class_install_property (object_class, PROP_LAYER,
properties[PROP_LAYER]); properties[PROP_LAYER]);
@ -1585,13 +1598,42 @@ ges_clip_is_moving_from_layer (GESClip * clip)
gboolean gboolean
ges_clip_move_to_layer (GESClip * clip, GESLayer * layer) ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
{ {
gboolean ret; gboolean ret = FALSE;
GESLayer *current_layer; GESLayer *current_layer;
GESTimeline *current_timeline = GES_TIMELINE_ELEMENT_TIMELINE (clip);
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE); g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
g_return_val_if_fail (GES_IS_LAYER (layer), FALSE); g_return_val_if_fail (GES_IS_LAYER (layer), FALSE);
current_layer = clip->priv->layer;
if (current_layer == layer) {
GST_INFO_OBJECT (clip, "Already in the layer %" GST_PTR_FORMAT, layer);
return TRUE;
}
gst_object_ref (clip);
if (current_layer == NULL) {
GST_DEBUG ("Not moving %p, only adding it to %p", clip, layer);
ret = ges_layer_add_clip (layer, clip);
goto done;
}
ELEMENT_SET_FLAG (clip, GES_CLIP_IS_MOVING); ELEMENT_SET_FLAG (clip, GES_CLIP_IS_MOVING);
if (current_timeline != layer->timeline) {
/* make sure we can perform the can_move_element_check in the timeline
* of the layer */
GST_WARNING_OBJECT (layer, "Cannot move clip %" GES_FORMAT " into "
"the layer because its timeline %" GST_PTR_FORMAT " does not "
"match the timeline of the layer %" GST_PTR_FORMAT,
GES_ARGS (clip), current_timeline, layer->timeline);
ret = FALSE;
goto done;
}
if (layer->timeline if (layer->timeline
&& !timeline_tree_can_move_element (timeline_get_tree (layer->timeline), && !timeline_tree_can_move_element (timeline_get_tree (layer->timeline),
GES_TIMELINE_ELEMENT (clip), GES_TIMELINE_ELEMENT (clip),
@ -1600,36 +1642,30 @@ ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
GES_TIMELINE_ELEMENT_DURATION (clip), NULL)) { GES_TIMELINE_ELEMENT_DURATION (clip), NULL)) {
GST_INFO_OBJECT (layer, "Clip %" GES_FORMAT " can't move to layer %d", GST_INFO_OBJECT (layer, "Clip %" GES_FORMAT " can't move to layer %d",
GES_ARGS (clip), ges_layer_get_priority (layer)); GES_ARGS (clip), ges_layer_get_priority (layer));
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING); goto done;
return FALSE;
}
current_layer = clip->priv->layer;
if (current_layer == NULL) {
GST_DEBUG ("Not moving %p, only adding it to %p", clip, layer);
return ges_layer_add_clip (layer, clip);
} }
GST_DEBUG_OBJECT (clip, "moving to layer %p, priority: %d", layer, GST_DEBUG_OBJECT (clip, "moving to layer %p, priority: %d", layer,
ges_layer_get_priority (layer)); ges_layer_get_priority (layer));
gst_object_ref (clip);
ret = ges_layer_remove_clip (current_layer, clip); ret = ges_layer_remove_clip (current_layer, clip);
if (!ret) { if (!ret) {
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING); goto done;
gst_object_unref (clip);
return FALSE;
} }
ret = ges_layer_add_clip (layer, clip); ret = ges_layer_add_clip (layer, clip);
if (ret) {
g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
} else {
/* try and move back into the original layer */
ges_layer_add_clip (current_layer, clip);
}
done:
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING); ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
gst_object_unref (clip); gst_object_unref (clip);
g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
return ret && (clip->priv->layer == layer); return ret && (clip->priv->layer == layer);
} }

View file

@ -690,29 +690,41 @@ ges_layer_add_clip (GESLayer * layer, GESClip * clip)
GESAsset *asset; GESAsset *asset;
GESLayerPrivate *priv; GESLayerPrivate *priv;
GESLayer *current_layer; GESLayer *current_layer;
GESTimeline *timeline = GES_TIMELINE_ELEMENT_TIMELINE (clip);
g_return_val_if_fail (GES_IS_LAYER (layer), FALSE); g_return_val_if_fail (GES_IS_LAYER (layer), FALSE);
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE); g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
GST_DEBUG_OBJECT (layer, "adding clip:%p", clip); GST_DEBUG_OBJECT (layer, "adding clip:%p", clip);
gst_object_ref_sink (clip);
priv = layer->priv; priv = layer->priv;
current_layer = ges_clip_get_layer (clip); current_layer = ges_clip_get_layer (clip);
if (G_UNLIKELY (current_layer)) { if (G_UNLIKELY (current_layer)) {
GST_WARNING ("Clip %p already belongs to another layer", clip); GST_WARNING_OBJECT (layer, "Clip %" GES_FORMAT " already belongs to "
gst_object_ref_sink (clip); "another layer", GES_ARGS (clip));
gst_object_unref (clip); gst_object_unref (clip);
gst_object_unref (current_layer); gst_object_unref (current_layer);
return FALSE; return FALSE;
} }
if (timeline && timeline != layer->timeline) {
/* if a clip is not in any layer, its timeline should not be set */
GST_ERROR_OBJECT (layer, "Clip %" GES_FORMAT " timeline %"
GST_PTR_FORMAT " does not match that of the layer %"
GST_PTR_FORMAT, GES_ARGS (clip), timeline, layer->timeline);
gst_object_unref (clip);
return FALSE;
}
timeline = layer->timeline;
asset = ges_extractable_get_asset (GES_EXTRACTABLE (clip)); asset = ges_extractable_get_asset (GES_EXTRACTABLE (clip));
if (asset == NULL) { if (asset == NULL) {
gchar *id; gchar *id;
NewAssetUData *mudata = g_slice_new (NewAssetUData); NewAssetUData *mudata = g_slice_new (NewAssetUData);
mudata->clip = gst_object_ref_sink (clip); mudata->clip = clip;
mudata->layer = layer; mudata->layer = layer;
GST_DEBUG_OBJECT (layer, "%" GST_PTR_FORMAT " as no reference to any " GST_DEBUG_OBJECT (layer, "%" GST_PTR_FORMAT " as no reference to any "
@ -741,8 +753,6 @@ ges_layer_add_clip (GESLayer * layer, GESClip * clip)
g_slice_free (NewAssetUData, mudata); g_slice_free (NewAssetUData, mudata);
gst_clear_object (&asset); gst_clear_object (&asset);
} else {
gst_object_ref_sink (clip);
} }
/* Take a reference to the clip and store it stored by start/priority */ /* Take a reference to the clip and store it stored by start/priority */
@ -775,9 +785,9 @@ ges_layer_add_clip (GESLayer * layer, GESClip * clip)
* invoked by ges_timeline_add_clip */ * invoked by ges_timeline_add_clip */
g_signal_emit (layer, ges_layer_signals[OBJECT_ADDED], 0, clip); g_signal_emit (layer, ges_layer_signals[OBJECT_ADDED], 0, clip);
if (layer->timeline && !ges_timeline_add_clip (layer->timeline, clip)) { if (timeline && !ges_timeline_add_clip (timeline, clip)) {
GST_WARNING_OBJECT (layer, "Could not add the clip %" GES_FORMAT GST_WARNING_OBJECT (layer, "Could not add the clip %" GES_FORMAT
" to the timeline %" GST_PTR_FORMAT, GES_ARGS (clip), layer->timeline); " to the timeline %" GST_PTR_FORMAT, GES_ARGS (clip), timeline);
/* FIXME: change emit signal to FALSE once we are able to delay the /* FIXME: change emit signal to FALSE once we are able to delay the
* "clip-added" signal until after ges_timeline_add_clip */ * "clip-added" signal until after ges_timeline_add_clip */
ges_layer_remove_clip_internal (layer, clip, TRUE); ges_layer_remove_clip_internal (layer, clip, TRUE);