More updates to the non-cothread scheduler

Original commit message from CVS:
More updates to the non-cothread scheduler
This commit is contained in:
Wim Taymans 2002-09-12 19:22:03 +00:00
parent 3a5b408709
commit 27301abdb3

View file

@ -28,6 +28,7 @@
#define GST_ELEMENT_SCHED_CONTEXT(elem) ((GstOptSchedulerCtx*) (GST_ELEMENT_CAST (elem)->sched_private)) #define GST_ELEMENT_SCHED_CONTEXT(elem) ((GstOptSchedulerCtx*) (GST_ELEMENT_CAST (elem)->sched_private))
#define GST_ELEMENT_SCHED_GROUP(elem) (GST_ELEMENT_SCHED_CONTEXT (elem)->group) #define GST_ELEMENT_SCHED_GROUP(elem) (GST_ELEMENT_SCHED_CONTEXT (elem)->group)
#define GST_PAD_BUFLIST(pad) ((GList*) (GST_REAL_PAD_CAST(pad)->sched_private))
#define GST_ELEMENT_COTHREAD_STOPPING GST_ELEMENT_SCHEDULER_PRIVATE1 #define GST_ELEMENT_COTHREAD_STOPPING GST_ELEMENT_SCHEDULER_PRIVATE1
#define GST_ELEMENT_IS_COTHREAD_STOPPING(element) GST_FLAG_IS_SET((element), GST_ELEMENT_COTHREAD_STOPPING) #define GST_ELEMENT_IS_COTHREAD_STOPPING(element) GST_FLAG_IS_SET((element), GST_ELEMENT_COTHREAD_STOPPING)
@ -58,15 +59,19 @@ typedef enum {
} GstOptSchedulerState; } GstOptSchedulerState;
struct _GstOptScheduler { struct _GstOptScheduler {
GstScheduler parent; GstScheduler parent;
GstOptSchedulerState state; GstOptSchedulerState state;
cothread_context *context; cothread_context *context;
gboolean use_cothreads; gboolean use_cothreads;
gint iterations;
GSList *elements; GSList *elements;
GSList *chains; GSList *chains;
GList *runqueue;
gint recursion;
}; };
struct _GstOptSchedulerClass { struct _GstOptSchedulerClass {
@ -154,10 +159,22 @@ struct _GstOptSchedulerCtx {
gint element_type; gint element_type;
}; };
enum
{
ARG_0,
ARG_USE_COTHREADS,
ARG_ITERATIONS,
};
static void gst_opt_scheduler_class_init (GstOptSchedulerClass *klass); static void gst_opt_scheduler_class_init (GstOptSchedulerClass *klass);
static void gst_opt_scheduler_init (GstOptScheduler *scheduler); static void gst_opt_scheduler_init (GstOptScheduler *scheduler);
static void gst_opt_scheduler_set_property (GObject *object, guint prop_id,
const GValue *value, GParamSpec *pspec);
static void gst_opt_scheduler_get_property (GObject *object, guint prop_id,
GValue *value, GParamSpec *pspec);
static void gst_opt_scheduler_dispose (GObject *object); static void gst_opt_scheduler_dispose (GObject *object);
static void gst_opt_scheduler_setup (GstScheduler *sched); static void gst_opt_scheduler_setup (GstScheduler *sched);
@ -218,8 +235,17 @@ gst_opt_scheduler_class_init (GstOptSchedulerClass *klass)
parent_class = g_type_class_ref (GST_TYPE_SCHEDULER); parent_class = g_type_class_ref (GST_TYPE_SCHEDULER);
gobject_class->set_property = GST_DEBUG_FUNCPTR (gst_opt_scheduler_set_property);
gobject_class->get_property = GST_DEBUG_FUNCPTR (gst_opt_scheduler_get_property);
gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_opt_scheduler_dispose); gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_opt_scheduler_dispose);
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_USE_COTHREADS,
g_param_spec_boolean ("use_cothreads", "Use cothreads", "Should this scheduler use cothreads",
TRUE, G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_USE_COTHREADS,
g_param_spec_int ("iterations", "Iterations", "Number of groups to schedule in one iteration (-1 == until EOS/error)",
-1, G_MAXINT, 1, G_PARAM_READWRITE));
gstscheduler_class->setup = GST_DEBUG_FUNCPTR (gst_opt_scheduler_setup); gstscheduler_class->setup = GST_DEBUG_FUNCPTR (gst_opt_scheduler_setup);
gstscheduler_class->reset = GST_DEBUG_FUNCPTR (gst_opt_scheduler_reset); gstscheduler_class->reset = GST_DEBUG_FUNCPTR (gst_opt_scheduler_reset);
gstscheduler_class->add_element = GST_DEBUG_FUNCPTR (gst_opt_scheduler_add_element); gstscheduler_class->add_element = GST_DEBUG_FUNCPTR (gst_opt_scheduler_add_element);
@ -242,8 +268,9 @@ static void
gst_opt_scheduler_init (GstOptScheduler *scheduler) gst_opt_scheduler_init (GstOptScheduler *scheduler)
{ {
scheduler->elements = NULL; scheduler->elements = NULL;
scheduler->use_cothreads = FALSE; //scheduler->use_cothreads = FALSE;
scheduler->use_cothreads = TRUE; scheduler->use_cothreads = TRUE;
scheduler->iterations = 1;
} }
static void static void
@ -279,31 +306,6 @@ GstPluginDesc plugin_desc = {
plugin_init plugin_init
}; };
/*
* Entry points for this scheduler.
*/
static void
gst_opt_scheduler_setup (GstScheduler *sched)
{
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
/* first create thread context */
if (osched->context == NULL && osched->use_cothreads) {
GST_DEBUG (GST_CAT_SCHEDULING, "initializing cothread context");
osched->context = do_cothread_context_init ();
}
}
static void
gst_opt_scheduler_reset (GstScheduler *sched)
{
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
if (osched->context && osched->use_cothreads) {
do_cothread_context_destroy (osched->context);
osched->context = NULL;
}
}
static void static void
delete_chain (GstOptScheduler *osched, GstOptSchedulerChain *chain) delete_chain (GstOptScheduler *osched, GstOptSchedulerChain *chain)
@ -506,6 +508,9 @@ remove_from_group (GstOptSchedulerGroup *group, GstElement *element)
} }
*/ */
/* this function enables/disables an element, it will set/clear a flag on the element
* and tells the chain that the group is enabled if all elements inside the group are
* enabled */
static void static void
group_element_set_enabled (GstOptSchedulerGroup *group, GstElement *element, gboolean enabled) group_element_set_enabled (GstOptSchedulerGroup *group, GstElement *element, gboolean enabled)
{ {
@ -531,24 +536,51 @@ group_element_set_enabled (GstOptSchedulerGroup *group, GstElement *element, gbo
} }
} }
static int /* a group is scheduled by doing a cothread switch to it or
* by calling the schedule function. In the non-cothread case
* we cannot run already running groups so we return FALSE here
* to indicate this to the caller */
static gboolean
schedule_group (GstOptSchedulerGroup *group) schedule_group (GstOptSchedulerGroup *group)
{ {
if (group->chain->sched->use_cothreads) { if (group->chain->sched->use_cothreads) {
if (group->cothread) if (group->cothread)
do_cothread_switch (group->cothread); do_cothread_switch (group->cothread);
return 1; return TRUE;
} }
else { else {
/* in the no cothread case, we cannot schedule already running groups */ group->schedulefunc (group->argc, group->argv);
if (!(group->flags & GST_OPT_SCHEDULER_GROUP_RUNNING)) { return TRUE;
group->schedulefunc (group->argc, group->argv);
return 1;
}
} }
return 0; return FALSE;
} }
static void
gst_opt_scheduler_schedule_run_queue (GstOptScheduler *osched)
{
GST_INFO (GST_CAT_SCHEDULING, "entering scheduler run queue recursion %d", osched->recursion);
osched->recursion++;
while (osched->runqueue) {
GstOptSchedulerGroup *group;
group = (GstOptSchedulerGroup *) osched->runqueue->data;
osched->runqueue = g_list_remove (osched->runqueue, group);
GST_INFO (GST_CAT_SCHEDULING, "scheduling %p", group);
schedule_group (group);
GST_INFO (GST_CAT_SCHEDULING, "done scheduling %p", group);
}
GST_INFO (GST_CAT_SCHEDULING, "run queue length after scheduling %d", g_list_length (osched->runqueue));
osched->recursion--;
}
/* a chain is scheduled by picking the first active group and scheduling it */
static void static void
schedule_chain (GstOptSchedulerChain *chain) schedule_chain (GstOptSchedulerChain *chain)
{ {
@ -556,61 +588,39 @@ schedule_chain (GstOptSchedulerChain *chain)
while (groups) { while (groups) {
GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) groups->data; GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) groups->data;
groups = g_slist_next (groups); groups = g_slist_next (groups);
if (!GST_OPT_SCHEDULER_GROUP_IS_DISABLED (group)) { if (!GST_OPT_SCHEDULER_GROUP_IS_DISABLED (group)) {
GstOptScheduler *osched;
osched = chain->sched;
GST_INFO (GST_CAT_SCHEDULING, "scheduling group %p in chain %p", GST_INFO (GST_CAT_SCHEDULING, "scheduling group %p in chain %p",
group, chain); group, chain);
schedule_group (group); if (osched->use_cothreads) {
schedule_group (group);
}
else {
osched->recursion = 0;
osched->runqueue = g_list_append (osched->runqueue, group);
gst_opt_scheduler_schedule_run_queue (osched);
}
GST_INFO (GST_CAT_SCHEDULING, "done scheduling group %p in chain %p",
group, chain);
break; break;
} }
} }
} }
static void /* a get-based group is scheduled by getting a buffer from the get based
gst_opt_scheduler_add_element (GstScheduler *sched, GstElement *element) * entry point and by pushing the buffer to the peer.
{ * We also set the running flag on this group for as long as this
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); * function is running. */
GstOptSchedulerCtx *ctx;
GST_INFO (GST_CAT_SCHEDULING, "adding element \"%s\" to scheduler", GST_ELEMENT_NAME (element));
if (GST_ELEMENT_IS_DECOUPLED (element))
return;
ctx = g_new0 (GstOptSchedulerCtx, 1);
GST_ELEMENT_SCHED_CONTEXT (element) = ctx;
if (element->loopfunc) {
GstOptSchedulerGroup *group;
GstOptSchedulerChain *chain;
chain = create_chain (osched);
group = create_group (chain, element);
group->entry = element;
group->type = GST_OPT_SCHEDULER_GROUP_LOOP;
GST_INFO (GST_CAT_SCHEDULING, "added element \"%s\" as loop based entry", GST_ELEMENT_NAME (element));
}
}
static void
gst_opt_scheduler_remove_element (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
GST_INFO (GST_CAT_SCHEDULING, "removing element \"%s\" from scheduler", GST_ELEMENT_NAME (element));
g_free (GST_ELEMENT_SCHED_CONTEXT (element));
GST_ELEMENT_SCHED_CONTEXT (element) = NULL;
g_warning ("remove implement me");
}
static int static int
get_wrapper (int argc, char *argv[]) get_group_schedule_function (int argc, char *argv[])
{ {
GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv; GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv;
const GList *pads = gst_element_get_pad_list (group->entry); const GList *pads = gst_element_get_pad_list (group->entry);
@ -641,8 +651,12 @@ get_wrapper (int argc, char *argv[])
return 0; return 0;
} }
/* a loop-based group is scheduled by calling the loop function
* on the entry point.
* We also set the running flag on this group for as long as this
* function is running. */
static int static int
loop_wrapper (int argc, char *argv[]) loop_group_schedule_function (int argc, char *argv[])
{ {
GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv; GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv;
GstElement *entry = group->entry; GstElement *entry = group->entry;
@ -662,8 +676,9 @@ loop_wrapper (int argc, char *argv[])
} }
/* the function to schedule an unkown group, which just gives an error */
static int static int
unkown_wrapper (int argc, char *argv[]) unkown_group_schedule_function (int argc, char *argv[])
{ {
GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv; GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv;
@ -674,59 +689,89 @@ unkown_wrapper (int argc, char *argv[])
return 0; return 0;
} }
/* this function is called when the first element of a chain-loop or a loop-loop
* connection performs a push to the loop element. We then schedule the
* group with the loop-based element until the bufpen is empty */
static void static void
gst_opt_scheduler_loop_wrapper (GstPad *sinkpad, GstBuffer *buffer) gst_opt_scheduler_loop_wrapper (GstPad *sinkpad, GstBuffer *buffer)
{ {
GstOptSchedulerGroup *group; GstOptSchedulerGroup *group;
GstOptScheduler *osched;
GST_INFO (GST_CAT_SCHEDULING, "loop wrapper, putting buffer in bufpen"); GST_INFO (GST_CAT_SCHEDULING, "loop wrapper, putting buffer in bufpen");
group = GST_ELEMENT_SCHED_GROUP (GST_PAD_PARENT (sinkpad)); group = GST_ELEMENT_SCHED_GROUP (GST_PAD_PARENT (sinkpad));
osched = group->chain->sched;
if (GST_RPAD_BUFPEN (GST_RPAD_PEER (sinkpad))) {
g_warning ("scheduling error, bufpen not empty, disabling group %p", group); if (osched->use_cothreads) {
chain_group_set_enabled (group->chain, group, FALSE); if (GST_PAD_BUFLIST (GST_RPAD_PEER (sinkpad))) {
group->chain->sched->state = GST_OPT_SCHEDULER_STATE_ERROR; g_warning ("deadlock detected, disabling group %p", group);
return; chain_group_set_enabled (group->chain, group, FALSE);
group->chain->sched->state = GST_OPT_SCHEDULER_STATE_ERROR;
}
else {
GST_PAD_BUFLIST (GST_RPAD_PEER (sinkpad)) = g_list_append (GST_PAD_BUFLIST (GST_RPAD_PEER (sinkpad)), buffer);
schedule_group (group);
}
} }
else if (!(group->flags & GST_OPT_SCHEDULER_GROUP_RUNNING)) {
GST_RPAD_BUFPEN (GST_RPAD_PEER (sinkpad)) = buffer; GST_PAD_BUFLIST (GST_RPAD_PEER (sinkpad)) = g_list_append (GST_PAD_BUFLIST (GST_RPAD_PEER (sinkpad)), buffer);
osched->runqueue = g_list_append (osched->runqueue, group);
while (GST_RPAD_BUFPEN (GST_RPAD_PEER (sinkpad))) {
if (!schedule_group (group))
break;
} }
GST_INFO (GST_CAT_SCHEDULING, "after loop wrapper buflist %d",
g_list_length (GST_PAD_BUFLIST (GST_RPAD_PEER (sinkpad))));
} }
/* this function is called by a loop based element that performs a
* pull on a sinkpad. We schedule the peer group until the bufpen
* is filled with the buffer so that this function can return */
static GstBuffer* static GstBuffer*
gst_opt_scheduler_get_wrapper (GstPad *srcpad) gst_opt_scheduler_get_wrapper (GstPad *srcpad)
{ {
GstBuffer *buffer; GstBuffer *buffer = NULL;
GST_INFO (GST_CAT_SCHEDULING, "get wrapper, removing buffer from bufpen"); GST_INFO (GST_CAT_SCHEDULING, "get wrapper, removing buffer from bufpen");
buffer = GST_RPAD_BUFPEN (srcpad); if (GST_PAD_BUFLIST (srcpad))
buffer = GST_PAD_BUFLIST (srcpad)->data;
while (!buffer) { while (!buffer) {
GstOptSchedulerGroup *group; GstOptSchedulerGroup *group;
GstOptScheduler *osched;
group = GST_ELEMENT_SCHED_GROUP (GST_PAD_PARENT (srcpad)); group = GST_ELEMENT_SCHED_GROUP (GST_PAD_PARENT (srcpad));
osched = group->chain->sched;
if (!schedule_group (group)) { if (osched->use_cothreads) {
schedule_group (group);
}
else if (!(group->flags & GST_OPT_SCHEDULER_GROUP_RUNNING)) {
osched->runqueue = g_list_append (osched->runqueue, group);
gst_opt_scheduler_schedule_run_queue (osched);
}
else {
g_warning ("deadlock detected, disabling group %p", group); g_warning ("deadlock detected, disabling group %p", group);
chain_group_set_enabled (group->chain, group, FALSE); chain_group_set_enabled (group->chain, group, FALSE);
group->chain->sched->state = GST_OPT_SCHEDULER_STATE_ERROR; group->chain->sched->state = GST_OPT_SCHEDULER_STATE_ERROR;
return NULL; return NULL;
} }
buffer = GST_RPAD_BUFPEN (srcpad); if (GST_PAD_BUFLIST (srcpad)) {
buffer = (GstBuffer *) GST_PAD_BUFLIST (srcpad)->data;
}
} }
GST_PAD_BUFLIST (srcpad) = g_list_remove (GST_PAD_BUFLIST (srcpad), buffer);
GST_RPAD_BUFPEN (srcpad) = NULL; GST_INFO (GST_CAT_SCHEDULING, "get wrapper, returning buffer %d",
g_list_length (GST_PAD_BUFLIST (srcpad)));
return buffer; return buffer;
} }
/* this function is a chain wrapper for non-event-aware plugins,
* it'll simply dispatch the events to the (default) event handler */
static void static void
gst_opt_scheduler_chain_wrapper (GstPad *sinkpad, GstBuffer *buffer) gst_opt_scheduler_chain_wrapper (GstPad *sinkpad, GstBuffer *buffer)
{ {
@ -738,17 +783,21 @@ gst_opt_scheduler_chain_wrapper (GstPad *sinkpad, GstBuffer *buffer)
} }
} }
/* setup the scheduler context for a group. The right schedule function
* is selected based on the group type and cothreads are created if
* needed */
static void static void
setup_group_scheduler (GstOptScheduler *osched, GstOptSchedulerGroup *group) setup_group_scheduler (GstOptScheduler *osched, GstOptSchedulerGroup *group)
{ {
GroupScheduleFunction wrapper; GroupScheduleFunction wrapper;
wrapper = unkown_wrapper; wrapper = unkown_group_schedule_function;
/* figure out the wrapper function for this group */
if (group->type == GST_OPT_SCHEDULER_GROUP_GET) if (group->type == GST_OPT_SCHEDULER_GROUP_GET)
wrapper = get_wrapper; wrapper = get_group_schedule_function;
else if (group->type == GST_OPT_SCHEDULER_GROUP_LOOP) else if (group->type == GST_OPT_SCHEDULER_GROUP_LOOP)
wrapper = loop_wrapper; wrapper = loop_group_schedule_function;
if (osched->use_cothreads) { if (osched->use_cothreads) {
if (!(group->flags & GST_OPT_SCHEDULER_GROUP_SCHEDULABLE)) { if (!(group->flags & GST_OPT_SCHEDULER_GROUP_SCHEDULABLE)) {
@ -773,9 +822,11 @@ gst_opt_scheduler_state_transition (GstScheduler *sched, GstElement *element, gi
{ {
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
GstOptSchedulerGroup *group; GstOptSchedulerGroup *group;
GstElementStateReturn res = GST_STATE_SUCCESS;
GST_INFO (GST_CAT_SCHEDULING, "element \"%s\" state change %d", GST_ELEMENT_NAME (element), transition); GST_INFO (GST_CAT_SCHEDULING, "element \"%s\" state change %d", GST_ELEMENT_NAME (element), transition);
/* we check the state of the managing pipeline here */
if (GST_IS_BIN (element)) { if (GST_IS_BIN (element)) {
if (GST_SCHEDULER_PARENT (sched) == element) { if (GST_SCHEDULER_PARENT (sched) == element) {
GST_INFO (GST_CAT_SCHEDULING, "parent \"%s\" changed state", GST_ELEMENT_NAME (element)); GST_INFO (GST_CAT_SCHEDULING, "parent \"%s\" changed state", GST_ELEMENT_NAME (element));
@ -793,58 +844,38 @@ gst_opt_scheduler_state_transition (GstScheduler *sched, GstElement *element, gi
GST_INFO (GST_CAT_SCHEDULING, "no interesting state change, doing nothing"); GST_INFO (GST_CAT_SCHEDULING, "no interesting state change, doing nothing");
} }
} }
return GST_STATE_SUCCESS; return res;
} }
/* we don't care about decoupled elements after this */
if (GST_ELEMENT_IS_DECOUPLED (element)) if (GST_ELEMENT_IS_DECOUPLED (element))
return GST_STATE_SUCCESS; return GST_STATE_SUCCESS;
/* get the group of the element */
group = GST_ELEMENT_SCHED_GROUP (element); group = GST_ELEMENT_SCHED_GROUP (element);
switch (transition) { switch (transition) {
case GST_STATE_PAUSED_TO_PLAYING: case GST_STATE_PAUSED_TO_PLAYING:
setup_group_scheduler (osched, group); /* an element withut a group has to be an unconnected src, sink
group_element_set_enabled (group, element, TRUE); * filter element */
if (!group)
res = GST_STATE_FAILURE;
/* else construct the scheduling context of this group and enable it */
else {
setup_group_scheduler (osched, group);
group_element_set_enabled (group, element, TRUE);
}
break; break;
case GST_STATE_PLAYING_TO_PAUSED: case GST_STATE_PLAYING_TO_PAUSED:
group_element_set_enabled (group, element, FALSE); /* if the element still has a group, we disable it */
if (group)
group_element_set_enabled (group, element, FALSE);
break; break;
default: default:
break; break;
} }
return GST_STATE_SUCCESS; return res;
}
static void
gst_opt_scheduler_lock_element (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
}
static void
gst_opt_scheduler_unlock_element (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
}
static void
gst_opt_scheduler_yield (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
}
static gboolean
gst_opt_scheduler_interrupt (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
return TRUE;
}
static void
gst_opt_scheduler_error (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
} }
/* /*
@ -917,6 +948,114 @@ typedef enum {
GST_OPT_LOOP_TO_LOOP, GST_OPT_LOOP_TO_LOOP,
} ConnectionType; } ConnectionType;
/*
* Entry points for this scheduler.
*/
static void
gst_opt_scheduler_setup (GstScheduler *sched)
{
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
/* first create thread context */
if (osched->context == NULL && osched->use_cothreads) {
GST_DEBUG (GST_CAT_SCHEDULING, "initializing cothread context");
osched->context = do_cothread_context_init ();
}
}
static void
gst_opt_scheduler_reset (GstScheduler *sched)
{
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
if (osched->context && osched->use_cothreads) {
do_cothread_context_destroy (osched->context);
osched->context = NULL;
}
}
static void
gst_opt_scheduler_add_element (GstScheduler *sched, GstElement *element)
{
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
GstOptSchedulerCtx *ctx;
GST_INFO (GST_CAT_SCHEDULING, "adding element \"%s\" to scheduler", GST_ELEMENT_NAME (element));
/* decoupled elements are not added to the scheduler lists */
if (GST_ELEMENT_IS_DECOUPLED (element))
return;
ctx = g_new0 (GstOptSchedulerCtx, 1);
GST_ELEMENT_SCHED_CONTEXT (element) = ctx;
/* loop based elements *always* end up in their own group. It can eventually
* be merged with another group when a connection is made */
if (element->loopfunc) {
GstOptSchedulerGroup *group;
GstOptSchedulerChain *chain;
chain = create_chain (osched);
group = create_group (chain, element);
group->entry = element;
group->type = GST_OPT_SCHEDULER_GROUP_LOOP;
GST_INFO (GST_CAT_SCHEDULING, "added element \"%s\" as loop based entry", GST_ELEMENT_NAME (element));
}
}
static void
gst_opt_scheduler_remove_element (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
GST_INFO (GST_CAT_SCHEDULING, "removing element \"%s\" from scheduler", GST_ELEMENT_NAME (element));
g_free (GST_ELEMENT_SCHED_CONTEXT (element));
GST_ELEMENT_SCHED_CONTEXT (element) = NULL;
g_warning ("remove implement me");
}
static void
gst_opt_scheduler_lock_element (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
g_warning ("lock element, implement me");
}
static void
gst_opt_scheduler_unlock_element (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
g_warning ("unlock element, implement me");
}
static void
gst_opt_scheduler_yield (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
}
static gboolean
gst_opt_scheduler_interrupt (GstScheduler *sched, GstElement *element)
{
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
g_warning ("interrupt element, implement me");
return TRUE;
}
static void
gst_opt_scheduler_error (GstScheduler *sched, GstElement *element)
{
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
osched->state = GST_OPT_SCHEDULER_STATE_ERROR;
}
/* connect pads, merge groups and chains */
static void static void
gst_opt_scheduler_pad_connect (GstScheduler *sched, GstPad *srcpad, GstPad *sinkpad) gst_opt_scheduler_pad_connect (GstScheduler *sched, GstPad *srcpad, GstPad *sinkpad)
{ {
@ -1056,6 +1195,9 @@ static GstPad*
gst_opt_scheduler_pad_select (GstScheduler *sched, GList *padlist) gst_opt_scheduler_pad_select (GstScheduler *sched, GList *padlist)
{ {
//GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); //GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
g_warning ("pad select, implement me");
return NULL; return NULL;
} }
@ -1066,35 +1208,51 @@ gst_opt_scheduler_clock_wait (GstScheduler *sched, GstElement *element,
return gst_clock_wait (clock, time, jitter); return gst_clock_wait (clock, time, jitter);
} }
/* a scheduler iteration is done by looping and scheduling the active chains */
static GstSchedulerState static GstSchedulerState
gst_opt_scheduler_iterate (GstScheduler *sched) gst_opt_scheduler_iterate (GstScheduler *sched)
{ {
GstSchedulerState state; GstSchedulerState state = GST_SCHEDULER_STATE_STOPPED;
GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched);
GSList *chains; gint iterations = osched->iterations;
gboolean scheduled = FALSE;
osched->state = GST_OPT_SCHEDULER_STATE_RUNNING; osched->state = GST_OPT_SCHEDULER_STATE_RUNNING;
chains = osched->chains; while (iterations) {
while (chains) { gboolean scheduled = FALSE;
GstOptSchedulerChain *chain = (GstOptSchedulerChain *) chains->data; GSList *chains;
chains = g_slist_next (chains);
if (!GST_OPT_SCHEDULER_CHAIN_IS_DISABLED (chain)) { /* we have to schedule each of the scheduler chains now */
schedule_chain (chain); chains = osched->chains;
scheduled = TRUE; while (chains) {
GstOptSchedulerChain *chain = (GstOptSchedulerChain *) chains->data;
chains = g_slist_next (chains);
/* if the chain is not disabled, schedule it */
if (!GST_OPT_SCHEDULER_CHAIN_IS_DISABLED (chain)) {
schedule_chain (chain);
scheduled = TRUE;
}
} }
}
if (osched->state == GST_OPT_SCHEDULER_STATE_ERROR) { /* at this point it's possible that the scheduler state is
state = GST_SCHEDULER_STATE_ERROR; * in error, we then return an error */
} if (osched->state == GST_OPT_SCHEDULER_STATE_ERROR) {
else { state = GST_SCHEDULER_STATE_ERROR;
if (scheduled) break;
state = GST_SCHEDULER_STATE (sched); }
else else {
state = GST_SCHEDULER_STATE_STOPPED; /* if chains were scheduled, return our current state */
if (scheduled)
state = GST_SCHEDULER_STATE (sched);
/* if no chains were scheduled, we say we are stopped */
else {
state = GST_SCHEDULER_STATE_STOPPED;
break;
}
}
if (iterations > 0)
iterations--;
} }
return state; return state;
@ -1134,3 +1292,50 @@ gst_opt_scheduler_show (GstScheduler *sched)
} }
} }
} }
static void
gst_opt_scheduler_get_property (GObject *object, guint prop_id,
GValue *value, GParamSpec *pspec)
{
GstOptScheduler *osched;
g_return_if_fail (GST_IS_OPT_SCHEDULER (object));
osched = GST_OPT_SCHEDULER_CAST (object);
switch (prop_id) {
case ARG_USE_COTHREADS:
g_value_set_boolean (value, osched->use_cothreads);
break;
case ARG_ITERATIONS:
g_value_set_int (value, osched->iterations);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
gst_opt_scheduler_set_property (GObject *object, guint prop_id,
const GValue *value, GParamSpec *pspec)
{
GstOptScheduler *osched;
g_return_if_fail (GST_IS_OPT_SCHEDULER (object));
osched = GST_OPT_SCHEDULER_CAST (object);
switch (prop_id) {
case ARG_USE_COTHREADS:
osched->use_cothreads = g_value_get_boolean (value);
break;
case ARG_ITERATIONS:
osched->iterations = g_value_get_int (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}