Re: gdk-pixbuf/test-loaders
- From: Soeren Sandmann <sandmann daimi au dk>
- To: Matthias Clasen <maclas gmx de>
- Cc: gtk-devel-list gnome org
- Subject: Re: gdk-pixbuf/test-loaders
- Date: 19 Mar 2002 03:56:16 +0100
Soeren Sandmann <sandmann daimi au dk> writes:
> Instead of doing what you suggest, why not split the program into four
> different ones:
In fact I have done this now. Attached are the source to the four
programs. There is also a program to dump the contents of
test-images.h to separate files.
Is it okay for me to create a test directory under gtk+/tests
containing these programs and the extraced images from test-images.h?
/* -*- Mode: C; c-basic-offset: 2; -*- */
/* GdkPixbuf library - assault loaders with random data
*
* Copyright (C) 2001 Søren Sandmann (sandmann daimi au dk)
*
* 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
*/
#include <config.h>
#include "gdk-pixbuf.h"
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <string.h>
static void
assault (const gchar *header, gsize header_size, int n_images)
{
FILE *f;
enum { N_CHARACTERS = 10000 };
int j;
for (j = 0; j < n_images; ++j)
{
GError *err = NULL;
int i;
GdkPixbufLoader *loader;
f = fopen ("test-random-image", "w");
if (!f)
{
perror ("fopen");
exit (EXIT_FAILURE);
}
loader = gdk_pixbuf_loader_new ();
gdk_pixbuf_loader_write (loader, header, header_size, &err);
if (err)
{
g_error_free (err);
continue;
}
for (i = 0; i < N_CHARACTERS; ++i)
{
int r = g_random_int ();
fwrite (&r, 1, sizeof (r), f);
if (ferror (f))
{
perror ("fwrite");
exit (EXIT_FAILURE);
}
gdk_pixbuf_loader_write (loader, (guchar *)&r, sizeof (r), &err);
if (err)
{
g_error_free (err);
err = NULL;
break;
}
}
fclose (f);
gdk_pixbuf_loader_close (loader, &err);
if (err)
{
g_error_free (err);
err = NULL;
}
g_object_unref (loader);
}
}
static void
write_seed (int seed)
{
FILE *f;
/* write this so you can reproduce failed tests */
f = fopen ("test-random-seed", "w");
if (!f)
{
perror ("fopen");
exit (EXIT_FAILURE);
}
if (fprintf (f, "%d\n", seed) < 0)
{
perror ("fprintf");
exit (EXIT_FAILURE);
}
if (fclose (f) < 0)
{
perror ("fclose");
exit (EXIT_FAILURE);
}
g_print ("seed: %d\n", seed);
}
int
main (int argc, char **argv)
{
int seed;
if (argc > 1)
seed = atoi (argv[1]);
else
{
seed = time (NULL);
write_seed (seed);
}
g_print ("the last tested image is saved to the file \"test-random-image\"\n\n");
g_type_init ();
g_log_set_always_fatal (G_LOG_LEVEL_WARNING | G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL);
g_random_set_seed (seed);
#define GIF_HEADER 'G', 'I', 'F', '8', '9', 'a'
#define PNG_HEADER 0x89, 'P', 'N', 'G', 0x0d, 0x0a, 0x1a, 0x0a
#define TIFF1_HEADER 'M', 'M', 0x00, 0x2a
#define TIFF2_HEADER 'I', 'I', 0x2a, 0x00
#define JPEG_HEADER 0xFF, 0xd8
#define PNM_HEADER 'P', '6'
#define XBM_HEADER '#', 'd', 'e', 'f', 'i', 'n', 'e', ' '
#define BMP_HEADER 'B', 'M'
#define XPM_HEADER '/', '*', ' ', 'X', 'P', 'M', ' ', '*', '/'
#define RAS_HEADER 0x59, 0xA6, 0x6A, 0x95
#define TEST_RANDOM(header, n_img) \
do { \
static guchar h[] = { header }; \
g_print (#header); \
fflush (stdout); \
assault (h, sizeof (h), n_img); \
g_print ("\t\tpassed\n"); \
} while (0)
for (;;)
{
TEST_RANDOM (GIF_HEADER, 150);
TEST_RANDOM (PNG_HEADER, 110);
TEST_RANDOM (JPEG_HEADER, 800);
TEST_RANDOM (TIFF1_HEADER, 150);
TEST_RANDOM (TIFF2_HEADER, 150);
TEST_RANDOM (PNM_HEADER, 150);
TEST_RANDOM (XBM_HEADER, 150);
TEST_RANDOM (BMP_HEADER, 150);
TEST_RANDOM (XPM_HEADER, 150);
TEST_RANDOM (RAS_HEADER, 300);
g_print ("===========================\n");
}
return 0;
}
/* -*- Mode: C; c-basic-offset: 2; -*- */
/* GdkPixbuf library - test loaders
*
* Copyright (C) 2001 Søren Sandmann (sandmann daimi au dk)
*
* 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
*/
#include <config.h>
#include "gdk-pixbuf.h"
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <string.h>
static void
disaster (const char *what)
{
perror (what);
exit (EXIT_FAILURE);
}
static void
randomly_modify (const guchar *image, guint size)
{
int i;
FILE *f;
guchar *img_copy = g_malloc (size);
for (i = 0; i < size; i++)
img_copy [i] = image[i];
f = fopen ("test-randomly-modified-image", "w");
if (!f)
disaster ("fopen");
fwrite (img_copy, size, 1, f);
if (ferror (f))
disaster ("fwrite");
fclose (f);
for (i = 0; i < size / 4; i++)
{
GdkPixbufLoader *loader;
guint index = g_random_int_range (0, size);
guchar byte = g_random_int_range (0, 256);
img_copy[index] = byte;
loader = gdk_pixbuf_loader_new ();
gdk_pixbuf_loader_write (loader, img_copy, size, NULL);
gdk_pixbuf_loader_close (loader, NULL);
}
g_free (img_copy);
}
static void
write_seed (int seed)
{
FILE *f;
/* write this so you can reproduce failed tests */
f = fopen ("test-randomly-modified-seed", "w");
if (!f)
disaster ("fopen");
if (fprintf (f, "%d\n", seed) < 0)
disaster ("fprintf");
if (fclose (f) < 0)
disaster ("fclose");
g_print ("seed: %d\n", seed);
}
int
main (int argc, char **argv)
{
int seed, i;
if (argc > 1)
seed = atoi (argv[1]);
else
{
seed = time (NULL);
write_seed (seed);
}
g_random_set_seed (seed);
g_print ("the last tested image is saved to test-randomly-modified-image\n");
/* Set a malloc which emulates low mem */
g_type_init ();
g_log_set_always_fatal (G_LOG_LEVEL_WARNING | G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL);
for (;;)
for (i = 1; i < argc; ++i)
{
gchar *contents;
gsize size;
GError *err = NULL;
fflush (stdout);
if (!g_file_get_contents (argv[i], &contents, &size, &err))
{
g_print ("%s: error: %s\n", argv[i], err->message);
}
else
{
g_print ("%s\t\t", argv[i]);
randomly_modify (contents, size);
g_print ("done\n");
g_free (contents);
}
}
return 0;
}
/* -*- Mode: C; c-basic-offset: 2; -*- */
/* GdkPixbuf library - test loaders
*
* Copyright (C) 2001 Søren Sandmann (sandmann daimi au dk)
*
* 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
*/
#include <config.h>
#include "gdk-pixbuf.h"
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <string.h>
#define PRETEND_MEM_SIZE (16 * 1024 * 1024)
#define REMAINING_MEM_SIZE 100000
static int current_allocation = 0;
static int max_allocation = 0;
#define HEADER_SPACE sizeof(void*)
static gpointer
record_bytes (gpointer mem, gsize bytes)
{
if (mem == NULL ||
(current_allocation + bytes) > max_allocation)
{
if (mem)
free (mem);
return NULL;
}
*(void **)mem = GINT_TO_POINTER (bytes);
g_assert (GPOINTER_TO_INT (*(void**)mem) == bytes);
g_assert (current_allocation >= 0);
current_allocation += bytes;
g_assert (current_allocation >= 0);
g_assert ( mem == (void*) ((((char*)mem) + HEADER_SPACE) - HEADER_SPACE) );
return ((char*)mem) + HEADER_SPACE;
}
static gpointer
limited_try_malloc (gsize n_bytes)
{
return record_bytes (malloc (n_bytes + HEADER_SPACE), n_bytes);
}
static gpointer
limited_malloc (gsize n_bytes)
{
return limited_try_malloc (n_bytes);
}
static gpointer
limited_calloc (gsize n_blocks,
gsize n_block_bytes)
{
int bytes = n_blocks * n_block_bytes + HEADER_SPACE;
gpointer mem = malloc (bytes);
memset (mem, 0, bytes);
return record_bytes (mem, n_blocks * n_block_bytes);
}
static void
limited_free (gpointer mem)
{
gpointer real = ((char*)mem) - HEADER_SPACE;
g_assert (current_allocation >= 0);
current_allocation -= GPOINTER_TO_INT (*(void**)real);
g_assert (current_allocation >= 0);
free (real);
}
static gpointer
limited_try_realloc (gpointer mem,
gsize n_bytes)
{
if (mem == NULL)
{
return limited_try_malloc (n_bytes);
}
else
{
gpointer real;
g_assert (mem);
real = ((char*)mem) - HEADER_SPACE;
g_assert (current_allocation >= 0);
current_allocation -= GPOINTER_TO_INT (*(void**)real);
g_assert (current_allocation >= 0);
return record_bytes (realloc (real, n_bytes + HEADER_SPACE), n_bytes);
}
}
static gpointer
limited_realloc (gpointer mem,
gsize n_bytes)
{
return limited_try_realloc (mem, n_bytes);
}
static GMemVTable limited_table = {
limited_malloc,
limited_realloc,
limited_free,
limited_calloc,
limited_try_malloc,
limited_try_realloc
};
static void
mem_test (const guchar *bytes, gsize len)
{
gboolean did_fail = FALSE;
GError *err = NULL;
GdkPixbufLoader *loader;
GList *loaders = NULL;
GList *i;
do {
loader = gdk_pixbuf_loader_new ();
gdk_pixbuf_loader_write (loader, bytes, len, &err);
if (err)
{
g_error_free (err);
err = NULL;
did_fail = TRUE;
}
gdk_pixbuf_loader_close (loader, NULL);
if (err)
{
g_error_free (err);
err = NULL;
did_fail = TRUE;
}
loaders = g_list_prepend (loaders, loader);
} while (!did_fail);
for (i = loaders; i != NULL; i = i->next)
g_object_unref (i->data);
g_list_free (loaders);
}
static void
almost_exhaust_memory (void)
{
gpointer x = g_malloc (REMAINING_MEM_SIZE);
while (g_try_malloc (REMAINING_MEM_SIZE / 10))
;
g_free (x);
}
static void
usage (void)
{
g_print ("usage: test-lowmem <pretend_memory_size> <files>\n");
exit (EXIT_FAILURE);
}
int
main (int argc, char **argv)
{
int i;
char *endptr;
if (argc <= 2)
usage();
max_allocation = strtol (argv[1], &endptr, 10);
if (endptr == argv[1])
usage();
/* Set a malloc which emulates low mem */
g_mem_set_vtable (&limited_table);
g_type_init ();
g_log_set_always_fatal (G_LOG_LEVEL_WARNING | G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL);
/* memory tests */
/* How do the loaders behave when memory is low?
It depends on the state the above tests left the
memory in.
- Sometimes the png loader tries to report an
"out of memory", but then g_strdup_printf() calls
g_malloc(), which fails.
- There are unchecked realloc()s inside libtiff, which means it
will never work with low memory, unless something drastic is
done, like allocating a lot of memory upfront and release it
before entering libtiff. Also, some TIFFReadRGBAImage calls
returns successfully, even though they have called the error
handler with an 'out of memory' message.
*/
almost_exhaust_memory ();
g_print ("Allocated %dK of %dK, %dK free during tests\n",
current_allocation / 1024, max_allocation / 1024,
(max_allocation - current_allocation) / 1024);
for (i = 2; i < argc; ++i)
{
gchar *contents;
gsize size;
GError *err = NULL;
if (!g_file_get_contents (argv[i], &contents, &size, &err))
{
g_print ("couldn't read %s: %s\n", argv[i], err->message);
exit (EXIT_FAILURE);
}
else
{
g_print ("%-40s memory ", argv[i]);
fflush (stdout);
mem_test (contents, size);
g_print ("\tpassed\n");
g_free (contents);
}
}
return 0;
}
/* -*- Mode: C; c-basic-offset: 2; -*- */
/* GdkPixbuf library - test loaders
*
* Copyright (C) 2001 Søren Sandmann (sandmann daimi au dk)
*
* 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
*/
#include <config.h>
#include "gdk-pixbuf.h"
#include <stdio.h>
static gboolean
test_loader (const guchar *bytes, gsize len, GError **err)
{
GdkPixbufLoader *loader;
loader = gdk_pixbuf_loader_new ();
gdk_pixbuf_loader_write (loader, bytes, len, err);
if (*err)
return FALSE;
gdk_pixbuf_loader_close (loader, err);
if (*err)
return FALSE;
return TRUE;
}
int
main (int argc, char **argv)
{
int i;
g_type_init ();
g_log_set_always_fatal (G_LOG_LEVEL_WARNING | G_LOG_LEVEL_ERROR | G_LOG_LEVEL_CRITICAL);
for (i = 1; i < argc; ++i)
{
gchar *contents;
gsize size;
GError *err = NULL;
g_print ("%s\t\t", argv[i]);
fflush (stdout);
if (!g_file_get_contents (argv[i], &contents, &size, &err))
{
fprintf (stderr, "%s: error: %s\n", argv[i], err->message);
}
else
{
err = NULL;
if (test_loader (contents, size, &err))
g_print ("success\n");
else
g_print ("error: %s\n", err->message);
g_free (contents);
}
}
return 0;
}
#include <stdio.h>
#include <test-images.h>
static void
write_file (const char *bytes, int n_bytes, const char *file_name)
{
FILE *f = fopen (file_name, "w");
int i;
for (i = 0; i < n_bytes; ++i)
fwrite (bytes + i, 1, 1, f);
fclose (f);
printf ("wrote %s\n", file_name);
}
int
main ()
{
#define WR(img) write_file (img, sizeof (img), #img)
WR (valid_gif_test);
WR (gif_test_1);
WR (gif_test_2);
WR (gif_test_3);
WR (gif_test_4);
WR (valid_png_test);
WR (png_test_1);
WR (png_test_2);
WR (valid_tiff1_test);
WR (tiff1_test_1);
WR (tiff1_test_2);
WR (ico_test_1);
WR (wbmp_test_1);
WR (wbmp_test_2);
WR (valid_jpeg_test);
WR (valid_tga_test);
WR (tga_test_1);
WR (valid_ppm_1);
WR (valid_ppm_2);
WR (valid_ppm_3);
WR (valid_ppm_4);
WR (invalid_ppm_1);
WR (invalid_ppm_2);
WR (invalid_ppm_3);
WR (invalid_ppm_4);
WR (invalid_ppm_5);
WR (invalid_ppm_6);
WR (invalid_ppm_7);
WR (invalid_ppm_8);
WR (xpm_test_1);
WR (ico_test_2);
WR (tiff1_test_3);
WR (valid_bmp_test);
WR (valid_xpm_test);
WR (valid_ras_test);
WR (invalid_bmp_1);
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]