devicemonitor: Only fail start() if no provider at all could be started

Also refactor various internals of the monitor code:
  - Don't allow starting twice but just return directly when starting a
    second time.
  - Don't end up in an inconsistent state if call start() a second time
    while the monitor is starting up.
  - Remove complicated cookie code: it was not possible to add/remove
    filters while the monitor was started anyway so this was only useful
    in the very small time-window while starting the monitor or while
    getting the devices. Instead disallow adding/removing filters while
    the monitor is starting, and when getting devices work on a snapshot
    of providers/filters.

Fixes https://gitlab.freedesktop.org/gstreamer/gstreamer/-/issues/667

Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/1189>
This commit is contained in:
Sebastian Dröge 2021-10-19 13:39:55 +03:00 committed by GStreamer Marge Bot
parent 139bfc8aeb
commit 1912bcbcc4

View file

@ -108,10 +108,10 @@ struct _GstDeviceMonitorPrivate
GstBus *bus; GstBus *bus;
GPtrArray *providers; GPtrArray *providers;
guint cookie;
GPtrArray *filters; GPtrArray *filters;
GList *started_providers;
guint last_id; guint last_id;
GList *hidden; GList *hidden;
gboolean show_all; gboolean show_all;
@ -129,6 +129,9 @@ G_DEFINE_TYPE_WITH_PRIVATE (GstDeviceMonitor, gst_device_monitor,
static void gst_device_monitor_dispose (GObject * object); static void gst_device_monitor_dispose (GObject * object);
static guint gst_device_monitor_add_filter_unlocked (GstDeviceMonitor * monitor,
const gchar * classes, GstCaps * caps);
struct DeviceFilter struct DeviceFilter
{ {
guint id; guint id;
@ -137,11 +140,24 @@ struct DeviceFilter
GstCaps *caps; GstCaps *caps;
}; };
static struct DeviceFilter *
device_filter_copy (struct DeviceFilter *filter)
{
struct DeviceFilter *copy = g_slice_new0 (struct DeviceFilter);
copy->classesv = g_strdupv (filter->classesv);
copy->caps = filter->caps ? gst_caps_ref (filter->caps) : NULL;
return copy;
}
static void static void
device_filter_free (struct DeviceFilter *filter) device_filter_free (struct DeviceFilter *filter)
{ {
g_strfreev (filter->classesv); g_strfreev (filter->classesv);
gst_caps_unref (filter->caps);
if (filter->caps)
gst_caps_unref (filter->caps);
g_slice_free (struct DeviceFilter, filter); g_slice_free (struct DeviceFilter, filter);
} }
@ -357,9 +373,11 @@ gst_device_monitor_dispose (GObject * object)
GList * GList *
gst_device_monitor_get_devices (GstDeviceMonitor * monitor) gst_device_monitor_get_devices (GstDeviceMonitor * monitor)
{ {
GList *devices = NULL, *hidden = NULL; GQueue providers = G_QUEUE_INIT, filters = G_QUEUE_INIT;
GList *hidden = NULL;
GQueue devices = G_QUEUE_INIT;
GList *l;
guint i; guint i;
guint cookie;
g_return_val_if_fail (GST_IS_DEVICE_MONITOR (monitor), NULL); g_return_val_if_fail (GST_IS_DEVICE_MONITOR (monitor), NULL);
@ -377,13 +395,6 @@ gst_device_monitor_get_devices (GstDeviceMonitor * monitor)
return NULL; return NULL;
} }
again:
g_list_free_full (devices, gst_object_unref);
g_list_free_full (hidden, g_free);
devices = NULL;
hidden = NULL;
for (i = 0; i < monitor->priv->providers->len; i++) { for (i = 0; i < monitor->priv->providers->len; i++) {
GstDeviceProvider *provider = GstDeviceProvider *provider =
g_ptr_array_index (monitor->priv->providers, i); g_ptr_array_index (monitor->priv->providers, i);
@ -391,37 +402,41 @@ again:
update_hidden_providers_list (&hidden, provider); update_hidden_providers_list (&hidden, provider);
} }
cookie = monitor->priv->cookie; /* Create a copy of all current providers and filters while keeping the lock
* and afterwards unlock and work with this snapshot */
for (i = 0; i < monitor->priv->providers->len; i++) { for (i = 0; i < monitor->priv->providers->len; i++) {
GList *tmpdev;
GstDeviceProvider *provider = GstDeviceProvider *provider =
gst_object_ref (g_ptr_array_index (monitor->priv->providers, i)); g_ptr_array_index (monitor->priv->providers, i);
GList *item;
if (!is_provider_hidden (monitor, hidden, provider)) { if (!is_provider_hidden (monitor, hidden, provider)) {
GST_OBJECT_UNLOCK (monitor); g_queue_push_tail (&providers, gst_object_ref (provider));
tmpdev = gst_device_provider_get_devices (provider);
GST_OBJECT_LOCK (monitor);
} else {
tmpdev = NULL;
} }
}
for (i = 0; i < monitor->priv->filters->len; i++) {
struct DeviceFilter *filter = g_ptr_array_index (monitor->priv->filters, i);
g_queue_push_tail (&filters, device_filter_copy (filter));
}
GST_OBJECT_UNLOCK (monitor);
for (l = providers.head; l; l = l->next) {
GstDeviceProvider *provider = l->data;
GList *tmpdev, *item, *filter_item;
tmpdev = gst_device_provider_get_devices (provider);
for (item = tmpdev; item; item = item->next) { for (item = tmpdev; item; item = item->next) {
GstDevice *dev = GST_DEVICE (item->data); GstDevice *dev = GST_DEVICE (item->data);
GstCaps *caps = gst_device_get_caps (dev); GstCaps *caps = gst_device_get_caps (dev);
guint j;
for (j = 0; j < monitor->priv->filters->len; j++) { for (filter_item = filters.head; filter_item;
struct DeviceFilter *filter = filter_item = filter_item->next) {
g_ptr_array_index (monitor->priv->filters, j); struct DeviceFilter *filter = filter_item->data;
if (gst_caps_can_intersect (filter->caps, caps) && if (gst_caps_can_intersect (filter->caps, caps) &&
gst_device_has_classesv (dev, filter->classesv)) { gst_device_has_classesv (dev, filter->classesv)) {
devices = g_list_prepend (devices, gst_object_ref (dev)); g_queue_push_tail (&devices, gst_object_ref (dev));
break; break;
} }
} }
@ -429,16 +444,13 @@ again:
} }
g_list_free_full (tmpdev, gst_object_unref); g_list_free_full (tmpdev, gst_object_unref);
gst_object_unref (provider);
if (monitor->priv->cookie != cookie)
goto again;
} }
g_list_free_full (hidden, g_free); g_list_free_full (hidden, g_free);
GST_OBJECT_UNLOCK (monitor); g_queue_clear_full (&providers, (GDestroyNotify) gst_object_unref);
g_queue_clear_full (&filters, (GDestroyNotify) device_filter_free);
return g_list_reverse (devices); return devices.head;
} }
/** /**
@ -449,7 +461,8 @@ again:
* %GST_MESSAGE_DEVICE_ADDED and %GST_MESSAGE_DEVICE_REMOVED messages * %GST_MESSAGE_DEVICE_ADDED and %GST_MESSAGE_DEVICE_REMOVED messages
* will be emitted on the bus when the list of devices changes. * will be emitted on the bus when the list of devices changes.
* *
* Returns: %TRUE if the device monitoring could be started * Returns: %TRUE if the device monitoring could be started, i.e. at least a
* single device provider was started successfully.
* *
* Since: 1.4 * Since: 1.4
*/ */
@ -457,19 +470,24 @@ again:
gboolean gboolean
gst_device_monitor_start (GstDeviceMonitor * monitor) gst_device_monitor_start (GstDeviceMonitor * monitor)
{ {
guint cookie, i; guint i;
GList *pending = NULL, *started = NULL, *removed = NULL; GQueue pending = G_QUEUE_INIT;
GList *started = NULL;
GstDeviceProvider *provider;
g_return_val_if_fail (GST_IS_DEVICE_MONITOR (monitor), FALSE); g_return_val_if_fail (GST_IS_DEVICE_MONITOR (monitor), FALSE);
GST_OBJECT_LOCK (monitor); GST_OBJECT_LOCK (monitor);
if (monitor->priv->filters->len == 0) { if (monitor->priv->started) {
GST_OBJECT_UNLOCK (monitor); GST_OBJECT_UNLOCK (monitor);
GST_DEBUG_OBJECT (monitor, "Monitor started already");
return TRUE;
}
if (monitor->priv->filters->len == 0) {
GST_WARNING_OBJECT (monitor, "No filters have been set, will expose all " GST_WARNING_OBJECT (monitor, "No filters have been set, will expose all "
"devices found"); "devices found");
gst_device_monitor_add_filter (monitor, NULL, NULL); gst_device_monitor_add_filter_unlocked (monitor, NULL, NULL);
GST_OBJECT_LOCK (monitor);
} }
if (monitor->priv->providers->len == 0) { if (monitor->priv->providers->len == 0) {
@ -478,74 +496,39 @@ gst_device_monitor_start (GstDeviceMonitor * monitor)
return FALSE; return FALSE;
} }
monitor->priv->started = TRUE;
gst_bus_set_flushing (monitor->priv->bus, FALSE); gst_bus_set_flushing (monitor->priv->bus, FALSE);
again:
cookie = monitor->priv->cookie;
g_list_free_full (pending, gst_object_unref);
pending = NULL;
removed = started;
started = NULL;
for (i = 0; i < monitor->priv->providers->len; i++) { for (i = 0; i < monitor->priv->providers->len; i++) {
GstDeviceProvider *provider; GstDeviceProvider *provider;
GList *find;
provider = g_ptr_array_index (monitor->priv->providers, i); provider = g_ptr_array_index (monitor->priv->providers, i);
g_queue_push_tail (&pending, gst_object_ref (provider));
find = g_list_find (removed, provider);
if (find) {
/* this was already started, move to started list */
removed = g_list_remove_link (removed, find);
started = g_list_concat (started, find);
} else {
/* not started, add to pending list */
pending = g_list_append (pending, gst_object_ref (provider));
}
} }
g_list_free_full (removed, gst_object_unref);
removed = NULL;
while (pending) {
GstDeviceProvider *provider = pending->data;
while ((provider = g_queue_pop_head (&pending))) {
GST_OBJECT_UNLOCK (monitor); GST_OBJECT_UNLOCK (monitor);
if (!gst_device_provider_start (provider)) if (gst_device_provider_start (provider)) {
goto start_failed; started = g_list_prepend (started, provider);
} else {
gst_object_unref (provider);
}
GST_OBJECT_LOCK (monitor); GST_OBJECT_LOCK (monitor);
started = g_list_prepend (started, provider);
pending = g_list_delete_link (pending, pending);
if (monitor->priv->cookie != cookie)
goto again;
} }
monitor->priv->started = TRUE;
if (started) {
monitor->priv->started_providers = started;
} else {
gst_bus_set_flushing (monitor->priv->bus, TRUE);
monitor->priv->started = FALSE;
}
GST_OBJECT_UNLOCK (monitor); GST_OBJECT_UNLOCK (monitor);
g_list_free_full (started, gst_object_unref); return started != NULL;
return TRUE;
start_failed:
{
GST_OBJECT_LOCK (monitor);
gst_bus_set_flushing (monitor->priv->bus, TRUE);
GST_OBJECT_UNLOCK (monitor);
while (started) {
GstDeviceProvider *provider = started->data;
gst_device_provider_stop (provider);
gst_object_unref (provider);
started = g_list_delete_link (started, started);
}
return FALSE;
}
} }
/** /**
@ -559,7 +542,6 @@ start_failed:
void void
gst_device_monitor_stop (GstDeviceMonitor * monitor) gst_device_monitor_stop (GstDeviceMonitor * monitor)
{ {
guint i;
GList *started = NULL; GList *started = NULL;
g_return_if_fail (GST_IS_DEVICE_MONITOR (monitor)); g_return_if_fail (GST_IS_DEVICE_MONITOR (monitor));
@ -567,13 +549,15 @@ gst_device_monitor_stop (GstDeviceMonitor * monitor)
gst_bus_set_flushing (monitor->priv->bus, TRUE); gst_bus_set_flushing (monitor->priv->bus, TRUE);
GST_OBJECT_LOCK (monitor); GST_OBJECT_LOCK (monitor);
for (i = 0; i < monitor->priv->providers->len; i++) { if (!monitor->priv->started) {
GstDeviceProvider *provider = GST_DEBUG_OBJECT (monitor, "Monitor was not started yet");
g_ptr_array_index (monitor->priv->providers, i); GST_OBJECT_UNLOCK (monitor);
return;
if (gst_device_provider_is_started (provider))
started = g_list_prepend (started, gst_object_ref (provider));
} }
started = monitor->priv->started_providers;
monitor->priv->started_providers = NULL;
monitor->priv->started = FALSE;
GST_OBJECT_UNLOCK (monitor); GST_OBJECT_UNLOCK (monitor);
while (started) { while (started) {
@ -584,11 +568,6 @@ gst_device_monitor_stop (GstDeviceMonitor * monitor)
started = g_list_delete_link (started, started); started = g_list_delete_link (started, started);
gst_object_unref (provider); gst_object_unref (provider);
} }
GST_OBJECT_LOCK (monitor);
monitor->priv->started = FALSE;
GST_OBJECT_UNLOCK (monitor);
} }
static void static void
@ -645,15 +624,26 @@ guint
gst_device_monitor_add_filter (GstDeviceMonitor * monitor, gst_device_monitor_add_filter (GstDeviceMonitor * monitor,
const gchar * classes, GstCaps * caps) const gchar * classes, GstCaps * caps)
{ {
GList *factories = NULL; guint id;
struct DeviceFilter *filter;
guint id = 0;
gboolean matched = FALSE;
g_return_val_if_fail (GST_IS_DEVICE_MONITOR (monitor), 0); g_return_val_if_fail (GST_IS_DEVICE_MONITOR (monitor), 0);
g_return_val_if_fail (!monitor->priv->started, 0); g_return_val_if_fail (!monitor->priv->started, 0);
GST_OBJECT_LOCK (monitor); GST_OBJECT_LOCK (monitor);
id = gst_device_monitor_add_filter_unlocked (monitor, classes, caps);
GST_OBJECT_UNLOCK (monitor);
return id;
}
static guint
gst_device_monitor_add_filter_unlocked (GstDeviceMonitor * monitor,
const gchar * classes, GstCaps * caps)
{
GList *factories = NULL;
struct DeviceFilter *filter;
guint id = 0;
gboolean matched = FALSE;
filter = g_slice_new0 (struct DeviceFilter); filter = g_slice_new0 (struct DeviceFilter);
filter->id = monitor->priv->last_id++; filter->id = monitor->priv->last_id++;
@ -702,7 +692,6 @@ gst_device_monitor_add_filter (GstDeviceMonitor * monitor,
G_CALLBACK (bus_sync_message), monitor); G_CALLBACK (bus_sync_message), monitor);
gst_object_unref (bus); gst_object_unref (bus);
g_ptr_array_add (monitor->priv->providers, provider); g_ptr_array_add (monitor->priv->providers, provider);
monitor->priv->cookie++;
} }
} }
@ -717,8 +706,6 @@ gst_device_monitor_add_filter (GstDeviceMonitor * monitor,
id = filter->id; id = filter->id;
g_ptr_array_add (monitor->priv->filters, filter); g_ptr_array_add (monitor->priv->filters, filter);
GST_OBJECT_UNLOCK (monitor);
return id; return id;
} }
@ -775,7 +762,6 @@ gst_device_monitor_remove_filter (GstDeviceMonitor * monitor, guint filter_id)
} }
if (!valid) { if (!valid) {
monitor->priv->cookie++;
gst_device_monitor_remove (monitor, i); gst_device_monitor_remove (monitor, i);
i--; i--;
} }