xref: /aosp_15_r20/external/google-benchmark/test/benchmark_random_interleaving_gtest.cc (revision dbb99499c3810fa1611fa2242a2fc446be01a57c)
1*dbb99499SAndroid Build Coastguard Worker #include <queue>
2*dbb99499SAndroid Build Coastguard Worker #include <string>
3*dbb99499SAndroid Build Coastguard Worker #include <vector>
4*dbb99499SAndroid Build Coastguard Worker 
5*dbb99499SAndroid Build Coastguard Worker #include "../src/commandlineflags.h"
6*dbb99499SAndroid Build Coastguard Worker #include "../src/string_util.h"
7*dbb99499SAndroid Build Coastguard Worker #include "benchmark/benchmark.h"
8*dbb99499SAndroid Build Coastguard Worker #include "gmock/gmock.h"
9*dbb99499SAndroid Build Coastguard Worker #include "gtest/gtest.h"
10*dbb99499SAndroid Build Coastguard Worker 
11*dbb99499SAndroid Build Coastguard Worker namespace benchmark {
12*dbb99499SAndroid Build Coastguard Worker 
13*dbb99499SAndroid Build Coastguard Worker BM_DECLARE_bool(benchmark_enable_random_interleaving);
14*dbb99499SAndroid Build Coastguard Worker BM_DECLARE_string(benchmark_filter);
15*dbb99499SAndroid Build Coastguard Worker BM_DECLARE_int32(benchmark_repetitions);
16*dbb99499SAndroid Build Coastguard Worker 
17*dbb99499SAndroid Build Coastguard Worker namespace internal {
18*dbb99499SAndroid Build Coastguard Worker namespace {
19*dbb99499SAndroid Build Coastguard Worker 
20*dbb99499SAndroid Build Coastguard Worker class EventQueue : public std::queue<std::string> {
21*dbb99499SAndroid Build Coastguard Worker  public:
Put(const std::string & event)22*dbb99499SAndroid Build Coastguard Worker   void Put(const std::string& event) { push(event); }
23*dbb99499SAndroid Build Coastguard Worker 
Clear()24*dbb99499SAndroid Build Coastguard Worker   void Clear() {
25*dbb99499SAndroid Build Coastguard Worker     while (!empty()) {
26*dbb99499SAndroid Build Coastguard Worker       pop();
27*dbb99499SAndroid Build Coastguard Worker     }
28*dbb99499SAndroid Build Coastguard Worker   }
29*dbb99499SAndroid Build Coastguard Worker 
Get()30*dbb99499SAndroid Build Coastguard Worker   std::string Get() {
31*dbb99499SAndroid Build Coastguard Worker     std::string event = front();
32*dbb99499SAndroid Build Coastguard Worker     pop();
33*dbb99499SAndroid Build Coastguard Worker     return event;
34*dbb99499SAndroid Build Coastguard Worker   }
35*dbb99499SAndroid Build Coastguard Worker };
36*dbb99499SAndroid Build Coastguard Worker 
37*dbb99499SAndroid Build Coastguard Worker EventQueue* queue = new EventQueue();
38*dbb99499SAndroid Build Coastguard Worker 
39*dbb99499SAndroid Build Coastguard Worker class NullReporter : public BenchmarkReporter {
40*dbb99499SAndroid Build Coastguard Worker  public:
ReportContext(const Context &)41*dbb99499SAndroid Build Coastguard Worker   bool ReportContext(const Context& /*context*/) override { return true; }
ReportRuns(const std::vector<Run> &)42*dbb99499SAndroid Build Coastguard Worker   void ReportRuns(const std::vector<Run>& /* report */) override {}
43*dbb99499SAndroid Build Coastguard Worker };
44*dbb99499SAndroid Build Coastguard Worker 
45*dbb99499SAndroid Build Coastguard Worker class BenchmarkTest : public testing::Test {
46*dbb99499SAndroid Build Coastguard Worker  public:
SetupHook(int)47*dbb99499SAndroid Build Coastguard Worker   static void SetupHook(int /* num_threads */) { queue->push("Setup"); }
48*dbb99499SAndroid Build Coastguard Worker 
TeardownHook(int)49*dbb99499SAndroid Build Coastguard Worker   static void TeardownHook(int /* num_threads */) { queue->push("Teardown"); }
50*dbb99499SAndroid Build Coastguard Worker 
Execute(const std::string & pattern)51*dbb99499SAndroid Build Coastguard Worker   void Execute(const std::string& pattern) {
52*dbb99499SAndroid Build Coastguard Worker     queue->Clear();
53*dbb99499SAndroid Build Coastguard Worker 
54*dbb99499SAndroid Build Coastguard Worker     std::unique_ptr<BenchmarkReporter> reporter(new NullReporter());
55*dbb99499SAndroid Build Coastguard Worker     FLAGS_benchmark_filter = pattern;
56*dbb99499SAndroid Build Coastguard Worker     RunSpecifiedBenchmarks(reporter.get());
57*dbb99499SAndroid Build Coastguard Worker 
58*dbb99499SAndroid Build Coastguard Worker     queue->Put("DONE");  // End marker
59*dbb99499SAndroid Build Coastguard Worker   }
60*dbb99499SAndroid Build Coastguard Worker };
61*dbb99499SAndroid Build Coastguard Worker 
BM_Match1(benchmark::State & state)62*dbb99499SAndroid Build Coastguard Worker void BM_Match1(benchmark::State& state) {
63*dbb99499SAndroid Build Coastguard Worker   const int64_t arg = state.range(0);
64*dbb99499SAndroid Build Coastguard Worker 
65*dbb99499SAndroid Build Coastguard Worker   for (auto _ : state) {
66*dbb99499SAndroid Build Coastguard Worker   }
67*dbb99499SAndroid Build Coastguard Worker   queue->Put(StrFormat("BM_Match1/%d", static_cast<int>(arg)));
68*dbb99499SAndroid Build Coastguard Worker }
69*dbb99499SAndroid Build Coastguard Worker BENCHMARK(BM_Match1)
70*dbb99499SAndroid Build Coastguard Worker     ->Iterations(100)
71*dbb99499SAndroid Build Coastguard Worker     ->Arg(1)
72*dbb99499SAndroid Build Coastguard Worker     ->Arg(2)
73*dbb99499SAndroid Build Coastguard Worker     ->Arg(3)
74*dbb99499SAndroid Build Coastguard Worker     ->Range(10, 80)
75*dbb99499SAndroid Build Coastguard Worker     ->Args({90})
76*dbb99499SAndroid Build Coastguard Worker     ->Args({100});
77*dbb99499SAndroid Build Coastguard Worker 
TEST_F(BenchmarkTest,Match1)78*dbb99499SAndroid Build Coastguard Worker TEST_F(BenchmarkTest, Match1) {
79*dbb99499SAndroid Build Coastguard Worker   Execute("BM_Match1");
80*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/1", queue->Get());
81*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/2", queue->Get());
82*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/3", queue->Get());
83*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/10", queue->Get());
84*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/64", queue->Get());
85*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/80", queue->Get());
86*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/90", queue->Get());
87*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/100", queue->Get());
88*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("DONE", queue->Get());
89*dbb99499SAndroid Build Coastguard Worker }
90*dbb99499SAndroid Build Coastguard Worker 
TEST_F(BenchmarkTest,Match1WithRepetition)91*dbb99499SAndroid Build Coastguard Worker TEST_F(BenchmarkTest, Match1WithRepetition) {
92*dbb99499SAndroid Build Coastguard Worker   FLAGS_benchmark_repetitions = 2;
93*dbb99499SAndroid Build Coastguard Worker 
94*dbb99499SAndroid Build Coastguard Worker   Execute("BM_Match1/(64|80)");
95*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/64", queue->Get());
96*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/64", queue->Get());
97*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/80", queue->Get());
98*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("BM_Match1/80", queue->Get());
99*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("DONE", queue->Get());
100*dbb99499SAndroid Build Coastguard Worker }
101*dbb99499SAndroid Build Coastguard Worker 
TEST_F(BenchmarkTest,Match1WithRandomInterleaving)102*dbb99499SAndroid Build Coastguard Worker TEST_F(BenchmarkTest, Match1WithRandomInterleaving) {
103*dbb99499SAndroid Build Coastguard Worker   FLAGS_benchmark_enable_random_interleaving = true;
104*dbb99499SAndroid Build Coastguard Worker   FLAGS_benchmark_repetitions = 100;
105*dbb99499SAndroid Build Coastguard Worker 
106*dbb99499SAndroid Build Coastguard Worker   std::map<std::string, int> element_count;
107*dbb99499SAndroid Build Coastguard Worker   std::map<std::string, int> interleaving_count;
108*dbb99499SAndroid Build Coastguard Worker   Execute("BM_Match1/(64|80)");
109*dbb99499SAndroid Build Coastguard Worker   for (int i = 0; i < 100; ++i) {
110*dbb99499SAndroid Build Coastguard Worker     std::vector<std::string> interleaving;
111*dbb99499SAndroid Build Coastguard Worker     interleaving.push_back(queue->Get());
112*dbb99499SAndroid Build Coastguard Worker     interleaving.push_back(queue->Get());
113*dbb99499SAndroid Build Coastguard Worker     element_count[interleaving[0]]++;
114*dbb99499SAndroid Build Coastguard Worker     element_count[interleaving[1]]++;
115*dbb99499SAndroid Build Coastguard Worker     interleaving_count[StrFormat("%s,%s", interleaving[0].c_str(),
116*dbb99499SAndroid Build Coastguard Worker                                  interleaving[1].c_str())]++;
117*dbb99499SAndroid Build Coastguard Worker   }
118*dbb99499SAndroid Build Coastguard Worker   EXPECT_EQ(element_count["BM_Match1/64"], 100) << "Unexpected repetitions.";
119*dbb99499SAndroid Build Coastguard Worker   EXPECT_EQ(element_count["BM_Match1/80"], 100) << "Unexpected repetitions.";
120*dbb99499SAndroid Build Coastguard Worker   EXPECT_GE(interleaving_count.size(), 2) << "Interleaving was not randomized.";
121*dbb99499SAndroid Build Coastguard Worker   ASSERT_EQ("DONE", queue->Get());
122*dbb99499SAndroid Build Coastguard Worker }
123*dbb99499SAndroid Build Coastguard Worker 
124*dbb99499SAndroid Build Coastguard Worker }  // namespace
125*dbb99499SAndroid Build Coastguard Worker }  // namespace internal
126*dbb99499SAndroid Build Coastguard Worker }  // namespace benchmark
127