#include <assert.h>
#include "cairo-perf.h"
#define MAX_SEGMENTS 2560
typedef enum {
PIXALIGN,
NONALIGN
} align_t;
typedef enum {
RECTCLOSE,
DIAGCLOSE
} close_t;
static cairo_perf_ticks_t
draw_spiral (cairo_t *cr,
cairo_fill_rule_t fill_rule,
align_t align,
close_t close,
int width, int height, int loops)
{
int i;
int n=0;
double x[MAX_SEGMENTS];
double y[MAX_SEGMENTS];
int step = 3;
int side = width < height ? width : height;
assert(5*(side/step/2+1)+2 < MAX_SEGMENTS);
#define L(x_,y_) (x[n] = (x_), y[n] = (y_), n++)
#define M(x_,y_) L(x_,y_)
#define v(t) L(x[n-1], y[n-1] + (t))
#define h(t) L(x[n-1] + (t), y[n-1])
switch (align) {
case PIXALIGN: M(0,0); break;
case NONALIGN: M(0.1415926, 0.7182818); break;
}
while (side >= step && side >= 0) {
v(side);
h(side);
v(-side);
h(-side+step);
v(step);
side -= 2*step;
}
switch (close) {
case RECTCLOSE: L(x[n-1],y[0]); break;
case DIAGCLOSE: L(x[0],y[0]); break;
}
assert(n < MAX_SEGMENTS);
cairo_save (cr);
cairo_set_source_rgb (cr, 0, 0, 0);
cairo_paint (cr);
cairo_translate (cr, 1, 1);
cairo_set_fill_rule (cr, fill_rule);
cairo_set_source_rgb (cr, 1, 0, 0);
cairo_new_path (cr);
cairo_move_to (cr, x[0], y[0]);
for (i = 1; i < n; i++) {
cairo_line_to (cr, x[i], y[i]);
}
cairo_close_path (cr);
cairo_perf_timer_start ();
while (loops--)
cairo_fill_preserve (cr);
cairo_perf_timer_stop ();
cairo_restore (cr);
return cairo_perf_timer_elapsed ();
}
static cairo_perf_ticks_t
draw_spiral_box (cairo_t *cr,
cairo_fill_rule_t fill_rule,
align_t align,
int width, int height, int loops)
{
const int step = 3;
int side = (width < height ? width : height) - 2;
cairo_save (cr);
cairo_set_source_rgb (cr, 0, 0, 0);
cairo_paint (cr);
cairo_set_source_rgb (cr, 1, 0, 0);
cairo_set_fill_rule (cr, fill_rule);
cairo_translate (cr, 1, 1);
if (align == NONALIGN)
cairo_translate (cr, 0.1415926, 0.7182818);
cairo_new_path (cr);
while (side >= step) {
cairo_rectangle (cr, 0, 0, side, side);
cairo_translate (cr, step, step);
side -= 2*step;
}
cairo_perf_timer_start ();
while (loops--)
cairo_fill_preserve (cr);
cairo_perf_timer_stop ();
cairo_restore (cr);
return cairo_perf_timer_elapsed ();
}
static cairo_perf_ticks_t
draw_spiral_stroke (cairo_t *cr,
align_t align,
int width, int height, int loops)
{
const int step = 3;
int side = width < height ? width : height;
cairo_save (cr);
cairo_set_source_rgb (cr, 0, 0, 0);
cairo_paint (cr);
cairo_translate (cr, 1, 1);
cairo_set_source_rgb (cr, 1, 0, 0);
cairo_set_line_width (cr, 4.);
cairo_set_line_join (cr, CAIRO_LINE_JOIN_MITER);
cairo_set_line_cap (cr, CAIRO_LINE_CAP_BUTT);
cairo_new_path (cr);
switch (align) {
case PIXALIGN: cairo_move_to (cr, 0,0); break;
case NONALIGN: cairo_move_to (cr, 0.1415926, 0.7182818); break;
}
while (side >= step) {
cairo_rel_line_to (cr, 0, side);
side -= step;
if (side <= 0)
break;
cairo_rel_line_to (cr, side, 0);
side -= step;
if (side <= 0)
break;
cairo_rel_line_to (cr, 0, -side);
side -= step;
if (side <= 0)
break;
cairo_rel_line_to (cr, -side, 0);
side -= step;
if (side <= 0)
break;
}
cairo_perf_timer_start ();
while (loops--)
cairo_stroke_preserve (cr);
cairo_perf_timer_stop ();
cairo_restore (cr);
return cairo_perf_timer_elapsed ();
}
static cairo_perf_ticks_t
draw_spiral_eo_pa_re (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_EVEN_ODD,
PIXALIGN,
RECTCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_nz_pa_re (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_WINDING,
PIXALIGN,
RECTCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_eo_na_re (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_EVEN_ODD,
NONALIGN,
RECTCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_nz_na_re (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_WINDING,
NONALIGN,
RECTCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_eo_pa_di (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_EVEN_ODD,
PIXALIGN,
DIAGCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_nz_pa_di (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_WINDING,
PIXALIGN,
DIAGCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_eo_na_di (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_EVEN_ODD,
NONALIGN,
DIAGCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_nz_na_di (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral (cr,
CAIRO_FILL_RULE_WINDING,
NONALIGN,
DIAGCLOSE,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_nz_pa_box (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral_box (cr,
CAIRO_FILL_RULE_WINDING, PIXALIGN,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_nz_na_box (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral_box (cr,
CAIRO_FILL_RULE_WINDING, NONALIGN,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_eo_pa_box (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral_box (cr,
CAIRO_FILL_RULE_EVEN_ODD, PIXALIGN,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_eo_na_box (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral_box (cr,
CAIRO_FILL_RULE_EVEN_ODD, NONALIGN,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_stroke_pa (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral_stroke (cr,
PIXALIGN,
width, height, loops);
}
static cairo_perf_ticks_t
draw_spiral_stroke_na (cairo_t *cr, int width, int height, int loops)
{
return draw_spiral_stroke (cr,
NONALIGN,
width, height, loops);
}
void
spiral (cairo_perf_t *perf, cairo_t *cr, int width, int height)
{
if (! cairo_perf_can_run (perf, "spiral", NULL))
return;
cairo_perf_run (perf, "spiral-box-nonalign-evenodd-fill", draw_spiral_eo_na_box, NULL);
cairo_perf_run (perf, "spiral-box-nonalign-nonzero-fill", draw_spiral_nz_na_box, NULL);
cairo_perf_run (perf, "spiral-box-pixalign-evenodd-fill", draw_spiral_eo_pa_box, NULL);
cairo_perf_run (perf, "spiral-box-pixalign-nonzero-fill", draw_spiral_nz_pa_box, NULL);
cairo_perf_run (perf, "spiral-diag-nonalign-evenodd-fill", draw_spiral_eo_na_di, NULL);
cairo_perf_run (perf, "spiral-diag-nonalign-nonzero-fill", draw_spiral_nz_na_di, NULL);
cairo_perf_run (perf, "spiral-diag-pixalign-evenodd-fill", draw_spiral_eo_pa_di, NULL);
cairo_perf_run (perf, "spiral-diag-pixalign-nonzero-fill", draw_spiral_nz_pa_di, NULL);
cairo_perf_run (perf, "spiral-rect-nonalign-evenodd-fill", draw_spiral_eo_na_re, NULL);
cairo_perf_run (perf, "spiral-rect-nonalign-nonzero-fill", draw_spiral_nz_na_re, NULL);
cairo_perf_run (perf, "spiral-rect-pixalign-evenodd-fill", draw_spiral_eo_pa_re, NULL);
cairo_perf_run (perf, "spiral-rect-pixalign-nonzero-fill", draw_spiral_nz_pa_re, NULL);
cairo_perf_run (perf, "spiral-nonalign-stroke", draw_spiral_stroke_na, NULL);
cairo_perf_run (perf, "spiral-pixalign-stroke", draw_spiral_stroke_pa, NULL);
}