[gnome-shell] Screenshot: Move filesystem I/O to a thread



commit 882fe48d80e790a0bafd86b7d04f4f326f3cfb19
Author: Adel Gadllah <adel gadllah gmail com>
Date:   Sun Jan 22 10:57:42 2012 +0100

    Screenshot: Move filesystem I/O to a thread
    
    Writting the screenshot to a file can take a relativly long time
    in which we block the compositor, so do that part in a separate
    thread to avoid the hang.
    
    https://bugzilla.gnome.org/show_bug.cgi?id=652952

 js/ui/shellDBus.js         |    9 +++-
 src/shell-global-private.h |    2 +
 src/shell-global.c         |  110 +++++++++++++++++++++++++++-----------------
 src/shell-global.h         |    5 +-
 4 files changed, 80 insertions(+), 46 deletions(-)
---
diff --git a/js/ui/shellDBus.js b/js/ui/shellDBus.js
index 275d57b..ee88594 100644
--- a/js/ui/shellDBus.js
+++ b/js/ui/shellDBus.js
@@ -141,8 +141,13 @@ const GnomeShell = new Lang.Class({
      * indicating whether the operation was successful or not.
      *
      */
-    ScreenshotWindow : function (include_frame, filename) {
-        return global.screenshot_window (include_frame, filename);
+    ScreenshotWindowAsync : function (params, invocation) {
+        let [include_frame, filename] = params;
+        global.screenshot_window (include_frame, filename,
+            function (obj, result) {
+                let retval = GLib.Variant.new('(b)', [result]);
+                invocation.return_value(retval);
+            });
     },
 
     /**
diff --git a/src/shell-global-private.h b/src/shell-global-private.h
index a44b4dc..5cb9d6a 100644
--- a/src/shell-global-private.h
+++ b/src/shell-global-private.h
@@ -30,6 +30,8 @@ typedef struct _screenshot_data {
   int width;
   int height;
 
+  cairo_surface_t *image;
+
   ShellGlobalScreenshotCallback callback;
 } _screenshot_data;
 
diff --git a/src/shell-global.c b/src/shell-global.c
index bc0faf1..13fb314 100644
--- a/src/shell-global.c
+++ b/src/shell-global.c
@@ -1966,24 +1966,51 @@ shell_global_launch_calendar_server (ShellGlobal *global)
 }
 
 static void
+on_screenshot_written (GObject *source,
+                       GAsyncResult *result,
+                       gpointer user_data)
+{
+  _screenshot_data *screenshot_data = (_screenshot_data*) user_data;
+  if (screenshot_data->callback)
+    screenshot_data->callback (screenshot_data->global, g_simple_async_result_get_op_res_gboolean (G_SIMPLE_ASYNC_RESULT (result)));
+
+  cairo_surface_destroy (screenshot_data->image);
+  g_free (screenshot_data->filename);
+  g_free (screenshot_data);
+}
+
+static void
+write_screenshot_thread (GSimpleAsyncResult *result,
+                         GObject *object,
+                         GCancellable *cancellable)
+{
+  cairo_status_t status;
+  _screenshot_data *screenshot_data = g_async_result_get_user_data (G_ASYNC_RESULT (result));
+  g_assert (screenshot_data != NULL);
+
+  status = cairo_surface_write_to_png (screenshot_data->image, screenshot_data->filename);
+  g_simple_async_result_set_op_res_gboolean (result, status == CAIRO_STATUS_SUCCESS);
+}
+
+static void
 grab_screenshot (ClutterActor *stage,
                  _screenshot_data *screenshot_data)
 {
   MetaScreen *screen = shell_global_get_screen (screenshot_data->global);
-  cairo_status_t status;
-  cairo_surface_t *image;
   guchar *data;
   int width, height;
 
+  GSimpleAsyncResult *result;
+
   meta_plugin_query_screen_size (screenshot_data->global->plugin, &width, &height);
-  image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, width, height);
-  data = cairo_image_surface_get_data (image);
+  screenshot_data->image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, width, height);
+  data = cairo_image_surface_get_data (screenshot_data->image);
 
   cogl_flush();
 
   cogl_read_pixels (0, 0, width, height, COGL_READ_PIXELS_COLOR_BUFFER, CLUTTER_CAIRO_FORMAT_ARGB32, data);
 
-  cairo_surface_mark_dirty (image);
+  cairo_surface_mark_dirty (screenshot_data->image);
 
   if (meta_screen_get_n_monitors (screen) > 1)
     {
@@ -2009,7 +2036,7 @@ grab_screenshot (ClutterActor *stage,
       cairo_region_xor (stage_region, screen_region);
       cairo_region_destroy (screen_region);
 
-      cr = cairo_create (image);
+      cr = cairo_create (screenshot_data->image);
 
       for (i = 0; i < cairo_region_num_rectangles (stage_region); i++)
         {
@@ -2023,41 +2050,34 @@ grab_screenshot (ClutterActor *stage,
       cairo_region_destroy (stage_region);
     }
 
-
-  status = cairo_surface_write_to_png (image, screenshot_data->filename);
-  cairo_surface_destroy (image);
-
-  if (screenshot_data->callback)
-    screenshot_data->callback (screenshot_data->global, status == CAIRO_STATUS_SUCCESS);
-
   g_signal_handlers_disconnect_by_func (stage, (void *)grab_screenshot, (gpointer)screenshot_data);
-  g_free (screenshot_data->filename);
-  g_free (screenshot_data);
+
+  result = g_simple_async_result_new (NULL, on_screenshot_written, (gpointer)screenshot_data, grab_screenshot);
+  g_simple_async_result_run_in_thread (result, write_screenshot_thread, G_PRIORITY_DEFAULT, NULL);
+  g_object_unref (result);
 }
 
 static void
 grab_area_screenshot (ClutterActor *stage,
                       _screenshot_data *screenshot_data)
 {
-  cairo_status_t status;
-  cairo_surface_t *image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, screenshot_data->width, screenshot_data->height);
-  guchar *data = cairo_image_surface_get_data (image);
+  GSimpleAsyncResult *result;
+  guchar *data;
+
+  screenshot_data->image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, screenshot_data->width, screenshot_data->height);
+  data = cairo_image_surface_get_data (screenshot_data->image);
 
   cogl_flush();
 
   cogl_read_pixels (screenshot_data->x, screenshot_data->y, screenshot_data->width, screenshot_data->height,
                     COGL_READ_PIXELS_COLOR_BUFFER, CLUTTER_CAIRO_FORMAT_ARGB32, data);
 
-  cairo_surface_mark_dirty (image);
-  status = cairo_surface_write_to_png (image, screenshot_data->filename);
-  cairo_surface_destroy (image);
-
-  if (screenshot_data->callback)
-    screenshot_data->callback (screenshot_data->global, status == CAIRO_STATUS_SUCCESS);
+  cairo_surface_mark_dirty (screenshot_data->image);
 
   g_signal_handlers_disconnect_by_func (stage, (void *)grab_area_screenshot, (gpointer)screenshot_data);
-  g_free (screenshot_data->filename);
-  g_free (screenshot_data);
+  result = g_simple_async_result_new (NULL, on_screenshot_written, (gpointer)screenshot_data, grab_area_screenshot);
+  g_simple_async_result_run_in_thread (result, write_screenshot_thread, G_PRIORITY_DEFAULT, NULL);
+  g_object_unref (result);
 }
 
 /**
@@ -2138,27 +2158,33 @@ shell_global_screenshot_area (ShellGlobal  *global,
  * @include_frame: Whether to include the frame or not
  *
  * @filename: The filename for the screenshot
+ * @callback: (scope async): function to call returning success or failure
+ * of the async grabbing
  *
  * Takes a screenshot of the focused window (optionally omitting the frame)
  * in @filename as png image.
  *
- * Return value: success or failure.
  */
-gboolean
+void
 shell_global_screenshot_window (ShellGlobal  *global,
                                 gboolean include_frame,
-                                const char *filename)
+                                const char *filename,
+                                ShellGlobalScreenshotCallback callback)
 {
   CoglHandle texture;
-  cairo_surface_t *image;
   guchar *data;
+  GSimpleAsyncResult *result;
+
+  _screenshot_data *screenshot_data = g_new0 (_screenshot_data, 1);
 
   MetaScreen *screen = meta_plugin_get_screen (global->plugin);
   MetaDisplay *display = meta_screen_get_display (screen);
   MetaWindow *window = meta_display_get_focus_window (display);
   ClutterActor *window_actor;
 
-  cairo_status_t status;
+  screenshot_data->global = global;
+  screenshot_data->filename = g_strdup (filename);
+  screenshot_data->callback = callback;
 
   window_actor = CLUTTER_ACTOR (meta_window_get_compositor_private (window));
   texture = clutter_texture_get_cogl_texture (CLUTTER_TEXTURE (meta_window_actor_get_texture (META_WINDOW_ACTOR (window_actor))));
@@ -2172,26 +2198,26 @@ shell_global_screenshot_window (ShellGlobal  *global,
                                                    window_rect->width,
                                                    window_rect->height);
 
-      image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
-                                          window_rect->width,
-                                          window_rect->height);
+      screenshot_data->image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
+                                                          window_rect->width,
+                                                          window_rect->height);
     }
   else
