2004-10-26 14:38:43 +00:00
|
|
|
/*
|
|
|
|
|
* Copyright © 2004 Red Hat, Inc.
|
|
|
|
|
*
|
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software
|
|
|
|
|
* and its documentation for any purpose is hereby granted without
|
|
|
|
|
* fee, provided that the above copyright notice appear in all copies
|
|
|
|
|
* and that both that copyright notice and this permission notice
|
|
|
|
|
* appear in supporting documentation, and that the name of
|
|
|
|
|
* Red Hat, Inc. not be used in advertising or publicity pertaining to
|
|
|
|
|
* distribution of the software without specific, written prior
|
|
|
|
|
* permission. Red Hat, Inc. makes no representations about the
|
|
|
|
|
* suitability of this software for any purpose. It is provided "as
|
|
|
|
|
* is" without express or implied warranty.
|
|
|
|
|
*
|
|
|
|
|
* RED HAT, INC. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS
|
|
|
|
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
|
* FITNESS, IN NO EVENT SHALL RED HAT, INC. BE LIABLE FOR ANY SPECIAL,
|
|
|
|
|
* INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
|
|
|
|
|
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
|
|
|
|
* OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
|
|
|
|
|
* IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
*
|
|
|
|
|
* Author: Carl D. Worth <cworth@cworth.org>
|
|
|
|
|
*/
|
|
|
|
|
|
2005-01-20 20:45:38 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <stdarg.h>
|
2004-10-26 14:38:43 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <errno.h>
|
2005-01-20 20:45:38 +00:00
|
|
|
#include <string.h>
|
2004-10-26 14:38:43 +00:00
|
|
|
|
2005-03-29 00:02:19 +00:00
|
|
|
#include "cairo-test.h"
|
2004-10-26 14:38:43 +00:00
|
|
|
|
2005-03-29 00:02:19 +00:00
|
|
|
#include "buffer-diff.h"
|
|
|
|
|
#include "read-png.h"
|
|
|
|
|
#include "write-png.h"
|
2004-10-26 14:38:43 +00:00
|
|
|
#include "xmalloc.h"
|
|
|
|
|
|
2005-03-08 13:44:14 +00:00
|
|
|
#define CAIRO_TEST_LOG_SUFFIX ".log"
|
2004-10-26 14:38:43 +00:00
|
|
|
#define CAIRO_TEST_PNG_SUFFIX "-out.png"
|
|
|
|
|
#define CAIRO_TEST_REF_SUFFIX "-ref.png"
|
|
|
|
|
#define CAIRO_TEST_DIFF_SUFFIX "-diff.png"
|
|
|
|
|
|
2005-04-04 09:47:12 +00:00
|
|
|
void
|
2004-10-26 18:19:55 +00:00
|
|
|
xasprintf (char **strp, const char *fmt, ...)
|
2004-10-26 14:38:43 +00:00
|
|
|
{
|
2005-02-01 21:45:51 +00:00
|
|
|
#ifdef HAVE_VASPRINTF
|
2004-10-26 18:19:55 +00:00
|
|
|
va_list va;
|
|
|
|
|
int ret;
|
2005-02-01 21:45:51 +00:00
|
|
|
|
2004-10-26 18:19:55 +00:00
|
|
|
va_start (va, fmt);
|
|
|
|
|
ret = vasprintf (strp, fmt, va);
|
|
|
|
|
va_end (va);
|
2004-10-26 14:38:43 +00:00
|
|
|
|
2004-10-26 18:19:55 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
|
fprintf (stderr, "Out of memory\n");
|
|
|
|
|
exit (1);
|
|
|
|
|
}
|
2005-02-01 21:45:51 +00:00
|
|
|
#else /* !HAVE_VASNPRINTF */
|
|
|
|
|
#define BUF_SIZE 1024
|
|
|
|
|
va_list va;
|
|
|
|
|
char buffer[BUF_SIZE];
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
va_start (va, fmt);
|
|
|
|
|
ret = vsnprintf (buffer, sizeof(buffer), fmt, va);
|
|
|
|
|
va_end (va);
|
|
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
fprintf (stderr, "Failure in vsnprintf\n");
|
|
|
|
|
exit (1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (strlen (buffer) == sizeof(buffer) - 1) {
|
|
|
|
|
fprintf (stderr, "Overflowed fixed buffer\n");
|
|
|
|
|
exit (1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*strp = strdup (buffer);
|
|
|
|
|
if (!*strp) {
|
|
|
|
|
fprintf (stderr, "Out of memory\n");
|
|
|
|
|
exit (1);
|
|
|
|
|
}
|
|
|
|
|
#endif /* !HAVE_VASNPRINTF */
|
2004-10-26 14:38:43 +00:00
|
|
|
}
|
|
|
|
|
|
2005-03-08 13:44:14 +00:00
|
|
|
static void
|
|
|
|
|
xunlink (const char *pathname)
|
|
|
|
|
{
|
|
|
|
|
if (unlink (pathname) < 0 && errno != ENOENT) {
|
|
|
|
|
fprintf (stderr, " Error: Cannot remove %s: %s\n",
|
|
|
|
|
pathname, strerror (errno));
|
|
|
|
|
exit (1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-14 10:02:58 +00:00
|
|
|
cairo_test_status_t
|
|
|
|
|
cairo_test_expect_failure (cairo_test_t *test,
|
|
|
|
|
cairo_test_draw_function_t draw,
|
|
|
|
|
const char *because)
|
|
|
|
|
{
|
2005-04-14 14:42:26 +00:00
|
|
|
printf ("%s is expected to fail:\n\t%s\n", test->name, because);
|
2005-04-14 10:02:58 +00:00
|
|
|
return cairo_test (test, draw);
|
|
|
|
|
}
|
|
|
|
|
|
2004-10-26 14:38:43 +00:00
|
|
|
cairo_test_status_t
|
|
|
|
|
cairo_test (cairo_test_t *test, cairo_test_draw_function_t draw)
|
|
|
|
|
{
|
2005-03-09 13:58:20 +00:00
|
|
|
cairo_test_status_t status;
|
2004-10-26 14:38:43 +00:00
|
|
|
cairo_t *cr;
|
|
|
|
|
int stride;
|
|
|
|
|
unsigned char *png_buf, *ref_buf, *diff_buf;
|
2005-03-08 13:44:14 +00:00
|
|
|
char *log_name, *png_name, *ref_name, *diff_name;
|
2004-10-26 18:19:55 +00:00
|
|
|
char *srcdir;
|
2004-10-26 14:38:43 +00:00
|
|
|
int pixels_changed;
|
2005-04-04 09:47:12 +00:00
|
|
|
unsigned int ref_width, ref_height, ref_stride;
|
2004-10-26 14:38:43 +00:00
|
|
|
read_png_status_t png_status;
|
|
|
|
|
cairo_test_status_t ret;
|
2005-01-25 14:45:31 +00:00
|
|
|
FILE *png_file;
|
2005-03-08 13:44:14 +00:00
|
|
|
FILE *log_file;
|
2004-10-26 14:38:43 +00:00
|
|
|
|
2005-03-09 14:34:26 +00:00
|
|
|
/* Get the strings ready that we'll need. */
|
|
|
|
|
srcdir = getenv ("srcdir");
|
|
|
|
|
if (!srcdir)
|
|
|
|
|
srcdir = ".";
|
|
|
|
|
xasprintf (&log_name, "%s%s", test->name, CAIRO_TEST_LOG_SUFFIX);
|
|
|
|
|
xasprintf (&png_name, "%s%s", test->name, CAIRO_TEST_PNG_SUFFIX);
|
|
|
|
|
xasprintf (&ref_name, "%s/%s%s", srcdir, test->name, CAIRO_TEST_REF_SUFFIX);
|
|
|
|
|
xasprintf (&diff_name, "%s%s", test->name, CAIRO_TEST_DIFF_SUFFIX);
|
|
|
|
|
|
2005-03-10 09:22:20 +00:00
|
|
|
xunlink (log_name);
|
|
|
|
|
|
2005-03-09 14:34:26 +00:00
|
|
|
/* Run the actual drawing code. */
|
2004-10-26 14:38:43 +00:00
|
|
|
cr = cairo_create ();
|
|
|
|
|
|
|
|
|
|
stride = 4 * test->width;
|
|
|
|
|
|
|
|
|
|
png_buf = xcalloc (stride * test->height, 1);
|
|
|
|
|
diff_buf = xcalloc (stride * test->height, 1);
|
|
|
|
|
|
|
|
|
|
cairo_set_target_image (cr, png_buf, CAIRO_FORMAT_ARGB32,
|
|
|
|
|
test->width, test->height, stride);
|
|
|
|
|
|
2005-03-09 13:58:20 +00:00
|
|
|
status = (draw) (cr, test->width, test->height);
|
2005-03-09 14:34:26 +00:00
|
|
|
|
|
|
|
|
/* Then, check all the different ways it could fail. */
|
2005-03-09 13:58:20 +00:00
|
|
|
if (status) {
|
|
|
|
|
log_file = fopen (log_name, "a");
|
|
|
|
|
fprintf (log_file, "Error: Function under test failed\n");
|
|
|
|
|
fclose (log_file);
|
|
|
|
|
return status;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (cairo_status (cr) != CAIRO_STATUS_SUCCESS) {
|
|
|
|
|
log_file = fopen (log_name, "a");
|
|
|
|
|
fprintf (log_file, "Error: Function under test left cairo status in an error state: %s\n", cairo_status_string (cr));
|
|
|
|
|
fclose (log_file);
|
|
|
|
|
return CAIRO_TEST_FAILURE;
|
|
|
|
|
}
|
2004-10-26 14:38:43 +00:00
|
|
|
|
2004-11-23 12:53:46 +00:00
|
|
|
/* Skip image check for tests with no image (width,height == 0,0) */
|
|
|
|
|
if (test->width == 0 || test->height == 0) {
|
2005-04-13 14:45:05 +00:00
|
|
|
cairo_destroy (cr);
|
2004-11-23 12:53:46 +00:00
|
|
|
free (png_buf);
|
|
|
|
|
free (diff_buf);
|
|
|
|
|
return CAIRO_TEST_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-26 09:43:39 +00:00
|
|
|
cairo_surface_write_to_png (cairo_get_target_surface (cr), png_name);
|
2004-10-26 14:38:43 +00:00
|
|
|
|
2005-04-04 09:47:12 +00:00
|
|
|
cairo_destroy (cr);
|
|
|
|
|
|
2004-10-26 14:38:43 +00:00
|
|
|
ref_buf = NULL;
|
|
|
|
|
png_status = (read_png_argb32 (ref_name, &ref_buf, &ref_width, &ref_height, &ref_stride));
|
|
|
|
|
if (png_status) {
|
2005-03-08 13:44:14 +00:00
|
|
|
log_file = fopen (log_name, "a");
|
2004-10-26 14:38:43 +00:00
|
|
|
switch (png_status)
|
|
|
|
|
{
|
|
|
|
|
case READ_PNG_FILE_NOT_FOUND:
|
2005-03-08 13:44:14 +00:00
|
|
|
fprintf (log_file, "Error: No reference image found: %s\n", ref_name);
|
2004-10-26 14:38:43 +00:00
|
|
|
break;
|
|
|
|
|
case READ_PNG_FILE_NOT_PNG:
|
2005-03-08 13:44:14 +00:00
|
|
|
fprintf (log_file, "Error: %s is not a png image\n", ref_name);
|
2004-10-26 14:38:43 +00:00
|
|
|
break;
|
|
|
|
|
default:
|
2005-03-08 13:44:14 +00:00
|
|
|
fprintf (log_file, "Error: Failed to read %s\n", ref_name);
|
2004-10-26 14:38:43 +00:00
|
|
|
}
|
2005-03-08 13:44:14 +00:00
|
|
|
fclose (log_file);
|
2004-10-26 14:38:43 +00:00
|
|
|
|
|
|
|
|
ret = CAIRO_TEST_FAILURE;
|
|
|
|
|
goto BAIL;
|
2005-03-08 13:44:14 +00:00
|
|
|
} else {
|
2004-10-26 14:38:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (test->width != ref_width || test->height != ref_height) {
|
2005-03-08 13:44:14 +00:00
|
|
|
log_file = fopen (log_name, "a");
|
|
|
|
|
fprintf (log_file,
|
|
|
|
|
"Error: Image size mismatch: (%dx%d) vs. (%dx%d)\n"
|
|
|
|
|
" for %s vs %s\n",
|
2004-10-26 14:38:43 +00:00
|
|
|
test->width, test->height,
|
|
|
|
|
ref_width, ref_height,
|
|
|
|
|
png_name, ref_name);
|
2005-03-08 13:44:14 +00:00
|
|
|
fclose (log_file);
|
|
|
|
|
|
2004-10-26 14:38:43 +00:00
|
|
|
ret = CAIRO_TEST_FAILURE;
|
|
|
|
|
goto BAIL;
|
|
|
|
|
}
|
|
|
|
|
|
2005-01-25 14:45:31 +00:00
|
|
|
pixels_changed = buffer_diff (png_buf, ref_buf, diff_buf,
|
|
|
|
|
test->width, test->height, stride);
|
2004-10-26 14:38:43 +00:00
|
|
|
if (pixels_changed) {
|
2005-03-08 13:44:14 +00:00
|
|
|
log_file = fopen (log_name, "a");
|
|
|
|
|
fprintf (log_file, "Error: %d pixels differ from reference image %s\n",
|
2004-10-26 14:38:43 +00:00
|
|
|
pixels_changed, ref_name);
|
2005-03-17 13:13:56 +00:00
|
|
|
png_file = fopen (diff_name, "wb");
|
2005-01-25 14:45:31 +00:00
|
|
|
write_png_argb32 (diff_buf, png_file, test->width, test->height, stride);
|
|
|
|
|
fclose (png_file);
|
2005-03-08 13:44:14 +00:00
|
|
|
fclose (log_file);
|
|
|
|
|
|
2004-10-26 14:38:43 +00:00
|
|
|
ret = CAIRO_TEST_FAILURE;
|
|
|
|
|
goto BAIL;
|
|
|
|
|
} else {
|
2005-03-08 13:44:14 +00:00
|
|
|
xunlink (diff_name);
|
2004-10-26 14:38:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = CAIRO_TEST_SUCCESS;
|
|
|
|
|
|
|
|
|
|
BAIL:
|
|
|
|
|
free (png_buf);
|
|
|
|
|
free (ref_buf);
|
|
|
|
|
free (diff_buf);
|
2005-03-08 13:44:14 +00:00
|
|
|
free (log_name);
|
2004-10-26 14:38:43 +00:00
|
|
|
free (png_name);
|
|
|
|
|
free (ref_name);
|
|
|
|
|
free (diff_name);
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
2005-01-26 13:41:55 +00:00
|
|
|
|
|
|
|
|
cairo_pattern_t *
|
|
|
|
|
cairo_test_create_png_pattern (cairo_t *cr, const char *filename)
|
|
|
|
|
{
|
|
|
|
|
cairo_surface_t *image;
|
|
|
|
|
cairo_pattern_t *pattern;
|
|
|
|
|
unsigned char *buffer;
|
2005-04-04 09:47:12 +00:00
|
|
|
unsigned int w, h, stride;
|
2005-01-26 13:41:55 +00:00
|
|
|
read_png_status_t status;
|
2005-03-08 19:25:39 +00:00
|
|
|
char *srcdir = getenv ("srcdir");
|
2005-01-26 13:41:55 +00:00
|
|
|
|
|
|
|
|
status = read_png_argb32 (filename, &buffer, &w,&h, &stride);
|
2005-03-08 19:25:39 +00:00
|
|
|
if (status != READ_PNG_SUCCESS) {
|
|
|
|
|
if (srcdir) {
|
|
|
|
|
char *srcdir_filename;
|
|
|
|
|
xasprintf (&srcdir_filename, "%s/%s", srcdir, filename);
|
|
|
|
|
status = read_png_argb32 (srcdir_filename, &buffer, &w,&h, &stride);
|
|
|
|
|
free (srcdir_filename);
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-01-26 13:41:55 +00:00
|
|
|
if (status != READ_PNG_SUCCESS)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
image = cairo_surface_create_for_image (buffer, CAIRO_FORMAT_ARGB32,
|
|
|
|
|
w, h, stride);
|
|
|
|
|
|
|
|
|
|
cairo_surface_set_repeat (image, 1);
|
|
|
|
|
|
|
|
|
|
pattern = cairo_pattern_create_for_surface (image);
|
|
|
|
|
|
|
|
|
|
return pattern;
|
|
|
|
|
}
|