xref: /aosp_15_r20/external/harfbuzz_ng/perf/benchmark-map.cc (revision 2d1272b857b1f7575e6e246373e1cb218663db8a)
1*2d1272b8SAndroid Build Coastguard Worker #include "hb-benchmark.hh"
2*2d1272b8SAndroid Build Coastguard Worker 
RandomMap(unsigned size,hb_map_t * out,hb_set_t * key_sample)3*2d1272b8SAndroid Build Coastguard Worker void RandomMap(unsigned size, hb_map_t* out, hb_set_t* key_sample) {
4*2d1272b8SAndroid Build Coastguard Worker   hb_map_clear(out);
5*2d1272b8SAndroid Build Coastguard Worker 
6*2d1272b8SAndroid Build Coastguard Worker   unsigned sample_denom = 1;
7*2d1272b8SAndroid Build Coastguard Worker   if (size > 10000)
8*2d1272b8SAndroid Build Coastguard Worker     sample_denom = size / 10000;
9*2d1272b8SAndroid Build Coastguard Worker 
10*2d1272b8SAndroid Build Coastguard Worker   srand(size);
11*2d1272b8SAndroid Build Coastguard Worker   for (unsigned i = 0; i < size; i++) {
12*2d1272b8SAndroid Build Coastguard Worker     while (true) {
13*2d1272b8SAndroid Build Coastguard Worker       hb_codepoint_t next = rand();
14*2d1272b8SAndroid Build Coastguard Worker       if (hb_map_has (out, next)) continue;
15*2d1272b8SAndroid Build Coastguard Worker 
16*2d1272b8SAndroid Build Coastguard Worker       hb_codepoint_t val = rand();
17*2d1272b8SAndroid Build Coastguard Worker       if (key_sample && val % sample_denom == 0)
18*2d1272b8SAndroid Build Coastguard Worker         hb_set_add (key_sample, next);
19*2d1272b8SAndroid Build Coastguard Worker       hb_map_set (out, next, val);
20*2d1272b8SAndroid Build Coastguard Worker       break;
21*2d1272b8SAndroid Build Coastguard Worker     }
22*2d1272b8SAndroid Build Coastguard Worker   }
23*2d1272b8SAndroid Build Coastguard Worker }
24*2d1272b8SAndroid Build Coastguard Worker 
25*2d1272b8SAndroid Build Coastguard Worker /* Insert a single value into map of varying sizes. */
BM_MapInsert(benchmark::State & state)26*2d1272b8SAndroid Build Coastguard Worker static void BM_MapInsert(benchmark::State& state) {
27*2d1272b8SAndroid Build Coastguard Worker   unsigned map_size = state.range(0);
28*2d1272b8SAndroid Build Coastguard Worker 
29*2d1272b8SAndroid Build Coastguard Worker   hb_map_t* original = hb_map_create ();
30*2d1272b8SAndroid Build Coastguard Worker   RandomMap(map_size, original, nullptr);
31*2d1272b8SAndroid Build Coastguard Worker   assert(hb_map_get_population(original) == map_size);
32*2d1272b8SAndroid Build Coastguard Worker 
33*2d1272b8SAndroid Build Coastguard Worker   unsigned mask = 1;
34*2d1272b8SAndroid Build Coastguard Worker   while (mask < map_size)
35*2d1272b8SAndroid Build Coastguard Worker     mask <<= 1;
36*2d1272b8SAndroid Build Coastguard Worker   mask--;
37*2d1272b8SAndroid Build Coastguard Worker 
38*2d1272b8SAndroid Build Coastguard Worker   auto needle = 0u;
39*2d1272b8SAndroid Build Coastguard Worker   for (auto _ : state) {
40*2d1272b8SAndroid Build Coastguard Worker     // TODO(garretrieger): create a copy of the original map.
41*2d1272b8SAndroid Build Coastguard Worker     //                     Needs a hb_map_copy(..) in public api.
42*2d1272b8SAndroid Build Coastguard Worker 
43*2d1272b8SAndroid Build Coastguard Worker     hb_map_set (original, needle++ & mask, 1);
44*2d1272b8SAndroid Build Coastguard Worker   }
45*2d1272b8SAndroid Build Coastguard Worker 
46*2d1272b8SAndroid Build Coastguard Worker   hb_map_destroy(original);
47*2d1272b8SAndroid Build Coastguard Worker }
48*2d1272b8SAndroid Build Coastguard Worker BENCHMARK(BM_MapInsert)
49*2d1272b8SAndroid Build Coastguard Worker     ->Range(1 << 4, 1 << 20);
50*2d1272b8SAndroid Build Coastguard Worker 
51*2d1272b8SAndroid Build Coastguard Worker /* Single value lookup on map of various sizes where the key is not present. */
BM_MapLookupMiss(benchmark::State & state)52*2d1272b8SAndroid Build Coastguard Worker static void BM_MapLookupMiss(benchmark::State& state) {
53*2d1272b8SAndroid Build Coastguard Worker   unsigned map_size = state.range(0);
54*2d1272b8SAndroid Build Coastguard Worker 
55*2d1272b8SAndroid Build Coastguard Worker   hb_map_t* original = hb_map_create ();
56*2d1272b8SAndroid Build Coastguard Worker   RandomMap(map_size, original, nullptr);
57*2d1272b8SAndroid Build Coastguard Worker   assert(hb_map_get_population(original) == map_size);
58*2d1272b8SAndroid Build Coastguard Worker 
59*2d1272b8SAndroid Build Coastguard Worker   auto needle = map_size / 2;
60*2d1272b8SAndroid Build Coastguard Worker 
61*2d1272b8SAndroid Build Coastguard Worker   for (auto _ : state) {
62*2d1272b8SAndroid Build Coastguard Worker     benchmark::DoNotOptimize(
63*2d1272b8SAndroid Build Coastguard Worker         hb_map_get (original, needle++));
64*2d1272b8SAndroid Build Coastguard Worker   }
65*2d1272b8SAndroid Build Coastguard Worker 
66*2d1272b8SAndroid Build Coastguard Worker   hb_map_destroy(original);
67*2d1272b8SAndroid Build Coastguard Worker }
68*2d1272b8SAndroid Build Coastguard Worker BENCHMARK(BM_MapLookupMiss)
69*2d1272b8SAndroid Build Coastguard Worker     ->Range(1 << 4, 1 << 20); // Map size
70*2d1272b8SAndroid Build Coastguard Worker 
71*2d1272b8SAndroid Build Coastguard Worker /* Single value lookup on map of various sizes. */
BM_MapLookupHit(benchmark::State & state)72*2d1272b8SAndroid Build Coastguard Worker static void BM_MapLookupHit(benchmark::State& state) {
73*2d1272b8SAndroid Build Coastguard Worker   unsigned map_size = state.range(0);
74*2d1272b8SAndroid Build Coastguard Worker 
75*2d1272b8SAndroid Build Coastguard Worker   hb_map_t* original = hb_map_create ();
76*2d1272b8SAndroid Build Coastguard Worker   hb_set_t* key_set = hb_set_create ();
77*2d1272b8SAndroid Build Coastguard Worker   RandomMap(map_size, original, key_set);
78*2d1272b8SAndroid Build Coastguard Worker   assert(hb_map_get_population(original) == map_size);
79*2d1272b8SAndroid Build Coastguard Worker 
80*2d1272b8SAndroid Build Coastguard Worker   unsigned num_keys = hb_set_get_population (key_set);
81*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t* key_array =
82*2d1272b8SAndroid Build Coastguard Worker     (hb_codepoint_t*) calloc (num_keys, sizeof(hb_codepoint_t));
83*2d1272b8SAndroid Build Coastguard Worker 
84*2d1272b8SAndroid Build Coastguard Worker   hb_codepoint_t cp = HB_SET_VALUE_INVALID;
85*2d1272b8SAndroid Build Coastguard Worker   unsigned i = 0;
86*2d1272b8SAndroid Build Coastguard Worker   while (hb_set_next (key_set, &cp))
87*2d1272b8SAndroid Build Coastguard Worker     key_array[i++] = cp;
88*2d1272b8SAndroid Build Coastguard Worker 
89*2d1272b8SAndroid Build Coastguard Worker   i = 0;
90*2d1272b8SAndroid Build Coastguard Worker   for (auto _ : state) {
91*2d1272b8SAndroid Build Coastguard Worker     benchmark::DoNotOptimize(
92*2d1272b8SAndroid Build Coastguard Worker         hb_map_get (original, key_array[i++ % num_keys]));
93*2d1272b8SAndroid Build Coastguard Worker   }
94*2d1272b8SAndroid Build Coastguard Worker 
95*2d1272b8SAndroid Build Coastguard Worker   hb_set_destroy (key_set);
96*2d1272b8SAndroid Build Coastguard Worker   free (key_array);
97*2d1272b8SAndroid Build Coastguard Worker   hb_map_destroy(original);
98*2d1272b8SAndroid Build Coastguard Worker }
99*2d1272b8SAndroid Build Coastguard Worker BENCHMARK(BM_MapLookupHit)
100*2d1272b8SAndroid Build Coastguard Worker     ->Range(1 << 4, 1 << 20); // Map size
101*2d1272b8SAndroid Build Coastguard Worker 
102*2d1272b8SAndroid Build Coastguard Worker 
103*2d1272b8SAndroid Build Coastguard Worker BENCHMARK_MAIN();
104