-    image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
-                                        clutter_actor_get_width (window_actor),
-                                        clutter_actor_get_height (window_actor));
+    screenshot_data->image = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
+                                                        clutter_actor_get_width (window_actor),
+                                                        clutter_actor_get_height (window_actor));
 
-  data = cairo_image_surface_get_data (image);
+  data = cairo_image_surface_get_data (screenshot_data->image);
 
   cogl_flush();
 
   cogl_texture_get_data (texture, CLUTTER_CAIRO_FORMAT_ARGB32, 0, data);
 
-  cairo_surface_mark_dirty (image);
-  status = cairo_surface_write_to_png (image, filename);
-  cairo_surface_destroy (image);
+  cairo_surface_mark_dirty (screenshot_data->image);
 
-  return status == CAIRO_STATUS_SUCCESS;
+  result = g_simple_async_result_new (NULL, on_screenshot_written, (gpointer)screenshot_data, shell_global_screenshot_window);
+  g_simple_async_result_run_in_thread (result, write_screenshot_thread, G_PRIORITY_DEFAULT, NULL);
+  g_object_unref (result);
 }
 
 /**
diff --git a/src/shell-global.h b/src/shell-global.h
index dcb1517..4433025 100644
--- a/src/shell-global.h
+++ b/src/shell-global.h
@@ -156,9 +156,10 @@ void    shell_global_screenshot_area           (ShellGlobal  *global,
                                                 const char *filename,
                                                 ShellGlobalScreenshotCallback callback);
 
-gboolean shell_global_screenshot_window         (ShellGlobal  *global,
+void    shell_global_screenshot_window         (ShellGlobal  *global,
                                                 gboolean include_frame,
-                                                const char *filename);
+                                                const char *filename,
+                                                ShellGlobalScreenshotCallback callback);
 
 void    shell_global_screenshot                (ShellGlobal  *global,
                                                 const char *filename,



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]