summaryrefslogtreecommitdiff
path: root/tests/pixbuf-gif.c
diff options
context:
space:
mode:
Diffstat (limited to 'tests/pixbuf-gif.c')
-rw-r--r--tests/pixbuf-gif.c222
1 files changed, 222 insertions, 0 deletions
diff --git a/tests/pixbuf-gif.c b/tests/pixbuf-gif.c
new file mode 100644
index 000000000..53b58d587
--- /dev/null
+++ b/tests/pixbuf-gif.c
@@ -0,0 +1,222 @@
+/* -*- Mode: C; c-basic-offset: 2; -*- */
+/* GdkPixbuf library - test loaders
+ *
+ * Copyright (C) 2018 Canonical Ltd.
+ *
+ * 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 2 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 <http://www.gnu.org/licenses/>.
+ */
+
+#include "gdk-pixbuf/gdk-pixbuf.h"
+#include "test-common.h"
+#include <string.h>
+
+const gchar *
+get_test_path (const gchar *filename)
+{
+ return g_test_get_filename (G_TEST_DIST, "test-images", "gif-test-suite", filename, NULL);
+}
+
+static gboolean
+pixels_match (GBytes *pixels0, GBytes *pixels1)
+{
+ gsize size0, size1, i;
+ guint8 *data0, *data1;
+
+ data0 = g_bytes_get_data (pixels0, &size0);
+ data1 = g_bytes_get_data (pixels1, &size1);
+
+ if (size0 != size1)
+ return FALSE;
+
+ for (i = 0; i < size0; i += 4)
+ {
+ guint8 red0, green0, blue0, alpha0;
+ guint8 red1, green1, blue1, alpha1;
+
+ red0 = data0[i + 0];
+ green0 = data0[i + 1];
+ blue0 = data0[i + 2];
+ alpha0 = data0[i + 3];
+ red1 = data1[i + 0];
+ green1 = data1[i + 1];
+ blue1 = data1[i + 2];
+ alpha1 = data1[i + 3];
+ if (alpha0 == 0 && alpha1 == 0)
+ ; /* Transparent, don't care what the RGB is set to */
+ else if (red0 != red1 || blue0 != blue1 || green0 != green1 || alpha0 != alpha1)
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+static void
+run_gif_test (gconstpointer data)
+{
+ const gchar *name = data;
+ GKeyFile *config_file;
+ gchar *config_filename, *input_filename;
+ gint width, height;
+ GFile *input_file;
+ GBytes *input_bytes;
+ GdkPixbufLoader *loader;
+ GdkPixbufAnimation *animation = NULL;
+ GdkPixbufAnimationIter *iter = NULL;
+ GTimeVal animation_time;
+ GStrv frames;
+ int i;
+ GError *error = NULL;
+
+ config_file = g_key_file_new ();
+ g_key_file_set_list_separator (config_file, ',');
+ config_filename = g_strdup_printf ("%s.conf", name);
+ g_key_file_load_from_file (config_file, get_test_path (config_filename), G_KEY_FILE_NONE, &error);
+ g_free (config_filename);
+ g_assert_no_error (error);
+
+ input_filename = g_key_file_get_string (config_file, "config", "input", &error);
+ g_assert_no_error (error);
+ input_file = g_file_new_for_path (get_test_path (input_filename));
+ g_free (input_filename);
+ input_bytes = g_file_load_bytes (input_file, NULL, NULL, &error);
+ g_clear_object (&input_file);
+ g_assert_no_error (error);
+
+ width = g_key_file_get_integer (config_file, "config", "width", &error);
+ g_assert_no_error (error);
+ height = g_key_file_get_integer (config_file, "config", "height", &error);
+ g_assert_no_error (error);
+
+ loader = gdk_pixbuf_loader_new ();
+ gdk_pixbuf_loader_write_bytes (loader, input_bytes, &error);
+ g_clear_pointer (&input_bytes, g_bytes_unref);
+ g_assert_no_error (error);
+
+ gdk_pixbuf_loader_close (loader, &error);
+ if (width == 0 || height == 0) {
+ g_assert_error (error, GDK_PIXBUF_ERROR, GDK_PIXBUF_ERROR_CORRUPT_IMAGE);
+ g_clear_error (&error);
+ }
+ else {
+ g_assert_no_error (error);
+ animation = gdk_pixbuf_loader_get_animation (loader);
+ g_assert_nonnull (animation);
+ }
+
+ frames = g_key_file_get_string_list (config_file, "config", "frames", NULL, &error);
+ g_assert_no_error (error);
+ animation_time.tv_sec = 0;
+ animation_time.tv_usec = 0;
+ for (i = 0; frames[i]; i++)
+ {
+ const gchar *frame = frames[i];
+ GdkPixbuf *pixbuf;
+ gint delay_time, expected_delay_time = 0;
+ gchar *pixels_filename;
+ GFile *pixels_file;
+ GBytes *expected_pixels, *pixels;
+
+ g_assert_nonnull (animation);
+
+ if (iter == NULL)
+ iter = gdk_pixbuf_animation_get_iter (animation, &animation_time);
+ else
+ gdk_pixbuf_animation_iter_advance (iter, &animation_time);
+ delay_time = gdk_pixbuf_animation_iter_get_delay_time (iter);
+ g_time_val_add (&animation_time, gdk_pixbuf_animation_iter_get_delay_time (iter) * 1000);
+
+ if (g_key_file_has_key (config_file, frame, "delay", &error))
+ expected_delay_time = g_key_file_get_integer (config_file, frame, "delay", &error) * 10;
+ g_assert_no_error (error);
+
+ /* gdk-pixbuf uses 20ms minimum delay when no delay specified */
+ if (expected_delay_time < 20)
+ expected_delay_time = 20;
+
+ /* gdk-pixbuf uses 100ms minimum when using Graphics Control Extension */
+ if (strcmp (name, "transparent") == 0 ||
+ strcmp (name, "invalid-transparent") == 0 ||
+ strcmp (name, "disabled-transparent") == 0 ||
+ strcmp (name, "animation-zero-delays") == 0)
+ if (expected_delay_time < 100)
+ expected_delay_time = 100;
+
+ g_assert_cmpint (delay_time, ==, expected_delay_time);
+
+ pixbuf = gdk_pixbuf_animation_iter_get_pixbuf (iter);
+
+ g_assert_cmpint (width, ==, gdk_pixbuf_get_width (pixbuf));
+ g_assert_cmpint (height, ==, gdk_pixbuf_get_height (pixbuf));
+
+ pixels_filename = g_key_file_get_string (config_file, frame, "pixels", &error);
+ g_assert_no_error (error);
+ pixels_file = g_file_new_for_path (get_test_path (pixels_filename));
+ g_free (pixels_filename);
+ expected_pixels = g_file_load_bytes (pixels_file, NULL, NULL, &error);
+ g_clear_object (&pixels_file);
+ g_assert_no_error (error);
+
+ g_assert_cmpint (gdk_pixbuf_get_colorspace (pixbuf), ==, GDK_COLORSPACE_RGB);
+ g_assert_cmpint (gdk_pixbuf_get_n_channels (pixbuf), ==, 4);
+ g_assert (gdk_pixbuf_get_has_alpha (pixbuf));
+ g_assert_cmpint (gdk_pixbuf_get_rowstride (pixbuf), ==, width * 4);
+ pixels = g_bytes_new_static (gdk_pixbuf_read_pixels (pixbuf), gdk_pixbuf_get_byte_length (pixbuf));
+ g_assert (pixels_match (pixels, expected_pixels));
+ g_clear_pointer (&pixels, g_bytes_unref);
+ g_clear_pointer (&expected_pixels, g_bytes_unref);
+ }
+ g_strfreev (frames);
+ g_clear_object (&iter);
+
+ /* FIXME: We should check here if there's more frames than we were expecting, but gdk-pixbuf doesn't return this information */
+
+ g_clear_object (&loader);
+ g_clear_pointer (&config_file, g_key_file_unref);
+}
+
+int
+main (int argc, char **argv)
+{
+ gchar *path, *contents;
+ GStrv lines;
+ int i, result;
+ GError *error = NULL;
+
+ g_test_init (&argc, &argv, NULL);
+
+ path = g_build_filename (g_test_get_dir (G_TEST_DIST), "test-images", "gif-test-suite", "TESTS", NULL);
+ g_file_get_contents (path, &contents, NULL, &error);
+ g_free (path);
+ g_assert_no_error (error);
+ lines = g_strsplit (contents, "\n", -1);
+ g_free (contents);
+ for (i = 0; lines[i]; i++)
+ {
+ const gchar *name = g_strstrip (lines[i]);
+ gchar *test_name;
+
+ if (g_strcmp0 (name, "") == 0 || name[0] == '#')
+ continue;
+
+ test_name = g_strdup_printf ("/pixbuf/gif/%s", name);
+ g_test_add_data_func (test_name, name, run_gif_test);
+ g_free (test_name);
+ }
+
+ result = g_test_run ();
+
+ g_strfreev (lines);
+
+ return result;
+}