[gegl] workshop: new op boxblur-1d
- From: Thomas Manni <tmanni src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gegl] workshop: new op boxblur-1d
- Date: Tue, 25 Jun 2019 10:50:17 +0000 (UTC)
commit 70d0729a209d8ca6b8d759c4cd6be3376c64c554
Author: Thomas Manni <thomas manni free fr>
Date: Tue Jun 25 12:38:52 2019 +0200
workshop: new op boxblur-1d
A one dimensional box blur filter which handles RGB, Y and CMYK color models
with appropriate multithreads splitting strategy.
operations/workshop/Makefile.am | 1 +
operations/workshop/boxblur-1d.c | 325 +++++++++++++++++++++++++++++++++++++++
2 files changed, 326 insertions(+)
---
diff --git a/operations/workshop/Makefile.am b/operations/workshop/Makefile.am
index faacec525..b805083ef 100644
--- a/operations/workshop/Makefile.am
+++ b/operations/workshop/Makefile.am
@@ -11,6 +11,7 @@ LIBS = $(op_libs)
ops = \
bayer-matrix.la \
bilateral-filter-fast.la \
+ boxblur-1d.la \
connected-components.la \
demosaic-bimedian.la \
demosaic-simple.la \
diff --git a/operations/workshop/boxblur-1d.c b/operations/workshop/boxblur-1d.c
new file mode 100644
index 000000000..b5e3cfde3
--- /dev/null
+++ b/operations/workshop/boxblur-1d.c
@@ -0,0 +1,325 @@
+/* This file is an image processing operation for GEGL
+ * GEGL is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * GEGL is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with GEGL; if not, see <https://www.gnu.org/licenses/>.
+ *
+ * Copyright 2019 Thomas Manni <thomas manni free fr>
+ *
+ * TODO:
+ * - manage mipmap level
+ * - add opencl support
+ */
+
+ #include "config.h"
+#include <glib/gi18n-lib.h>
+
+#ifdef GEGL_PROPERTIES
+
+property_int (radius, _("Radius"), 4)
+ description(_("Radius of row pixel region, (size will be radius*2+1)"))
+ value_range (0, 1000)
+ ui_range (0, 100)
+ ui_gamma (1.5)
+
+property_enum (orientation, _("Orientation"),
+ GeglOrientation, gegl_orientation,
+ GEGL_ORIENTATION_HORIZONTAL)
+ description (_("The orientation of the blur - hor/ver"))
+
+#else
+
+#define GEGL_OP_AREA_FILTER
+#define GEGL_OP_NAME boxblur_1d
+#define GEGL_OP_C_SOURCE boxblur-1d.c
+
+#include "gegl-op.h"
+
+static void
+prepare (GeglOperation *operation)
+{
+ GeglProperties *o = GEGL_PROPERTIES (operation);
+ GeglOperationAreaFilter *op_area = GEGL_OPERATION_AREA_FILTER (operation);
+ const Babl *space = gegl_operation_get_source_space (operation, "input");
+ const Babl *src_format = gegl_operation_get_source_format (operation, "input");
+ const char *format = "RaGaBaA float";
+
+ if (o->orientation == GEGL_ORIENTATION_HORIZONTAL)
+ {
+ op_area->left = op_area->right = o->radius;
+ }
+ else
+ {
+ op_area->top = op_area->bottom = o->radius;
+ }
+
+ if (src_format)
+ {
+ const Babl *model = babl_format_get_model (src_format);
+
+ if (babl_model_is (model, "RGB") || babl_model_is (model, "R'G'B'"))
+ {
+ format = "RGB float";
+ }
+ else if (babl_model_is (model, "Y") || babl_model_is (model, "Y'"))
+ {
+ format = "Y float";
+ }
+ else if (babl_model_is (model, "YA") || babl_model_is (model, "Y'A") ||
+ babl_model_is (model, "YaA") || babl_model_is (model, "Y'aA"))
+ {
+ format = "YaA float";
+ }
+ else if (babl_model_is (model, "cmyk"))
+ {
+ format = "cmyk float";
+ }
+ else if (babl_model_is (model, "CMYK"))
+ {
+ format = "CMYK float";
+ }
+ else if (babl_model_is (model, "cmykA") ||
+ babl_model_is (model, "camayakaA") ||
+ babl_model_is (model, "CMYKA") ||
+ babl_model_is (model, "CaMaYaKaA"))
+ {
+ format = "camayakaA float";
+ }
+ }
+
+ gegl_operation_set_format (operation, "input",
+ babl_format_with_space (format, space));
+ gegl_operation_set_format (operation, "output",
+ babl_format_with_space (format, space));
+}
+
+static GeglSplitStrategy
+get_split_strategy (GeglOperation *operation,
+ GeglOperationContext *context,
+ const gchar *output_prop,
+ const GeglRectangle *result,
+ gint level)
+{
+ GeglProperties *o = GEGL_PROPERTIES (operation);
+
+ if (o->orientation == GEGL_ORIENTATION_HORIZONTAL)
+ return GEGL_SPLIT_STRATEGY_HORIZONTAL;
+ else
+ return GEGL_SPLIT_STRATEGY_VERTICAL;
+}
+
+static GeglRectangle
+get_bounding_box (GeglOperation *operation)
+{
+ GeglRectangle *in_rect;
+
+ in_rect = gegl_operation_source_get_bounding_box (operation,"input");
+
+ if (! in_rect)
+ return *GEGL_RECTANGLE (0, 0, 0, 0);
+
+ return *in_rect;
+}
+
+static GeglRectangle
+get_cached_region (GeglOperation *operation,
+ const GeglRectangle *output_roi)
+{
+ GeglProperties *o = GEGL_PROPERTIES (operation);
+ const GeglRectangle in_rect = get_bounding_box (operation);
+ GeglRectangle cached_region = *output_roi;
+
+ if (! gegl_rectangle_is_empty (&in_rect) &&
+ ! gegl_rectangle_is_infinite_plane (&in_rect))
+ {
+ if (o->orientation == GEGL_ORIENTATION_HORIZONTAL)
+ {
+ cached_region.x = in_rect.x;
+ cached_region.width = in_rect.width;
+ }
+ else
+ {
+ cached_region.y = in_rect.y;
+ cached_region.height = in_rect.height;
+ }
+ }
+
+ return cached_region;
+}
+
+static inline void
+box_blur_1d (gfloat *src_buf,
+ gfloat *dst_buf,
+ gint dst_size,
+ gint radius,
+ gint n_components)
+{
+ gint n, i;
+ gint src_offset;
+ gint dst_offset = 0;
+ gint prev_rad = radius * n_components + n_components;
+ gint next_rad = radius * n_components;
+ gfloat rad1 = 1.f / (gfloat)(radius * 2 + 1);
+
+ for (i = 0; i < (2 * radius + 1); i++)
+ {
+ src_offset = i * n_components;
+ for (n = 0; n < n_components; n++)
+ {
+ dst_buf[dst_offset + n] += src_buf[src_offset + n] * rad1;
+ }
+ }
+
+ dst_offset += n_components;
+
+ for (i = 1; i < dst_size; i++)
+ {
+ src_offset = (i + radius) * n_components;
+ for (n = 0; n < n_components; n++)
+ {
+ dst_buf[dst_offset] = dst_buf[dst_offset - n_components]
+ - src_buf[src_offset - prev_rad] * rad1
+ + src_buf[src_offset + next_rad] * rad1;
+ src_offset++;
+ dst_offset++;
+ }
+ }
+}
+
+static gboolean
+process (GeglOperation *operation,
+ GeglBuffer *input,
+ GeglBuffer *output,
+ const GeglRectangle *roi,
+ gint level)
+{
+ GeglProperties *o = GEGL_PROPERTIES (operation);
+ const Babl *format = gegl_operation_get_format (operation, "input");
+ gint n_components = babl_format_get_n_components (format);
+ GeglRectangle src_rect;
+ GeglRectangle dst_rect;
+ gfloat *src_buf;
+ gfloat *dst_buf;
+
+ if (o->orientation == GEGL_ORIENTATION_HORIZONTAL)
+ {
+ src_rect.x = roi->x - o->radius;
+ src_rect.width = roi->width + 2 * o->radius;
+ src_rect.height = 1;
+
+ dst_rect.x = roi->x;
+ dst_rect.width = roi->width;
+ dst_rect.height = 1;
+
+ src_buf = g_new (gfloat, src_rect.width * n_components);
+
+ for (src_rect.y = roi->y; src_rect.y < roi->y + roi->height; src_rect.y++)
+ {
+ dst_rect.y = src_rect.y;
+ dst_buf = g_new0 (gfloat, dst_rect.width * n_components);
+
+ gegl_buffer_get (input, &src_rect, 1.0, format, src_buf,
+ GEGL_AUTO_ROWSTRIDE, GEGL_ABYSS_CLAMP);
+
+ box_blur_1d (src_buf, dst_buf, dst_rect.width, o->radius, n_components);
+
+ gegl_buffer_set (output, &dst_rect, 0, format, dst_buf,
+ GEGL_AUTO_ROWSTRIDE);
+ g_free (dst_buf);
+ }
+ }
+ else
+ {
+ src_rect.y = roi->y - o->radius;
+ src_rect.width = 1;
+ src_rect.height = roi->height + 2 * o->radius;
+
+ dst_rect.y = roi->y;
+ dst_rect.width = 1;
+ dst_rect.height = roi->height;
+
+ src_buf = g_new (gfloat, src_rect.height * n_components);
+
+ for (src_rect.x = roi->x; src_rect.x < roi->x + roi->width; src_rect.x++)
+ {
+ dst_rect.x = src_rect.x;
+ dst_buf = g_new0 (gfloat, dst_rect.height * n_components);
+
+ gegl_buffer_get (input, &src_rect, 1.0, format, src_buf,
+ GEGL_AUTO_ROWSTRIDE, GEGL_ABYSS_CLAMP);
+
+ box_blur_1d (src_buf, dst_buf, dst_rect.height, o->radius, n_components);
+
+ gegl_buffer_set (output, &dst_rect, 0, format, dst_buf,
+ GEGL_AUTO_ROWSTRIDE);
+ g_free (dst_buf);
+ }
+ }
+
+ g_free (src_buf);
+
+ return TRUE;
+}
+
+/* Pass-through if radius is zero
+ */
+static gboolean
+operation_process (GeglOperation *operation,
+ GeglOperationContext *context,
+ const gchar *output_prop,
+ const GeglRectangle *result,
+ gint level)
+{
+ GeglOperationClass *operation_class;
+ GeglProperties *o = GEGL_PROPERTIES (operation);
+
+ operation_class = GEGL_OPERATION_CLASS (gegl_op_parent_class);
+
+ if (! o->radius)
+ {
+ gpointer in = gegl_operation_context_get_object (context, "input");
+ gegl_operation_context_take_object (context, "output",
+ g_object_ref (G_OBJECT (in)));
+ return TRUE;
+ }
+
+ return operation_class->process (operation, context, output_prop, result,
+ gegl_operation_context_get_level (context));
+}
+
+static void
+gegl_op_class_init (GeglOpClass *klass)
+{
+ GeglOperationClass *operation_class;
+ GeglOperationFilterClass *filter_class;
+
+ operation_class = GEGL_OPERATION_CLASS (klass);
+ filter_class = GEGL_OPERATION_FILTER_CLASS (klass);
+
+ filter_class->get_split_strategy = get_split_strategy;
+ filter_class->process = process;
+
+ operation_class->get_bounding_box = get_bounding_box;
+ operation_class->get_cached_region = get_cached_region;
+ operation_class->opencl_support = FALSE;
+ operation_class->prepare = prepare;
+ operation_class->process = operation_process;
+
+ gegl_operation_class_set_keys (operation_class,
+ "name", "gegl:boxblur-1d",
+ "categories", "hidden:blur",
+ "title", _("1D Box Blur"),
+ "description", _("Blur resulting from averaging the colors of a row "
+ "neighbourhood."),
+ NULL);
+}
+
+#endif
\ No newline at end of file
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]