summaryrefslogtreecommitdiffstats
path: root/app/xcf/xcf-read.c
diff options
context:
space:
mode:
Diffstat (limited to 'app/xcf/xcf-read.c')
-rw-r--r--app/xcf/xcf-read.c274
1 files changed, 274 insertions, 0 deletions
diff --git a/app/xcf/xcf-read.c b/app/xcf/xcf-read.c
new file mode 100644
index 0000000..1c53436
--- /dev/null
+++ b/app/xcf/xcf-read.c
@@ -0,0 +1,274 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include <gio/gio.h>
+
+#include "libgimpbase/gimpbase.h"
+
+#include "core/core-types.h"
+
+#include "xcf-private.h"
+#include "xcf-read.h"
+
+#include "gimp-intl.h"
+
+
+#define MAX_XCF_STRING_LEN (16L * 1024 * 1024)
+
+
+guint
+xcf_read_int8 (XcfInfo *info,
+ guint8 *data,
+ gint count)
+{
+ gsize bytes_read = 0;
+
+ /* we allow for 'data == NULL && count == 0', which g_input_stream_read_all()
+ * rejects.
+ */
+ if (count > 0)
+ {
+ g_input_stream_read_all (info->input, data, count,
+ &bytes_read, NULL, NULL);
+
+ info->cp += bytes_read;
+ }
+
+ return bytes_read;
+}
+
+guint
+xcf_read_int16 (XcfInfo *info,
+ guint16 *data,
+ gint count)
+{
+ guint total = 0;
+
+ if (count > 0)
+ {
+ total += xcf_read_int8 (info, (guint8 *) data, count * 2);
+
+ while (count--)
+ {
+ *data = g_ntohs (*data);
+ data++;
+ }
+ }
+
+ return total;
+}
+
+guint
+xcf_read_int32 (XcfInfo *info,
+ guint32 *data,
+ gint count)
+{
+ guint total = 0;
+
+ if (count > 0)
+ {
+ total += xcf_read_int8 (info, (guint8 *) data, count * 4);
+
+ while (count--)
+ {
+ *data = g_ntohl (*data);
+ data++;
+ }
+ }
+
+ return total;
+}
+
+guint
+xcf_read_int64 (XcfInfo *info,
+ guint64 *data,
+ gint count)
+{
+ guint total = 0;
+
+ if (count > 0)
+ {
+ total += xcf_read_int8 (info, (guint8 *) data, count * 8);
+
+ while (count--)
+ {
+ *data = GINT64_FROM_BE (*data);
+ data++;
+ }
+ }
+
+ return total;
+}
+
+guint
+xcf_read_offset (XcfInfo *info,
+ goffset *data,
+ gint count)
+{
+ guint total = 0;
+
+ if (count > 0)
+ {
+ if (info->bytes_per_offset == 4)
+ {
+ gint32 *int_offsets = g_alloca (count * sizeof (gint32));
+
+ total += xcf_read_int8 (info, (guint8 *) int_offsets, count * 4);
+
+ while (count--)
+ {
+ *data = g_ntohl (*int_offsets);
+ int_offsets++;
+ data++;
+ }
+ }
+ else
+ {
+ total += xcf_read_int8 (info, (guint8 *) data, count * 8);
+
+ while (count--)
+ {
+ *data = GINT64_FROM_BE (*data);
+ data++;
+ }
+ }
+ }
+
+ return total;
+}
+
+guint
+xcf_read_float (XcfInfo *info,
+ gfloat *data,
+ gint count)
+{
+ return xcf_read_int32 (info, (guint32 *) ((void *) data), count);
+}
+
+guint
+xcf_read_string (XcfInfo *info,
+ gchar **data,
+ gint count)
+{
+ guint total = 0;
+ gint i;
+
+ for (i = 0; i < count; i++)
+ {
+ guint32 tmp;
+
+ total += xcf_read_int32 (info, &tmp, 1);
+
+ if (tmp > MAX_XCF_STRING_LEN)
+ {
+ g_warning ("Maximum string length (%ld bytes) exceeded. "
+ "Possibly corrupt XCF file.", MAX_XCF_STRING_LEN);
+ data[i] = NULL;
+ }
+ else if (tmp > 0)
+ {
+ gchar *str;
+
+ str = g_new (gchar, tmp);
+ total += xcf_read_int8 (info, (guint8*) str, tmp);
+
+ if (str[tmp - 1] != '\0')
+ str[tmp - 1] = '\0';
+
+ data[i] = gimp_any_to_utf8 (str, -1,
+ _("Invalid UTF-8 string in XCF file"));
+
+ g_free (str);
+ }
+ else
+ {
+ data[i] = NULL;
+ }
+ }
+
+ return total;
+}
+
+guint
+xcf_read_component (XcfInfo *info,
+ gint bpc,
+ guint8 *data,
+ gint count)
+{
+ switch (bpc)
+ {
+ case 1:
+ return xcf_read_int8 (info, data, count);
+
+ case 2:
+ return xcf_read_int16 (info, (guint16 *) data, count);
+
+ case 4:
+ return xcf_read_int32 (info, (guint32 *) data, count);
+
+ case 8:
+ return xcf_read_int64 (info, (guint64 *) data, count);
+
+ default:
+ break;
+ }
+
+ return 0;
+}
+
+void
+xcf_read_from_be (gint bpc,
+ guint8 *data,
+ gint count)
+{
+ gint i;
+
+ switch (bpc)
+ {
+ case 1:
+ break;
+
+ case 2:
+ {
+ guint16 *d = (guint16 *) data;
+
+ for (i = 0; i < count; i++)
+ d[i] = g_ntohs (d[i]);
+ }
+ break;
+
+ case 4:
+ {
+ guint32 *d = (guint32 *) data;
+
+ for (i = 0; i < count; i++)
+ d[i] = g_ntohl (d[i]);
+ }
+ break;
+
+ case 8:
+ {
+ guint64 *d = (guint64 *) data;
+
+ for (i = 0; i < count; i++)
+ d[i] = GINT64_FROM_BE (d[i]);
+ }
+ break;
+ }
+}