[gimp] app: various GimpAsync improvements
- From: N/A <ell src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gimp] app: various GimpAsync improvements
- Date: Sun, 13 May 2018 19:21:53 +0000 (UTC)
commit 16a24ff0817df52461060e065380ff8a58dc2ebc
Author: Ell <ell_se yahoo com>
Date: Sun May 13 14:16:57 2018 -0400
app: various GimpAsync improvements
Code cleanup.
Improve function descriptions.
Improve function precondition checks.
Make sure the main thread is properly synced with the async thread
before calling the completion callbacks in the idle function.
Guarantee that all callbacks are called in FIFO order, even
callbacks added during the execution of other callbacks.
app/core/gimpasync.c | 128 ++++++++++++++++++++++++--------------------------
1 files changed, 61 insertions(+), 67 deletions(-)
---
diff --git a/app/core/gimpasync.c b/app/core/gimpasync.c
index 692554b..6b1970c 100644
--- a/app/core/gimpasync.c
+++ b/app/core/gimpasync.c
@@ -63,7 +63,7 @@ struct _GimpAsyncPrivate
GMutex mutex;
GCond cond;
- GSList *callbacks;
+ GQueue callbacks;
gpointer result;
GDestroyNotify result_destroy_func;
@@ -78,12 +78,11 @@ struct _GimpAsyncPrivate
/* local function prototypes */
-static void gimp_async_dispose (GObject *object);
-static void gimp_async_finalize (GObject *object);
+static void gimp_async_finalize (GObject *object);
-static gboolean gimp_async_idle (GimpAsync *async);
+static gboolean gimp_async_idle (GimpAsync *async);
-static void gimp_async_stop (GimpAsync *async);
+static void gimp_async_stop (GimpAsync *async);
G_DEFINE_TYPE (GimpAsync, gimp_async, G_TYPE_OBJECT)
@@ -99,7 +98,6 @@ gimp_async_class_init (GimpAsyncClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
- object_class->dispose = gimp_async_dispose;
object_class->finalize = gimp_async_finalize;
g_type_class_add_private (klass, sizeof (GimpAsyncPrivate));
@@ -114,14 +112,18 @@ gimp_async_init (GimpAsync *async)
g_mutex_init (&async->priv->mutex);
g_cond_init (&async->priv->cond);
+
+ g_queue_init (&async->priv->callbacks);
}
static void
-gimp_async_dispose (GObject *object)
+gimp_async_finalize (GObject *object)
{
GimpAsync *async = GIMP_ASYNC (object);
- g_return_if_fail (async->priv->stopped);
+ g_warn_if_fail (async->priv->stopped);
+ g_warn_if_fail (async->priv->idle_id == 0);
+ g_warn_if_fail (g_queue_is_empty (&async->priv->callbacks));
if (async->priv->finished &&
async->priv->result &&
@@ -132,14 +134,6 @@ gimp_async_dispose (GObject *object)
async->priv->result = NULL;
}
- G_OBJECT_CLASS (parent_class)->dispose (object);
-}
-
-static void
-gimp_async_finalize (GObject *object)
-{
- GimpAsync *async = GIMP_ASYNC (object);
-
g_cond_clear (&async->priv->cond);
g_mutex_clear (&async->priv->mutex);
@@ -149,25 +143,7 @@ gimp_async_finalize (GObject *object)
static gboolean
gimp_async_idle (GimpAsync *async)
{
- GSList *iter;
-
- async->priv->idle_id = 0;
-
- async->priv->callbacks = g_slist_reverse (async->priv->callbacks);
-
- for (iter = async->priv->callbacks; iter; iter = g_slist_next (iter))
- {
- GimpAsyncCallbackInfo *callback_info = iter->data;
-
- callback_info->callback (async, callback_info->data);
-
- g_slice_free (GimpAsyncCallbackInfo, callback_info);
- }
-
- g_slist_free (async->priv->callbacks);
- async->priv->callbacks = NULL;
-
- g_object_unref (async);
+ gimp_async_wait (async);
return G_SOURCE_REMOVE;
}
@@ -175,7 +151,7 @@ gimp_async_idle (GimpAsync *async)
static void
gimp_async_stop (GimpAsync *async)
{
- if (async->priv->callbacks)
+ if (! g_queue_is_empty (&async->priv->callbacks))
{
g_object_ref (async);
@@ -232,19 +208,29 @@ gimp_async_wait (GimpAsync *async)
if (async->priv->idle_id)
{
+ GimpAsyncCallbackInfo *callback_info;
+
g_source_remove (async->priv->idle_id);
+ async->priv->idle_id = 0;
+
+ while ((callback_info = g_queue_pop_head (&async->priv->callbacks)))
+ {
+ callback_info->callback (async, callback_info->data);
- gimp_async_idle (async);
+ g_slice_free (GimpAsyncCallbackInfo, callback_info);
+ }
+
+ g_object_unref (async);
}
}
/* registers a callback to be called when 'async' transitions to the "stopped"
- * state. if 'async' is already stopped, the callback is called directly.
+ * state. if 'async' is already stopped, the callback may be called directly.
*
- * callbacks added before 'async' was stopped are called in the order in which
- * they were added. 'async' is guaranteed to be kept alive, even without an
- * external reference, between the point where it was stopped, and until all
- * previously added callbacks have been called.
+ * callbacks are called in the order in which they were added. 'async' is
+ * guaranteed to be kept alive, even without an external reference, between the
+ * point where it was stopped, and until all callbacks added while 'async' was
+ * externally referenced have been called.
*
* the callback is guaranteed to be called on the main thread.
*
@@ -262,7 +248,7 @@ gimp_async_add_callback (GimpAsync *async,
g_mutex_lock (&async->priv->mutex);
- if (async->priv->stopped && ! async->priv->idle_id)
+ if (async->priv->stopped && g_queue_is_empty (&async->priv->callbacks))
{
g_mutex_unlock (&async->priv->mutex);
@@ -275,8 +261,7 @@ gimp_async_add_callback (GimpAsync *async,
callback_info->callback = callback;
callback_info->data = data;
- async->priv->callbacks = g_slist_prepend (async->priv->callbacks,
- callback_info);
+ g_queue_push_tail (&async->priv->callbacks, callback_info);
g_mutex_unlock (&async->priv->mutex);
}
@@ -329,6 +314,7 @@ gboolean
gimp_async_is_finished (GimpAsync *async)
{
g_return_val_if_fail (GIMP_IS_ASYNC (async), FALSE);
+ g_return_val_if_fail (async->priv->stopped, FALSE);
return async->priv->finished;
}
@@ -344,33 +330,12 @@ gpointer
gimp_async_get_result (GimpAsync *async)
{
g_return_val_if_fail (GIMP_IS_ASYNC (async), NULL);
+ g_return_val_if_fail (async->priv->stopped, NULL);
g_return_val_if_fail (async->priv->finished, NULL);
return async->priv->result;
}
-/* requests the cancellation of the task managed by 'async'. */
-void
-gimp_async_cancel (GimpAsync *async)
-{
- g_return_if_fail (GIMP_IS_ASYNC (async));
-
- async->priv->canceled = TRUE;
-}
-
-/* checks if cancellation of 'async' have been requested.
- *
- * note that a return value of TRUE only indicates that 'gimp_async_cancel()'
- * has been called for 'async', and not that 'async' is stopped.
- */
-gboolean
-gimp_async_is_canceled (GimpAsync *async)
-{
- g_return_val_if_fail (GIMP_IS_ASYNC (async), FALSE);
-
- return async->priv->canceled;
-}
-
/* transitions 'async' to the "stopped" state, indicating that the task
* was stopped before completion (normally, in response to a
* 'gimp_async_cancel()' call).
@@ -391,3 +356,32 @@ gimp_async_abort (GimpAsync *async)
g_mutex_unlock (&async->priv->mutex);
}
+
+/* requests the cancellation of the task managed by 'async'.
+ *
+ * note that 'gimp_async_cancel()' doesn't directly cause 'async' to be
+ * stopped, nor does it cause the main thread to become synced with the async
+ * thread. furthermore, 'async' may still complete successfully even when
+ * cancellation has been requested.
+ */
+void
+gimp_async_cancel (GimpAsync *async)
+{
+ g_return_if_fail (GIMP_IS_ASYNC (async));
+
+ async->priv->canceled = TRUE;
+}
+
+/* checks if cancellation of 'async' has been requested.
+ *
+ * note that a return value of TRUE only indicates that 'gimp_async_cancel()'
+ * has been called for 'async', and not that 'async' is stopped, or, if it is
+ * stopped, that it was aborted.
+ */
+gboolean
+gimp_async_is_canceled (GimpAsync *async)
+{
+ g_return_val_if_fail (GIMP_IS_ASYNC (async), FALSE);
+
+ return async->priv->canceled;
+}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]