2006-08-31 07:19:05 -07:00
|
|
|
/*
|
2006-08-31 10:39:24 -07:00
|
|
|
* Copyright © 2006 Mozilla Corporation
|
2006-08-31 07:19:05 -07:00
|
|
|
* Copyright © 2006 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
|
2006-08-31 10:39:24 -07:00
|
|
|
* the authors not be used in advertising or publicity pertaining to
|
2006-08-31 07:19:05 -07:00
|
|
|
* distribution of the software without specific, written prior
|
2006-08-31 10:39:24 -07:00
|
|
|
* permission. The authors make no representations about the
|
2006-08-31 07:19:05 -07:00
|
|
|
* suitability of this software for any purpose. It is provided "as
|
|
|
|
|
* is" without express or implied warranty.
|
|
|
|
|
*
|
2006-08-31 10:39:24 -07:00
|
|
|
* THE AUTHORS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
|
2006-08-31 07:19:05 -07:00
|
|
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
2006-08-31 10:39:24 -07:00
|
|
|
* FITNESS, IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY SPECIAL,
|
2006-08-31 07:19:05 -07:00
|
|
|
* 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.
|
|
|
|
|
*
|
2006-08-31 10:39:24 -07:00
|
|
|
* Authors: Vladimir Vukicevic <vladimir@pobox.com>
|
|
|
|
|
* Carl Worth <cworth@cworth.org>
|
2006-08-31 07:19:05 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "cairo-perf.h"
|
|
|
|
|
|
2006-08-31 14:08:43 -07:00
|
|
|
double cairo_perf_duration = 1;
|
2006-08-31 13:22:17 -07:00
|
|
|
|
|
|
|
|
int cairo_perf_iterations = 10;
|
2006-08-31 10:39:24 -07:00
|
|
|
|
2006-08-31 11:02:20 -07:00
|
|
|
int cairo_perf_alarm_expired = 0;
|
2006-08-31 07:19:05 -07:00
|
|
|
|
|
|
|
|
typedef struct _cairo_perf {
|
|
|
|
|
const char *name;
|
|
|
|
|
cairo_perf_func_t run;
|
|
|
|
|
unsigned int min_size;
|
|
|
|
|
unsigned int max_size;
|
|
|
|
|
} cairo_perf_t;
|
|
|
|
|
|
2006-08-31 11:51:28 -07:00
|
|
|
cairo_perf_t perfs[];
|
2006-08-31 07:19:05 -07:00
|
|
|
|
2006-08-31 08:53:58 -07:00
|
|
|
/* Some targets just aren't that interesting for performance testing,
|
|
|
|
|
* (not least because many of these surface types use a meta-surface
|
|
|
|
|
* and as such defer the "real" rendering to later, so our timing
|
|
|
|
|
* loops wouldn't count the real work, just the recording by the
|
|
|
|
|
* meta-surface. */
|
|
|
|
|
static cairo_bool_t
|
|
|
|
|
target_is_measurable (cairo_test_target_t *target)
|
|
|
|
|
{
|
|
|
|
|
switch (target->expected_type) {
|
|
|
|
|
case CAIRO_SURFACE_TYPE_IMAGE:
|
2006-08-31 11:53:16 -07:00
|
|
|
if (strcmp (target->name, "pdf") == 0 ||
|
|
|
|
|
strcmp (target->name, "ps") == 0)
|
2006-08-31 08:53:58 -07:00
|
|
|
{
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return TRUE;
|
|
|
|
|
}
|
|
|
|
|
case CAIRO_SURFACE_TYPE_XLIB:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_XCB:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_GLITZ:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_QUARTZ:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_WIN32:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_BEOS:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_DIRECTFB:
|
|
|
|
|
return TRUE;
|
|
|
|
|
case CAIRO_SURFACE_TYPE_PDF:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_PS:
|
|
|
|
|
case CAIRO_SURFACE_TYPE_SVG:
|
|
|
|
|
default:
|
|
|
|
|
return FALSE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2006-08-31 12:34:21 -07:00
|
|
|
static const char *
|
|
|
|
|
_content_to_string (cairo_content_t content)
|
|
|
|
|
{
|
|
|
|
|
switch (content) {
|
|
|
|
|
case CAIRO_CONTENT_COLOR:
|
|
|
|
|
return "rgb";
|
|
|
|
|
case CAIRO_CONTENT_ALPHA:
|
|
|
|
|
return "a";
|
|
|
|
|
case CAIRO_CONTENT_COLOR_ALPHA:
|
|
|
|
|
return "rgba";
|
|
|
|
|
default:
|
|
|
|
|
return "<unknown_content>";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2006-08-31 13:22:17 -07:00
|
|
|
typedef struct _stats
|
|
|
|
|
{
|
|
|
|
|
double mean;
|
|
|
|
|
double std_dev;
|
|
|
|
|
} stats_t;
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
_compute_stats (double *values, int num_values, stats_t *stats)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
double sum, delta;
|
|
|
|
|
|
|
|
|
|
sum = 0.0;
|
|
|
|
|
for (i = 0; i < num_values; i++)
|
|
|
|
|
sum += values[i];
|
|
|
|
|
|
|
|
|
|
stats->mean = sum / num_values;
|
|
|
|
|
|
|
|
|
|
sum = 0.0;
|
|
|
|
|
for (i = 0; i < num_values; i++) {
|
|
|
|
|
delta = values[i] - stats->mean;
|
|
|
|
|
sum += delta * delta;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Let's use a std. deviation normalized to the mean for easier
|
|
|
|
|
* comparison. */
|
|
|
|
|
stats->std_dev = sqrt(sum / num_values) / stats->mean;
|
|
|
|
|
}
|
|
|
|
|
|
2006-08-31 07:19:05 -07:00
|
|
|
int
|
|
|
|
|
main (int argc, char *argv[])
|
|
|
|
|
{
|
2006-08-31 13:22:17 -07:00
|
|
|
int i, j, k;
|
2006-08-31 07:19:05 -07:00
|
|
|
cairo_test_target_t *target;
|
|
|
|
|
cairo_perf_t *perf;
|
|
|
|
|
cairo_surface_t *surface;
|
|
|
|
|
cairo_t *cr;
|
|
|
|
|
unsigned int size;
|
2006-08-31 13:22:17 -07:00
|
|
|
double *rates;
|
|
|
|
|
stats_t stats;
|
2006-08-31 07:19:05 -07:00
|
|
|
|
2006-08-31 11:19:48 -07:00
|
|
|
if (getenv("CAIRO_PERF_DURATION"))
|
2006-08-31 14:08:43 -07:00
|
|
|
cairo_perf_duration = strtod(getenv("CAIRO_PERF_DURATION"), NULL);
|
2006-08-31 11:19:48 -07:00
|
|
|
|
2006-08-31 13:22:17 -07:00
|
|
|
if (getenv("CAIRO_PERF_ITERATIONS"))
|
|
|
|
|
cairo_perf_iterations = strtol(getenv("CAIRO_PERF_ITERATIONS"), NULL, 0);
|
|
|
|
|
|
|
|
|
|
rates = xmalloc (cairo_perf_iterations * sizeof (double));
|
|
|
|
|
|
2006-08-31 07:19:05 -07:00
|
|
|
for (i = 0; targets[i].name; i++) {
|
|
|
|
|
target = &targets[i];
|
2006-08-31 08:53:58 -07:00
|
|
|
if (! target_is_measurable (target))
|
|
|
|
|
continue;
|
2006-08-31 07:19:05 -07:00
|
|
|
for (j = 0; perfs[j].name; j++) {
|
|
|
|
|
perf = &perfs[j];
|
|
|
|
|
for (size = perf->min_size; size <= perf->max_size; size *= 2) {
|
|
|
|
|
surface = (target->create_target_surface) (perf->name,
|
|
|
|
|
target->content,
|
|
|
|
|
size, size,
|
|
|
|
|
&target->closure);
|
|
|
|
|
cr = cairo_create (surface);
|
2006-08-31 13:22:17 -07:00
|
|
|
for (k =0; k < cairo_perf_iterations; k++)
|
|
|
|
|
rates[k] = perf->run (cr, size, size);
|
|
|
|
|
_compute_stats (rates, cairo_perf_iterations, &stats);
|
|
|
|
|
if (i==0 && j==0 && size == perf->min_size)
|
|
|
|
|
printf ("backend-content\ttest-size\trate\tstd dev.\titerations\n");
|
|
|
|
|
printf ("%s-%s\t%s-%d\t%g\t%g%%\t%d\n",
|
|
|
|
|
target->name, _content_to_string (target->content),
|
|
|
|
|
perf->name, size,
|
|
|
|
|
stats.mean, stats.std_dev * 100.0, cairo_perf_iterations);
|
2006-08-31 07:19:05 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2006-08-31 11:51:28 -07:00
|
|
|
cairo_perf_t perfs[] = {
|
|
|
|
|
{ "paint", paint, 32, 1024 },
|
|
|
|
|
{ "paint_alpha", paint_alpha, 32, 1024 },
|
|
|
|
|
{ NULL }
|
|
|
|
|
};
|