multiqueue: Improve iterate internal links function

Pads have their GstSingleQueue stored as element private data
so there's no need to iterate over the list of single queues
every time. Also every pad only has a single internal link so
use a single iterator instead of a complex custom iterator.

Set the element private data of the pad to NULL when freeing the
single queue.
This commit is contained in:
Sebastian Dröge 2009-09-26 11:43:06 +02:00
parent 920e9b569d
commit d6de0e4cce

View file

@ -465,95 +465,36 @@ gst_multi_queue_get_property (GObject * object, guint prop_id,
GST_MULTI_QUEUE_MUTEX_UNLOCK (mq); GST_MULTI_QUEUE_MUTEX_UNLOCK (mq);
} }
typedef struct
{
GstIterator parent;
GstMultiQueue *mqueue;
GstPad *pad;
GList *current;
} GstMultiQueueIterator;
static void
_iterate_free (GstIterator * it)
{
GstMultiQueueIterator *mit = (GstMultiQueueIterator *) it;
g_object_unref (mit->mqueue);
g_object_unref (mit->pad);
g_free (it);
}
static GstIteratorResult
_iterate_next (GstIterator * it, gpointer * result)
{
GstMultiQueueIterator *mit = (GstMultiQueueIterator *) it;
GList *current;
GstPad *res = NULL;
/* Find which single queue it belongs to */
for (current = mit->current; !res && current; current = current->next) {
GstSingleQueue *sq = (GstSingleQueue *) current->data;
if (sq->sinkpad == mit->pad)
res = sq->srcpad;
if (sq->srcpad == mit->pad)
res = sq->sinkpad;
}
*result = res;
mit->current = current;
return res ? GST_ITERATOR_OK : GST_ITERATOR_DONE;
}
static GstIteratorItem
_iterate_item (GstIterator * it, gpointer item)
{
GstPad *pad = (GstPad *) item;
gst_object_ref (pad);
return GST_ITERATOR_ITEM_PASS;
}
static void
_iterate_resync (GstIterator * it)
{
GstMultiQueueIterator *mit = (GstMultiQueueIterator *) it;
mit->current = mit->mqueue->queues;
}
static GstIterator * static GstIterator *
gst_multi_queue_iterate_internal_links (GstPad * pad) gst_multi_queue_iterate_internal_links (GstPad * pad)
{ {
GstMultiQueue *mqueue; GstIterator *it = NULL;
GstMultiQueueIterator *ret; GstPad *opad;
GstSingleQueue *squeue;
GstMultiQueue *mq = GST_MULTI_QUEUE (gst_pad_get_parent (pad));
g_return_val_if_fail (GST_IS_PAD (pad), NULL); GST_MULTI_QUEUE_MUTEX_LOCK (mq);
mqueue = GST_MULTI_QUEUE (GST_PAD_PARENT (pad)); squeue = gst_pad_get_element_private (pad);
if (!mqueue) if (!squeue)
goto no_parent; goto out;
ret = (GstMultiQueueIterator *) if (squeue->sinkpad == pad)
gst_iterator_new (sizeof (GstMultiQueueIterator), opad = gst_object_ref (squeue->srcpad);
GST_TYPE_PAD, else if (squeue->srcpad == pad)
mqueue->qlock, opad = gst_object_ref (squeue->sinkpad);
&mqueue->queues_cookie, else
_iterate_next, _iterate_item, _iterate_resync, _iterate_free); goto out;
ret->mqueue = g_object_ref (mqueue);
ret->current = mqueue->queues;
ret->pad = g_object_ref (pad);
return (GstIterator *) ret; it = gst_iterator_new_single (GST_TYPE_PAD, opad,
(GstCopyFunction) gst_object_ref, (GFreeFunc) gst_object_unref);
no_parent: gst_object_unref (opad);
{
GST_DEBUG_OBJECT (pad, "no parent"); out:
return NULL; GST_MULTI_QUEUE_MUTEX_UNLOCK (mq);
} gst_object_unref (mq);
return it;
} }
@ -623,6 +564,8 @@ gst_multi_queue_release_pad (GstElement * element, GstPad * pad)
gst_pad_set_active (sq->srcpad, FALSE); gst_pad_set_active (sq->srcpad, FALSE);
gst_pad_set_active (sq->sinkpad, FALSE); gst_pad_set_active (sq->sinkpad, FALSE);
gst_pad_set_element_private (sq->srcpad, NULL);
gst_pad_set_element_private (sq->sinkpad, NULL);
gst_element_remove_pad (element, sq->srcpad); gst_element_remove_pad (element, sq->srcpad);
gst_element_remove_pad (element, sq->sinkpad); gst_element_remove_pad (element, sq->sinkpad);
gst_single_queue_free (sq); gst_single_queue_free (sq);