| /* GStreamer |
| * Copyright (C) 1999,2000 Erik Walthinsen <omega@cse.ogi.edu> |
| * 2000 Wim Taymans <wtay@chello.be> |
| * |
| * gstscheduler.c: Default scheduling code for most cases |
| * |
| * 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., 59 Temple Place - Suite 330, |
| * Boston, MA 02111-1307, USA. |
| */ |
| |
| /*#define GST_DEBUG_ENABLED */ |
| #include <gst/gst.h> |
| |
| #include "cothreads_compat.h" |
| |
| |
| #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_PAD_BUFLIST(pad) ((GList*) (GST_REAL_PAD_CAST(pad)->sched_private)) |
| |
| #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_INTERRUPTED GST_ELEMENT_SCHEDULER_PRIVATE2 |
| #define GST_ELEMENT_IS_INTERRUPTED(element) GST_FLAG_IS_SET((element), GST_ELEMENT_INTERRUPTED) |
| |
| typedef struct _GstOptScheduler GstOptScheduler; |
| typedef struct _GstOptSchedulerClass GstOptSchedulerClass; |
| |
| #define GST_TYPE_OPT_SCHEDULER \ |
| (gst_opt_scheduler_get_type()) |
| #define GST_OPT_SCHEDULER(obj) \ |
| (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_OPT_SCHEDULER,GstOptScheduler)) |
| #define GST_OPT_SCHEDULER_CLASS(klass) \ |
| (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_OPT_SCHEDULER,GstOptSchedulerClass)) |
| #define GST_IS_OPT_SCHEDULER(obj) \ |
| (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_OPT_SCHEDULER)) |
| #define GST_IS_OPT_SCHEDULER_CLASS(obj) \ |
| (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_OPT_SCHEDULER)) |
| |
| #define GST_OPT_SCHEDULER_CAST(sched) ((GstOptScheduler *)(sched)) |
| |
| typedef enum { |
| GST_OPT_SCHEDULER_STATE_NONE, |
| GST_OPT_SCHEDULER_STATE_STOPPED, |
| GST_OPT_SCHEDULER_STATE_ERROR, |
| GST_OPT_SCHEDULER_STATE_RUNNING, |
| } GstOptSchedulerState; |
| |
| struct _GstOptScheduler { |
| GstScheduler parent; |
| |
| GstOptSchedulerState state; |
| |
| cothread_context *context; |
| gboolean use_cothreads; |
| gint iterations; |
| |
| GSList *elements; |
| GSList *chains; |
| |
| GList *runqueue; |
| gint recursion; |
| }; |
| |
| struct _GstOptSchedulerClass { |
| GstSchedulerClass parent_class; |
| }; |
| |
| static GType _gst_opt_scheduler_type = 0; |
| |
| typedef enum { |
| GST_OPT_SCHEDULER_CHAIN_DIRTY = (1 << 1), |
| GST_OPT_SCHEDULER_CHAIN_DISABLED = (1 << 2), |
| GST_OPT_SCHEDULER_CHAIN_RUNNING = (1 << 3), |
| } GstOptSchedulerChainFlags; |
| |
| #define GST_OPT_SCHEDULER_CHAIN_DISABLE(chain) ((chain)->flags |= GST_OPT_SCHEDULER_CHAIN_DISABLED) |
| #define GST_OPT_SCHEDULER_CHAIN_ENABLE(chain) ((chain)->flags &= ~GST_OPT_SCHEDULER_CHAIN_DISABLED) |
| #define GST_OPT_SCHEDULER_CHAIN_IS_DISABLED(chain) ((chain)->flags & GST_OPT_SCHEDULER_CHAIN_DISABLED) |
| |
| typedef struct _GstOptSchedulerChain GstOptSchedulerChain; |
| |
| struct _GstOptSchedulerChain { |
| GstOptScheduler *sched; |
| |
| GstOptSchedulerChainFlags flags; |
| |
| GSList *groups; /* the groups in this chain */ |
| gint num_groups; |
| gint num_enabled; |
| }; |
| |
| /* |
| * elements that are scheduled in one cothread |
| */ |
| typedef enum { |
| GST_OPT_SCHEDULER_GROUP_DIRTY = (1 << 1), /* this group has been modified */ |
| GST_OPT_SCHEDULER_GROUP_COTHREAD_STOPPING = (1 << 2), /* the group's cothread stops after one iteration */ |
| GST_OPT_SCHEDULER_GROUP_DISABLED = (1 << 3), /* this group is disabled */ |
| GST_OPT_SCHEDULER_GROUP_RUNNING = (1 << 4), /* this group is running */ |
| GST_OPT_SCHEDULER_GROUP_SCHEDULABLE = (1 << 5), /* this group is schedulable */ |
| } GstOptSchedulerGroupFlags; |
| |
| typedef enum { |
| GST_OPT_SCHEDULER_GROUP_GET = 1, |
| GST_OPT_SCHEDULER_GROUP_LOOP = 2, |
| } GstOptSchedulerGroupType; |
| |
| #define GST_OPT_SCHEDULER_GROUP_DISABLE(group) ((group)->flags |= GST_OPT_SCHEDULER_GROUP_DISABLED) |
| #define GST_OPT_SCHEDULER_GROUP_ENABLE(group) ((group)->flags &= ~GST_OPT_SCHEDULER_GROUP_DISABLED) |
| #define GST_OPT_SCHEDULER_GROUP_IS_ENABLED(group) (!((group)->flags & GST_OPT_SCHEDULER_GROUP_DISABLED)) |
| #define GST_OPT_SCHEDULER_GROUP_IS_DISABLED(group) ((group)->flags & GST_OPT_SCHEDULER_GROUP_DISABLED) |
| |
| typedef struct _GstOptSchedulerGroup GstOptSchedulerGroup; |
| |
| typedef int (*GroupScheduleFunction) (int argc, char *argv[]); |
| |
| struct _GstOptSchedulerGroup { |
| GstOptSchedulerChain *chain; /* the chain this group belongs to */ |
| GstOptSchedulerGroupFlags flags; /* flags for this group */ |
| GstOptSchedulerGroupType type; /* flags for this group */ |
| |
| GSList *elements; /* elements of this group */ |
| gint num_elements; |
| gint num_enabled; |
| GstElement *entry; /* the group's entry point */ |
| |
| GSList *providers; /* other groups that provide data |
| for this group */ |
| |
| cothread *cothread; /* the cothread of this group */ |
| GroupScheduleFunction schedulefunc; |
| int argc; |
| char **argv; |
| }; |
| |
| /* |
| * Scheduler private data for an element |
| */ |
| typedef struct _GstOptSchedulerCtx GstOptSchedulerCtx; |
| |
| typedef enum { |
| GST_OPT_SCHEDULER_CTX_DISABLED = (1 << 1), /* the element is disabled */ |
| } GstOptSchedulerCtxFlags; |
| |
| struct _GstOptSchedulerCtx { |
| GstOptSchedulerGroup *group; /* the group this element belongs to */ |
| |
| GstOptSchedulerCtxFlags flags; /* flags for this element */ |
| 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_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_setup (GstScheduler *sched); |
| static void gst_opt_scheduler_reset (GstScheduler *sched); |
| static void gst_opt_scheduler_add_element (GstScheduler *sched, GstElement *element); |
| static void gst_opt_scheduler_remove_element (GstScheduler *sched, GstElement *element); |
| static GstElementStateReturn |
| gst_opt_scheduler_state_transition (GstScheduler *sched, GstElement *element, gint transition); |
| static void gst_opt_scheduler_lock_element (GstScheduler *sched, GstElement *element); |
| static void gst_opt_scheduler_unlock_element (GstScheduler *sched, GstElement *element); |
| static void gst_opt_scheduler_yield (GstScheduler *sched, GstElement *element); |
| static gboolean gst_opt_scheduler_interrupt (GstScheduler *sched, GstElement *element); |
| static void gst_opt_scheduler_error (GstScheduler *sched, GstElement *element); |
| static void gst_opt_scheduler_pad_connect (GstScheduler *sched, GstPad *srcpad, GstPad *sinkpad); |
| static void gst_opt_scheduler_pad_disconnect (GstScheduler *sched, GstPad *srcpad, GstPad *sinkpad); |
| static GstPad* gst_opt_scheduler_pad_select (GstScheduler *sched, GList *padlist); |
| static GstClockReturn gst_opt_scheduler_clock_wait (GstScheduler *sched, GstElement *element, |
| GstClock *clock, GstClockTime time, GstClockTimeDiff *jitter); |
| static GstSchedulerState |
| gst_opt_scheduler_iterate (GstScheduler *sched); |
| |
| static void gst_opt_scheduler_show (GstScheduler *sched); |
| |
| static GstSchedulerClass *parent_class = NULL; |
| |
| static GType |
| gst_opt_scheduler_get_type (void) |
| { |
| if (!_gst_opt_scheduler_type) { |
| static const GTypeInfo scheduler_info = { |
| sizeof (GstOptSchedulerClass), |
| NULL, |
| NULL, |
| (GClassInitFunc) gst_opt_scheduler_class_init, |
| NULL, |
| NULL, |
| sizeof (GstOptScheduler), |
| 0, |
| (GInstanceInitFunc) gst_opt_scheduler_init, |
| NULL |
| }; |
| |
| _gst_opt_scheduler_type = g_type_register_static (GST_TYPE_SCHEDULER, |
| "GstOpt"COTHREADS_NAME_CAPITAL"Scheduler", &scheduler_info, 0); |
| } |
| return _gst_opt_scheduler_type; |
| } |
| |
| static void |
| gst_opt_scheduler_class_init (GstOptSchedulerClass *klass) |
| { |
| GObjectClass *gobject_class; |
| GstObjectClass *gstobject_class; |
| GstSchedulerClass *gstscheduler_class; |
| |
| gobject_class = (GObjectClass*)klass; |
| gstobject_class = (GstObjectClass*)klass; |
| gstscheduler_class = (GstSchedulerClass*)klass; |
| |
| 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); |
| |
| 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->reset = GST_DEBUG_FUNCPTR (gst_opt_scheduler_reset); |
| gstscheduler_class->add_element = GST_DEBUG_FUNCPTR (gst_opt_scheduler_add_element); |
| gstscheduler_class->remove_element = GST_DEBUG_FUNCPTR (gst_opt_scheduler_remove_element); |
| gstscheduler_class->state_transition = GST_DEBUG_FUNCPTR (gst_opt_scheduler_state_transition); |
| gstscheduler_class->lock_element = GST_DEBUG_FUNCPTR (gst_opt_scheduler_lock_element); |
| gstscheduler_class->unlock_element = GST_DEBUG_FUNCPTR (gst_opt_scheduler_unlock_element); |
| gstscheduler_class->yield = GST_DEBUG_FUNCPTR (gst_opt_scheduler_yield); |
| gstscheduler_class->interrupt = GST_DEBUG_FUNCPTR (gst_opt_scheduler_interrupt); |
| gstscheduler_class->error = GST_DEBUG_FUNCPTR (gst_opt_scheduler_error); |
| gstscheduler_class->pad_connect = GST_DEBUG_FUNCPTR (gst_opt_scheduler_pad_connect); |
| gstscheduler_class->pad_disconnect = GST_DEBUG_FUNCPTR (gst_opt_scheduler_pad_disconnect); |
| gstscheduler_class->pad_select = GST_DEBUG_FUNCPTR (gst_opt_scheduler_pad_select); |
| gstscheduler_class->clock_wait = GST_DEBUG_FUNCPTR (gst_opt_scheduler_clock_wait); |
| gstscheduler_class->iterate = GST_DEBUG_FUNCPTR (gst_opt_scheduler_iterate); |
| gstscheduler_class->show = GST_DEBUG_FUNCPTR (gst_opt_scheduler_show); |
| } |
| |
| static void |
| gst_opt_scheduler_init (GstOptScheduler *scheduler) |
| { |
| scheduler->elements = NULL; |
| //scheduler->use_cothreads = FALSE; |
| scheduler->use_cothreads = TRUE; |
| scheduler->iterations = 1; |
| } |
| |
| static void |
| gst_opt_scheduler_dispose (GObject *object) |
| { |
| G_OBJECT_CLASS (parent_class)->dispose (object); |
| } |
| |
| static gboolean |
| plugin_init (GModule *module, GstPlugin *plugin) |
| { |
| GstSchedulerFactory *factory; |
| |
| gst_plugin_set_longname (plugin, "An optimal scheduler"); |
| |
| factory = gst_scheduler_factory_new ("opt"COTHREADS_NAME, |
| "An optimal scheduler using "COTHREADS_NAME" cothreads", |
| gst_opt_scheduler_get_type()); |
| |
| if (factory != NULL) { |
| gst_plugin_add_feature (plugin, GST_PLUGIN_FEATURE (factory)); |
| } |
| else { |
| g_warning ("could not register scheduler: optimal"); |
| } |
| return TRUE; |
| } |
| |
| GstPluginDesc plugin_desc = { |
| GST_VERSION_MAJOR, |
| GST_VERSION_MINOR, |
| "gstopt"COTHREADS_NAME"scheduler", |
| plugin_init |
| }; |
| |
| |
| static void |
| delete_chain (GstOptScheduler *osched, GstOptSchedulerChain *chain) |
| { |
| GSList *groups; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "delete chain %p", chain); |
| |
| g_assert (chain->sched == osched); |
| |
| osched->chains = g_slist_remove (osched->chains, chain); |
| |
| groups = chain->groups; |
| while (groups) { |
| GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) groups->data; |
| |
| group->chain = NULL; |
| |
| groups = g_slist_next (groups); |
| } |
| |
| g_slist_free (chain->groups); |
| g_free (chain); |
| } |
| |
| static void |
| add_to_chain (GstOptSchedulerChain *chain, GstOptSchedulerGroup *group) |
| { |
| GST_INFO (GST_CAT_SCHEDULING, "adding group %p to chain %p", group, chain); |
| |
| g_assert (group->chain == NULL); |
| |
| chain->groups = g_slist_prepend (chain->groups, group); |
| group->chain = chain; |
| chain->num_groups++; |
| } |
| |
| static GstOptSchedulerChain* |
| create_chain (GstOptScheduler *osched) |
| { |
| GstOptSchedulerChain *chain; |
| |
| chain = g_new0 (GstOptSchedulerChain, 1); |
| chain->sched = osched; |
| |
| osched->chains = g_slist_prepend (osched->chains, chain); |
| |
| GST_INFO (GST_CAT_SCHEDULING, "new chain %p", chain); |
| |
| return chain; |
| } |
| |
| static void |
| remove_from_chain (GstOptSchedulerChain *chain, GstOptSchedulerGroup *group) |
| { |
| GST_INFO (GST_CAT_SCHEDULING, "removing group %p from chain %p", group, chain); |
| |
| g_assert (group->chain == chain); |
| |
| chain->groups = g_slist_remove (chain->groups, group); |
| chain->num_groups--; |
| |
| group->chain = NULL; |
| } |
| |
| static void |
| merge_chains (GstOptSchedulerChain *chain1, GstOptSchedulerChain *chain2) |
| { |
| GSList *walk; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "mergin chain %p and %p", chain1, chain2); |
| |
| if (chain1 == chain2) |
| return; |
| |
| walk = chain2->groups; |
| while (walk) { |
| GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) walk->data; |
| |
| group->chain = NULL; |
| add_to_chain (chain1, group); |
| walk = g_slist_next (walk); |
| } |
| delete_chain (chain2->sched, chain2); |
| } |
| |
| static void |
| chain_group_set_enabled (GstOptSchedulerChain *chain, GstOptSchedulerGroup *group, gboolean enabled) |
| { |
| if (enabled) { |
| chain->num_enabled++; |
| GST_INFO (GST_CAT_SCHEDULING, "enable group %p in chain %p, now %d groups enabled out of %d", group, chain, |
| chain->num_enabled, chain->num_groups); |
| if (chain->num_enabled == chain->num_groups) { |
| GST_INFO (GST_CAT_SCHEDULING, "enable chain %p", chain); |
| GST_OPT_SCHEDULER_CHAIN_ENABLE (chain); |
| } |
| } |
| else { |
| chain->num_enabled--; |
| GST_INFO (GST_CAT_SCHEDULING, "disable group %p in chain %p, now %d groups enabled out of %d", group, chain, |
| chain->num_enabled, chain->num_groups); |
| if (chain->num_enabled == 0) { |
| GST_INFO (GST_CAT_SCHEDULING, "disable chain %p", chain); |
| GST_OPT_SCHEDULER_CHAIN_DISABLE (chain); |
| } |
| } |
| } |
| |
| static void |
| add_to_group (GstOptSchedulerGroup *group, GstElement *element) |
| { |
| GST_INFO (GST_CAT_SCHEDULING, "adding element \"%s\" to group %p", GST_ELEMENT_NAME (element), group); |
| |
| if (GST_ELEMENT_IS_DECOUPLED (element)) { |
| GST_INFO (GST_CAT_SCHEDULING, "element \"%s\" is decoupled, not adding to group %p", GST_ELEMENT_NAME (element), group); |
| return; |
| } |
| |
| g_assert (GST_ELEMENT_SCHED_GROUP (element) == NULL); |
| |
| group->elements = g_slist_prepend (group->elements, element); |
| group->num_elements++; |
| |
| GST_ELEMENT_SCHED_GROUP (element) = group; |
| } |
| |
| static GstOptSchedulerGroup* |
| create_group (GstOptSchedulerChain *chain, GstElement *element) |
| { |
| GstOptSchedulerGroup *group; |
| |
| group = g_new0 (GstOptSchedulerGroup, 1); |
| GST_INFO (GST_CAT_SCHEDULING, "new group %p", group); |
| |
| add_to_group (group, element); |
| add_to_chain (chain, group); |
| |
| return group; |
| } |
| |
| static void |
| destroy_group_scheduler (GstOptSchedulerGroup *group) |
| { |
| if (group->flags & GST_OPT_SCHEDULER_GROUP_RUNNING) |
| g_warning ("removing running element"); |
| |
| if (group->cothread) { |
| do_cothread_destroy (group->cothread); |
| } |
| else { |
| group->schedulefunc = NULL; |
| group->argc = 0; |
| group->argv = NULL; |
| } |
| |
| group->flags &= ~GST_OPT_SCHEDULER_GROUP_SCHEDULABLE; |
| } |
| |
| static void |
| delete_group (GstOptSchedulerGroup *group) |
| { |
| GSList *elements; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "delete group %p", group); |
| |
| g_assert (group->chain == NULL); |
| |
| if (group->flags & GST_OPT_SCHEDULER_GROUP_SCHEDULABLE) |
| destroy_group_scheduler (group); |
| |
| /* remove all elements from the group */ |
| elements = group->elements; |
| while (elements) { |
| GstElement *element = GST_ELEMENT (elements->data); |
| |
| GST_ELEMENT_SCHED_GROUP (element) = NULL; |
| |
| elements = g_slist_next (elements); |
| } |
| |
| g_slist_free (group->elements); |
| g_free (group); |
| } |
| |
| static void |
| merge_groups (GstOptSchedulerGroup *group1, GstOptSchedulerGroup *group2) |
| { |
| GSList *walk; |
| GstOptSchedulerChain *chain1; |
| GstOptSchedulerChain *chain2; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "merging groups %p and %p", group1, group2); |
| |
| if (group1 == group2) |
| return; |
| |
| walk = group2->elements; |
| while (walk) { |
| add_to_group (group1, (GstElement *)walk->data); |
| walk = g_slist_next (walk); |
| } |
| |
| chain1 = group1->chain; |
| chain2 = group2->chain; |
| |
| remove_from_chain (chain2, group2); |
| delete_group (group2); |
| |
| merge_chains (chain1, chain2); |
| } |
| |
| /* |
| static void |
| remove_from_group (GstOptSchedulerGroup *group, GstElement *element) |
| { |
| GST_INFO (GST_CAT_SCHEDULING, "removing element \"%s\" to group %p", GST_ELEMENT_NAME (element), group); |
| |
| group->elements = g_slist_remove (group->elements, element); |
| group->num_elements--; |
| |
| GST_ELEMENT_SCHED_GROUP (element) = NULL; |
| } |
| */ |
| |
| /* 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 |
| group_element_set_enabled (GstOptSchedulerGroup *group, GstElement *element, gboolean enabled) |
| { |
| if (enabled) { |
| group->num_enabled++; |
| GST_INFO (GST_CAT_SCHEDULING, "enable element %s in group %p, now %d elements enabled out of %d", |
| GST_ELEMENT_NAME (element), group, group->num_enabled, group->num_elements); |
| if (group->num_enabled == group->num_elements) { |
| GST_INFO (GST_CAT_SCHEDULING, "enable group %p", group); |
| GST_OPT_SCHEDULER_GROUP_ENABLE (group); |
| chain_group_set_enabled (group->chain, group, TRUE); |
| } |
| } |
| else { |
| group->num_enabled--; |
| GST_INFO (GST_CAT_SCHEDULING, "disable element %s in group %p, now %d elements enabled out of %d", |
| GST_ELEMENT_NAME (element), group, group->num_enabled, group->num_elements); |
| if (group->num_enabled == 0) { |
| GST_INFO (GST_CAT_SCHEDULING, "disable group %p", group); |
| GST_OPT_SCHEDULER_GROUP_DISABLE (group); |
| chain_group_set_enabled (group->chain, group, FALSE); |
| } |
| } |
| } |
| |
| /* 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) |
| { |
| if (group->chain->sched->use_cothreads) { |
| if (group->cothread) |
| do_cothread_switch (group->cothread); |
| return TRUE; |
| } |
| else { |
| group->schedulefunc (group->argc, group->argv); |
| return TRUE; |
| } |
| 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 |
| schedule_chain (GstOptSchedulerChain *chain) |
| { |
| GSList *groups = chain->groups; |
| |
| while (groups) { |
| GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) groups->data; |
| |
| groups = g_slist_next (groups); |
| |
| if (!GST_OPT_SCHEDULER_GROUP_IS_DISABLED (group)) { |
| GstOptScheduler *osched; |
| |
| osched = chain->sched; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "scheduling group %p in chain %p", |
| group, chain); |
| |
| 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; |
| } |
| } |
| } |
| |
| /* a get-based group is scheduled by getting a buffer from the get based |
| * entry point and by pushing the buffer to the peer. |
| * We also set the running flag on this group for as long as this |
| * function is running. */ |
| static int |
| get_group_schedule_function (int argc, char *argv[]) |
| { |
| GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv; |
| const GList *pads = gst_element_get_pad_list (group->entry); |
| |
| GST_INFO (GST_CAT_SCHEDULING, "get wrapper of group %p", group); |
| |
| group->flags |= GST_OPT_SCHEDULER_GROUP_RUNNING; |
| |
| while (pads) { |
| GstBuffer *buffer; |
| GstPad *pad = GST_PAD_CAST (pads->data); |
| pads = g_list_next (pads); |
| |
| /* skip sinks and ghostpads */ |
| if (!GST_PAD_IS_SRC (pad) || !GST_IS_REAL_PAD (pad)) |
| continue; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "doing get and push on pad \"%s:%s\" in group %p", |
| GST_DEBUG_PAD_NAME (pad), group); |
| |
| buffer = GST_RPAD_GETFUNC (pad) (pad); |
| if (buffer) |
| gst_pad_push (pad, buffer); |
| } |
| |
| group->flags &= ~GST_OPT_SCHEDULER_GROUP_RUNNING; |
| |
| 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 |
| loop_group_schedule_function (int argc, char *argv[]) |
| { |
| GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv; |
| GstElement *entry = group->entry; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "loop wrapper of group %p", group); |
| |
| group->flags |= GST_OPT_SCHEDULER_GROUP_RUNNING; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "calling loopfunc of element %s in group %p", |
| GST_ELEMENT_NAME (entry), group); |
| |
| entry->loopfunc (entry); |
| |
| group->flags &= ~GST_OPT_SCHEDULER_GROUP_RUNNING; |
| |
| return 0; |
| |
| } |
| |
| /* the function to schedule an unkown group, which just gives an error */ |
| static int |
| unkown_group_schedule_function (int argc, char *argv[]) |
| { |
| GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) argv; |
| |
| g_warning ("(internal error) unkown group type %d, disabling\n", group->type); |
| chain_group_set_enabled (group->chain, group, FALSE); |
| group->chain->sched->state = GST_OPT_SCHEDULER_STATE_ERROR; |
| |
| 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 |
| gst_opt_scheduler_loop_wrapper (GstPad *sinkpad, GstBuffer *buffer) |
| { |
| GstOptSchedulerGroup *group; |
| GstOptScheduler *osched; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "loop wrapper, putting buffer in bufpen"); |
| |
| group = GST_ELEMENT_SCHED_GROUP (GST_PAD_PARENT (sinkpad)); |
| osched = group->chain->sched; |
| |
| |
| if (osched->use_cothreads) { |
| if (GST_PAD_BUFLIST (GST_RPAD_PEER (sinkpad))) { |
| g_warning ("deadlock detected, disabling group %p", group); |
| 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_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); |
| } |
| |
| 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* |
| gst_opt_scheduler_get_wrapper (GstPad *srcpad) |
| { |
| GstBuffer *buffer = NULL; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "get wrapper, removing buffer from bufpen"); |
| |
| if (GST_PAD_BUFLIST (srcpad)) |
| buffer = GST_PAD_BUFLIST (srcpad)->data; |
| |
| while (!buffer) { |
| GstOptSchedulerGroup *group; |
| GstOptScheduler *osched; |
| |
| group = GST_ELEMENT_SCHED_GROUP (GST_PAD_PARENT (srcpad)); |
| osched = group->chain->sched; |
| |
| 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); |
| chain_group_set_enabled (group->chain, group, FALSE); |
| group->chain->sched->state = GST_OPT_SCHEDULER_STATE_ERROR; |
| return NULL; |
| } |
| |
| 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_INFO (GST_CAT_SCHEDULING, "get wrapper, returning buffer %d", |
| g_list_length (GST_PAD_BUFLIST (srcpad))); |
| |
| 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 |
| gst_opt_scheduler_chain_wrapper (GstPad *sinkpad, GstBuffer *buffer) |
| { |
| if (GST_IS_EVENT (buffer)) { |
| gst_pad_send_event (sinkpad, GST_EVENT (buffer)); |
| } |
| else { |
| GST_RPAD_CHAINFUNC (sinkpad) (sinkpad, 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 |
| setup_group_scheduler (GstOptScheduler *osched, GstOptSchedulerGroup *group) |
| { |
| GroupScheduleFunction wrapper; |
| |
| wrapper = unkown_group_schedule_function; |
| |
| /* figure out the wrapper function for this group */ |
| if (group->type == GST_OPT_SCHEDULER_GROUP_GET) |
| wrapper = get_group_schedule_function; |
| else if (group->type == GST_OPT_SCHEDULER_GROUP_LOOP) |
| wrapper = loop_group_schedule_function; |
| |
| if (osched->use_cothreads) { |
| if (!(group->flags & GST_OPT_SCHEDULER_GROUP_SCHEDULABLE)) { |
| do_cothread_create (group->cothread, osched->context, |
| (cothread_func) wrapper, 0, (char **) group); |
| } |
| else { |
| do_cothread_setfunc (group->cothread, osched->context, |
| (cothread_func) wrapper, 0, (char **) group); |
| } |
| } |
| else { |
| group->schedulefunc = wrapper; |
| group->argc = 0; |
| group->argv = (char **) group; |
| } |
| group->flags |= GST_OPT_SCHEDULER_GROUP_SCHEDULABLE; |
| } |
| |
| static GstElementStateReturn |
| gst_opt_scheduler_state_transition (GstScheduler *sched, GstElement *element, gint transition) |
| { |
| GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); |
| GstOptSchedulerGroup *group; |
| GstElementStateReturn res = GST_STATE_SUCCESS; |
| |
| 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_SCHEDULER_PARENT (sched) == element) { |
| GST_INFO (GST_CAT_SCHEDULING, "parent \"%s\" changed state", GST_ELEMENT_NAME (element)); |
| |
| switch (transition) { |
| case GST_STATE_PLAYING_TO_PAUSED: |
| GST_INFO (GST_CAT_SCHEDULING, "setting scheduler state to stopped"); |
| GST_SCHEDULER_STATE (sched) = GST_SCHEDULER_STATE_STOPPED; |
| break; |
| case GST_STATE_PAUSED_TO_PLAYING: |
| GST_INFO (GST_CAT_SCHEDULING, "setting scheduler state to running"); |
| GST_SCHEDULER_STATE (sched) = GST_SCHEDULER_STATE_RUNNING; |
| break; |
| default: |
| GST_INFO (GST_CAT_SCHEDULING, "no interesting state change, doing nothing"); |
| } |
| } |
| return res; |
| } |
| |
| /* we don't care about decoupled elements after this */ |
| if (GST_ELEMENT_IS_DECOUPLED (element)) |
| return GST_STATE_SUCCESS; |
| |
| /* get the group of the element */ |
| group = GST_ELEMENT_SCHED_GROUP (element); |
| |
| switch (transition) { |
| case GST_STATE_PAUSED_TO_PLAYING: |
| /* an element withut a group has to be an unconnected src, sink |
| * 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; |
| case GST_STATE_PLAYING_TO_PAUSED: |
| /* if the element still has a group, we disable it */ |
| if (group) |
| group_element_set_enabled (group, element, FALSE); |
| break; |
| default: |
| break; |
| } |
| |
| return res; |
| } |
| |
| static void |
| get_group (GstElement *element, GstOptSchedulerGroup **group) |
| { |
| GstOptSchedulerCtx *ctx; |
| |
| ctx = GST_ELEMENT_SCHED_CONTEXT (element); |
| if (ctx) |
| *group = ctx->group; |
| else |
| *group = NULL; |
| } |
| |
| /* |
| * the idea is to put the two elements into the same group. |
| * - When no element is inside a group, we create a new group and add |
| * the elements to it. |
| * - When one of the elements has a group, add the other element to |
| * that group |
| * - if both of the elements have a group, we merge the groups, which |
| * will also merge the chains. |
| */ |
| static GstOptSchedulerGroup* |
| group_elements (GstOptScheduler *osched, GstElement *element1, GstElement *element2) |
| { |
| GstOptSchedulerGroup *group1, *group2, *group = NULL; |
| |
| get_group (element1, &group1); |
| get_group (element2, &group2); |
| |
| /* none of the elements is added to a group, create a new group |
| * and chain to add the elements to */ |
| if (!group1 && !group2) { |
| GstOptSchedulerChain *chain; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "creating new group to hold \"%s\" and \"%s\"", |
| GST_ELEMENT_NAME (element1), GST_ELEMENT_NAME (element2)); |
| |
| chain = create_chain (osched); |
| group = create_group (chain, element1); |
| add_to_group (group, element2); |
| } |
| /* the first element has a group */ |
| else if (group1) { |
| GST_INFO (GST_CAT_SCHEDULING, "adding \"%s\" to \"%s\"'s group", |
| GST_ELEMENT_NAME (element2), GST_ELEMENT_NAME (element1)); |
| |
| /* the second element also has a group, merge */ |
| if (group2) |
| merge_groups (group1, group2); |
| /* the second element has no group, add it to the group |
| * of the first element */ |
| else |
| add_to_group (group1, element2); |
| |
| group = group1; |
| } |
| /* element1 has no group, element2 does. Add element1 to the |
| * group of element2 */ |
| else { |
| GST_INFO (GST_CAT_SCHEDULING, "adding \"%s\" to \"%s\"'s group", |
| GST_ELEMENT_NAME (element1), GST_ELEMENT_NAME (element2)); |
| add_to_group (group2, element1); |
| group = group2; |
| } |
| return group; |
| } |
| |
| typedef enum { |
| GST_OPT_INVALID, |
| GST_OPT_GET_TO_CHAIN, |
| GST_OPT_LOOP_TO_CHAIN, |
| GST_OPT_GET_TO_LOOP, |
| GST_OPT_CHAIN_TO_CHAIN, |
| GST_OPT_CHAIN_TO_LOOP, |
| GST_OPT_LOOP_TO_LOOP, |
| } 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); |
| GstOptSchedulerGroup *group; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "removing element \"%s\" from scheduler", GST_ELEMENT_NAME (element)); |
| |
| /* decoupled elements are not added to the scheduler lists and should therefor |
| * no be removed */ |
| if (GST_ELEMENT_IS_DECOUPLED (element)) |
| return; |
| |
| /* the element is guaranteed to live in it's own group/chain now */ |
| get_group (element, &group); |
| if (group) { |
| |
| GST_ELEMENT_SCHED_GROUP (element) = NULL; |
| |
| if (group->chain) { |
| GstOptSchedulerChain *chain; |
| |
| chain = group->chain; |
| |
| remove_from_chain (chain, group); |
| delete_chain (osched, chain); |
| } |
| |
| delete_group (group); |
| } |
| |
| g_free (GST_ELEMENT_SCHED_CONTEXT (element)); |
| GST_ELEMENT_SCHED_CONTEXT (element) = NULL; |
| } |
| |
| 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 |
| gst_opt_scheduler_pad_connect (GstScheduler *sched, GstPad *srcpad, GstPad *sinkpad) |
| { |
| GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); |
| ConnectionType type = GST_OPT_INVALID; |
| GstElement *element1, *element2; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "pad connect between \"%s:%s\" and \"%s:%s\"", |
| GST_DEBUG_PAD_NAME (srcpad), GST_DEBUG_PAD_NAME (sinkpad)); |
| |
| element1 = GST_PAD_PARENT (srcpad); |
| element2 = GST_PAD_PARENT (sinkpad); |
| |
| /* first we need to figure out whar type of connection we're dealing |
| * with */ |
| if (element1->loopfunc && element2->loopfunc) |
| type = GST_OPT_LOOP_TO_LOOP; |
| else { |
| if (element1->loopfunc) { |
| if (GST_RPAD_CHAINFUNC (sinkpad)) |
| type = GST_OPT_LOOP_TO_CHAIN; |
| } |
| else if (element2->loopfunc) { |
| if (GST_RPAD_GETFUNC (srcpad)) { |
| type = GST_OPT_GET_TO_LOOP; |
| /* this could be tricky, the get based source could |
| * already be part of a loop based group in another pad, |
| * we assert on that for now */ |
| if (GST_ELEMENT_SCHED_CONTEXT (element1) && |
| GST_ELEMENT_SCHED_GROUP (element1) != NULL) |
| { |
| g_warning ("internal error: cannot schedule get to loop with get in group"); |
| return; |
| } |
| } |
| else |
| type = GST_OPT_CHAIN_TO_LOOP; |
| } |
| else { |
| if (GST_RPAD_GETFUNC (srcpad) && GST_RPAD_CHAINFUNC (sinkpad)) { |
| type = GST_OPT_GET_TO_CHAIN; |
| /* the get based source could already be part of a loop |
| * based group in another pad, |
| * we assert on that for now */ |
| if (GST_ELEMENT_SCHED_CONTEXT (element1) && |
| GST_ELEMENT_SCHED_GROUP (element1) != NULL) |
| { |
| g_warning ("internal error: cannot schedule get to loop with get in group"); |
| return; |
| } |
| } |
| else |
| type = GST_OPT_CHAIN_TO_CHAIN; |
| } |
| } |
| |
| /* for each connection type, perform specific actions */ |
| switch (type) { |
| case GST_OPT_GET_TO_CHAIN: |
| { |
| GstOptSchedulerGroup *group = NULL; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "get to chain based connection"); |
| |
| /* setup get/chain handlers */ |
| GST_RPAD_GETHANDLER (srcpad) = GST_RPAD_GETFUNC (srcpad); |
| if (GST_ELEMENT_IS_EVENT_AWARE (element2)) |
| GST_RPAD_CHAINHANDLER (sinkpad) = GST_RPAD_CHAINFUNC (sinkpad); |
| else |
| GST_RPAD_CHAINHANDLER (sinkpad) = gst_opt_scheduler_chain_wrapper; |
| |
| /* the two elements should be put into the same group, |
| * this also means that they are in the same chain automatically */ |
| group = group_elements (osched, element1, element2); |
| |
| /* if there is not yet an entry in the group, select the source |
| * element as the entry point */ |
| if (!group->entry) { |
| group->entry = element1; |
| group->type = GST_OPT_SCHEDULER_GROUP_GET; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "setting \"%s\" as entry point of _get-based group %p", |
| GST_ELEMENT_NAME (element1), group); |
| } |
| break; |
| } |
| case GST_OPT_LOOP_TO_CHAIN: |
| case GST_OPT_CHAIN_TO_CHAIN: |
| GST_INFO (GST_CAT_SCHEDULING, "loop/chain to chain based connection"); |
| |
| if (GST_ELEMENT_IS_EVENT_AWARE (element2)) |
| GST_RPAD_CHAINHANDLER (sinkpad) = GST_RPAD_CHAINFUNC (sinkpad); |
| else |
| GST_RPAD_CHAINHANDLER (sinkpad) = gst_opt_scheduler_chain_wrapper; |
| |
| /* the two elements should be put into the same group, |
| * this also means that they are in the same chain automatically, |
| * in case of a loop-based element1, there will be a group for element1 and |
| * element2 will be added to it. */ |
| group_elements (osched, element1, element2); |
| break; |
| case GST_OPT_GET_TO_LOOP: |
| GST_INFO (GST_CAT_SCHEDULING, "get to loop based connection"); |
| |
| GST_RPAD_GETHANDLER (srcpad) = GST_RPAD_GETFUNC (srcpad); |
| |
| /* the two elements should be put into the same group, |
| * this also means that they are in the same chain automatically, |
| * element2 is loop-based so it already has a group where element1 |
| * will be added to */ |
| group_elements (osched, element1, element2); |
| break; |
| case GST_OPT_CHAIN_TO_LOOP: |
| case GST_OPT_LOOP_TO_LOOP: |
| { |
| GstOptSchedulerGroup *group1, *group2; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "chain/loop to loop based connection"); |
| |
| GST_RPAD_CHAINHANDLER (sinkpad) = gst_opt_scheduler_loop_wrapper; |
| GST_RPAD_GETHANDLER (srcpad) = gst_opt_scheduler_get_wrapper; |
| |
| group1 = GST_ELEMENT_SCHED_GROUP (element1); |
| group2 = GST_ELEMENT_SCHED_GROUP (element2); |
| /* group2 is guaranteed to exist as it contains a loop-based element. |
| * group1 only exists if element1 is connected to some other element */ |
| if (!group1) { |
| /* create a new group for element1 as it cannot be merged into another group |
| * here. we create the group in the same chain as the loop-based element. */ |
| GST_INFO (GST_CAT_SCHEDULING, "creating new group for element %s", GST_ELEMENT_NAME (element1)); |
| group1 = create_group (group2->chain, element1); |
| } |
| else { |
| /* both elements are already in a group, make sure they are added to |
| * the same chain */ |
| merge_chains (group1->chain, group2->chain); |
| } |
| break; |
| } |
| case GST_OPT_INVALID: |
| g_warning ("(internal error) invalid element connection"); |
| break; |
| } |
| } |
| |
| static gboolean |
| element_has_connection_with_group (GstElement *element, GstOptSchedulerGroup *group) |
| { |
| gboolean connected = FALSE; |
| const GList *pads; |
| |
| /* see if the element has no more connections to the peer group */ |
| pads = gst_element_get_pad_list (element); |
| while (pads && !connected) { |
| GstPad *pad = GST_PAD_CAST (pads->data); |
| pads = g_list_next (pads); |
| |
| /* we only operate on real pads */ |
| if (!GST_IS_REAL_PAD (pad)) |
| continue; |
| |
| if (GST_PAD_PEER (pad)) { |
| } |
| } |
| return connected; |
| } |
| |
| static void |
| gst_opt_scheduler_pad_disconnect (GstScheduler *sched, GstPad *srcpad, GstPad *sinkpad) |
| { |
| //GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); |
| GstElement *element1, *element2; |
| GstOptSchedulerGroup *group1, *group2; |
| gboolean still_connect; |
| |
| GST_INFO (GST_CAT_SCHEDULING, "pad disconnect between \"%s:%s\" and \"%s:%s\"", |
| GST_DEBUG_PAD_NAME (srcpad), GST_DEBUG_PAD_NAME (sinkpad)); |
| |
| element1 = GST_PAD_PARENT (srcpad); |
| element2 = GST_PAD_PARENT (sinkpad); |
| |
| get_group (element1, &group1); |
| get_group (element2, &group2); |
| |
| /* having no groups is pretty bad, this means that two decoupled |
| * elements were connected or something */ |
| if (!group1 && !group2) { |
| g_warning ("internal error: cannot disconnect pads"); |
| return; |
| } |
| |
| /* see if the group has to be broken up */ |
| if (group1) |
| still_connect = element_has_connection_with_group (element2, group1); |
| else |
| still_connect = element_has_connection_with_group (element1, group2); |
| |
| /* if there is still a connection, we don't need to break this group */ |
| if (still_connect) |
| return; |
| |
| /* if they are equal, they both are non zero */ |
| if (group1 == group2) { |
| g_warning ("pad disconnect: implement me"); |
| } |
| else if (group1) { |
| g_warning ("pad disconnect: implement me"); |
| } |
| else { |
| /* there was no group for element1, see if the element |
| * was an entry point for group2 */ |
| if (group2) { |
| if (group2->entry == element1) { |
| group2->entry = NULL; |
| } |
| } |
| } |
| } |
| |
| static GstPad* |
| gst_opt_scheduler_pad_select (GstScheduler *sched, GList *padlist) |
| { |
| //GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); |
| |
| g_warning ("pad select, implement me"); |
| |
| return NULL; |
| } |
| |
| static GstClockReturn |
| gst_opt_scheduler_clock_wait (GstScheduler *sched, GstElement *element, |
| GstClock *clock, GstClockTime time, GstClockTimeDiff *jitter) |
| { |
| GstClockID id; |
| |
| id = gst_clock_new_single_shot_id (clock, time); |
| |
| return gst_clock_id_wait (id, jitter); |
| } |
| |
| /* a scheduler iteration is done by looping and scheduling the active chains */ |
| static GstSchedulerState |
| gst_opt_scheduler_iterate (GstScheduler *sched) |
| { |
| GstSchedulerState state = GST_SCHEDULER_STATE_STOPPED; |
| GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); |
| gint iterations = osched->iterations; |
| |
| osched->state = GST_OPT_SCHEDULER_STATE_RUNNING; |
| |
| while (iterations) { |
| gboolean scheduled = FALSE; |
| GSList *chains; |
| |
| /* we have to schedule each of the scheduler chains now */ |
| chains = osched->chains; |
| 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; |
| } |
| } |
| |
| /* at this point it's possible that the scheduler state is |
| * in error, we then return an error */ |
| if (osched->state == GST_OPT_SCHEDULER_STATE_ERROR) { |
| state = GST_SCHEDULER_STATE_ERROR; |
| break; |
| } |
| else { |
| /* 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; |
| } |
| |
| |
| static void |
| gst_opt_scheduler_show (GstScheduler *sched) |
| { |
| GstOptScheduler *osched = GST_OPT_SCHEDULER_CAST (sched); |
| GSList *chains; |
| |
| chains = osched->chains; |
| while (chains) { |
| GstOptSchedulerChain *chain = (GstOptSchedulerChain *) chains->data; |
| GSList *groups = chain->groups; |
| chains = g_slist_next (chains); |
| |
| g_print ("+- chain %p: %d groups, %d enabled, flags %d\n", chain, chain->num_groups, chain->num_enabled, chain->flags); |
| |
| while (groups) { |
| GstOptSchedulerGroup *group = (GstOptSchedulerGroup *) groups->data; |
| GSList *elements = group->elements; |
| groups = g_slist_next (groups); |
| |
| g_print (" +- group %p: %d elements, %d enabled, flags %d, entry %s, %s\n", |
| group, group->num_elements, group->num_enabled, group->flags, |
| (group->entry ? GST_ELEMENT_NAME (group->entry): "(none)"), |
| (group->type == GST_OPT_SCHEDULER_GROUP_GET ? "get-based" : "loop-based") ); |
| |
| while (elements) { |
| GstElement *element = (GstElement *) elements->data; |
| elements = g_slist_next (elements); |
| |
| g_print (" +- element %s\n", GST_ELEMENT_NAME (element)); |
| } |
| } |
| } |
| } |
| |
| 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; |
| } |
| } |
| |