[gegl] Extract a function to do the preprocessing for the mesh rendering
- From: Michael Natterer <mitch src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gegl] Extract a function to do the preprocessing for the mesh rendering
- Date: Tue, 14 May 2013 22:16:42 +0000 (UTC)
commit 25d3ea917ac825518f7ffbe96431a92b80d4727e
Author: Barak Itkin <lightningismyname gmail com>
Date: Thu Aug 18 23:00:45 2011 +0300
Extract a function to do the preprocessing for the mesh rendering
.../seamless-clone/poly2tri-c/render/mesh-render.c | 95 +++++++++++++++-----
.../seamless-clone/poly2tri-c/render/mesh-render.h | 18 ++++
2 files changed, 90 insertions(+), 23 deletions(-)
---
diff --git a/operations/common/seamless-clone/poly2tri-c/render/mesh-render.c
b/operations/common/seamless-clone/poly2tri-c/render/mesh-render.c
index 34317ac..656c9ec 100755
--- a/operations/common/seamless-clone/poly2tri-c/render/mesh-render.c
+++ b/operations/common/seamless-clone/poly2tri-c/render/mesh-render.c
@@ -191,32 +191,72 @@ void p2tr_test_point_to_color (P2tRPoint* point, gfloat *dest, gpointer user_dat
dest[2] = 1;
}
+#define uvt3_u(ptr) (((ptr)+0)->u)
+#define uvt3_v(ptr) (((ptr)+1)->v)
+#define uvt3_t(ptr) (((ptr)+2)->tri)
+
void
-p2tr_mesh_render_scanline (P2tRTriangulation *T,
- gfloat *dest,
- P2tRImageConfig *config,
- P2tRPointToColorFunc pt2col,
- gpointer pt2col_user_data)
+p2tr_mesh_render_cache_uvt (P2tRTriangulation *T,
+ P2tRuvt *dest,
+ P2tRImageConfig *config)
{
- gdouble *uvbuf = g_new (gdouble, 2 * config->x_samples * config->y_samples), *Puv = uvbuf;
- P2tRTriangle **tribuf = g_new (P2tRTriangle*, config->x_samples * config->y_samples), **Ptri = tribuf;
- P2tRTriangle *tr_prev = NULL, *tr_now;
-
gint x, y;
+ P2tRuvt *uvt = dest;
+ P2tRTriangle *tr_prev = NULL;
- tribuf[0] = p2tr_triangulation_locate_point2 (T, config->min_x, config->min_y, NULL, &uvbuf[0], &uvbuf[1]);
-
+ uvt3_t(uvt) = p2tr_triangulation_locate_point2 (T, config->min_x, config->min_y, NULL, &uvt3_u(uvt),
&uvt3_v(uvt));
+ tr_prev = uvt3_t(uvt);
+
for (y = 0; y < config->y_samples; y++)
for (x = 0; x < config->x_samples; x++)
{
gdouble Px = config->min_x + x * config->step_x;
gdouble Py = config->min_y + y * config->step_y;
- *Ptri++ = p2tr_triangulation_locate_point2 (T, Px, Py, *Ptri, &Puv[0], &Puv[1]);
- Puv+=2;
+ uvt3_t(uvt) = p2tr_triangulation_locate_point2 (T, Px, Py, tr_prev, &uvt3_u(uvt), &uvt3_v(uvt));
+ tr_prev = uvt3_t(uvt);
+ uvt += 3;
}
+}
- Puv = uvbuf;
- Ptri = tribuf;
+void
+p2tr_mesh_render_scanline (P2tRTriangulation *T,
+ gfloat *dest,
+ P2tRImageConfig *config,
+ P2tRPointToColorFunc pt2col,
+ gpointer pt2col_user_data)
+{
+ P2tRuvt *uvt_cache = g_new (P2tRuvt, 3 * config->x_samples * config->y_samples);
+ GTimer *timer = g_timer_new ();
+
+ g_timer_start (timer);
+ p2tr_mesh_render_cache_uvt (T, uvt_cache, config);
+ g_timer_stop (timer);
+ g_debug ("Mesh preprocessing took %f seconds\n", g_timer_elapsed (timer, NULL));
+
+ g_timer_start (timer);
+ p2tr_mesh_render_scanline2 (uvt_cache, dest, config, pt2col, pt2col_user_data);
+ g_timer_stop (timer);
+ g_debug ("Mesh rendering took %f seconds\n", g_timer_elapsed (timer, NULL));
+
+ g_timer_destroy (timer);
+ g_free (uvt_cache);
+
+}
+
+
+void
+p2tr_mesh_render_scanline2 (P2tRuvt *uvt_cache,
+ gfloat *dest,
+ P2tRImageConfig *config,
+ P2tRPointToColorFunc pt2col,
+ gpointer pt2col_user_data)
+{
+ P2tRuvt *uvt_p = uvt_cache;
+
+ gdouble u, v;
+ P2tRTriangle *tr_prev = NULL, *tr_now;
+
+ gint x, y;
P2tRPoint *A = NULL, *B = NULL, *C = NULL;
@@ -227,16 +267,17 @@ p2tr_mesh_render_scanline (P2tRTriangulation *T,
gfloat *pixel = dest;
- GTimer *timer = g_timer_new ();
- g_timer_start (timer);
-
for (y = 0; y < config->y_samples; y++)
for (x = 0; x < config->x_samples; x++)
{
- gdouble u = Puv[0], v = Puv[1];
- tr_now = *Ptri++;
- Puv += 2;
+ u = uvt3_u (uvt_p);
+ v = uvt3_v (uvt_p);
+ tr_now = uvt3_t (uvt_p);
+ uvt_p += 3;
+
+ /* If we are outside of the triangulation, set alpha to zero and
+ * continue */
if (tr_now == NULL)
{
pixel[3] = 0;
@@ -244,23 +285,31 @@ p2tr_mesh_render_scanline (P2tRTriangulation *T,
}
else
{
+ /* If the triangle hasn't changed since the previous pixel,
+ * then don't sample the color at the vertices again, since
+ * that is an expensive process! */
if (tr_now != tr_prev)
{
+ /* Get the points of the triangle in some fixed order,
+ * just to make sure that the computation goes the same
+ * everywhere */
p2tr_triangle_barycentric_get_points (tr_now, &A, &B, &C);
+ /* At each point X sample the color into colX */
pt2col (A, colA, pt2col_user_data);
pt2col (B, colB, pt2col_user_data);
pt2col (C, colC, pt2col_user_data);
+ /* Set the current triangle */
tr_now = tr_prev;
}
+ /* Interpolate the color using barycentric coodinates */
*pixel++ = USE_BARYCENTRIC (u,v,colA[0],colB[0],colC[0]);
*pixel++ = USE_BARYCENTRIC (u,v,colA[1],colB[1],colC[1]);
*pixel++ = USE_BARYCENTRIC (u,v,colA[2],colB[2],colC[2]);
+ /* Finally, set as opaque since we are inside the mesh */
*pixel++ = 1;
}
}
- g_timer_stop (timer);
- p2tr_debug ("Mesh rendering took %f seconds\n", g_timer_elapsed (timer, NULL));
}
void
diff --git a/operations/common/seamless-clone/poly2tri-c/render/mesh-render.h
b/operations/common/seamless-clone/poly2tri-c/render/mesh-render.h
index 6d86985..3f99828 100755
--- a/operations/common/seamless-clone/poly2tri-c/render/mesh-render.h
+++ b/operations/common/seamless-clone/poly2tri-c/render/mesh-render.h
@@ -27,15 +27,33 @@ typedef struct {
typedef void (*P2tRPointToColorFunc) (P2tRPoint* point, gfloat *dest, gpointer user_data);
+typedef union {
+ P2tRTriangle *tri;
+ gdouble u;
+ gdouble v;
+} P2tRuvt;
+
void p2tr_test_point_to_color (P2tRPoint* point, gfloat *dest, gpointer user_data);
void
+p2tr_mesh_render_cache_uvt (P2tRTriangulation *T,
+ P2tRuvt *dest,
+ P2tRImageConfig *config);
+
+void
p2tr_mesh_render_scanline (P2tRTriangulation *T,
gfloat *dest,
P2tRImageConfig *config,
P2tRPointToColorFunc pt2col,
gpointer pt2col_user_data);
+void
+p2tr_mesh_render_scanline2 (P2tRuvt *uvt_cache,
+ gfloat *dest,
+ P2tRImageConfig *config,
+ P2tRPointToColorFunc pt2col,
+ gpointer pt2col_user_data);
+
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]