uridownloader: make cancelled state 'permanent' until a reset
authorAndre Moreira Magalhaes (andrunko) <andre.magalhaes@collabora.co.uk>
Tue, 16 Apr 2013 20:23:02 +0000 (17:23 -0300)
committerThiago Santos <thiago.sousa.santos@collabora.com>
Tue, 7 May 2013 23:02:41 +0000 (20:02 -0300)
When downloading and cancelling quickly the uridownloader object and the
element using it could miss the cancelled window and the uridownloader
would fetch the wrong URI and block on subsequent fetches.

This was also problematic when stopping elements, while one task would
call the cancel, another element thread could issue a new fetch_uri. As
the cancel state isn't 'permanent' this fetch_uri would block and
prevent the whole element from stopping and going to NULL.

This patch makes the 'cancelled' state permanent until a
gst_uri_downloader_reset is called. This way the element knows the
window where the uridownloader isn't active and only reactivate it when
ready.

gst-libs/gst/uridownloader/gsturidownloader.c
gst-libs/gst/uridownloader/gsturidownloader.h

index 5384141ae0566dcdaa71aaff14902763712dba02..4ca0dc70ec3effa83cf545467bcfe8b22129bd66 100644 (file)
@@ -41,6 +41,7 @@ struct _GstUriDownloaderPrivate
   GstFragment *download;
   GMutex lock;
   GCond cond;
   GstFragment *download;
   GMutex lock;
   GCond cond;
+  gboolean cancelled;
 };
 
 static void gst_uri_downloader_finalize (GObject * object);
 };
 
 static void gst_uri_downloader_finalize (GObject * object);
@@ -263,6 +264,16 @@ gst_uri_downloader_stop (GstUriDownloader * downloader)
       GST_CLOCK_TIME_NONE);
 }
 
       GST_CLOCK_TIME_NONE);
 }
 
+void
+gst_uri_downloader_reset (GstUriDownloader * downloader)
+{
+  g_return_if_fail (downloader != NULL);
+
+  GST_OBJECT_LOCK (downloader);
+  downloader->priv->cancelled = FALSE;
+  GST_OBJECT_UNLOCK (downloader);
+}
+
 void
 gst_uri_downloader_cancel (GstUriDownloader * downloader)
 {
 void
 gst_uri_downloader_cancel (GstUriDownloader * downloader)
 {
@@ -271,15 +282,21 @@ gst_uri_downloader_cancel (GstUriDownloader * downloader)
     GST_DEBUG_OBJECT (downloader, "Cancelling download");
     g_object_unref (downloader->priv->download);
     downloader->priv->download = NULL;
     GST_DEBUG_OBJECT (downloader, "Cancelling download");
     g_object_unref (downloader->priv->download);
     downloader->priv->download = NULL;
+    downloader->priv->cancelled = TRUE;
     GST_OBJECT_UNLOCK (downloader);
     GST_DEBUG_OBJECT (downloader, "Signaling chain funtion");
     g_mutex_lock (&downloader->priv->lock);
     g_cond_signal (&downloader->priv->cond);
     g_mutex_unlock (&downloader->priv->lock);
   } else {
     GST_OBJECT_UNLOCK (downloader);
     GST_DEBUG_OBJECT (downloader, "Signaling chain funtion");
     g_mutex_lock (&downloader->priv->lock);
     g_cond_signal (&downloader->priv->cond);
     g_mutex_unlock (&downloader->priv->lock);
   } else {
+    gboolean cancelled;
+
+    cancelled = downloader->priv->cancelled;
+    downloader->priv->cancelled = TRUE;
     GST_OBJECT_UNLOCK (downloader);
     GST_OBJECT_UNLOCK (downloader);
-    GST_DEBUG_OBJECT (downloader,
-        "Trying to cancell a download that was alredy cancelled");
+    if (cancelled)
+      GST_DEBUG_OBJECT (downloader,
+          "Trying to cancell a download that was alredy cancelled");
   }
 }
 
   }
 }
 
@@ -319,6 +336,10 @@ gst_uri_downloader_fetch_uri (GstUriDownloader * downloader, const gchar * uri)
 
   g_mutex_lock (&downloader->priv->lock);
 
 
   g_mutex_lock (&downloader->priv->lock);
 
+  if (downloader->priv->cancelled) {
+    goto quit;
+  }
+
   if (!gst_uri_downloader_set_uri (downloader, uri)) {
     goto quit;
   }
   if (!gst_uri_downloader_set_uri (downloader, uri)) {
     goto quit;
   }
@@ -337,9 +358,17 @@ gst_uri_downloader_fetch_uri (GstUriDownloader * downloader, const gchar * uri)
    *   - the download failed (Error message on the fetcher bus)
    *   - the download was canceled
    */
    *   - the download failed (Error message on the fetcher bus)
    *   - the download was canceled
    */
-  GST_DEBUG_OBJECT (downloader, "Waiting to fetch the URI");
+  GST_DEBUG_OBJECT (downloader, "Waiting to fetch the URI %s", uri);
   g_cond_wait (&downloader->priv->cond, &downloader->priv->lock);
 
   g_cond_wait (&downloader->priv->cond, &downloader->priv->lock);
 
+  if (downloader->priv->cancelled) {
+    if (downloader->priv->download) {
+      g_object_unref (downloader->priv->download);
+      downloader->priv->download = NULL;
+    }
+    goto quit;
+  }
+
   GST_OBJECT_LOCK (downloader);
   download = downloader->priv->download;
   downloader->priv->download = NULL;
   GST_OBJECT_LOCK (downloader);
   download = downloader->priv->download;
   downloader->priv->download = NULL;
index 8c1a6ec2847e11f2a3131f9dfed5a728b8df24ab..c29cb52b39142eb378ef36cef702c284bf19d8d6 100644 (file)
@@ -57,6 +57,7 @@ GType gst_uri_downloader_get_type (void);
 
 GstUriDownloader * gst_uri_downloader_new (void);
 GstFragment * gst_uri_downloader_fetch_uri (GstUriDownloader * downloader, const gchar * uri);
 
 GstUriDownloader * gst_uri_downloader_new (void);
 GstFragment * gst_uri_downloader_fetch_uri (GstUriDownloader * downloader, const gchar * uri);
+void gst_uri_downloader_reset (GstUriDownloader *downloader);
 void gst_uri_downloader_cancel (GstUriDownloader *downloader);
 void gst_uri_downloader_free (GstUriDownloader *downloader);
 
 void gst_uri_downloader_cancel (GstUriDownloader *downloader);
 void gst_uri_downloader_free (GstUriDownloader *downloader);