[gimp] app: rename GimpProjection's IdleRender stuff to ChunkRender
- From: Michael Natterer <mitch src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gimp] app: rename GimpProjection's IdleRender stuff to ChunkRender
- Date: Sun, 28 Jul 2013 14:04:25 +0000 (UTC)
commit 3b12541d61a4be973611d3d776a77ab331f3b4bf
Author: Michael Natterer <mitch gimp org>
Date: Sun Jul 28 15:57:28 2013 +0200
app: rename GimpProjection's IdleRender stuff to ChunkRender
and separate the chunk rendering logic from the fact that it's invoked
from an idle callback. Idle rendering no longer works in GTK+ 3.8 and
later because the expose logic was changed from an idle function to a
frame clock, and when we switch to that in gtk3-port, the diff to
master should stay readable (and rebasable) in this critical part of
the code.
app/core/gimpprojection.c | 202 +++++++++++++++++++++++++--------------------
app/core/gimpprojection.h | 35 ++++----
2 files changed, 132 insertions(+), 105 deletions(-)
---
diff --git a/app/core/gimpprojection.c b/app/core/gimpprojection.c
index 1a11412..7708b76 100644
--- a/app/core/gimpprojection.c
+++ b/app/core/gimpprojection.c
@@ -37,12 +37,12 @@
/* halfway between G_PRIORITY_HIGH_IDLE and G_PRIORITY_DEFAULT_IDLE */
-#define GIMP_PROJECTION_IDLE_PRIORITY ((G_PRIORITY_HIGH_IDLE + \
- G_PRIORITY_DEFAULT_IDLE) / 2)
+#define GIMP_PROJECTION_IDLE_PRIORITY ((G_PRIORITY_HIGH_IDLE + \
+ G_PRIORITY_DEFAULT_IDLE) / 2)
-/* chunk size for one iteration of the idle renderer */
-#define GIMP_PROJECTION_IDLE_CHUNK_WIDTH 256
-#define GIMP_PROJECTION_IDLE_CHUNK_HEIGHT 128
+/* chunk size for one iteration of the chunk renderer */
+#define GIMP_PROJECTION_CHUNK_WIDTH 256
+#define GIMP_PROJECTION_CHUNK_HEIGHT 128
enum
@@ -82,9 +82,12 @@ static void gimp_projection_add_update_area (GimpProjection *proj,
gint h);
static void gimp_projection_flush_whenever (GimpProjection *proj,
gboolean now);
-static void gimp_projection_idle_render_init (GimpProjection *proj);
-static gboolean gimp_projection_idle_render_callback (gpointer data);
-static gboolean gimp_projection_idle_render_next_area (GimpProjection *proj);
+static void gimp_projection_chunk_render_start (GimpProjection *proj);
+static void gimp_projection_chunk_render_stop (GimpProjection *proj);
+static gboolean gimp_projection_chunk_render_callback (gpointer data);
+static void gimp_projection_chunk_render_init (GimpProjection *proj);
+static gboolean gimp_projection_chunk_render_iteration(GimpProjection *proj);
+static gboolean gimp_projection_chunk_render_next_area(GimpProjection *proj);
static void gimp_projection_paint_area (GimpProjection *proj,
gboolean now,
gint x,
@@ -166,17 +169,14 @@ gimp_projection_finalize (GObject *object)
{
GimpProjection *proj = GIMP_PROJECTION (object);
- if (proj->idle_render.idle_id)
- {
- g_source_remove (proj->idle_render.idle_id);
- proj->idle_render.idle_id = 0;
- }
+ if (proj->chunk_render.running)
+ gimp_projection_chunk_render_stop (proj);
gimp_area_list_free (proj->update_areas);
proj->update_areas = NULL;
- gimp_area_list_free (proj->idle_render.update_areas);
- proj->idle_render.update_areas = NULL;
+ gimp_area_list_free (proj->chunk_render.update_areas);
+ proj->chunk_render.update_areas = NULL;
gimp_projection_free_buffer (proj);
@@ -362,7 +362,7 @@ gimp_projection_flush (GimpProjection *proj)
{
g_return_if_fail (GIMP_IS_PROJECTION (proj));
- /* Construct on idle time */
+ /* Construct in chunks */
gimp_projection_flush_whenever (proj, FALSE);
}
@@ -380,12 +380,11 @@ gimp_projection_finish_draw (GimpProjection *proj)
{
g_return_if_fail (GIMP_IS_PROJECTION (proj));
- if (proj->idle_render.idle_id)
+ if (proj->chunk_render.running)
{
- g_source_remove (proj->idle_render.idle_id);
- proj->idle_render.idle_id = 0;
+ gimp_projection_chunk_render_stop (proj);
- while (gimp_projection_idle_render_callback (proj));
+ while (gimp_projection_chunk_render_iteration (proj));
}
}
@@ -468,7 +467,7 @@ gimp_projection_flush_whenever (GimpProjection *proj,
}
else /* Asynchronous */
{
- gimp_projection_idle_render_init (proj);
+ gimp_projection_chunk_render_init (proj);
}
/* Free the update lists */
@@ -487,108 +486,137 @@ gimp_projection_flush_whenever (GimpProjection *proj,
}
static void
-gimp_projection_idle_render_init (GimpProjection *proj)
+gimp_projection_chunk_render_start (GimpProjection *proj)
+{
+ g_return_if_fail (proj->chunk_render.running == FALSE);
+
+ proj->chunk_render_idle_id =
+ g_idle_add_full (GIMP_PROJECTION_IDLE_PRIORITY,
+ gimp_projection_chunk_render_callback, proj,
+ NULL);
+
+ proj->chunk_render.running = TRUE;
+}
+
+static void
+gimp_projection_chunk_render_stop (GimpProjection *proj)
+{
+ g_return_if_fail (proj->chunk_render.running == TRUE);
+
+ g_source_remove (proj->chunk_render_idle_id);
+ proj->chunk_render_idle_id = 0;
+
+ proj->chunk_render.running = FALSE;
+}
+
+static gboolean
+gimp_projection_chunk_render_callback (gpointer data)
+{
+ GimpProjection *proj = data;
+
+ if (! gimp_projection_chunk_render_iteration (proj))
+ {
+ gimp_projection_chunk_render_stop (proj);
+
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+static void
+gimp_projection_chunk_render_init (GimpProjection *proj)
{
GSList *list;
- /* We need to merge the IdleRender's and the GimpProjection's update_areas
- * list to keep track of which of the updates have been flushed and hence
- * need to be drawn.
+ /* We need to merge the ChunkRender's and the GimpProjection's
+ * update_areas list to keep track of which of the updates have been
+ * flushed and hence need to be drawn.
*/
for (list = proj->update_areas; list; list = g_slist_next (list))
{
GimpArea *area = list->data;
- proj->idle_render.update_areas =
- gimp_area_list_process (proj->idle_render.update_areas,
+ proj->chunk_render.update_areas =
+ gimp_area_list_process (proj->chunk_render.update_areas,
gimp_area_new (area->x1, area->y1,
area->x2, area->y2));
}
- /* If an idlerender was already running, merge the remainder of its
- * unrendered area with the update_areas list, and make it start work
- * on the next unrendered area in the list.
+ /* If a chunk renderer was already running, merge the remainder of
+ * its unrendered area with the update_areas list, and make it start
+ * work on the next unrendered area in the list.
*/
- if (proj->idle_render.idle_id)
+ if (proj->chunk_render.running)
{
GimpArea *area =
- gimp_area_new (proj->idle_render.base_x,
- proj->idle_render.y,
- proj->idle_render.base_x + proj->idle_render.width,
- proj->idle_render.y + (proj->idle_render.height -
- (proj->idle_render.y -
- proj->idle_render.base_y)));
+ gimp_area_new (proj->chunk_render.base_x,
+ proj->chunk_render.y,
+ proj->chunk_render.base_x + proj->chunk_render.width,
+ proj->chunk_render.y + (proj->chunk_render.height -
+ (proj->chunk_render.y -
+ proj->chunk_render.base_y)));
- proj->idle_render.update_areas =
- gimp_area_list_process (proj->idle_render.update_areas, area);
+ proj->chunk_render.update_areas =
+ gimp_area_list_process (proj->chunk_render.update_areas, area);
- gimp_projection_idle_render_next_area (proj);
+ gimp_projection_chunk_render_next_area (proj);
}
else
{
- if (proj->idle_render.update_areas == NULL)
+ if (proj->chunk_render.update_areas == NULL)
{
- g_warning ("%s: wanted to start idle render with no update_areas",
+ g_warning ("%s: wanted to start chunk render with no update_areas",
G_STRFUNC);
return;
}
- gimp_projection_idle_render_next_area (proj);
+ gimp_projection_chunk_render_next_area (proj);
- proj->idle_render.idle_id =
- g_idle_add_full (GIMP_PROJECTION_IDLE_PRIORITY,
- gimp_projection_idle_render_callback, proj,
- NULL);
+ gimp_projection_chunk_render_start (proj);
}
}
/* Unless specified otherwise, projection re-rendering is organised by
- * IdleRender, which amalgamates areas to be re-rendered and breaks
- * them into bite-sized chunks which are chewed on in a low- priority
- * idle thread. This greatly improves responsiveness for many GIMP
+ * ChunkRender, which amalgamates areas to be re-rendered and breaks
+ * them into bite-sized chunks which are chewed on in an idle
+ * function. This greatly improves responsiveness for many GIMP
* operations. -- Adam
*/
static gboolean
-gimp_projection_idle_render_callback (gpointer data)
+gimp_projection_chunk_render_iteration (GimpProjection *proj)
{
- GimpProjection *proj = data;
- gint workx, worky;
- gint workw, workh;
+ gint workx = proj->chunk_render.x;
+ gint worky = proj->chunk_render.y;
+ gint workw = GIMP_PROJECTION_CHUNK_WIDTH;
+ gint workh = GIMP_PROJECTION_CHUNK_HEIGHT;
- workw = GIMP_PROJECTION_IDLE_CHUNK_WIDTH;
- workh = GIMP_PROJECTION_IDLE_CHUNK_HEIGHT;
- workx = proj->idle_render.x;
- worky = proj->idle_render.y;
-
- if (workx + workw > proj->idle_render.base_x + proj->idle_render.width)
+ if (workx + workw > proj->chunk_render.base_x + proj->chunk_render.width)
{
- workw = proj->idle_render.base_x + proj->idle_render.width - workx;
+ workw = proj->chunk_render.base_x + proj->chunk_render.width - workx;
}
- if (worky + workh > proj->idle_render.base_y + proj->idle_render.height)
+ if (worky + workh > proj->chunk_render.base_y + proj->chunk_render.height)
{
- workh = proj->idle_render.base_y + proj->idle_render.height - worky;
+ workh = proj->chunk_render.base_y + proj->chunk_render.height - worky;
}
gimp_projection_paint_area (proj, TRUE /* sic! */,
workx, worky, workw, workh);
- proj->idle_render.x += GIMP_PROJECTION_IDLE_CHUNK_WIDTH;
+ proj->chunk_render.x += GIMP_PROJECTION_CHUNK_WIDTH;
- if (proj->idle_render.x >=
- proj->idle_render.base_x + proj->idle_render.width)
+ if (proj->chunk_render.x >=
+ proj->chunk_render.base_x + proj->chunk_render.width)
{
- proj->idle_render.x = proj->idle_render.base_x;
- proj->idle_render.y += GIMP_PROJECTION_IDLE_CHUNK_HEIGHT;
+ proj->chunk_render.x = proj->chunk_render.base_x;
+ proj->chunk_render.y += GIMP_PROJECTION_CHUNK_HEIGHT;
- if (proj->idle_render.y >=
- proj->idle_render.base_y + proj->idle_render.height)
+ if (proj->chunk_render.y >=
+ proj->chunk_render.base_y + proj->chunk_render.height)
{
- if (! gimp_projection_idle_render_next_area (proj))
+ if (! gimp_projection_chunk_render_next_area (proj))
{
- /* FINISHED */
- proj->idle_render.idle_id = 0;
-
if (proj->invalidate_preview)
{
/* invalidate the preview here since it is constructed from
@@ -599,6 +627,7 @@ gimp_projection_idle_render_callback (gpointer data)
gimp_projectable_invalidate_preview (proj->projectable);
}
+ /* FINISHED */
return FALSE;
}
}
@@ -609,22 +638,22 @@ gimp_projection_idle_render_callback (gpointer data)
}
static gboolean
-gimp_projection_idle_render_next_area (GimpProjection *proj)
+gimp_projection_chunk_render_next_area (GimpProjection *proj)
{
GimpArea *area;
- if (! proj->idle_render.update_areas)
+ if (! proj->chunk_render.update_areas)
return FALSE;
- area = proj->idle_render.update_areas->data;
+ area = proj->chunk_render.update_areas->data;
- proj->idle_render.update_areas =
- g_slist_remove (proj->idle_render.update_areas, area);
+ proj->chunk_render.update_areas =
+ g_slist_remove (proj->chunk_render.update_areas, area);
- proj->idle_render.x = proj->idle_render.base_x = area->x1;
- proj->idle_render.y = proj->idle_render.base_y = area->y1;
- proj->idle_render.width = area->x2 - area->x1;
- proj->idle_render.height = area->y2 - area->y1;
+ proj->chunk_render.x = proj->chunk_render.base_x = area->x1;
+ proj->chunk_render.y = proj->chunk_render.base_y = area->y1;
+ proj->chunk_render.width = area->x2 - area->x1;
+ proj->chunk_render.height = area->y2 - area->y1;
gimp_area_free (area);
@@ -710,11 +739,8 @@ gimp_projection_projectable_changed (GimpProjectable *projectable,
gint off_x, off_y;
gint width, height;
- if (proj->idle_render.idle_id)
- {
- g_source_remove (proj->idle_render.idle_id);
- proj->idle_render.idle_id = 0;
- }
+ if (proj->chunk_render.running)
+ gimp_projection_chunk_render_stop (proj);
gimp_area_list_free (proj->update_areas);
proj->update_areas = NULL;
diff --git a/app/core/gimpprojection.h b/app/core/gimpprojection.h
index d6af5dc..1b9b2d8 100644
--- a/app/core/gimpprojection.h
+++ b/app/core/gimpprojection.h
@@ -22,18 +22,18 @@
#include "gimpobject.h"
-typedef struct _GimpProjectionIdleRender GimpProjectionIdleRender;
+typedef struct _GimpProjectionChunkRender GimpProjectionChunkRender;
-struct _GimpProjectionIdleRender
+struct _GimpProjectionChunkRender
{
- gint width;
- gint height;
- gint x;
- gint y;
- gint base_x;
- gint base_y;
- guint idle_id;
- GSList *update_areas; /* flushed update areas */
+ gboolean running;
+ gint width;
+ gint height;
+ gint x;
+ gint y;
+ gint base_x;
+ gint base_y;
+ GSList *update_areas; /* flushed update areas */
};
@@ -50,17 +50,18 @@ typedef struct _GimpProjectionClass GimpProjectionClass;
struct _GimpProjection
{
- GimpObject parent_instance;
+ GimpObject parent_instance;
- GimpProjectable *projectable;
+ GimpProjectable *projectable;
- GeglBuffer *buffer;
- gpointer validate_handler;
+ GeglBuffer *buffer;
+ gpointer validate_handler;
- GSList *update_areas;
- GimpProjectionIdleRender idle_render;
+ GSList *update_areas;
+ GimpProjectionChunkRender chunk_render;
+ guint chunk_render_idle_id;
- gboolean invalidate_preview;
+ gboolean invalidate_preview;
};
struct _GimpProjectionClass
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]