xref: /aosp_15_r20/external/abseil-cpp/absl/strings/internal/memutil_benchmark.cc (revision 9356374a3709195abf420251b3e825997ff56c0f)
1*9356374aSAndroid Build Coastguard Worker // Copyright 2018 The Abseil Authors.
2*9356374aSAndroid Build Coastguard Worker //
3*9356374aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*9356374aSAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*9356374aSAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*9356374aSAndroid Build Coastguard Worker //
7*9356374aSAndroid Build Coastguard Worker //      https://www.apache.org/licenses/LICENSE-2.0
8*9356374aSAndroid Build Coastguard Worker //
9*9356374aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*9356374aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*9356374aSAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*9356374aSAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*9356374aSAndroid Build Coastguard Worker // limitations under the License.
14*9356374aSAndroid Build Coastguard Worker 
15*9356374aSAndroid Build Coastguard Worker #include "absl/strings/internal/memutil.h"
16*9356374aSAndroid Build Coastguard Worker 
17*9356374aSAndroid Build Coastguard Worker #include <algorithm>
18*9356374aSAndroid Build Coastguard Worker #include <cstdlib>
19*9356374aSAndroid Build Coastguard Worker 
20*9356374aSAndroid Build Coastguard Worker #include "benchmark/benchmark.h"
21*9356374aSAndroid Build Coastguard Worker #include "absl/strings/ascii.h"
22*9356374aSAndroid Build Coastguard Worker 
23*9356374aSAndroid Build Coastguard Worker // We fill the haystack with aaaaaaaaaaaaaaaaaa...aaaab.
24*9356374aSAndroid Build Coastguard Worker // That gives us:
25*9356374aSAndroid Build Coastguard Worker // - an easy search: 'b'
26*9356374aSAndroid Build Coastguard Worker // - a medium search: 'ab'.  That means every letter is a possible match.
27*9356374aSAndroid Build Coastguard Worker // - a pathological search: 'aaaaaa.......aaaaab' (half as many a's as haytack)
28*9356374aSAndroid Build Coastguard Worker 
29*9356374aSAndroid Build Coastguard Worker namespace {
30*9356374aSAndroid Build Coastguard Worker 
31*9356374aSAndroid Build Coastguard Worker constexpr int kHaystackSize = 10000;
32*9356374aSAndroid Build Coastguard Worker constexpr int64_t kHaystackSize64 = kHaystackSize;
MakeHaystack()33*9356374aSAndroid Build Coastguard Worker const char* MakeHaystack() {
34*9356374aSAndroid Build Coastguard Worker   char* haystack = new char[kHaystackSize];
35*9356374aSAndroid Build Coastguard Worker   for (int i = 0; i < kHaystackSize - 1; ++i) haystack[i] = 'a';
36*9356374aSAndroid Build Coastguard Worker   haystack[kHaystackSize - 1] = 'b';
37*9356374aSAndroid Build Coastguard Worker   return haystack;
38*9356374aSAndroid Build Coastguard Worker }
39*9356374aSAndroid Build Coastguard Worker const char* const kHaystack = MakeHaystack();
40*9356374aSAndroid Build Coastguard Worker 
case_eq(const char a,const char b)41*9356374aSAndroid Build Coastguard Worker bool case_eq(const char a, const char b) {
42*9356374aSAndroid Build Coastguard Worker   return absl::ascii_tolower(a) == absl::ascii_tolower(b);
43*9356374aSAndroid Build Coastguard Worker }
44*9356374aSAndroid Build Coastguard Worker 
BM_Searchcase(benchmark::State & state)45*9356374aSAndroid Build Coastguard Worker void BM_Searchcase(benchmark::State& state) {
46*9356374aSAndroid Build Coastguard Worker   for (auto _ : state) {
47*9356374aSAndroid Build Coastguard Worker     benchmark::DoNotOptimize(std::search(kHaystack, kHaystack + kHaystackSize,
48*9356374aSAndroid Build Coastguard Worker                                          kHaystack + kHaystackSize - 1,
49*9356374aSAndroid Build Coastguard Worker                                          kHaystack + kHaystackSize, case_eq));
50*9356374aSAndroid Build Coastguard Worker   }
51*9356374aSAndroid Build Coastguard Worker   state.SetBytesProcessed(kHaystackSize64 * state.iterations());
52*9356374aSAndroid Build Coastguard Worker }
53*9356374aSAndroid Build Coastguard Worker BENCHMARK(BM_Searchcase);
54*9356374aSAndroid Build Coastguard Worker 
BM_SearchcaseMedium(benchmark::State & state)55*9356374aSAndroid Build Coastguard Worker void BM_SearchcaseMedium(benchmark::State& state) {
56*9356374aSAndroid Build Coastguard Worker   for (auto _ : state) {
57*9356374aSAndroid Build Coastguard Worker     benchmark::DoNotOptimize(std::search(kHaystack, kHaystack + kHaystackSize,
58*9356374aSAndroid Build Coastguard Worker                                          kHaystack + kHaystackSize - 2,
59*9356374aSAndroid Build Coastguard Worker                                          kHaystack + kHaystackSize, case_eq));
60*9356374aSAndroid Build Coastguard Worker   }
61*9356374aSAndroid Build Coastguard Worker   state.SetBytesProcessed(kHaystackSize64 * state.iterations());
62*9356374aSAndroid Build Coastguard Worker }
63*9356374aSAndroid Build Coastguard Worker BENCHMARK(BM_SearchcaseMedium);
64*9356374aSAndroid Build Coastguard Worker 
BM_SearchcasePathological(benchmark::State & state)65*9356374aSAndroid Build Coastguard Worker void BM_SearchcasePathological(benchmark::State& state) {
66*9356374aSAndroid Build Coastguard Worker   for (auto _ : state) {
67*9356374aSAndroid Build Coastguard Worker     benchmark::DoNotOptimize(std::search(kHaystack, kHaystack + kHaystackSize,
68*9356374aSAndroid Build Coastguard Worker                                          kHaystack + kHaystackSize / 2,
69*9356374aSAndroid Build Coastguard Worker                                          kHaystack + kHaystackSize, case_eq));
70*9356374aSAndroid Build Coastguard Worker   }
71*9356374aSAndroid Build Coastguard Worker   state.SetBytesProcessed(kHaystackSize64 * state.iterations());
72*9356374aSAndroid Build Coastguard Worker }
73*9356374aSAndroid Build Coastguard Worker BENCHMARK(BM_SearchcasePathological);
74*9356374aSAndroid Build Coastguard Worker 
memcasechr(const char * s,int c,size_t slen)75*9356374aSAndroid Build Coastguard Worker char* memcasechr(const char* s, int c, size_t slen) {
76*9356374aSAndroid Build Coastguard Worker   c = absl::ascii_tolower(c);
77*9356374aSAndroid Build Coastguard Worker   for (; slen; ++s, --slen) {
78*9356374aSAndroid Build Coastguard Worker     if (absl::ascii_tolower(*s) == c) return const_cast<char*>(s);
79*9356374aSAndroid Build Coastguard Worker   }
80*9356374aSAndroid Build Coastguard Worker   return nullptr;
81*9356374aSAndroid Build Coastguard Worker }
82*9356374aSAndroid Build Coastguard Worker 
memcasematch(const char * phaystack,size_t haylen,const char * pneedle,size_t neelen)83*9356374aSAndroid Build Coastguard Worker const char* memcasematch(const char* phaystack, size_t haylen,
84*9356374aSAndroid Build Coastguard Worker                          const char* pneedle, size_t neelen) {
85*9356374aSAndroid Build Coastguard Worker   if (0 == neelen) {
86*9356374aSAndroid Build Coastguard Worker     return phaystack;  // even if haylen is 0
87*9356374aSAndroid Build Coastguard Worker   }
88*9356374aSAndroid Build Coastguard Worker   if (haylen < neelen) return nullptr;
89*9356374aSAndroid Build Coastguard Worker 
90*9356374aSAndroid Build Coastguard Worker   const char* match;
91*9356374aSAndroid Build Coastguard Worker   const char* hayend = phaystack + haylen - neelen + 1;
92*9356374aSAndroid Build Coastguard Worker   while ((match = static_cast<char*>(
93*9356374aSAndroid Build Coastguard Worker               memcasechr(phaystack, pneedle[0], hayend - phaystack)))) {
94*9356374aSAndroid Build Coastguard Worker     if (absl::strings_internal::memcasecmp(match, pneedle, neelen) == 0)
95*9356374aSAndroid Build Coastguard Worker       return match;
96*9356374aSAndroid Build Coastguard Worker     else
97*9356374aSAndroid Build Coastguard Worker       phaystack = match + 1;
98*9356374aSAndroid Build Coastguard Worker   }
99*9356374aSAndroid Build Coastguard Worker   return nullptr;
100*9356374aSAndroid Build Coastguard Worker }
101*9356374aSAndroid Build Coastguard Worker 
BM_Memcasematch(benchmark::State & state)102*9356374aSAndroid Build Coastguard Worker void BM_Memcasematch(benchmark::State& state) {
103*9356374aSAndroid Build Coastguard Worker   for (auto _ : state) {
104*9356374aSAndroid Build Coastguard Worker     benchmark::DoNotOptimize(memcasematch(kHaystack, kHaystackSize, "b", 1));
105*9356374aSAndroid Build Coastguard Worker   }
106*9356374aSAndroid Build Coastguard Worker   state.SetBytesProcessed(kHaystackSize64 * state.iterations());
107*9356374aSAndroid Build Coastguard Worker }
108*9356374aSAndroid Build Coastguard Worker BENCHMARK(BM_Memcasematch);
109*9356374aSAndroid Build Coastguard Worker 
BM_MemcasematchMedium(benchmark::State & state)110*9356374aSAndroid Build Coastguard Worker void BM_MemcasematchMedium(benchmark::State& state) {
111*9356374aSAndroid Build Coastguard Worker   for (auto _ : state) {
112*9356374aSAndroid Build Coastguard Worker     benchmark::DoNotOptimize(memcasematch(kHaystack, kHaystackSize, "ab", 2));
113*9356374aSAndroid Build Coastguard Worker   }
114*9356374aSAndroid Build Coastguard Worker   state.SetBytesProcessed(kHaystackSize64 * state.iterations());
115*9356374aSAndroid Build Coastguard Worker }
116*9356374aSAndroid Build Coastguard Worker BENCHMARK(BM_MemcasematchMedium);
117*9356374aSAndroid Build Coastguard Worker 
BM_MemcasematchPathological(benchmark::State & state)118*9356374aSAndroid Build Coastguard Worker void BM_MemcasematchPathological(benchmark::State& state) {
119*9356374aSAndroid Build Coastguard Worker   for (auto _ : state) {
120*9356374aSAndroid Build Coastguard Worker     benchmark::DoNotOptimize(memcasematch(kHaystack, kHaystackSize,
121*9356374aSAndroid Build Coastguard Worker                                           kHaystack + kHaystackSize / 2,
122*9356374aSAndroid Build Coastguard Worker                                           kHaystackSize - kHaystackSize / 2));
123*9356374aSAndroid Build Coastguard Worker   }
124*9356374aSAndroid Build Coastguard Worker   state.SetBytesProcessed(kHaystackSize64 * state.iterations());
125*9356374aSAndroid Build Coastguard Worker }
126*9356374aSAndroid Build Coastguard Worker BENCHMARK(BM_MemcasematchPathological);
127*9356374aSAndroid Build Coastguard Worker 
128*9356374aSAndroid Build Coastguard Worker }  // namespace
129