2004-10-26 14:38:43 +00:00
|
|
|
|
/*
|
2005-05-14 13:51:59 +00:00
|
|
|
|
* Copyright <EFBFBD> 2004 Red Hat, Inc.
|
2004-10-26 14:38:43 +00:00
|
|
|
|
*
|
|
|
|
|
|
* 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>
|
2005-08-05 07:48:18 +00:00
|
|
|
|
#ifdef HAVE_UNISTD_H
|
2004-10-26 14:38:43 +00:00
|
|
|
|
#include <unistd.h>
|
2005-08-05 07:48:18 +00:00
|
|
|
|
#endif
|
2004-10-26 14:38:43 +00:00
|
|
|
|
#include <errno.h>
|
2005-01-20 20:45:38 +00:00
|
|
|
|
#include <string.h>
|
2005-08-01 13:33:47 +00:00
|
|
|
|
#include <fontconfig/fontconfig.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-08-05 07:48:18 +00:00
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
|
#define vsnprintf _vsnprintf
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
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-05-10 20:25:38 +00:00
|
|
|
|
/* Static data is messy, but we're coding for tests here, not a
|
|
|
|
|
|
* general-purpose library, and it keeps the tests cleaner to avoid a
|
|
|
|
|
|
* context object there, (though not a whole lot). */
|
|
|
|
|
|
FILE *cairo_test_log_file;
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
|
cairo_test_log (const char *fmt, ...)
|
|
|
|
|
|
{
|
|
|
|
|
|
va_list va;
|
|
|
|
|
|
|
|
|
|
|
|
va_start (va, fmt);
|
|
|
|
|
|
vfprintf (cairo_test_log_file, fmt, va);
|
|
|
|
|
|
va_end (va);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
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) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Out of memory\n");
|
2004-10-26 18:19:55 +00:00
|
|
|
|
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) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Failure in vsnprintf\n");
|
2005-02-01 21:45:51 +00:00
|
|
|
|
exit (1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (strlen (buffer) == sizeof(buffer) - 1) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Overflowed fixed buffer\n");
|
2005-02-01 21:45:51 +00:00
|
|
|
|
exit (1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
*strp = strdup (buffer);
|
|
|
|
|
|
if (!*strp) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Out of memory\n");
|
2005-02-01 21:45:51 +00:00
|
|
|
|
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) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log (" Error: Cannot remove %s: %s\n",
|
|
|
|
|
|
pathname, strerror (errno));
|
2005-03-08 13:44:14 +00:00
|
|
|
|
exit (1);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
typedef cairo_surface_t *
|
|
|
|
|
|
(*cairo_test_create_target_surface_t) (int width, int height, void **closure);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
|
|
|
|
|
typedef void
|
|
|
|
|
|
(*cairo_test_cleanup_target_t) (void *closure);
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct _cairo_test_target
|
|
|
|
|
|
{
|
2005-05-06 13:23:41 +00:00
|
|
|
|
const char *name;
|
|
|
|
|
|
cairo_test_create_target_surface_t create_target_surface;
|
|
|
|
|
|
cairo_test_cleanup_target_t cleanup_target;
|
|
|
|
|
|
void *closure;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
} cairo_test_target_t;
|
|
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
static cairo_surface_t *
|
|
|
|
|
|
create_image_surface (int width, int height, void **closure)
|
2005-04-14 10:02:58 +00:00
|
|
|
|
{
|
2005-04-27 13:33:25 +00:00
|
|
|
|
int stride = 4 * width;
|
2005-05-06 13:23:41 +00:00
|
|
|
|
unsigned char *buf;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
*closure = buf = xcalloc (stride * height, 1);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
return cairo_image_surface_create_for_data (buf,
|
|
|
|
|
|
CAIRO_FORMAT_ARGB32,
|
|
|
|
|
|
width, height, stride);
|
2005-04-14 10:02:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2005-04-27 13:33:25 +00:00
|
|
|
|
static void
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cleanup_image (void *closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
2005-05-06 13:23:41 +00:00
|
|
|
|
unsigned char *buf = closure;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
free (buf);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* XXX: Someone who knows glitz better than I do should fix this up to
|
|
|
|
|
|
* work. */
|
|
|
|
|
|
#if 0 /* #ifdef CAIRO_HAS_GLITZ_SURFACE */
|
2005-05-06 13:23:41 +00:00
|
|
|
|
static cairo_surface_t *
|
|
|
|
|
|
create_glitz_surface (int width, int height, void **closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
#error Not yet implemented
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cleanup_glitz (cairo_t *cr)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
#error Not yet implemented
|
|
|
|
|
|
}
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
2005-05-06 21:33:22 +00:00
|
|
|
|
#if CAIRO_HAS_QUARTZ_SURFACE
|
2005-05-06 13:23:41 +00:00
|
|
|
|
static cairo_surface_t *
|
|
|
|
|
|
create_quartz_surface (int width, int height, void **closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
#error Not yet implemented
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cleanup_quartz (void *closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
#error Not yet implemented
|
|
|
|
|
|
}
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
2005-05-11 16:01:56 +00:00
|
|
|
|
/* Testing the win32 surface isn't interesting, since for
|
|
|
|
|
|
* ARGB images it just chains to the image backend
|
|
|
|
|
|
*/
|
|
|
|
|
|
#if 0 && CAIRO_HAS_WIN32_SURFACE
|
2005-05-06 13:23:41 +00:00
|
|
|
|
static cairo_surface_t *
|
|
|
|
|
|
create_win32_surface (int width, int height, void **closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
#error Not yet implemented
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cleanup_win32 (void *closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
#error Not yet implemented
|
|
|
|
|
|
}
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
2005-05-06 21:33:22 +00:00
|
|
|
|
#if CAIRO_HAS_XCB_SURFACE
|
2005-05-17 06:11:32 +00:00
|
|
|
|
#include "cairo-xcb-xrender.h"
|
2005-05-14 14:01:46 +00:00
|
|
|
|
typedef struct _xcb_target_closure
|
|
|
|
|
|
{
|
|
|
|
|
|
XCBConnection *c;
|
2005-05-17 06:11:32 +00:00
|
|
|
|
XCBDRAWABLE drawable;
|
2005-05-14 14:01:46 +00:00
|
|
|
|
} xcb_target_closure_t;
|
|
|
|
|
|
|
2005-05-17 06:11:32 +00:00
|
|
|
|
/* XXX: This is a nasty hack. Something like this should be in XCB's
|
|
|
|
|
|
* bindings for Render, not here in this test. */
|
|
|
|
|
|
static XCBRenderPICTFORMINFO
|
|
|
|
|
|
_format_from_cairo(XCBConnection *c, cairo_format_t fmt)
|
|
|
|
|
|
{
|
|
|
|
|
|
XCBRenderPICTFORMINFO ret = {{ 0 }};
|
|
|
|
|
|
struct tmpl_t {
|
|
|
|
|
|
XCBRenderDIRECTFORMAT direct;
|
|
|
|
|
|
CARD8 depth;
|
|
|
|
|
|
};
|
|
|
|
|
|
static const struct tmpl_t templates[] = {
|
|
|
|
|
|
/* CAIRO_FORMAT_ARGB32 */
|
|
|
|
|
|
{
|
|
|
|
|
|
{
|
|
|
|
|
|
16, 0xff,
|
|
|
|
|
|
8, 0xff,
|
|
|
|
|
|
0, 0xff,
|
|
|
|
|
|
24, 0xff
|
|
|
|
|
|
},
|
|
|
|
|
|
32
|
|
|
|
|
|
},
|
|
|
|
|
|
/* CAIRO_FORMAT_RGB24 */
|
|
|
|
|
|
{
|
|
|
|
|
|
{
|
|
|
|
|
|
16, 0xff,
|
|
|
|
|
|
8, 0xff,
|
|
|
|
|
|
0, 0xff,
|
|
|
|
|
|
0, 0x00
|
|
|
|
|
|
},
|
|
|
|
|
|
24
|
|
|
|
|
|
},
|
|
|
|
|
|
/* CAIRO_FORMAT_A8 */
|
|
|
|
|
|
{
|
|
|
|
|
|
{
|
|
|
|
|
|
0, 0x00,
|
|
|
|
|
|
0, 0x00,
|
|
|
|
|
|
0, 0x00,
|
|
|
|
|
|
0, 0xff
|
|
|
|
|
|
},
|
|
|
|
|
|
8
|
|
|
|
|
|
},
|
|
|
|
|
|
/* CAIRO_FORMAT_A1 */
|
|
|
|
|
|
{
|
|
|
|
|
|
{
|
|
|
|
|
|
0, 0x00,
|
|
|
|
|
|
0, 0x00,
|
|
|
|
|
|
0, 0x00,
|
|
|
|
|
|
0, 0x01
|
|
|
|
|
|
},
|
|
|
|
|
|
1
|
|
|
|
|
|
},
|
|
|
|
|
|
};
|
|
|
|
|
|
const struct tmpl_t *tmpl;
|
|
|
|
|
|
XCBRenderQueryPictFormatsRep *r;
|
|
|
|
|
|
XCBRenderPICTFORMINFOIter fi;
|
|
|
|
|
|
|
|
|
|
|
|
if(fmt < 0 || fmt >= (sizeof(templates) / sizeof(*templates)))
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
tmpl = templates + fmt;
|
|
|
|
|
|
|
|
|
|
|
|
r = XCBRenderQueryPictFormatsReply(c, XCBRenderQueryPictFormats(c), 0);
|
|
|
|
|
|
if(!r)
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
|
|
|
|
for(fi = XCBRenderQueryPictFormatsFormatsIter(r); fi.rem; XCBRenderPICTFORMINFONext(&fi))
|
|
|
|
|
|
{
|
|
|
|
|
|
const XCBRenderDIRECTFORMAT *t, *f;
|
|
|
|
|
|
if(fi.data->type != XCBRenderPictTypeDirect)
|
|
|
|
|
|
continue;
|
|
|
|
|
|
if(fi.data->depth != tmpl->depth)
|
|
|
|
|
|
continue;
|
|
|
|
|
|
t = &tmpl->direct;
|
|
|
|
|
|
f = &fi.data->direct;
|
|
|
|
|
|
if(t->red_mask && (t->red_mask != f->red_mask || t->red_shift != f->red_shift))
|
|
|
|
|
|
continue;
|
|
|
|
|
|
if(t->green_mask && (t->green_mask != f->green_mask || t->green_shift != f->green_shift))
|
|
|
|
|
|
continue;
|
|
|
|
|
|
if(t->blue_mask && (t->blue_mask != f->blue_mask || t->blue_shift != f->blue_shift))
|
|
|
|
|
|
continue;
|
|
|
|
|
|
if(t->alpha_mask && (t->alpha_mask != f->alpha_mask || t->alpha_shift != f->alpha_shift))
|
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
|
|
ret = *fi.data;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
free(r);
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
static cairo_surface_t *
|
|
|
|
|
|
create_xcb_surface (int width, int height, void **closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
2005-05-14 14:01:46 +00:00
|
|
|
|
XCBSCREEN *root;
|
|
|
|
|
|
xcb_target_closure_t *xtc;
|
|
|
|
|
|
cairo_surface_t *surface;
|
|
|
|
|
|
XCBConnection *c;
|
2005-05-17 06:11:32 +00:00
|
|
|
|
XCBRenderPICTFORMINFO render_format;
|
2005-05-14 14:01:46 +00:00
|
|
|
|
|
|
|
|
|
|
*closure = xtc = xmalloc (sizeof (xcb_target_closure_t));
|
|
|
|
|
|
|
|
|
|
|
|
if (width == 0)
|
|
|
|
|
|
width = 1;
|
|
|
|
|
|
if (height == 0)
|
|
|
|
|
|
height = 1;
|
|
|
|
|
|
|
|
|
|
|
|
xtc->c = c = XCBConnectBasic();
|
|
|
|
|
|
if (c == NULL) {
|
|
|
|
|
|
cairo_test_log ("Failed to connect to X server through XCB\n");
|
|
|
|
|
|
return NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2005-05-17 06:11:32 +00:00
|
|
|
|
root = XCBConnSetupSuccessRepRootsIter(XCBGetSetup(c)).data;
|
|
|
|
|
|
|
|
|
|
|
|
xtc->drawable.pixmap = XCBPIXMAPNew (c);
|
2005-05-14 14:01:46 +00:00
|
|
|
|
{
|
|
|
|
|
|
XCBDRAWABLE root_drawable;
|
|
|
|
|
|
root_drawable.window = root->root;
|
2005-05-17 06:11:32 +00:00
|
|
|
|
XCBCreatePixmap (c, 32, xtc->drawable.pixmap, root_drawable,
|
|
|
|
|
|
width, height);
|
2005-05-14 14:01:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2005-05-17 06:11:32 +00:00
|
|
|
|
render_format = _format_from_cairo (c, CAIRO_FORMAT_ARGB32);
|
|
|
|
|
|
if (render_format.id.xid == 0)
|
|
|
|
|
|
return NULL;
|
|
|
|
|
|
surface = cairo_xcb_surface_create_with_xrender_format (c, xtc->drawable,
|
|
|
|
|
|
&render_format,
|
|
|
|
|
|
width, height);
|
2005-05-14 14:01:46 +00:00
|
|
|
|
|
|
|
|
|
|
return surface;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cleanup_xcb (void *closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
2005-05-14 14:01:46 +00:00
|
|
|
|
xcb_target_closure_t *xtc = closure;
|
|
|
|
|
|
|
2005-05-17 06:11:32 +00:00
|
|
|
|
XCBFreePixmap (xtc->c, xtc->drawable.pixmap);
|
2005-05-14 14:01:46 +00:00
|
|
|
|
XCBDisconnect (xtc->c);
|
2005-07-14 11:11:15 +00:00
|
|
|
|
free (xtc);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
2005-05-06 21:33:22 +00:00
|
|
|
|
#if CAIRO_HAS_XLIB_SURFACE
|
2005-05-17 06:05:13 +00:00
|
|
|
|
#include "cairo-xlib-xrender.h"
|
2005-04-27 13:33:25 +00:00
|
|
|
|
typedef struct _xlib_target_closure
|
|
|
|
|
|
{
|
|
|
|
|
|
Display *dpy;
|
|
|
|
|
|
Pixmap pixmap;
|
|
|
|
|
|
} xlib_target_closure_t;
|
|
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
static cairo_surface_t *
|
|
|
|
|
|
create_xlib_surface (int width, int height, void **closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
xlib_target_closure_t *xtc;
|
|
|
|
|
|
cairo_surface_t *surface;
|
|
|
|
|
|
Display *dpy;
|
2005-05-17 06:05:13 +00:00
|
|
|
|
XRenderPictFormat *xrender_format;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
|
|
|
|
|
*closure = xtc = xmalloc (sizeof (xlib_target_closure_t));
|
|
|
|
|
|
|
2005-04-27 14:09:41 +00:00
|
|
|
|
if (width == 0)
|
|
|
|
|
|
width = 1;
|
|
|
|
|
|
if (height == 0)
|
|
|
|
|
|
height = 1;
|
|
|
|
|
|
|
2005-04-27 13:33:25 +00:00
|
|
|
|
xtc->dpy = dpy = XOpenDisplay (0);
|
|
|
|
|
|
if (xtc->dpy == NULL) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Failed to open display: %s\n", XDisplayName(0));
|
2005-05-06 13:23:41 +00:00
|
|
|
|
return NULL;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2005-05-17 06:05:13 +00:00
|
|
|
|
xrender_format = XRenderFindStandardFormat (dpy, PictStandardARGB32);
|
|
|
|
|
|
|
2005-04-27 13:33:25 +00:00
|
|
|
|
xtc->pixmap = XCreatePixmap (dpy, DefaultRootWindow (dpy),
|
2005-05-17 06:05:13 +00:00
|
|
|
|
width, height, xrender_format->depth);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
2005-05-17 06:05:13 +00:00
|
|
|
|
surface = cairo_xlib_surface_create_with_xrender_format (dpy, xtc->pixmap,
|
2005-07-20 18:52:31 +00:00
|
|
|
|
DefaultScreenOfDisplay (dpy),
|
2005-05-17 06:05:13 +00:00
|
|
|
|
xrender_format,
|
|
|
|
|
|
width, height);
|
2005-05-06 13:23:41 +00:00
|
|
|
|
return surface;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cleanup_xlib (void *closure)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
xlib_target_closure_t *xtc = closure;
|
|
|
|
|
|
|
|
|
|
|
|
XFreePixmap (xtc->dpy, xtc->pixmap);
|
|
|
|
|
|
XCloseDisplay (xtc->dpy);
|
2005-07-14 11:11:15 +00:00
|
|
|
|
free (xtc);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static cairo_test_status_t
|
|
|
|
|
|
cairo_test_for_target (cairo_test_t *test,
|
|
|
|
|
|
cairo_test_draw_function_t draw,
|
|
|
|
|
|
cairo_test_target_t *target)
|
2004-10-26 14:38:43 +00:00
|
|
|
|
{
|
2005-03-09 13:58:20 +00:00
|
|
|
|
cairo_test_status_t status;
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cairo_surface_t *surface;
|
2004-10-26 14:38:43 +00:00
|
|
|
|
cairo_t *cr;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
char *png_name, *ref_name, *diff_name;
|
2004-10-26 18:19:55 +00:00
|
|
|
|
char *srcdir;
|
2004-10-26 14:38:43 +00:00
|
|
|
|
cairo_test_status_t ret;
|
|
|
|
|
|
|
2005-03-09 14:34:26 +00:00
|
|
|
|
/* Get the strings ready that we'll need. */
|
|
|
|
|
|
srcdir = getenv ("srcdir");
|
|
|
|
|
|
if (!srcdir)
|
|
|
|
|
|
srcdir = ".";
|
2005-04-28 11:15:47 +00:00
|
|
|
|
xasprintf (&png_name, "%s-%s%s", test->name,
|
2005-04-27 13:33:25 +00:00
|
|
|
|
target->name, CAIRO_TEST_PNG_SUFFIX);
|
|
|
|
|
|
xasprintf (&ref_name, "%s/%s%s", srcdir, test->name,
|
|
|
|
|
|
CAIRO_TEST_REF_SUFFIX);
|
2005-04-28 11:15:47 +00:00
|
|
|
|
xasprintf (&diff_name, "%s-%s%s", test->name,
|
2005-04-27 13:33:25 +00:00
|
|
|
|
target->name, CAIRO_TEST_DIFF_SUFFIX);
|
2005-03-10 09:22:20 +00:00
|
|
|
|
|
2005-03-09 14:34:26 +00:00
|
|
|
|
/* Run the actual drawing code. */
|
2005-05-06 13:23:41 +00:00
|
|
|
|
surface = (target->create_target_surface) (test->width, test->height,
|
|
|
|
|
|
&target->closure);
|
|
|
|
|
|
if (surface == NULL) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Error: Failed to set %s target\n", target->name);
|
2005-07-14 12:20:42 +00:00
|
|
|
|
ret = CAIRO_TEST_UNTESTED;
|
2005-07-14 11:11:15 +00:00
|
|
|
|
goto UNWIND_STRINGS;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
2004-10-26 14:38:43 +00:00
|
|
|
|
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cr = cairo_create (surface);
|
|
|
|
|
|
|
2005-04-27 13:33:25 +00:00
|
|
|
|
cairo_save (cr);
|
|
|
|
|
|
cairo_set_source_rgba (cr, 0, 0, 0, 0);
|
2005-05-06 13:26:16 +00:00
|
|
|
|
cairo_set_operator (cr, CAIRO_OPERATOR_SOURCE);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
cairo_paint (cr);
|
|
|
|
|
|
cairo_restore (cr);
|
2004-10-26 14:38:43 +00:00
|
|
|
|
|
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) {
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Error: Function under test failed\n");
|
2005-07-14 11:11:15 +00:00
|
|
|
|
ret = status;
|
|
|
|
|
|
goto UNWIND_CAIRO;
|
2005-03-09 13:58:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (cairo_status (cr) != CAIRO_STATUS_SUCCESS) {
|
2005-06-15 10:56:31 +00:00
|
|
|
|
cairo_test_log ("Error: Function under test left cairo status in an error state: %s\n",
|
2005-06-15 10:58:52 +00:00
|
|
|
|
cairo_status_to_string (cairo_status (cr)));
|
2005-07-14 11:11:15 +00:00
|
|
|
|
ret = CAIRO_TEST_FAILURE;
|
|
|
|
|
|
goto UNWIND_CAIRO;
|
2005-03-09 13:58:20 +00:00
|
|
|
|
}
|
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) */
|
2005-07-14 11:11:15 +00:00
|
|
|
|
if (test->width != 0 && test->height != 0) {
|
|
|
|
|
|
int pixels_changed;
|
|
|
|
|
|
cairo_surface_write_to_png (surface, png_name);
|
|
|
|
|
|
pixels_changed = image_diff (png_name, ref_name, diff_name);
|
|
|
|
|
|
if (pixels_changed) {
|
|
|
|
|
|
if (pixels_changed > 0)
|
|
|
|
|
|
cairo_test_log ("Error: %d pixels differ from reference image %s\n",
|
|
|
|
|
|
pixels_changed, ref_name);
|
|
|
|
|
|
ret = CAIRO_TEST_FAILURE;
|
|
|
|
|
|
goto UNWIND_CAIRO;
|
|
|
|
|
|
}
|
2004-11-23 12:53:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2005-07-14 11:11:15 +00:00
|
|
|
|
ret = CAIRO_TEST_SUCCESS;
|
2004-10-26 14:38:43 +00:00
|
|
|
|
|
2005-07-14 11:11:15 +00:00
|
|
|
|
UNWIND_CAIRO:
|
2005-04-04 09:47:12 +00:00
|
|
|
|
cairo_destroy (cr);
|
2005-05-06 13:23:41 +00:00
|
|
|
|
cairo_surface_destroy (surface);
|
2005-08-01 13:33:47 +00:00
|
|
|
|
|
|
|
|
|
|
cairo_debug_reset_static_data ();
|
|
|
|
|
|
|
2005-04-27 13:33:25 +00:00
|
|
|
|
target->cleanup_target (target->closure);
|
2005-03-08 13:44:14 +00:00
|
|
|
|
|
2005-07-14 11:11:15 +00:00
|
|
|
|
UNWIND_STRINGS:
|
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
|
|
|
|
|
2005-04-27 13:33:25 +00:00
|
|
|
|
static cairo_test_status_t
|
2005-07-14 12:50:28 +00:00
|
|
|
|
cairo_test_expecting (cairo_test_t *test, cairo_test_draw_function_t draw,
|
|
|
|
|
|
cairo_test_status_t expectation)
|
2005-04-27 13:33:25 +00:00
|
|
|
|
{
|
|
|
|
|
|
int i;
|
|
|
|
|
|
cairo_test_status_t status, ret;
|
|
|
|
|
|
cairo_test_target_t targets[] =
|
|
|
|
|
|
{
|
2005-05-06 13:23:41 +00:00
|
|
|
|
{ "image", create_image_surface, cleanup_image},
|
2005-04-27 13:33:25 +00:00
|
|
|
|
#if 0 /* #ifdef CAIRO_HAS_GLITZ_SURFACE */
|
2005-05-06 13:23:41 +00:00
|
|
|
|
{ "glitz", create_glitz_surface, cleanup_glitz},
|
2005-04-27 13:33:25 +00:00
|
|
|
|
#endif
|
2005-05-06 21:33:22 +00:00
|
|
|
|
#if CAIRO_HAS_QUARTZ_SURFACE
|
2005-05-06 13:23:41 +00:00
|
|
|
|
{ "quartz", create_quartz_surface, cleanup_quartz},
|
2005-04-27 13:33:25 +00:00
|
|
|
|
#endif
|
2005-05-11 16:01:56 +00:00
|
|
|
|
#if 0 && CAIRO_HAS_WIN32_SURFACE
|
2005-05-06 13:23:41 +00:00
|
|
|
|
{ "win32", create_win32_surface, cleanup_win32},
|
2005-04-27 13:33:25 +00:00
|
|
|
|
#endif
|
2005-05-06 21:33:22 +00:00
|
|
|
|
#if CAIRO_HAS_XCB_SURFACE
|
2005-05-06 13:23:41 +00:00
|
|
|
|
{ "xcb", create_xcb_surface, cleanup_xcb},
|
2005-04-27 13:33:25 +00:00
|
|
|
|
#endif
|
2005-05-06 21:33:22 +00:00
|
|
|
|
#if CAIRO_HAS_XLIB_SURFACE
|
2005-05-06 13:23:41 +00:00
|
|
|
|
{ "xlib", create_xlib_surface, cleanup_xlib},
|
2005-04-27 13:33:25 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
};
|
|
|
|
|
|
char *log_name;
|
|
|
|
|
|
|
|
|
|
|
|
xasprintf (&log_name, "%s%s", test->name, CAIRO_TEST_LOG_SUFFIX);
|
|
|
|
|
|
xunlink (log_name);
|
|
|
|
|
|
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log_file = fopen (log_name, "a");
|
2005-05-14 13:51:59 +00:00
|
|
|
|
if (cairo_test_log_file == NULL) {
|
|
|
|
|
|
fprintf (stderr, "Error opening log file: %s\n", log_name);
|
|
|
|
|
|
cairo_test_log_file = stderr;
|
|
|
|
|
|
}
|
2005-07-14 11:11:15 +00:00
|
|
|
|
free (log_name);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
2005-07-14 12:20:42 +00:00
|
|
|
|
/* The intended logic here is that we return overall SUCCESS
|
2005-07-14 12:25:41 +00:00
|
|
|
|
* iff. there is at least one tested backend and that all tested
|
|
|
|
|
|
* backends return SUCCESS. In other words:
|
2005-07-14 12:20:42 +00:00
|
|
|
|
*
|
|
|
|
|
|
* if any backend FAILURE
|
|
|
|
|
|
* -> FAILURE
|
|
|
|
|
|
* else if all backends UNTESTED
|
|
|
|
|
|
* -> FAILURE
|
|
|
|
|
|
* else (== some backend SUCCESS)
|
|
|
|
|
|
* -> SUCCESS
|
|
|
|
|
|
*/
|
|
|
|
|
|
ret = CAIRO_TEST_UNTESTED;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
for (i=0; i < sizeof(targets)/sizeof(targets[0]); i++) {
|
|
|
|
|
|
cairo_test_target_t *target = &targets[i];
|
2005-05-10 20:25:38 +00:00
|
|
|
|
cairo_test_log ("Testing %s with %s target\n", test->name, target->name);
|
2005-05-03 08:33:32 +00:00
|
|
|
|
printf ("%s-%s:\t", test->name, target->name);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
status = cairo_test_for_target (test, draw, target);
|
2005-07-14 12:20:42 +00:00
|
|
|
|
switch (status) {
|
|
|
|
|
|
case CAIRO_TEST_SUCCESS:
|
|
|
|
|
|
printf ("PASS\n");
|
|
|
|
|
|
if (ret == CAIRO_TEST_UNTESTED)
|
|
|
|
|
|
ret = CAIRO_TEST_SUCCESS;
|
|
|
|
|
|
break;
|
|
|
|
|
|
case CAIRO_TEST_UNTESTED:
|
|
|
|
|
|
printf ("UNTESTED\n");
|
|
|
|
|
|
break;
|
|
|
|
|
|
default:
|
|
|
|
|
|
case CAIRO_TEST_FAILURE:
|
2005-07-14 12:50:28 +00:00
|
|
|
|
if (expectation == CAIRO_TEST_FAILURE)
|
|
|
|
|
|
printf ("XFAIL\n");
|
|
|
|
|
|
else
|
|
|
|
|
|
printf ("FAIL\n");
|
2005-04-27 13:33:25 +00:00
|
|
|
|
ret = status;
|
2005-07-14 12:20:42 +00:00
|
|
|
|
break;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2005-07-14 12:20:42 +00:00
|
|
|
|
if (ret == CAIRO_TEST_UNTESTED)
|
|
|
|
|
|
ret = CAIRO_TEST_FAILURE;
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
2005-05-10 20:25:38 +00:00
|
|
|
|
fclose (cairo_test_log_file);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
|
2005-08-03 10:32:50 +00:00
|
|
|
|
#if HAVE_FCFINI
|
2005-08-01 13:33:47 +00:00
|
|
|
|
FcFini ();
|
2005-08-03 10:32:50 +00:00
|
|
|
|
#endif
|
2005-08-01 13:33:47 +00:00
|
|
|
|
|
2005-04-27 13:33:25 +00:00
|
|
|
|
return ret;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
cairo_test_status_t
|
|
|
|
|
|
cairo_test_expect_failure (cairo_test_t *test,
|
|
|
|
|
|
cairo_test_draw_function_t draw,
|
|
|
|
|
|
const char *because)
|
|
|
|
|
|
{
|
|
|
|
|
|
printf ("\n%s is expected to fail:\n\t%s\n", test->name, because);
|
2005-07-14 12:50:28 +00:00
|
|
|
|
return cairo_test_expecting (test, draw, CAIRO_TEST_FAILURE);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
cairo_test_status_t
|
|
|
|
|
|
cairo_test (cairo_test_t *test, cairo_test_draw_function_t draw)
|
|
|
|
|
|
{
|
|
|
|
|
|
printf ("\n");
|
2005-07-14 12:50:28 +00:00
|
|
|
|
return cairo_test_expecting (test, draw, CAIRO_TEST_SUCCESS);
|
2005-04-27 13:33:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2005-07-18 08:04:16 +00:00
|
|
|
|
cairo_surface_t *
|
|
|
|
|
|
cairo_test_create_surface_from_png (const char *filename)
|
2005-01-26 13:41:55 +00:00
|
|
|
|
{
|
|
|
|
|
|
cairo_surface_t *image;
|
2005-03-08 19:25:39 +00:00
|
|
|
|
char *srcdir = getenv ("srcdir");
|
2005-01-26 13:41:55 +00:00
|
|
|
|
|
2005-07-14 16:18:39 +00:00
|
|
|
|
image = cairo_image_surface_create_from_png (filename);
|
2005-07-28 11:22:36 +00:00
|
|
|
|
if (cairo_surface_status(image)) {
|
|
|
|
|
|
/* expect not found when running with srcdir != builddir
|
|
|
|
|
|
* such as when 'make distcheck' is run
|
|
|
|
|
|
*/
|
2005-03-08 19:25:39 +00:00
|
|
|
|
if (srcdir) {
|
|
|
|
|
|
char *srcdir_filename;
|
|
|
|
|
|
xasprintf (&srcdir_filename, "%s/%s", srcdir, filename);
|
2005-07-14 16:18:39 +00:00
|
|
|
|
image = cairo_image_surface_create_from_png (srcdir_filename);
|
2005-03-08 19:25:39 +00:00
|
|
|
|
free (srcdir_filename);
|
|
|
|
|
|
}
|
2005-07-28 11:22:36 +00:00
|
|
|
|
if (cairo_surface_status(image))
|
2005-07-14 16:18:39 +00:00
|
|
|
|
return NULL;
|
2005-03-08 19:25:39 +00:00
|
|
|
|
}
|
2005-01-26 13:41:55 +00:00
|
|
|
|
|
2005-07-18 08:04:16 +00:00
|
|
|
|
return image;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
cairo_pattern_t *
|
|
|
|
|
|
cairo_test_create_pattern_from_png (const char *filename)
|
|
|
|
|
|
{
|
|
|
|
|
|
cairo_surface_t *image;
|
|
|
|
|
|
cairo_pattern_t *pattern;
|
|
|
|
|
|
|
|
|
|
|
|
image = cairo_test_create_surface_from_png (filename);
|
|
|
|
|
|
|
2005-01-26 13:41:55 +00:00
|
|
|
|
pattern = cairo_pattern_create_for_surface (image);
|
2005-07-14 16:18:39 +00:00
|
|
|
|
|
2005-05-06 13:32:53 +00:00
|
|
|
|
cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
|
2005-01-26 13:41:55 +00:00
|
|
|
|
|
2005-07-14 16:18:39 +00:00
|
|
|
|
cairo_surface_destroy (image);
|
|
|
|
|
|
|
2005-01-26 13:41:55 +00:00
|
|
|
|
return pattern;
|
|
|
|
|
|
}
|