summaryrefslogtreecommitdiff
path: root/perf
diff options
context:
space:
mode:
authorAndrea Canciani <ranma42@gmail.com>2011-08-03 09:49:08 +0200
committerAndrea Canciani <ranma42@gmail.com>2011-08-03 12:31:41 +0200
commit374b26ff03b9f36a7be974e65e42938a3c11b04c (patch)
tree3ce12cfe8a4b7b6e49b2bead595df27e14323e07 /perf
parent7b5b29cc0ffc44066589d371d872e355ce56343b (diff)
downloadcairo-374b26ff03b9f36a7be974e65e42938a3c11b04c.tar.gz
perf: Add hash table benchmark
A benchmark to test the speed of hash tables when inserting and removing a huge number of elements. Although originally hash tables were assumed not to get many deletions, in practice they are now being used as caches in multiple places. This means that they often have a fixed number of live elements and an element is evicted whenever a new element is inserted (this happens explicitly for cairo_cache_t objects, but also, for example, in scaled_font_map + holdovers). This access pattern is very inefficient with the current implementation.
Diffstat (limited to 'perf')
-rw-r--r--perf/cairo-perf-micro.c1
-rw-r--r--perf/cairo-perf.h1
-rw-r--r--perf/micro/Makefile.sources1
-rw-r--r--perf/micro/hash-table.c107
4 files changed, 110 insertions, 0 deletions
diff --git a/perf/cairo-perf-micro.c b/perf/cairo-perf-micro.c
index ee5269a97..ca447defb 100644
--- a/perf/cairo-perf-micro.c
+++ b/perf/cairo-perf-micro.c
@@ -548,6 +548,7 @@ const cairo_perf_case_t perf_cases[] = {
{ hatching, 64, 512},
{ tessellate, 100, 100},
{ subimage_copy, 16, 512},
+ { hash_table, 16, 16},
{ pattern_create_radial, 16, 16},
{ zrusin, 415, 415},
{ world_map, 800, 800},
diff --git a/perf/cairo-perf.h b/perf/cairo-perf.h
index 59a0d226c..32edc744f 100644
--- a/perf/cairo-perf.h
+++ b/perf/cairo-perf.h
@@ -193,6 +193,7 @@ CAIRO_PERF_DECL (hatching);
CAIRO_PERF_DECL (tessellate);
CAIRO_PERF_DECL (text);
CAIRO_PERF_DECL (glyphs);
+CAIRO_PERF_DECL (hash_table);
CAIRO_PERF_DECL (pattern_create_radial);
CAIRO_PERF_DECL (zrusin);
CAIRO_PERF_DECL (world_map);
diff --git a/perf/micro/Makefile.sources b/perf/micro/Makefile.sources
index aae9e2137..90f4bb6ff 100644
--- a/perf/micro/Makefile.sources
+++ b/perf/micro/Makefile.sources
@@ -5,6 +5,7 @@ libcairo_perf_micro_sources = \
disjoint.c \
fill.c \
hatching.c \
+ hash-table.c \
long-lines.c \
mosaic.c \
paint.c \
diff --git a/perf/micro/hash-table.c b/perf/micro/hash-table.c
new file mode 100644
index 000000000..79978f55a
--- /dev/null
+++ b/perf/micro/hash-table.c
@@ -0,0 +1,107 @@
+/*
+ * Copyright © 2011 Andrea Canciani
+ *
+ * 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: Andrea Canciani <ranma42@gmail.com>
+ */
+
+#include "cairo-perf.h"
+
+#define ITER 1000
+#define HOLDOVERS 256
+#define LIVE_ENTRIES 257
+#define ACTIVE_FONTS (LIVE_ENTRIES - HOLDOVERS - 1)
+
+/*
+ * The original implementation of hash tables was very inefficient, as
+ * pointed out in https://bugs.freedesktop.org/show_bug.cgi?id=17399
+ *
+ * This benchmark tries to fill up the scaled_font_map hash table to
+ * show the O(n) behavior.
+ */
+
+static cairo_perf_ticks_t
+do_hash_table (cairo_t *cr, int width, int height, int loops)
+{
+ cairo_scaled_font_t *active_fonts[ACTIVE_FONTS];
+ cairo_matrix_t m;
+ int i;
+
+ cairo_matrix_init_identity (&m);
+
+ /* Touch HOLDOVERS scaled fonts to fill up the holdover list. */
+ for (i = 0; i < HOLDOVERS; i++) {
+ m.yy = m.xx * (i + 1);
+ cairo_set_font_matrix (cr, &m);
+ cairo_get_scaled_font (cr);
+ }
+
+ /*
+ * Reference some scaled fonts so that they will be kept in the
+ * scaled fonts map. We want LIVE_ENTRIES elements in the font
+ * map, but cairo keeps HOLDOVERS recently used fonts in it and we
+ * will be activating a new font in the cr context, so we just
+ * keep references to ACTIVE_FONTS fonts.
+ *
+ * Note: setting LIVE_ENTRIES == HOLDOVERS+1 means that we keep no
+ * font in active_fonts and the slowness is caused by the holdover
+ * fonts only.
+ */
+ for (i = 0; i < ACTIVE_FONTS; i++) {
+ cairo_scaled_font_t *scaled_font;
+
+ m.yy = m.xx * (i + 1);
+ cairo_set_font_matrix (cr, &m);
+
+ scaled_font = cairo_get_scaled_font (cr);
+ active_fonts[i] = cairo_scaled_font_reference (scaled_font);
+ }
+
+ cairo_perf_timer_start ();
+
+ while (loops--) {
+ m.xx += 1.0;
+
+ /* Generate ITER new scaled fonts per loop */
+ for (i = 0; i < ITER; i++) {
+ m.yy = m.xx * (i + 1);
+ cairo_set_font_matrix (cr, &m);
+ cairo_get_scaled_font (cr);
+ }
+ }
+
+ cairo_perf_timer_stop ();
+
+ for (i = 0; i < ACTIVE_FONTS; i++)
+ cairo_scaled_font_destroy (active_fonts[i]);
+
+ return cairo_perf_timer_elapsed ();
+}
+
+void
+hash_table (cairo_perf_t *perf, cairo_t *cr, int width, int height)
+{
+ if (! cairo_perf_can_run (perf, "hash-table", NULL))
+ return;
+
+ cairo_perf_cover_sources_and_operators (perf, "hash-table",
+ do_hash_table, NULL);
+}