[gegl] tiff: Add a gegl:tiff-save operation
- From: Øyvind Kolås <ok src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gegl] tiff: Add a gegl:tiff-save operation
- Date: Sat, 21 Nov 2015 03:23:30 +0000 (UTC)
commit 319a91e3f09a58ed3fd42a71d256725f0efdd7b5
Author: Martin Blanchard <tchaik gmx com>
Date: Fri Oct 2 22:38:58 2015 +0200
tiff: Add a gegl:tiff-save operation
New GIO based TIFF file saver using libtiff. Operation will try to
preserve the original buffer's data as much as possible, avoiding
conversions. Based on GIMP's file-tiff-save plugin.
https://bugzilla.gnome.org/show_bug.cgi?id=739124
operations/external/Makefile.am | 5 +
operations/external/tiff-save.c | 600 +++++++++++++++++++++++++++++++++++++++
2 files changed, 605 insertions(+), 0 deletions(-)
---
diff --git a/operations/external/Makefile.am b/operations/external/Makefile.am
index 89633bb..4e40a39 100644
--- a/operations/external/Makefile.am
+++ b/operations/external/Makefile.am
@@ -134,6 +134,11 @@ ops += tiff-load.la
tiff_load_la_SOURCES = tiff-load.c
tiff_load_la_LIBADD = $(op_libs) $(TIFF_LIBS)
tiff_load_la_CFLAGS = $(AM_CFLAGS) $(TIFF_CFLAGS)
+
+ops += tiff-save.la
+tiff_save_la_SOURCES = tiff-save.c
+tiff_save_la_LIBADD = $(op_libs) $(TIFF_LIBS)
+tiff_save_la_CFLAGS = $(AM_CFLAGS) $(TIFF_CFLAGS)
endif
if HAVE_WEBP
diff --git a/operations/external/tiff-save.c b/operations/external/tiff-save.c
new file mode 100644
index 0000000..e8d3a45
--- /dev/null
+++ b/operations/external/tiff-save.c
@@ -0,0 +1,600 @@
+/* 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 <http://www.gnu.org/licenses/>.
+ *
+ * Copyright 2015 Martin Blanchard <tchaik gmx com>
+ */
+
+#include "config.h"
+#include <glib/gi18n-lib.h>
+
+
+#ifdef GEGL_PROPERTIES
+
+property_file_path (path, _("File"), "")
+ description (_("Target path and filename, use '-' for stdout"))
+
+#else
+
+#define GEGL_OP_SINK
+#define GEGL_OP_C_SOURCE tiff-save.c
+
+#include <gegl-op.h>
+#include <gegl-gio-private.h>
+#include <tiffio.h>
+
+typedef struct
+{
+ GFile *file;
+ GOutputStream *stream;
+ gboolean can_seek;
+
+ gchar *buffer;
+ gsize allocated;
+ gsize position;
+
+ TIFF *tiff;
+} Priv;
+
+static void
+cleanup(GeglOperation *operation)
+{
+ GeglProperties *o = GEGL_PROPERTIES(operation);
+ Priv *p = (Priv*) o->user_data;
+
+ if (p != NULL)
+ {
+ if (p->tiff != NULL)
+ TIFFClose(p->tiff);
+ else if (p->stream != NULL)
+ g_output_stream_close(G_OUTPUT_STREAM(p->stream), NULL, NULL);
+ if (p->stream != NULL)
+ g_clear_object(&p->stream);
+ p->tiff = NULL;
+
+ if (p->file != NULL)
+ g_clear_object(&p->file);
+ }
+}
+
+static GSeekType
+lseek_to_seek_type(int whence)
+{
+ switch (whence)
+ {
+ default:
+ case SEEK_SET:
+ return G_SEEK_SET;
+
+ case SEEK_CUR:
+ return G_SEEK_CUR;
+
+ case G_SEEK_END:
+ return SEEK_END;
+ }
+}
+
+static tsize_t
+read_from_stream(thandle_t handle,
+ tdata_t buffer,
+ tsize_t size)
+{
+ Priv *p = (Priv*) handle;
+
+ g_assert(p->stream && FALSE);
+
+ return -1;
+}
+
+static tsize_t
+write_to_stream(thandle_t handle,
+ tdata_t buffer,
+ tsize_t size)
+{
+ Priv *p = (Priv*) handle;
+ GError *error = NULL;
+ gchar *new_buffer;
+ gsize new_size;
+ gssize written = -1;
+
+ g_assert(p->stream);
+
+ if (p->can_seek)
+ {
+ written = g_output_stream_write(G_OUTPUT_STREAM(p->stream),
+ (void *) buffer, (gsize) size,
+ NULL, &error);
+ if (written < 0)
+ {
+ g_warning(error->message);
+ g_error_free(error);
+ }
+ }
+ else
+ {
+ if (p->position + size > p->allocated)
+ {
+ new_size = p->position + size;
+ new_buffer = g_try_realloc(p->buffer, new_size);
+ if (!new_buffer)
+ return -1;
+
+ p->allocated = new_size;
+ p->buffer = new_buffer;
+ }
+
+ g_assert(p->position + size >= p->allocated);
+
+ memcpy(p->buffer + p->position, buffer, size);
+ p->position += size;
+ written = size;
+ }
+
+ return (tsize_t) written;
+}
+
+static toff_t
+seek_in_stream(thandle_t handle,
+ toff_t offset,
+ int whence)
+{
+ Priv *p = (Priv*) handle;
+ GError *error = NULL;
+ gboolean sought = FALSE;
+ goffset position = -1;
+
+ g_assert(p->stream);
+
+ if (p->can_seek)
+ {
+ sought = g_seekable_seek(G_SEEKABLE(p->stream),
+ (goffset) offset, lseek_to_seek_type(whence),
+ NULL, &error);
+ if (sought)
+ position = g_seekable_tell(G_SEEKABLE(p->stream));
+ else
+ {
+ g_warning(error->message);
+ g_error_free(error);
+ }
+ }
+ else
+ {
+ switch (whence)
+ {
+ default:
+ case SEEK_SET:
+ if (offset <= p->allocated)
+ position = p->position = offset;
+ break;
+
+ case SEEK_CUR:
+ if (p->position + offset <= p->allocated)
+ position = p->position += offset;
+ break;
+
+ case G_SEEK_END:
+ position = p->position = p->allocated + offset;
+ break;
+ }
+ }
+
+ return (toff_t) position;
+}
+
+static int
+close_stream(thandle_t handle)
+{
+ Priv *p = (Priv*) handle;
+ GError *error = NULL;
+ gsize total = 0;
+ gboolean closed = FALSE;
+
+ g_assert(p->stream);
+
+ /* !can_seek: file content is now fully cached, time to write it down. */
+ if (!p->can_seek && p->buffer != NULL && p->allocated > 0)
+ {
+ while (total < p->allocated)
+ {
+ gssize written;
+
+ written = g_output_stream_write(G_OUTPUT_STREAM(p->stream),
+ (void *) (p->buffer + total),
+ p->allocated - total,
+ NULL, &error);
+ if (written < 0)
+ {
+ g_warning(error->message);
+ g_error_free(error);
+ break;
+ }
+
+ total += written;
+ }
+ }
+
+ closed = g_output_stream_close(G_OUTPUT_STREAM(p->stream),
+ NULL, &error);
+ if (!closed)
+ {
+ g_warning(error->message);
+ g_error_free(error);
+ }
+
+ g_clear_object(&p->stream);
+
+ p->position = 0;
+
+ if (p->buffer != NULL)
+ g_free(p->buffer);
+ p->buffer = NULL;
+
+ p->allocated = 0;
+
+ return (closed) ? 0 : -1;
+}
+
+static toff_t
+get_file_size(thandle_t handle)
+{
+ Priv *p = (Priv*) handle;
+ GError *error = NULL;
+ GFileInfo *info;
+ goffset size;
+
+ g_assert(p->stream);
+
+ size = p->allocated;
+
+ if (p->file != NULL)
+ {
+ info = g_file_query_info(p->file,
+ G_FILE_ATTRIBUTE_STANDARD_SIZE,
+ G_FILE_QUERY_INFO_NONE,
+ NULL, &error);
+ if (info == NULL)
+ {
+ g_warning(error->message);
+ g_error_free(error);
+ }
+ else
+ {
+ if (g_file_info_has_attribute(info, G_FILE_ATTRIBUTE_STANDARD_SIZE))
+ size = g_file_info_get_size(info);
+ g_object_unref(info);
+ }
+ }
+
+ return (toff_t) size;
+}
+
+static gint
+save_contiguous(GeglOperation *operation,
+ GeglBuffer *input,
+ const GeglRectangle *result,
+ const Babl *format)
+{
+ GeglProperties *o = GEGL_PROPERTIES(operation);
+ Priv *p = (Priv*) o->user_data;
+ gint bytes_per_pixel, bytes_per_row;
+ gint tile_width = result->width;
+ gint tile_height = result->height;
+ guchar *buffer;
+ gint x, y;
+
+ g_return_val_if_fail(p->tiff != NULL, -1);
+
+ bytes_per_pixel = babl_format_get_bytes_per_pixel(format);
+ bytes_per_row = bytes_per_pixel * tile_width;
+
+ buffer = g_try_new(guchar, bytes_per_row * tile_height);
+
+ g_assert(buffer != NULL);
+
+ for (y = result->y; y < result->y + tile_height; y += tile_height)
+ {
+ for (x = result->x; x < result->x + tile_width; x += tile_width)
+ {
+ GeglRectangle tile = { x, y, tile_width, tile_height };
+ gint row;
+
+ gegl_buffer_get(input, &tile, 1.0, format, buffer,
+ GEGL_AUTO_ROWSTRIDE, GEGL_ABYSS_NONE);
+
+ for (row = y; row < y + tile_height; row++)
+ {
+ guchar *tile_row = buffer + (bytes_per_row * (row - y));
+ gint written;
+
+ written = TIFFWriteScanline(p->tiff, tile_row, row, 0);
+
+ if (!written)
+ {
+ g_critical("failed a scanline write on row %d", row);
+ continue;
+ }
+ }
+ }
+ }
+
+ TIFFFlushData(p->tiff);
+
+ g_free(buffer);
+ return 0;
+}
+
+static int
+export_tiff (GeglOperation *operation,
+ GeglBuffer *input,
+ const GeglRectangle *result)
+{
+ GeglProperties *o = GEGL_PROPERTIES(operation);
+ Priv *p = (Priv*) o->user_data;
+ gshort color_space, compression = COMPRESSION_NONE;
+ gushort bits_per_sample, samples_per_pixel;
+ gboolean has_alpha, alpha_is_premultiplied = FALSE;
+ gushort sample_format, predictor = 0;
+ gushort extra_types[1];
+ glong rows_per_stripe = 1;
+ gint bytes_per_row;
+ const Babl *type, *model;
+ gchar format_string[32];
+ const Babl *format;
+
+ g_return_val_if_fail(p->tiff != NULL, -1);
+
+ TIFFSetField(p->tiff, TIFFTAG_SUBFILETYPE, 0);
+ TIFFSetField(p->tiff, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
+
+ TIFFSetField(p->tiff, TIFFTAG_IMAGEWIDTH, result->width);
+ TIFFSetField(p->tiff, TIFFTAG_IMAGELENGTH, result->height);
+
+ format = gegl_buffer_get_format(input);
+
+ model = babl_format_get_model(format);
+ type = babl_format_get_type(format, 0);
+
+ if (model == babl_model("Y") || model == babl_model("Y'"))
+ {
+ has_alpha = FALSE;
+ color_space = PHOTOMETRIC_MINISBLACK;
+ model = babl_model("Y'");
+ samples_per_pixel = 1;
+ }
+ else if (model == babl_model("YA") || model == babl_model("Y'A"))
+ {
+ has_alpha = TRUE;
+ alpha_is_premultiplied = FALSE;
+ color_space = PHOTOMETRIC_MINISBLACK;
+ model = babl_model("Y'A");
+ samples_per_pixel = 2;
+ }
+ else if (model == babl_model("YaA") || model == babl_model("Y'aA"))
+ {
+ has_alpha = TRUE;
+ alpha_is_premultiplied = TRUE;
+ color_space = PHOTOMETRIC_MINISBLACK;
+ model = babl_model("Y'aA");
+ samples_per_pixel = 2;
+ }
+ else if (model == babl_model("RGB") || model == babl_model("R'G'B'"))
+ {
+ has_alpha = FALSE;
+ color_space = PHOTOMETRIC_RGB;
+ model = babl_model("R'G'B'");
+ samples_per_pixel = 3;
+ predictor = 2;
+ }
+ else if (model == babl_model("RGBA") || model == babl_model("R'G'B'A"))
+ {
+ has_alpha = TRUE;
+ alpha_is_premultiplied = FALSE;
+ color_space = PHOTOMETRIC_RGB;
+ model = babl_model("R'G'B'A");
+ samples_per_pixel = 4;
+ predictor = 2;
+ }
+ else if (model == babl_model("RaGaBaA") || model == babl_model("R'aG'aB'aA"))
+ {
+ has_alpha = TRUE;
+ alpha_is_premultiplied = TRUE;
+ color_space = PHOTOMETRIC_RGB;
+ model = babl_model("R'aG'aB'aA");
+ samples_per_pixel = 4;
+ predictor = 2;
+ }
+ else
+ {
+ g_warning("color space not supported: %s", babl_get_name(model));
+
+ has_alpha = TRUE;
+ alpha_is_premultiplied = TRUE;
+ color_space = PHOTOMETRIC_RGB;
+ model = babl_model("R'aG'aB'aA");
+ samples_per_pixel = 4;
+ predictor = 2;
+ }
+
+ TIFFSetField(p->tiff, TIFFTAG_PHOTOMETRIC, color_space);
+ TIFFSetField(p->tiff, TIFFTAG_SAMPLESPERPIXEL, samples_per_pixel);
+ TIFFSetField(p->tiff, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
+
+ if (has_alpha)
+ {
+ if (alpha_is_premultiplied)
+ extra_types[0] = EXTRASAMPLE_ASSOCALPHA;
+ else
+ extra_types[0] = EXTRASAMPLE_UNASSALPHA;
+
+ TIFFSetField(p->tiff, TIFFTAG_EXTRASAMPLES, 1, extra_types);
+ }
+
+ if (predictor != 0)
+ {
+ if (compression == COMPRESSION_LZW)
+ TIFFSetField(p->tiff, TIFFTAG_PREDICTOR, predictor);
+ else if (compression == COMPRESSION_ADOBE_DEFLATE)
+ TIFFSetField(p->tiff, TIFFTAG_PREDICTOR, predictor);
+ }
+
+ if (type == babl_type("u8"))
+ {
+ sample_format = SAMPLEFORMAT_UINT;
+ bits_per_sample = 8;
+ }
+ else if (type == babl_type("half"))
+ {
+ sample_format = SAMPLEFORMAT_IEEEFP;
+ bits_per_sample = 16;
+ }
+ else if (type == babl_type("u16"))
+ {
+ sample_format = SAMPLEFORMAT_UINT;
+ bits_per_sample = 16;
+ }
+ else if (type == babl_type("float"))
+ {
+ sample_format = SAMPLEFORMAT_IEEEFP;
+ bits_per_sample = 32;
+ }
+ else if (type == babl_type("u32"))
+ {
+ sample_format = SAMPLEFORMAT_UINT;
+ bits_per_sample = 32;
+ }
+ else if (type == babl_type("double"))
+ {
+ sample_format = SAMPLEFORMAT_IEEEFP;
+ bits_per_sample = 64;
+ }
+ else
+ {
+ g_warning("sample format not supported: %s", babl_get_name(type));
+
+ sample_format = SAMPLEFORMAT_UINT;
+ type = babl_type("u8");
+ bits_per_sample = 8;
+ }
+
+ TIFFSetField(p->tiff, TIFFTAG_BITSPERSAMPLE, bits_per_sample);
+ TIFFSetField(p->tiff, TIFFTAG_SAMPLEFORMAT, sample_format);
+
+ TIFFSetField(p->tiff, TIFFTAG_COMPRESSION, compression);
+
+ if ((compression == COMPRESSION_CCITTFAX3 ||
+ compression == COMPRESSION_CCITTFAX4) &&
+ (bits_per_sample != 1 || samples_per_pixel != 1))
+ {
+ g_critical("only monochrome pictures can be compressed "
+ "with \"CCITT Group 4\" or \"CCITT Group 3\"");
+ return -1;
+ }
+
+ g_snprintf(format_string, 32, "%s %s",
+ babl_get_name(model), babl_get_name(type));
+
+ format = babl_format(format_string);
+
+ /* "Choose RowsPerStrip such that each strip is about 8K bytes." */
+ bytes_per_row = babl_format_get_bytes_per_pixel(format) * result->width;
+ while (bytes_per_row * rows_per_stripe <= 8192)
+ rows_per_stripe++;
+
+ rows_per_stripe = MIN(rows_per_stripe, result->height);
+
+ TIFFSetField(p->tiff, TIFFTAG_ROWSPERSTRIP, rows_per_stripe);
+
+ return save_contiguous(operation, input, result, format);
+}
+
+static gboolean
+process(GeglOperation *operation,
+ GeglBuffer *input,
+ const GeglRectangle *result,
+ int level)
+{
+ GeglProperties *o = GEGL_PROPERTIES(operation);
+ Priv *p = g_new0(Priv, 1);
+ gboolean status = TRUE;
+ GError *error = NULL;
+
+ g_assert(p != NULL);
+
+ o->user_data = (void*) p;
+
+ p->stream = gegl_gio_open_output_stream(NULL, o->path, &p->file, &error);
+ if (p->stream != NULL && p->file != NULL)
+ p->can_seek = g_seekable_can_seek(G_SEEKABLE(p->stream));
+ if (p->stream == NULL)
+ {
+ status = FALSE;
+ g_warning(error->message);
+ goto cleanup;
+ }
+
+ p->tiff = TIFFClientOpen("GEGL-tiff-save", "w", (thandle_t) p,
+ read_from_stream, write_to_stream,
+ seek_in_stream, close_stream,
+ get_file_size, NULL, NULL);
+ if (p->tiff == NULL)
+ {
+ status = FALSE;
+ g_warning("failed to openi TIFF from %s", o->path);
+ goto cleanup;
+ }
+
+ if (export_tiff(operation, input, result))
+ {
+ status = FALSE;
+ g_warning("could not export TIFF file");
+ goto cleanup;
+ }
+
+cleanup:
+ cleanup(operation);
+ if (o->user_data != NULL)
+ g_free(o->user_data);
+ o->user_data = NULL;
+
+ if (error != NULL)
+ g_error_free(error);
+
+ return status;
+}
+
+static void
+gegl_op_class_init(GeglOpClass *klass)
+{
+ GeglOperationClass *operation_class;
+ GeglOperationSinkClass *sink_class;
+
+ operation_class = GEGL_OPERATION_CLASS(klass);
+ sink_class = GEGL_OPERATION_SINK_CLASS(klass);
+
+ sink_class->needs_full = TRUE;
+ sink_class->process = process;
+
+ gegl_operation_class_set_keys(operation_class,
+ "name", "gegl:tiff-save",
+ "title", _("TIFF File Saver"),
+ "categories", "output",
+ "description", _("TIFF image saver using libtiff"),
+ NULL);
+
+ gegl_extension_handler_register_saver(".tiff", "gegl:tiff-save");
+ gegl_extension_handler_register_saver(".tif", "gegl:tiff-save");
+}
+
+#endif
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]