#include "benchmark/benchmark.h" #include #include #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "hb.h" #include "hb-ot.h" #ifdef HAVE_FREETYPE #include "hb-ft.h" #endif #define SUBSET_FONT_BASE_PATH "test/subset/data/fonts/" struct test_input_t { bool is_variable; const char *font_path; } default_tests[] = { {false, SUBSET_FONT_BASE_PATH "Roboto-Regular.ttf"}, {true , SUBSET_FONT_BASE_PATH "RobotoFlex-Variable.ttf"}, {false, SUBSET_FONT_BASE_PATH "SourceSansPro-Regular.otf"}, {true , SUBSET_FONT_BASE_PATH "AdobeVFPrototype.otf"}, {true , SUBSET_FONT_BASE_PATH "SourceSerifVariable-Roman.ttf"}, {false, SUBSET_FONT_BASE_PATH "Comfortaa-Regular-new.ttf"}, {false, SUBSET_FONT_BASE_PATH "NotoNastaliqUrdu-Regular.ttf"}, {false, SUBSET_FONT_BASE_PATH "NotoSerifMyanmar-Regular.otf"}, }; static test_input_t *tests = default_tests; static unsigned num_tests = sizeof (default_tests) / sizeof (default_tests[0]); enum backend_t { HARFBUZZ, FREETYPE }; enum operation_t { nominal_glyphs, glyph_h_advances, glyph_extents, draw_glyph, paint_glyph, load_face_and_shape, }; static void _hb_move_to (hb_draw_funcs_t *, void *draw_data, hb_draw_state_t *, float x, float y, void *) { float &i = * (float *) draw_data; i += x + y; } static void _hb_line_to (hb_draw_funcs_t *, void *draw_data, hb_draw_state_t *, float x, float y, void *) { float &i = * (float *) draw_data; i += x + y; } static void _hb_quadratic_to (hb_draw_funcs_t *, void *draw_data, hb_draw_state_t *, float cx, float cy, float x, float y, void *) { float &i = * (float *) draw_data; i += cx + cy + x + y; } static void _hb_cubic_to (hb_draw_funcs_t *, void *draw_data, hb_draw_state_t *, float cx1, float cy1, float cx2, float cy2, float x, float y, void *) { float &i = * (float *) draw_data; i += cx1 + cy1 + cx2 + cy2 + x + y; } static void _hb_close_path (hb_draw_funcs_t *, void *draw_data, hb_draw_state_t *, void *) { float &i = * (float *) draw_data; i += 1.0; } static hb_draw_funcs_t * _draw_funcs_create (void) { hb_draw_funcs_t *draw_funcs = hb_draw_funcs_create (); hb_draw_funcs_set_move_to_func (draw_funcs, _hb_move_to, nullptr, nullptr); hb_draw_funcs_set_line_to_func (draw_funcs, _hb_line_to, nullptr, nullptr); hb_draw_funcs_set_quadratic_to_func (draw_funcs, _hb_quadratic_to, nullptr, nullptr); hb_draw_funcs_set_cubic_to_func (draw_funcs, _hb_cubic_to, nullptr, nullptr); hb_draw_funcs_set_close_path_func (draw_funcs, _hb_close_path, nullptr, nullptr); return draw_funcs; } static void BM_Font (benchmark::State &state, bool is_var, backend_t backend, operation_t operation, const test_input_t &test_input) { hb_font_t *font; unsigned num_glyphs; { hb_blob_t *blob = hb_blob_create_from_file_or_fail (test_input.font_path); assert (blob); hb_face_t *face = hb_face_create (blob, 0); hb_blob_destroy (blob); num_glyphs = hb_face_get_glyph_count (face); font = hb_font_create (face); hb_face_destroy (face); } if (is_var) { hb_variation_t wght = {HB_TAG ('w','g','h','t'), 500}; hb_font_set_variations (font, &wght, 1); } switch (backend) { case HARFBUZZ: hb_ot_font_set_funcs (font); break; case FREETYPE: #ifdef HAVE_FREETYPE hb_ft_font_set_funcs (font); #endif break; } switch (operation) { case nominal_glyphs: { hb_set_t *set = hb_set_create (); hb_face_collect_unicodes (hb_font_get_face (font), set); unsigned pop = hb_set_get_population (set); hb_codepoint_t *unicodes = (hb_codepoint_t *) calloc (pop, sizeof (hb_codepoint_t)); hb_codepoint_t *glyphs = (hb_codepoint_t *) calloc (pop, sizeof (hb_codepoint_t)); hb_codepoint_t *p = unicodes; for (hb_codepoint_t u = HB_SET_VALUE_INVALID; hb_set_next (set, &u);) *p++ = u; assert (p == unicodes + pop); for (auto _ : state) hb_font_get_nominal_glyphs (font, pop, unicodes, sizeof (*unicodes), glyphs, sizeof (*glyphs)); free (glyphs); free (unicodes); hb_set_destroy (set); break; } case glyph_h_advances: { hb_codepoint_t *glyphs = (hb_codepoint_t *) calloc (num_glyphs, sizeof (hb_codepoint_t)); hb_position_t *advances = (hb_position_t *) calloc (num_glyphs, sizeof (hb_codepoint_t)); for (unsigned g = 0; g < num_glyphs; g++) glyphs[g] = g; for (auto _ : state) hb_font_get_glyph_h_advances (font, num_glyphs, glyphs, sizeof (*glyphs), advances, sizeof (*advances)); free (advances); free (glyphs); break; } case glyph_extents: { hb_glyph_extents_t extents; for (auto _ : state) for (unsigned gid = 0; gid < num_glyphs; ++gid) hb_font_get_glyph_extents (font, gid, &extents); break; } case draw_glyph: { hb_draw_funcs_t *draw_funcs = _draw_funcs_create (); for (auto _ : state) { float i = 0; for (unsigned gid = 0; gid < num_glyphs; ++gid) hb_font_draw_glyph (font, gid, draw_funcs, &i); } hb_draw_funcs_destroy (draw_funcs); break; } case paint_glyph: { hb_paint_funcs_t *paint_funcs = hb_paint_funcs_create (); for (auto _ : state) { for (unsigned gid = 0; gid < num_glyphs; ++gid) hb_font_paint_glyph (font, gid, paint_funcs, nullptr, 0, 0); } hb_paint_funcs_destroy (paint_funcs); break; } case load_face_and_shape: { for (auto _ : state) { hb_blob_t *blob = hb_blob_create_from_file_or_fail (test_input.font_path); assert (blob); hb_face_t *face = hb_face_create (blob, 0); hb_blob_destroy (blob); hb_font_t *font = hb_font_create (face); hb_face_destroy (face); switch (backend) { case HARFBUZZ: hb_ot_font_set_funcs (font); break; case FREETYPE: #ifdef HAVE_FREETYPE hb_ft_font_set_funcs (font); #endif break; } hb_buffer_t *buffer = hb_buffer_create (); hb_buffer_add_utf8 (buffer, " ", -1, 0, -1); hb_buffer_guess_segment_properties (buffer); hb_shape (font, buffer, nullptr, 0); hb_buffer_destroy (buffer); hb_font_destroy (font); } break; } } hb_font_destroy (font); } static void test_backend (backend_t backend, const char *backend_name, bool variable, operation_t op, const char *op_name, benchmark::TimeUnit time_unit, const test_input_t &test_input) { char name[1024] = "BM_Font/"; strcat (name, op_name); strcat (name, "/"); const char *p = strrchr (test_input.font_path, '/'); strcat (name, p ? p + 1 : test_input.font_path); strcat (name, variable ? "/var" : ""); strcat (name, "/"); strcat (name, backend_name); benchmark::RegisterBenchmark (name, BM_Font, variable, backend, op, test_input) ->Unit(time_unit); } static void test_operation (operation_t op, const char *op_name, benchmark::TimeUnit time_unit) { for (unsigned i = 0; i < num_tests; i++) { auto& test_input = tests[i]; for (int variable = 0; variable < int (test_input.is_variable) + 1; variable++) { bool is_var = (bool) variable; test_backend (HARFBUZZ, "hb", is_var, op, op_name, time_unit, test_input); #ifdef HAVE_FREETYPE test_backend (FREETYPE, "ft", is_var, op, op_name, time_unit, test_input); #endif } } } int main(int argc, char** argv) { benchmark::Initialize(&argc, argv); if (argc > 1) { num_tests = argc - 1; tests = (test_input_t *) calloc (num_tests, sizeof (test_input_t)); for (unsigned i = 0; i < num_tests; i++) { tests[i].is_variable = true; tests[i].font_path = argv[i + 1]; } } #define TEST_OPERATION(op, time_unit) test_operation (op, #op, time_unit) TEST_OPERATION (nominal_glyphs, benchmark::kMicrosecond); TEST_OPERATION (glyph_h_advances, benchmark::kMicrosecond); TEST_OPERATION (glyph_extents, benchmark::kMicrosecond); TEST_OPERATION (draw_glyph, benchmark::kMicrosecond); TEST_OPERATION (paint_glyph, benchmark::kMillisecond); TEST_OPERATION (load_face_and_shape, benchmark::kMicrosecond); #undef TEST_OPERATION benchmark::RunSpecifiedBenchmarks(); benchmark::Shutdown(); if (tests != default_tests) free (tests); }