[gnome-shell] Screenshot: Move filesystem I/O to a thread
- From: Adel Gadllah <agadllah src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-shell] Screenshot: Move filesystem I/O to a thread
- Date: Sun, 22 Jan 2012 10:47:26 +0000 (UTC)
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]