blob: 44cbeaca7c4d4bcd0232628be8df68e0a3801879 [file] [log] [blame]
#include "hb-benchmark.hh"
#include <glib.h>
#define SUBSET_FONT_BASE_PATH "test/subset/data/fonts/"
struct test_input_t
{
const char *font_path;
const char *text_path;
} default_tests[] =
{
{"perf/fonts/NotoNastaliqUrdu-Regular.ttf",
"perf/texts/fa-thelittleprince.txt"},
{"perf/fonts/NotoNastaliqUrdu-Regular.ttf",
"perf/texts/fa-words.txt"},
{"perf/fonts/Amiri-Regular.ttf",
"perf/texts/fa-thelittleprince.txt"},
{SUBSET_FONT_BASE_PATH "NotoSansDevanagari-Regular.ttf",
"perf/texts/hi-words.txt"},
{"perf/fonts/Roboto-Regular.ttf",
"perf/texts/en-thelittleprince.txt"},
{"perf/fonts/Roboto-Regular.ttf",
"perf/texts/en-words.txt"},
{SUBSET_FONT_BASE_PATH "SourceSerifVariable-Roman.ttf",
"perf/texts/react-dom.txt"},
};
static test_input_t *tests = default_tests;
static unsigned num_tests = sizeof (default_tests) / sizeof (default_tests[0]);
const char *variation = nullptr;
const char *direction = nullptr;
static bool shape (hb_buffer_t *buf,
hb_font_t *font,
const char *text,
unsigned text_length,
const char *shaper)
{
const char *end;
while ((end = (const char *) memchr (text, '\n', text_length)))
{
hb_buffer_clear_contents (buf);
hb_buffer_add_utf8 (buf, text, text_length, 0, end - text);
hb_buffer_guess_segment_properties (buf);
if (direction)
hb_buffer_set_direction (buf, hb_direction_from_string (direction, -1));
const char *shaper_list[] = {shaper, nullptr};
if (!hb_shape_full (font, buf, nullptr, 0, shaper_list))
return false;
unsigned skip = end - text + 1;
text_length -= skip;
text += skip;
}
return true;
}
static void BM_Shape (benchmark::State &state,
const char *shaper,
const test_input_t &input)
{
hb_font_t *font;
{
hb_face_t *face = hb_benchmark_face_create_from_file_or_fail (input.font_path, 0);
assert (face);
font = hb_font_create (face);
hb_face_destroy (face);
}
if (variation)
{
hb_variation_t var;
hb_variation_from_string (variation, -1, &var);
hb_font_set_variations (font, &var, 1);
}
hb_blob_t *text_blob = hb_blob_create_from_file_or_fail (input.text_path);
assert (text_blob);
unsigned text_length;
const char *text = hb_blob_get_data (text_blob, &text_length);
hb_buffer_t *buf = hb_buffer_create ();
// Shape once, to warm up the font and buffer.
bool ret = shape (buf, font, text, text_length, shaper);
if (!ret)
{
state.SkipWithMessage ("Shaping failed.");
goto done;
}
for (auto _ : state)
{
bool ret = shape (buf, font, text, text_length, shaper);
assert (ret);
}
done:
hb_buffer_destroy (buf);
hb_blob_destroy (text_blob);
hb_font_destroy (font);
}
static void test_shaper (const char *shaper,
const test_input_t &test_input)
{
char name[1024] = "BM_Shape";
const char *p;
strcat (name, "/");
p = strrchr (test_input.font_path, '/');
strcat (name, p ? p + 1 : test_input.font_path);
strcat (name, "/");
p = strrchr (test_input.text_path, '/');
strcat (name, p ? p + 1 : test_input.text_path);
strcat (name, "/");
strcat (name, shaper);
benchmark::RegisterBenchmark (name, BM_Shape, shaper, test_input)
->Unit(benchmark::kMillisecond);
}
static const char *font_file = nullptr;
static const char *text_file = nullptr;
static GOptionEntry entries[] =
{
{"font-file", 0, 0, G_OPTION_ARG_STRING, &font_file, "Font file-path to benchmark", "FONTFILE"},
{"text-file", 0, 0, G_OPTION_ARG_STRING, &text_file, "Text file-path to benchmark", "TEXTFILE"},
{"variations", 0, 0, G_OPTION_ARG_STRING, &variation, "Variations to apply during shaping", "VARIATIONS"},
{"direction", 0, 0, G_OPTION_ARG_STRING, &direction, "Direction to apply during shaping", "DIRECTION"},
{nullptr}
};
static void print_usage (const char *prgname)
{
g_print ("Usage: %s [OPTIONS] [FONTFILE]\n", prgname);
}
int main(int argc, char** argv)
{
const char *prgname = g_path_get_basename (argv[0]);
GOptionContext *context = g_option_context_new ("");
g_option_context_set_summary (context, "Benchmark text shaping with different shapers");
g_option_context_set_description (context, "Benchmark Options:");
g_option_context_add_main_entries (context, entries, nullptr);
// if --help is provided, we need to write help for both option parsers.
bool show_help = false;
for (int i = 1; i < argc; i++)
{
if (strcmp(argv[i], "--help") == 0 || strcmp(argv[i], "-h") == 0)
{
argv[i] = (char *) "--help"; // Ensure it is recognized by both google-benchmark
show_help = true;
break;
}
}
if (show_help)
{
print_usage (prgname);
gchar *help_text = g_option_context_get_help (context, false, nullptr);
help_text = strstr (help_text, "\n\n");
g_print ("%s", help_text);
benchmark::Initialize(&argc, argv); // This shows the help for google-benchmark
}
benchmark::Initialize(&argc, argv);
g_option_context_parse (context, &argc, &argv, nullptr);
g_option_context_free (context);
argc--;
argv++;
if (!font_file && argc)
{
font_file = *argv;
argv++;
argc--;
}
if (!text_file && argc)
{
text_file = *argv;
argv++;
argc--;
}
if (argc)
{
g_printerr ("Unexpected arguments: ");
for (int i = 0; i < argc; i++)
g_printerr ("%s ", argv[i]);
g_printerr ("\n\n");
print_usage (prgname);
return 1;
}
test_input_t static_test = {};
if (font_file && text_file)
{
if (font_file && text_file)
{
static_test.font_path = font_file;
static_test.text_path = text_file;
tests = &static_test;
num_tests = 1;
}
}
for (unsigned i = 0; i < num_tests; i++)
{
auto& test_input = tests[i];
const char **shapers = hb_shape_list_shapers ();
for (const char **shaper = shapers; *shaper; shaper++)
test_shaper (*shaper, test_input);
}
benchmark::RunSpecifiedBenchmarks();
benchmark::Shutdown();
}