1*dbb99499SAndroid Build Coastguard Worker // Copyright 2015 Google Inc. All rights reserved.
2*dbb99499SAndroid Build Coastguard Worker //
3*dbb99499SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*dbb99499SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*dbb99499SAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*dbb99499SAndroid Build Coastguard Worker //
7*dbb99499SAndroid Build Coastguard Worker // http://www.apache.org/licenses/LICENSE-2.0
8*dbb99499SAndroid Build Coastguard Worker //
9*dbb99499SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*dbb99499SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*dbb99499SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*dbb99499SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*dbb99499SAndroid Build Coastguard Worker // limitations under the License.
14*dbb99499SAndroid Build Coastguard Worker
15*dbb99499SAndroid Build Coastguard Worker #include "benchmark/benchmark.h"
16*dbb99499SAndroid Build Coastguard Worker
17*dbb99499SAndroid Build Coastguard Worker #include "benchmark_api_internal.h"
18*dbb99499SAndroid Build Coastguard Worker #include "benchmark_runner.h"
19*dbb99499SAndroid Build Coastguard Worker #include "internal_macros.h"
20*dbb99499SAndroid Build Coastguard Worker
21*dbb99499SAndroid Build Coastguard Worker #ifndef BENCHMARK_OS_WINDOWS
22*dbb99499SAndroid Build Coastguard Worker #if !defined(BENCHMARK_OS_FUCHSIA) && !defined(BENCHMARK_OS_QURT)
23*dbb99499SAndroid Build Coastguard Worker #include <sys/resource.h>
24*dbb99499SAndroid Build Coastguard Worker #endif
25*dbb99499SAndroid Build Coastguard Worker #include <sys/time.h>
26*dbb99499SAndroid Build Coastguard Worker #include <unistd.h>
27*dbb99499SAndroid Build Coastguard Worker #endif
28*dbb99499SAndroid Build Coastguard Worker
29*dbb99499SAndroid Build Coastguard Worker #include <algorithm>
30*dbb99499SAndroid Build Coastguard Worker #include <atomic>
31*dbb99499SAndroid Build Coastguard Worker #include <condition_variable>
32*dbb99499SAndroid Build Coastguard Worker #include <cstdio>
33*dbb99499SAndroid Build Coastguard Worker #include <cstdlib>
34*dbb99499SAndroid Build Coastguard Worker #include <fstream>
35*dbb99499SAndroid Build Coastguard Worker #include <iostream>
36*dbb99499SAndroid Build Coastguard Worker #include <limits>
37*dbb99499SAndroid Build Coastguard Worker #include <map>
38*dbb99499SAndroid Build Coastguard Worker #include <memory>
39*dbb99499SAndroid Build Coastguard Worker #include <random>
40*dbb99499SAndroid Build Coastguard Worker #include <string>
41*dbb99499SAndroid Build Coastguard Worker #include <thread>
42*dbb99499SAndroid Build Coastguard Worker #include <utility>
43*dbb99499SAndroid Build Coastguard Worker
44*dbb99499SAndroid Build Coastguard Worker #include "check.h"
45*dbb99499SAndroid Build Coastguard Worker #include "colorprint.h"
46*dbb99499SAndroid Build Coastguard Worker #include "commandlineflags.h"
47*dbb99499SAndroid Build Coastguard Worker #include "complexity.h"
48*dbb99499SAndroid Build Coastguard Worker #include "counter.h"
49*dbb99499SAndroid Build Coastguard Worker #include "internal_macros.h"
50*dbb99499SAndroid Build Coastguard Worker #include "log.h"
51*dbb99499SAndroid Build Coastguard Worker #include "mutex.h"
52*dbb99499SAndroid Build Coastguard Worker #include "perf_counters.h"
53*dbb99499SAndroid Build Coastguard Worker #include "re.h"
54*dbb99499SAndroid Build Coastguard Worker #include "statistics.h"
55*dbb99499SAndroid Build Coastguard Worker #include "string_util.h"
56*dbb99499SAndroid Build Coastguard Worker #include "thread_manager.h"
57*dbb99499SAndroid Build Coastguard Worker #include "thread_timer.h"
58*dbb99499SAndroid Build Coastguard Worker
59*dbb99499SAndroid Build Coastguard Worker namespace benchmark {
60*dbb99499SAndroid Build Coastguard Worker // Print a list of benchmarks. This option overrides all other options.
61*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_bool(benchmark_list_tests, false);
62*dbb99499SAndroid Build Coastguard Worker
63*dbb99499SAndroid Build Coastguard Worker // A regular expression that specifies the set of benchmarks to execute. If
64*dbb99499SAndroid Build Coastguard Worker // this flag is empty, or if this flag is the string \"all\", all benchmarks
65*dbb99499SAndroid Build Coastguard Worker // linked into the binary are run.
66*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_filter, "");
67*dbb99499SAndroid Build Coastguard Worker
68*dbb99499SAndroid Build Coastguard Worker // Specification of how long to run the benchmark.
69*dbb99499SAndroid Build Coastguard Worker //
70*dbb99499SAndroid Build Coastguard Worker // It can be either an exact number of iterations (specified as `<integer>x`),
71*dbb99499SAndroid Build Coastguard Worker // or a minimum number of seconds (specified as `<float>s`). If the latter
72*dbb99499SAndroid Build Coastguard Worker // format (ie., min seconds) is used, the system may run the benchmark longer
73*dbb99499SAndroid Build Coastguard Worker // until the results are considered significant.
74*dbb99499SAndroid Build Coastguard Worker //
75*dbb99499SAndroid Build Coastguard Worker // For backward compatibility, the `s` suffix may be omitted, in which case,
76*dbb99499SAndroid Build Coastguard Worker // the specified number is interpreted as the number of seconds.
77*dbb99499SAndroid Build Coastguard Worker //
78*dbb99499SAndroid Build Coastguard Worker // For cpu-time based tests, this is the lower bound
79*dbb99499SAndroid Build Coastguard Worker // on the total cpu time used by all threads that make up the test. For
80*dbb99499SAndroid Build Coastguard Worker // real-time based tests, this is the lower bound on the elapsed time of the
81*dbb99499SAndroid Build Coastguard Worker // benchmark execution, regardless of number of threads.
82*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_min_time, kDefaultMinTimeStr);
83*dbb99499SAndroid Build Coastguard Worker
84*dbb99499SAndroid Build Coastguard Worker // Minimum number of seconds a benchmark should be run before results should be
85*dbb99499SAndroid Build Coastguard Worker // taken into account. This e.g can be necessary for benchmarks of code which
86*dbb99499SAndroid Build Coastguard Worker // needs to fill some form of cache before performance is of interest.
87*dbb99499SAndroid Build Coastguard Worker // Note: results gathered within this period are discarded and not used for
88*dbb99499SAndroid Build Coastguard Worker // reported result.
89*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_double(benchmark_min_warmup_time, 0.0);
90*dbb99499SAndroid Build Coastguard Worker
91*dbb99499SAndroid Build Coastguard Worker // The number of runs of each benchmark. If greater than 1, the mean and
92*dbb99499SAndroid Build Coastguard Worker // standard deviation of the runs will be reported.
93*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_int32(benchmark_repetitions, 1);
94*dbb99499SAndroid Build Coastguard Worker
95*dbb99499SAndroid Build Coastguard Worker // If set, enable random interleaving of repetitions of all benchmarks.
96*dbb99499SAndroid Build Coastguard Worker // See http://github.com/google/benchmark/issues/1051 for details.
97*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_bool(benchmark_enable_random_interleaving, false);
98*dbb99499SAndroid Build Coastguard Worker
99*dbb99499SAndroid Build Coastguard Worker // Report the result of each benchmark repetitions. When 'true' is specified
100*dbb99499SAndroid Build Coastguard Worker // only the mean, standard deviation, and other statistics are reported for
101*dbb99499SAndroid Build Coastguard Worker // repeated benchmarks. Affects all reporters.
102*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_bool(benchmark_report_aggregates_only, false);
103*dbb99499SAndroid Build Coastguard Worker
104*dbb99499SAndroid Build Coastguard Worker // Display the result of each benchmark repetitions. When 'true' is specified
105*dbb99499SAndroid Build Coastguard Worker // only the mean, standard deviation, and other statistics are displayed for
106*dbb99499SAndroid Build Coastguard Worker // repeated benchmarks. Unlike benchmark_report_aggregates_only, only affects
107*dbb99499SAndroid Build Coastguard Worker // the display reporter, but *NOT* file reporter, which will still contain
108*dbb99499SAndroid Build Coastguard Worker // all the output.
109*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_bool(benchmark_display_aggregates_only, false);
110*dbb99499SAndroid Build Coastguard Worker
111*dbb99499SAndroid Build Coastguard Worker // The format to use for console output.
112*dbb99499SAndroid Build Coastguard Worker // Valid values are 'console', 'json', or 'csv'.
113*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_format, "console");
114*dbb99499SAndroid Build Coastguard Worker
115*dbb99499SAndroid Build Coastguard Worker // The format to use for file output.
116*dbb99499SAndroid Build Coastguard Worker // Valid values are 'console', 'json', or 'csv'.
117*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_out_format, "json");
118*dbb99499SAndroid Build Coastguard Worker
119*dbb99499SAndroid Build Coastguard Worker // The file to write additional output to.
120*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_out, "");
121*dbb99499SAndroid Build Coastguard Worker
122*dbb99499SAndroid Build Coastguard Worker // Whether to use colors in the output. Valid values:
123*dbb99499SAndroid Build Coastguard Worker // 'true'/'yes'/1, 'false'/'no'/0, and 'auto'. 'auto' means to use colors if
124*dbb99499SAndroid Build Coastguard Worker // the output is being sent to a terminal and the TERM environment variable is
125*dbb99499SAndroid Build Coastguard Worker // set to a terminal type that supports colors.
126*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_color, "auto");
127*dbb99499SAndroid Build Coastguard Worker
128*dbb99499SAndroid Build Coastguard Worker // Whether to use tabular format when printing user counters to the console.
129*dbb99499SAndroid Build Coastguard Worker // Valid values: 'true'/'yes'/1, 'false'/'no'/0. Defaults to false.
130*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_bool(benchmark_counters_tabular, false);
131*dbb99499SAndroid Build Coastguard Worker
132*dbb99499SAndroid Build Coastguard Worker // List of additional perf counters to collect, in libpfm format. For more
133*dbb99499SAndroid Build Coastguard Worker // information about libpfm: https://man7.org/linux/man-pages/man3/libpfm.3.html
134*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_perf_counters, "");
135*dbb99499SAndroid Build Coastguard Worker
136*dbb99499SAndroid Build Coastguard Worker // Extra context to include in the output formatted as comma-separated key-value
137*dbb99499SAndroid Build Coastguard Worker // pairs. Kept internal as it's only used for parsing from env/command line.
138*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_kvpairs(benchmark_context, {});
139*dbb99499SAndroid Build Coastguard Worker
140*dbb99499SAndroid Build Coastguard Worker // Set the default time unit to use for reports
141*dbb99499SAndroid Build Coastguard Worker // Valid values are 'ns', 'us', 'ms' or 's'
142*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_string(benchmark_time_unit, "");
143*dbb99499SAndroid Build Coastguard Worker
144*dbb99499SAndroid Build Coastguard Worker // The level of verbose logging to output
145*dbb99499SAndroid Build Coastguard Worker BM_DEFINE_int32(v, 0);
146*dbb99499SAndroid Build Coastguard Worker
147*dbb99499SAndroid Build Coastguard Worker namespace internal {
148*dbb99499SAndroid Build Coastguard Worker
149*dbb99499SAndroid Build Coastguard Worker std::map<std::string, std::string>* global_context = nullptr;
150*dbb99499SAndroid Build Coastguard Worker
GetGlobalContext()151*dbb99499SAndroid Build Coastguard Worker BENCHMARK_EXPORT std::map<std::string, std::string>*& GetGlobalContext() {
152*dbb99499SAndroid Build Coastguard Worker return global_context;
153*dbb99499SAndroid Build Coastguard Worker }
154*dbb99499SAndroid Build Coastguard Worker
155*dbb99499SAndroid Build Coastguard Worker static void const volatile* volatile global_force_escape_pointer;
156*dbb99499SAndroid Build Coastguard Worker
157*dbb99499SAndroid Build Coastguard Worker // FIXME: Verify if LTO still messes this up?
UseCharPointer(char const volatile * const v)158*dbb99499SAndroid Build Coastguard Worker void UseCharPointer(char const volatile* const v) {
159*dbb99499SAndroid Build Coastguard Worker // We want to escape the pointer `v` so that the compiler can not eliminate
160*dbb99499SAndroid Build Coastguard Worker // computations that produced it. To do that, we escape the pointer by storing
161*dbb99499SAndroid Build Coastguard Worker // it into a volatile variable, since generally, volatile store, is not
162*dbb99499SAndroid Build Coastguard Worker // something the compiler is allowed to elide.
163*dbb99499SAndroid Build Coastguard Worker global_force_escape_pointer = reinterpret_cast<void const volatile*>(v);
164*dbb99499SAndroid Build Coastguard Worker }
165*dbb99499SAndroid Build Coastguard Worker
166*dbb99499SAndroid Build Coastguard Worker } // namespace internal
167*dbb99499SAndroid Build Coastguard Worker
State(std::string name,IterationCount max_iters,const std::vector<int64_t> & ranges,int thread_i,int n_threads,internal::ThreadTimer * timer,internal::ThreadManager * manager,internal::PerfCountersMeasurement * perf_counters_measurement,ProfilerManager * profiler_manager)168*dbb99499SAndroid Build Coastguard Worker State::State(std::string name, IterationCount max_iters,
169*dbb99499SAndroid Build Coastguard Worker const std::vector<int64_t>& ranges, int thread_i, int n_threads,
170*dbb99499SAndroid Build Coastguard Worker internal::ThreadTimer* timer, internal::ThreadManager* manager,
171*dbb99499SAndroid Build Coastguard Worker internal::PerfCountersMeasurement* perf_counters_measurement,
172*dbb99499SAndroid Build Coastguard Worker ProfilerManager* profiler_manager)
173*dbb99499SAndroid Build Coastguard Worker : total_iterations_(0),
174*dbb99499SAndroid Build Coastguard Worker batch_leftover_(0),
175*dbb99499SAndroid Build Coastguard Worker max_iterations(max_iters),
176*dbb99499SAndroid Build Coastguard Worker started_(false),
177*dbb99499SAndroid Build Coastguard Worker finished_(false),
178*dbb99499SAndroid Build Coastguard Worker skipped_(internal::NotSkipped),
179*dbb99499SAndroid Build Coastguard Worker range_(ranges),
180*dbb99499SAndroid Build Coastguard Worker complexity_n_(0),
181*dbb99499SAndroid Build Coastguard Worker name_(std::move(name)),
182*dbb99499SAndroid Build Coastguard Worker thread_index_(thread_i),
183*dbb99499SAndroid Build Coastguard Worker threads_(n_threads),
184*dbb99499SAndroid Build Coastguard Worker timer_(timer),
185*dbb99499SAndroid Build Coastguard Worker manager_(manager),
186*dbb99499SAndroid Build Coastguard Worker perf_counters_measurement_(perf_counters_measurement),
187*dbb99499SAndroid Build Coastguard Worker profiler_manager_(profiler_manager) {
188*dbb99499SAndroid Build Coastguard Worker BM_CHECK(max_iterations != 0) << "At least one iteration must be run";
189*dbb99499SAndroid Build Coastguard Worker BM_CHECK_LT(thread_index_, threads_)
190*dbb99499SAndroid Build Coastguard Worker << "thread_index must be less than threads";
191*dbb99499SAndroid Build Coastguard Worker
192*dbb99499SAndroid Build Coastguard Worker // Add counters with correct flag now. If added with `counters[name]` in
193*dbb99499SAndroid Build Coastguard Worker // `PauseTiming`, a new `Counter` will be inserted the first time, which
194*dbb99499SAndroid Build Coastguard Worker // won't have the flag. Inserting them now also reduces the allocations
195*dbb99499SAndroid Build Coastguard Worker // during the benchmark.
196*dbb99499SAndroid Build Coastguard Worker if (perf_counters_measurement_) {
197*dbb99499SAndroid Build Coastguard Worker for (const std::string& counter_name :
198*dbb99499SAndroid Build Coastguard Worker perf_counters_measurement_->names()) {
199*dbb99499SAndroid Build Coastguard Worker counters[counter_name] = Counter(0.0, Counter::kAvgIterations);
200*dbb99499SAndroid Build Coastguard Worker }
201*dbb99499SAndroid Build Coastguard Worker }
202*dbb99499SAndroid Build Coastguard Worker
203*dbb99499SAndroid Build Coastguard Worker // Note: The use of offsetof below is technically undefined until C++17
204*dbb99499SAndroid Build Coastguard Worker // because State is not a standard layout type. However, all compilers
205*dbb99499SAndroid Build Coastguard Worker // currently provide well-defined behavior as an extension (which is
206*dbb99499SAndroid Build Coastguard Worker // demonstrated since constexpr evaluation must diagnose all undefined
207*dbb99499SAndroid Build Coastguard Worker // behavior). However, GCC and Clang also warn about this use of offsetof,
208*dbb99499SAndroid Build Coastguard Worker // which must be suppressed.
209*dbb99499SAndroid Build Coastguard Worker #if defined(__INTEL_COMPILER)
210*dbb99499SAndroid Build Coastguard Worker #pragma warning push
211*dbb99499SAndroid Build Coastguard Worker #pragma warning(disable : 1875)
212*dbb99499SAndroid Build Coastguard Worker #elif defined(__GNUC__) || defined(__clang__)
213*dbb99499SAndroid Build Coastguard Worker #pragma GCC diagnostic push
214*dbb99499SAndroid Build Coastguard Worker #pragma GCC diagnostic ignored "-Winvalid-offsetof"
215*dbb99499SAndroid Build Coastguard Worker #endif
216*dbb99499SAndroid Build Coastguard Worker #if defined(__NVCC__)
217*dbb99499SAndroid Build Coastguard Worker #pragma nv_diagnostic push
218*dbb99499SAndroid Build Coastguard Worker #pragma nv_diag_suppress 1427
219*dbb99499SAndroid Build Coastguard Worker #endif
220*dbb99499SAndroid Build Coastguard Worker #if defined(__NVCOMPILER)
221*dbb99499SAndroid Build Coastguard Worker #pragma diagnostic push
222*dbb99499SAndroid Build Coastguard Worker #pragma diag_suppress offset_in_non_POD_nonstandard
223*dbb99499SAndroid Build Coastguard Worker #endif
224*dbb99499SAndroid Build Coastguard Worker // Offset tests to ensure commonly accessed data is on the first cache line.
225*dbb99499SAndroid Build Coastguard Worker const int cache_line_size = 64;
226*dbb99499SAndroid Build Coastguard Worker static_assert(
227*dbb99499SAndroid Build Coastguard Worker offsetof(State, skipped_) <= (cache_line_size - sizeof(skipped_)), "");
228*dbb99499SAndroid Build Coastguard Worker #if defined(__INTEL_COMPILER)
229*dbb99499SAndroid Build Coastguard Worker #pragma warning pop
230*dbb99499SAndroid Build Coastguard Worker #elif defined(__GNUC__) || defined(__clang__)
231*dbb99499SAndroid Build Coastguard Worker #pragma GCC diagnostic pop
232*dbb99499SAndroid Build Coastguard Worker #endif
233*dbb99499SAndroid Build Coastguard Worker #if defined(__NVCC__)
234*dbb99499SAndroid Build Coastguard Worker #pragma nv_diagnostic pop
235*dbb99499SAndroid Build Coastguard Worker #endif
236*dbb99499SAndroid Build Coastguard Worker #if defined(__NVCOMPILER)
237*dbb99499SAndroid Build Coastguard Worker #pragma diagnostic pop
238*dbb99499SAndroid Build Coastguard Worker #endif
239*dbb99499SAndroid Build Coastguard Worker }
240*dbb99499SAndroid Build Coastguard Worker
PauseTiming()241*dbb99499SAndroid Build Coastguard Worker void State::PauseTiming() {
242*dbb99499SAndroid Build Coastguard Worker // Add in time accumulated so far
243*dbb99499SAndroid Build Coastguard Worker BM_CHECK(started_ && !finished_ && !skipped());
244*dbb99499SAndroid Build Coastguard Worker timer_->StopTimer();
245*dbb99499SAndroid Build Coastguard Worker if (perf_counters_measurement_) {
246*dbb99499SAndroid Build Coastguard Worker std::vector<std::pair<std::string, double>> measurements;
247*dbb99499SAndroid Build Coastguard Worker if (!perf_counters_measurement_->Stop(measurements)) {
248*dbb99499SAndroid Build Coastguard Worker BM_CHECK(false) << "Perf counters read the value failed.";
249*dbb99499SAndroid Build Coastguard Worker }
250*dbb99499SAndroid Build Coastguard Worker for (const auto& name_and_measurement : measurements) {
251*dbb99499SAndroid Build Coastguard Worker const std::string& name = name_and_measurement.first;
252*dbb99499SAndroid Build Coastguard Worker const double measurement = name_and_measurement.second;
253*dbb99499SAndroid Build Coastguard Worker // Counter was inserted with `kAvgIterations` flag by the constructor.
254*dbb99499SAndroid Build Coastguard Worker assert(counters.find(name) != counters.end());
255*dbb99499SAndroid Build Coastguard Worker counters[name].value += measurement;
256*dbb99499SAndroid Build Coastguard Worker }
257*dbb99499SAndroid Build Coastguard Worker }
258*dbb99499SAndroid Build Coastguard Worker }
259*dbb99499SAndroid Build Coastguard Worker
ResumeTiming()260*dbb99499SAndroid Build Coastguard Worker void State::ResumeTiming() {
261*dbb99499SAndroid Build Coastguard Worker BM_CHECK(started_ && !finished_ && !skipped());
262*dbb99499SAndroid Build Coastguard Worker timer_->StartTimer();
263*dbb99499SAndroid Build Coastguard Worker if (perf_counters_measurement_) {
264*dbb99499SAndroid Build Coastguard Worker perf_counters_measurement_->Start();
265*dbb99499SAndroid Build Coastguard Worker }
266*dbb99499SAndroid Build Coastguard Worker }
267*dbb99499SAndroid Build Coastguard Worker
SkipWithMessage(const std::string & msg)268*dbb99499SAndroid Build Coastguard Worker void State::SkipWithMessage(const std::string& msg) {
269*dbb99499SAndroid Build Coastguard Worker skipped_ = internal::SkippedWithMessage;
270*dbb99499SAndroid Build Coastguard Worker {
271*dbb99499SAndroid Build Coastguard Worker MutexLock l(manager_->GetBenchmarkMutex());
272*dbb99499SAndroid Build Coastguard Worker if (internal::NotSkipped == manager_->results.skipped_) {
273*dbb99499SAndroid Build Coastguard Worker manager_->results.skip_message_ = msg;
274*dbb99499SAndroid Build Coastguard Worker manager_->results.skipped_ = skipped_;
275*dbb99499SAndroid Build Coastguard Worker }
276*dbb99499SAndroid Build Coastguard Worker }
277*dbb99499SAndroid Build Coastguard Worker total_iterations_ = 0;
278*dbb99499SAndroid Build Coastguard Worker if (timer_->running()) timer_->StopTimer();
279*dbb99499SAndroid Build Coastguard Worker }
280*dbb99499SAndroid Build Coastguard Worker
SkipWithError(const std::string & msg)281*dbb99499SAndroid Build Coastguard Worker void State::SkipWithError(const std::string& msg) {
282*dbb99499SAndroid Build Coastguard Worker skipped_ = internal::SkippedWithError;
283*dbb99499SAndroid Build Coastguard Worker {
284*dbb99499SAndroid Build Coastguard Worker MutexLock l(manager_->GetBenchmarkMutex());
285*dbb99499SAndroid Build Coastguard Worker if (internal::NotSkipped == manager_->results.skipped_) {
286*dbb99499SAndroid Build Coastguard Worker manager_->results.skip_message_ = msg;
287*dbb99499SAndroid Build Coastguard Worker manager_->results.skipped_ = skipped_;
288*dbb99499SAndroid Build Coastguard Worker }
289*dbb99499SAndroid Build Coastguard Worker }
290*dbb99499SAndroid Build Coastguard Worker total_iterations_ = 0;
291*dbb99499SAndroid Build Coastguard Worker if (timer_->running()) timer_->StopTimer();
292*dbb99499SAndroid Build Coastguard Worker }
293*dbb99499SAndroid Build Coastguard Worker
SetIterationTime(double seconds)294*dbb99499SAndroid Build Coastguard Worker void State::SetIterationTime(double seconds) {
295*dbb99499SAndroid Build Coastguard Worker timer_->SetIterationTime(seconds);
296*dbb99499SAndroid Build Coastguard Worker }
297*dbb99499SAndroid Build Coastguard Worker
SetLabel(const std::string & label)298*dbb99499SAndroid Build Coastguard Worker void State::SetLabel(const std::string& label) {
299*dbb99499SAndroid Build Coastguard Worker MutexLock l(manager_->GetBenchmarkMutex());
300*dbb99499SAndroid Build Coastguard Worker manager_->results.report_label_ = label;
301*dbb99499SAndroid Build Coastguard Worker }
302*dbb99499SAndroid Build Coastguard Worker
StartKeepRunning()303*dbb99499SAndroid Build Coastguard Worker void State::StartKeepRunning() {
304*dbb99499SAndroid Build Coastguard Worker BM_CHECK(!started_ && !finished_);
305*dbb99499SAndroid Build Coastguard Worker started_ = true;
306*dbb99499SAndroid Build Coastguard Worker total_iterations_ = skipped() ? 0 : max_iterations;
307*dbb99499SAndroid Build Coastguard Worker if (BENCHMARK_BUILTIN_EXPECT(profiler_manager_ != nullptr, false))
308*dbb99499SAndroid Build Coastguard Worker profiler_manager_->AfterSetupStart();
309*dbb99499SAndroid Build Coastguard Worker manager_->StartStopBarrier();
310*dbb99499SAndroid Build Coastguard Worker if (!skipped()) ResumeTiming();
311*dbb99499SAndroid Build Coastguard Worker }
312*dbb99499SAndroid Build Coastguard Worker
FinishKeepRunning()313*dbb99499SAndroid Build Coastguard Worker void State::FinishKeepRunning() {
314*dbb99499SAndroid Build Coastguard Worker BM_CHECK(started_ && (!finished_ || skipped()));
315*dbb99499SAndroid Build Coastguard Worker if (!skipped()) {
316*dbb99499SAndroid Build Coastguard Worker PauseTiming();
317*dbb99499SAndroid Build Coastguard Worker }
318*dbb99499SAndroid Build Coastguard Worker // Total iterations has now wrapped around past 0. Fix this.
319*dbb99499SAndroid Build Coastguard Worker total_iterations_ = 0;
320*dbb99499SAndroid Build Coastguard Worker finished_ = true;
321*dbb99499SAndroid Build Coastguard Worker manager_->StartStopBarrier();
322*dbb99499SAndroid Build Coastguard Worker if (BENCHMARK_BUILTIN_EXPECT(profiler_manager_ != nullptr, false))
323*dbb99499SAndroid Build Coastguard Worker profiler_manager_->BeforeTeardownStop();
324*dbb99499SAndroid Build Coastguard Worker }
325*dbb99499SAndroid Build Coastguard Worker
326*dbb99499SAndroid Build Coastguard Worker namespace internal {
327*dbb99499SAndroid Build Coastguard Worker namespace {
328*dbb99499SAndroid Build Coastguard Worker
329*dbb99499SAndroid Build Coastguard Worker // Flushes streams after invoking reporter methods that write to them. This
330*dbb99499SAndroid Build Coastguard Worker // ensures users get timely updates even when streams are not line-buffered.
FlushStreams(BenchmarkReporter * reporter)331*dbb99499SAndroid Build Coastguard Worker void FlushStreams(BenchmarkReporter* reporter) {
332*dbb99499SAndroid Build Coastguard Worker if (!reporter) return;
333*dbb99499SAndroid Build Coastguard Worker std::flush(reporter->GetOutputStream());
334*dbb99499SAndroid Build Coastguard Worker std::flush(reporter->GetErrorStream());
335*dbb99499SAndroid Build Coastguard Worker }
336*dbb99499SAndroid Build Coastguard Worker
337*dbb99499SAndroid Build Coastguard Worker // Reports in both display and file reporters.
Report(BenchmarkReporter * display_reporter,BenchmarkReporter * file_reporter,const RunResults & run_results)338*dbb99499SAndroid Build Coastguard Worker void Report(BenchmarkReporter* display_reporter,
339*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter* file_reporter, const RunResults& run_results) {
340*dbb99499SAndroid Build Coastguard Worker auto report_one = [](BenchmarkReporter* reporter, bool aggregates_only,
341*dbb99499SAndroid Build Coastguard Worker const RunResults& results) {
342*dbb99499SAndroid Build Coastguard Worker assert(reporter);
343*dbb99499SAndroid Build Coastguard Worker // If there are no aggregates, do output non-aggregates.
344*dbb99499SAndroid Build Coastguard Worker aggregates_only &= !results.aggregates_only.empty();
345*dbb99499SAndroid Build Coastguard Worker if (!aggregates_only) reporter->ReportRuns(results.non_aggregates);
346*dbb99499SAndroid Build Coastguard Worker if (!results.aggregates_only.empty())
347*dbb99499SAndroid Build Coastguard Worker reporter->ReportRuns(results.aggregates_only);
348*dbb99499SAndroid Build Coastguard Worker };
349*dbb99499SAndroid Build Coastguard Worker
350*dbb99499SAndroid Build Coastguard Worker report_one(display_reporter, run_results.display_report_aggregates_only,
351*dbb99499SAndroid Build Coastguard Worker run_results);
352*dbb99499SAndroid Build Coastguard Worker if (file_reporter)
353*dbb99499SAndroid Build Coastguard Worker report_one(file_reporter, run_results.file_report_aggregates_only,
354*dbb99499SAndroid Build Coastguard Worker run_results);
355*dbb99499SAndroid Build Coastguard Worker
356*dbb99499SAndroid Build Coastguard Worker FlushStreams(display_reporter);
357*dbb99499SAndroid Build Coastguard Worker FlushStreams(file_reporter);
358*dbb99499SAndroid Build Coastguard Worker }
359*dbb99499SAndroid Build Coastguard Worker
RunBenchmarks(const std::vector<BenchmarkInstance> & benchmarks,BenchmarkReporter * display_reporter,BenchmarkReporter * file_reporter)360*dbb99499SAndroid Build Coastguard Worker void RunBenchmarks(const std::vector<BenchmarkInstance>& benchmarks,
361*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter* display_reporter,
362*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter* file_reporter) {
363*dbb99499SAndroid Build Coastguard Worker // Note the file_reporter can be null.
364*dbb99499SAndroid Build Coastguard Worker BM_CHECK(display_reporter != nullptr);
365*dbb99499SAndroid Build Coastguard Worker
366*dbb99499SAndroid Build Coastguard Worker // Determine the width of the name field using a minimum width of 10.
367*dbb99499SAndroid Build Coastguard Worker bool might_have_aggregates = FLAGS_benchmark_repetitions > 1;
368*dbb99499SAndroid Build Coastguard Worker size_t name_field_width = 10;
369*dbb99499SAndroid Build Coastguard Worker size_t stat_field_width = 0;
370*dbb99499SAndroid Build Coastguard Worker for (const BenchmarkInstance& benchmark : benchmarks) {
371*dbb99499SAndroid Build Coastguard Worker name_field_width =
372*dbb99499SAndroid Build Coastguard Worker std::max<size_t>(name_field_width, benchmark.name().str().size());
373*dbb99499SAndroid Build Coastguard Worker might_have_aggregates |= benchmark.repetitions() > 1;
374*dbb99499SAndroid Build Coastguard Worker
375*dbb99499SAndroid Build Coastguard Worker for (const auto& Stat : benchmark.statistics())
376*dbb99499SAndroid Build Coastguard Worker stat_field_width = std::max<size_t>(stat_field_width, Stat.name_.size());
377*dbb99499SAndroid Build Coastguard Worker }
378*dbb99499SAndroid Build Coastguard Worker if (might_have_aggregates) name_field_width += 1 + stat_field_width;
379*dbb99499SAndroid Build Coastguard Worker
380*dbb99499SAndroid Build Coastguard Worker // Print header here
381*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter::Context context;
382*dbb99499SAndroid Build Coastguard Worker context.name_field_width = name_field_width;
383*dbb99499SAndroid Build Coastguard Worker
384*dbb99499SAndroid Build Coastguard Worker // Keep track of running times of all instances of each benchmark family.
385*dbb99499SAndroid Build Coastguard Worker std::map<int /*family_index*/, BenchmarkReporter::PerFamilyRunReports>
386*dbb99499SAndroid Build Coastguard Worker per_family_reports;
387*dbb99499SAndroid Build Coastguard Worker
388*dbb99499SAndroid Build Coastguard Worker if (display_reporter->ReportContext(context) &&
389*dbb99499SAndroid Build Coastguard Worker (!file_reporter || file_reporter->ReportContext(context))) {
390*dbb99499SAndroid Build Coastguard Worker FlushStreams(display_reporter);
391*dbb99499SAndroid Build Coastguard Worker FlushStreams(file_reporter);
392*dbb99499SAndroid Build Coastguard Worker
393*dbb99499SAndroid Build Coastguard Worker size_t num_repetitions_total = 0;
394*dbb99499SAndroid Build Coastguard Worker
395*dbb99499SAndroid Build Coastguard Worker // This perfcounters object needs to be created before the runners vector
396*dbb99499SAndroid Build Coastguard Worker // below so it outlasts their lifetime.
397*dbb99499SAndroid Build Coastguard Worker PerfCountersMeasurement perfcounters(
398*dbb99499SAndroid Build Coastguard Worker StrSplit(FLAGS_benchmark_perf_counters, ','));
399*dbb99499SAndroid Build Coastguard Worker
400*dbb99499SAndroid Build Coastguard Worker // Vector of benchmarks to run
401*dbb99499SAndroid Build Coastguard Worker std::vector<internal::BenchmarkRunner> runners;
402*dbb99499SAndroid Build Coastguard Worker runners.reserve(benchmarks.size());
403*dbb99499SAndroid Build Coastguard Worker
404*dbb99499SAndroid Build Coastguard Worker // Count the number of benchmarks with threads to warn the user in case
405*dbb99499SAndroid Build Coastguard Worker // performance counters are used.
406*dbb99499SAndroid Build Coastguard Worker int benchmarks_with_threads = 0;
407*dbb99499SAndroid Build Coastguard Worker
408*dbb99499SAndroid Build Coastguard Worker // Loop through all benchmarks
409*dbb99499SAndroid Build Coastguard Worker for (const BenchmarkInstance& benchmark : benchmarks) {
410*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter::PerFamilyRunReports* reports_for_family = nullptr;
411*dbb99499SAndroid Build Coastguard Worker if (benchmark.complexity() != oNone)
412*dbb99499SAndroid Build Coastguard Worker reports_for_family = &per_family_reports[benchmark.family_index()];
413*dbb99499SAndroid Build Coastguard Worker benchmarks_with_threads += (benchmark.threads() > 1);
414*dbb99499SAndroid Build Coastguard Worker runners.emplace_back(benchmark, &perfcounters, reports_for_family);
415*dbb99499SAndroid Build Coastguard Worker int num_repeats_of_this_instance = runners.back().GetNumRepeats();
416*dbb99499SAndroid Build Coastguard Worker num_repetitions_total +=
417*dbb99499SAndroid Build Coastguard Worker static_cast<size_t>(num_repeats_of_this_instance);
418*dbb99499SAndroid Build Coastguard Worker if (reports_for_family)
419*dbb99499SAndroid Build Coastguard Worker reports_for_family->num_runs_total += num_repeats_of_this_instance;
420*dbb99499SAndroid Build Coastguard Worker }
421*dbb99499SAndroid Build Coastguard Worker assert(runners.size() == benchmarks.size() && "Unexpected runner count.");
422*dbb99499SAndroid Build Coastguard Worker
423*dbb99499SAndroid Build Coastguard Worker // The use of performance counters with threads would be unintuitive for
424*dbb99499SAndroid Build Coastguard Worker // the average user so we need to warn them about this case
425*dbb99499SAndroid Build Coastguard Worker if ((benchmarks_with_threads > 0) && (perfcounters.num_counters() > 0)) {
426*dbb99499SAndroid Build Coastguard Worker GetErrorLogInstance()
427*dbb99499SAndroid Build Coastguard Worker << "***WARNING*** There are " << benchmarks_with_threads
428*dbb99499SAndroid Build Coastguard Worker << " benchmarks with threads and " << perfcounters.num_counters()
429*dbb99499SAndroid Build Coastguard Worker << " performance counters were requested. Beware counters will "
430*dbb99499SAndroid Build Coastguard Worker "reflect the combined usage across all "
431*dbb99499SAndroid Build Coastguard Worker "threads.\n";
432*dbb99499SAndroid Build Coastguard Worker }
433*dbb99499SAndroid Build Coastguard Worker
434*dbb99499SAndroid Build Coastguard Worker std::vector<size_t> repetition_indices;
435*dbb99499SAndroid Build Coastguard Worker repetition_indices.reserve(num_repetitions_total);
436*dbb99499SAndroid Build Coastguard Worker for (size_t runner_index = 0, num_runners = runners.size();
437*dbb99499SAndroid Build Coastguard Worker runner_index != num_runners; ++runner_index) {
438*dbb99499SAndroid Build Coastguard Worker const internal::BenchmarkRunner& runner = runners[runner_index];
439*dbb99499SAndroid Build Coastguard Worker std::fill_n(std::back_inserter(repetition_indices),
440*dbb99499SAndroid Build Coastguard Worker runner.GetNumRepeats(), runner_index);
441*dbb99499SAndroid Build Coastguard Worker }
442*dbb99499SAndroid Build Coastguard Worker assert(repetition_indices.size() == num_repetitions_total &&
443*dbb99499SAndroid Build Coastguard Worker "Unexpected number of repetition indexes.");
444*dbb99499SAndroid Build Coastguard Worker
445*dbb99499SAndroid Build Coastguard Worker if (FLAGS_benchmark_enable_random_interleaving) {
446*dbb99499SAndroid Build Coastguard Worker std::random_device rd;
447*dbb99499SAndroid Build Coastguard Worker std::mt19937 g(rd());
448*dbb99499SAndroid Build Coastguard Worker std::shuffle(repetition_indices.begin(), repetition_indices.end(), g);
449*dbb99499SAndroid Build Coastguard Worker }
450*dbb99499SAndroid Build Coastguard Worker
451*dbb99499SAndroid Build Coastguard Worker for (size_t repetition_index : repetition_indices) {
452*dbb99499SAndroid Build Coastguard Worker internal::BenchmarkRunner& runner = runners[repetition_index];
453*dbb99499SAndroid Build Coastguard Worker runner.DoOneRepetition();
454*dbb99499SAndroid Build Coastguard Worker if (runner.HasRepeatsRemaining()) continue;
455*dbb99499SAndroid Build Coastguard Worker // FIXME: report each repetition separately, not all of them in bulk.
456*dbb99499SAndroid Build Coastguard Worker
457*dbb99499SAndroid Build Coastguard Worker display_reporter->ReportRunsConfig(
458*dbb99499SAndroid Build Coastguard Worker runner.GetMinTime(), runner.HasExplicitIters(), runner.GetIters());
459*dbb99499SAndroid Build Coastguard Worker if (file_reporter)
460*dbb99499SAndroid Build Coastguard Worker file_reporter->ReportRunsConfig(
461*dbb99499SAndroid Build Coastguard Worker runner.GetMinTime(), runner.HasExplicitIters(), runner.GetIters());
462*dbb99499SAndroid Build Coastguard Worker
463*dbb99499SAndroid Build Coastguard Worker RunResults run_results = runner.GetResults();
464*dbb99499SAndroid Build Coastguard Worker
465*dbb99499SAndroid Build Coastguard Worker // Maybe calculate complexity report
466*dbb99499SAndroid Build Coastguard Worker if (const auto* reports_for_family = runner.GetReportsForFamily()) {
467*dbb99499SAndroid Build Coastguard Worker if (reports_for_family->num_runs_done ==
468*dbb99499SAndroid Build Coastguard Worker reports_for_family->num_runs_total) {
469*dbb99499SAndroid Build Coastguard Worker auto additional_run_stats = ComputeBigO(reports_for_family->Runs);
470*dbb99499SAndroid Build Coastguard Worker run_results.aggregates_only.insert(run_results.aggregates_only.end(),
471*dbb99499SAndroid Build Coastguard Worker additional_run_stats.begin(),
472*dbb99499SAndroid Build Coastguard Worker additional_run_stats.end());
473*dbb99499SAndroid Build Coastguard Worker per_family_reports.erase(
474*dbb99499SAndroid Build Coastguard Worker static_cast<int>(reports_for_family->Runs.front().family_index));
475*dbb99499SAndroid Build Coastguard Worker }
476*dbb99499SAndroid Build Coastguard Worker }
477*dbb99499SAndroid Build Coastguard Worker
478*dbb99499SAndroid Build Coastguard Worker Report(display_reporter, file_reporter, run_results);
479*dbb99499SAndroid Build Coastguard Worker }
480*dbb99499SAndroid Build Coastguard Worker }
481*dbb99499SAndroid Build Coastguard Worker display_reporter->Finalize();
482*dbb99499SAndroid Build Coastguard Worker if (file_reporter) file_reporter->Finalize();
483*dbb99499SAndroid Build Coastguard Worker FlushStreams(display_reporter);
484*dbb99499SAndroid Build Coastguard Worker FlushStreams(file_reporter);
485*dbb99499SAndroid Build Coastguard Worker }
486*dbb99499SAndroid Build Coastguard Worker
487*dbb99499SAndroid Build Coastguard Worker // Disable deprecated warnings temporarily because we need to reference
488*dbb99499SAndroid Build Coastguard Worker // CSVReporter but don't want to trigger -Werror=-Wdeprecated-declarations
489*dbb99499SAndroid Build Coastguard Worker BENCHMARK_DISABLE_DEPRECATED_WARNING
490*dbb99499SAndroid Build Coastguard Worker
CreateReporter(std::string const & name,ConsoleReporter::OutputOptions output_opts)491*dbb99499SAndroid Build Coastguard Worker std::unique_ptr<BenchmarkReporter> CreateReporter(
492*dbb99499SAndroid Build Coastguard Worker std::string const& name, ConsoleReporter::OutputOptions output_opts) {
493*dbb99499SAndroid Build Coastguard Worker typedef std::unique_ptr<BenchmarkReporter> PtrType;
494*dbb99499SAndroid Build Coastguard Worker if (name == "console") {
495*dbb99499SAndroid Build Coastguard Worker return PtrType(new ConsoleReporter(output_opts));
496*dbb99499SAndroid Build Coastguard Worker }
497*dbb99499SAndroid Build Coastguard Worker if (name == "json") {
498*dbb99499SAndroid Build Coastguard Worker return PtrType(new JSONReporter());
499*dbb99499SAndroid Build Coastguard Worker }
500*dbb99499SAndroid Build Coastguard Worker if (name == "csv") {
501*dbb99499SAndroid Build Coastguard Worker return PtrType(new CSVReporter());
502*dbb99499SAndroid Build Coastguard Worker }
503*dbb99499SAndroid Build Coastguard Worker std::cerr << "Unexpected format: '" << name << "'\n";
504*dbb99499SAndroid Build Coastguard Worker std::exit(1);
505*dbb99499SAndroid Build Coastguard Worker }
506*dbb99499SAndroid Build Coastguard Worker
507*dbb99499SAndroid Build Coastguard Worker BENCHMARK_RESTORE_DEPRECATED_WARNING
508*dbb99499SAndroid Build Coastguard Worker
509*dbb99499SAndroid Build Coastguard Worker } // end namespace
510*dbb99499SAndroid Build Coastguard Worker
IsZero(double n)511*dbb99499SAndroid Build Coastguard Worker bool IsZero(double n) {
512*dbb99499SAndroid Build Coastguard Worker return std::abs(n) < std::numeric_limits<double>::epsilon();
513*dbb99499SAndroid Build Coastguard Worker }
514*dbb99499SAndroid Build Coastguard Worker
GetOutputOptions(bool force_no_color)515*dbb99499SAndroid Build Coastguard Worker ConsoleReporter::OutputOptions GetOutputOptions(bool force_no_color) {
516*dbb99499SAndroid Build Coastguard Worker int output_opts = ConsoleReporter::OO_Defaults;
517*dbb99499SAndroid Build Coastguard Worker auto is_benchmark_color = [force_no_color]() -> bool {
518*dbb99499SAndroid Build Coastguard Worker if (force_no_color) {
519*dbb99499SAndroid Build Coastguard Worker return false;
520*dbb99499SAndroid Build Coastguard Worker }
521*dbb99499SAndroid Build Coastguard Worker if (FLAGS_benchmark_color == "auto") {
522*dbb99499SAndroid Build Coastguard Worker return IsColorTerminal();
523*dbb99499SAndroid Build Coastguard Worker }
524*dbb99499SAndroid Build Coastguard Worker return IsTruthyFlagValue(FLAGS_benchmark_color);
525*dbb99499SAndroid Build Coastguard Worker };
526*dbb99499SAndroid Build Coastguard Worker if (is_benchmark_color()) {
527*dbb99499SAndroid Build Coastguard Worker output_opts |= ConsoleReporter::OO_Color;
528*dbb99499SAndroid Build Coastguard Worker } else {
529*dbb99499SAndroid Build Coastguard Worker output_opts &= ~ConsoleReporter::OO_Color;
530*dbb99499SAndroid Build Coastguard Worker }
531*dbb99499SAndroid Build Coastguard Worker if (FLAGS_benchmark_counters_tabular) {
532*dbb99499SAndroid Build Coastguard Worker output_opts |= ConsoleReporter::OO_Tabular;
533*dbb99499SAndroid Build Coastguard Worker } else {
534*dbb99499SAndroid Build Coastguard Worker output_opts &= ~ConsoleReporter::OO_Tabular;
535*dbb99499SAndroid Build Coastguard Worker }
536*dbb99499SAndroid Build Coastguard Worker return static_cast<ConsoleReporter::OutputOptions>(output_opts);
537*dbb99499SAndroid Build Coastguard Worker }
538*dbb99499SAndroid Build Coastguard Worker
539*dbb99499SAndroid Build Coastguard Worker } // end namespace internal
540*dbb99499SAndroid Build Coastguard Worker
CreateDefaultDisplayReporter()541*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter* CreateDefaultDisplayReporter() {
542*dbb99499SAndroid Build Coastguard Worker static auto default_display_reporter =
543*dbb99499SAndroid Build Coastguard Worker internal::CreateReporter(FLAGS_benchmark_format,
544*dbb99499SAndroid Build Coastguard Worker internal::GetOutputOptions())
545*dbb99499SAndroid Build Coastguard Worker .release();
546*dbb99499SAndroid Build Coastguard Worker return default_display_reporter;
547*dbb99499SAndroid Build Coastguard Worker }
548*dbb99499SAndroid Build Coastguard Worker
RunSpecifiedBenchmarks()549*dbb99499SAndroid Build Coastguard Worker size_t RunSpecifiedBenchmarks() {
550*dbb99499SAndroid Build Coastguard Worker return RunSpecifiedBenchmarks(nullptr, nullptr, FLAGS_benchmark_filter);
551*dbb99499SAndroid Build Coastguard Worker }
552*dbb99499SAndroid Build Coastguard Worker
RunSpecifiedBenchmarks(std::string spec)553*dbb99499SAndroid Build Coastguard Worker size_t RunSpecifiedBenchmarks(std::string spec) {
554*dbb99499SAndroid Build Coastguard Worker return RunSpecifiedBenchmarks(nullptr, nullptr, std::move(spec));
555*dbb99499SAndroid Build Coastguard Worker }
556*dbb99499SAndroid Build Coastguard Worker
RunSpecifiedBenchmarks(BenchmarkReporter * display_reporter)557*dbb99499SAndroid Build Coastguard Worker size_t RunSpecifiedBenchmarks(BenchmarkReporter* display_reporter) {
558*dbb99499SAndroid Build Coastguard Worker return RunSpecifiedBenchmarks(display_reporter, nullptr,
559*dbb99499SAndroid Build Coastguard Worker FLAGS_benchmark_filter);
560*dbb99499SAndroid Build Coastguard Worker }
561*dbb99499SAndroid Build Coastguard Worker
RunSpecifiedBenchmarks(BenchmarkReporter * display_reporter,std::string spec)562*dbb99499SAndroid Build Coastguard Worker size_t RunSpecifiedBenchmarks(BenchmarkReporter* display_reporter,
563*dbb99499SAndroid Build Coastguard Worker std::string spec) {
564*dbb99499SAndroid Build Coastguard Worker return RunSpecifiedBenchmarks(display_reporter, nullptr, std::move(spec));
565*dbb99499SAndroid Build Coastguard Worker }
566*dbb99499SAndroid Build Coastguard Worker
RunSpecifiedBenchmarks(BenchmarkReporter * display_reporter,BenchmarkReporter * file_reporter)567*dbb99499SAndroid Build Coastguard Worker size_t RunSpecifiedBenchmarks(BenchmarkReporter* display_reporter,
568*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter* file_reporter) {
569*dbb99499SAndroid Build Coastguard Worker return RunSpecifiedBenchmarks(display_reporter, file_reporter,
570*dbb99499SAndroid Build Coastguard Worker FLAGS_benchmark_filter);
571*dbb99499SAndroid Build Coastguard Worker }
572*dbb99499SAndroid Build Coastguard Worker
RunSpecifiedBenchmarks(BenchmarkReporter * display_reporter,BenchmarkReporter * file_reporter,std::string spec)573*dbb99499SAndroid Build Coastguard Worker size_t RunSpecifiedBenchmarks(BenchmarkReporter* display_reporter,
574*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter* file_reporter,
575*dbb99499SAndroid Build Coastguard Worker std::string spec) {
576*dbb99499SAndroid Build Coastguard Worker if (spec.empty() || spec == "all")
577*dbb99499SAndroid Build Coastguard Worker spec = "."; // Regexp that matches all benchmarks
578*dbb99499SAndroid Build Coastguard Worker
579*dbb99499SAndroid Build Coastguard Worker // Setup the reporters
580*dbb99499SAndroid Build Coastguard Worker std::ofstream output_file;
581*dbb99499SAndroid Build Coastguard Worker std::unique_ptr<BenchmarkReporter> default_display_reporter;
582*dbb99499SAndroid Build Coastguard Worker std::unique_ptr<BenchmarkReporter> default_file_reporter;
583*dbb99499SAndroid Build Coastguard Worker if (!display_reporter) {
584*dbb99499SAndroid Build Coastguard Worker default_display_reporter.reset(CreateDefaultDisplayReporter());
585*dbb99499SAndroid Build Coastguard Worker display_reporter = default_display_reporter.get();
586*dbb99499SAndroid Build Coastguard Worker }
587*dbb99499SAndroid Build Coastguard Worker auto& Out = display_reporter->GetOutputStream();
588*dbb99499SAndroid Build Coastguard Worker auto& Err = display_reporter->GetErrorStream();
589*dbb99499SAndroid Build Coastguard Worker
590*dbb99499SAndroid Build Coastguard Worker std::string const& fname = FLAGS_benchmark_out;
591*dbb99499SAndroid Build Coastguard Worker if (fname.empty() && file_reporter) {
592*dbb99499SAndroid Build Coastguard Worker Err << "A custom file reporter was provided but "
593*dbb99499SAndroid Build Coastguard Worker "--benchmark_out=<file> was not specified."
594*dbb99499SAndroid Build Coastguard Worker << std::endl;
595*dbb99499SAndroid Build Coastguard Worker Out.flush();
596*dbb99499SAndroid Build Coastguard Worker Err.flush();
597*dbb99499SAndroid Build Coastguard Worker std::exit(1);
598*dbb99499SAndroid Build Coastguard Worker }
599*dbb99499SAndroid Build Coastguard Worker if (!fname.empty()) {
600*dbb99499SAndroid Build Coastguard Worker output_file.open(fname);
601*dbb99499SAndroid Build Coastguard Worker if (!output_file.is_open()) {
602*dbb99499SAndroid Build Coastguard Worker Err << "invalid file name: '" << fname << "'" << std::endl;
603*dbb99499SAndroid Build Coastguard Worker Out.flush();
604*dbb99499SAndroid Build Coastguard Worker Err.flush();
605*dbb99499SAndroid Build Coastguard Worker std::exit(1);
606*dbb99499SAndroid Build Coastguard Worker }
607*dbb99499SAndroid Build Coastguard Worker if (!file_reporter) {
608*dbb99499SAndroid Build Coastguard Worker default_file_reporter = internal::CreateReporter(
609*dbb99499SAndroid Build Coastguard Worker FLAGS_benchmark_out_format, FLAGS_benchmark_counters_tabular
610*dbb99499SAndroid Build Coastguard Worker ? ConsoleReporter::OO_Tabular
611*dbb99499SAndroid Build Coastguard Worker : ConsoleReporter::OO_None);
612*dbb99499SAndroid Build Coastguard Worker file_reporter = default_file_reporter.get();
613*dbb99499SAndroid Build Coastguard Worker }
614*dbb99499SAndroid Build Coastguard Worker file_reporter->SetOutputStream(&output_file);
615*dbb99499SAndroid Build Coastguard Worker file_reporter->SetErrorStream(&output_file);
616*dbb99499SAndroid Build Coastguard Worker }
617*dbb99499SAndroid Build Coastguard Worker
618*dbb99499SAndroid Build Coastguard Worker std::vector<internal::BenchmarkInstance> benchmarks;
619*dbb99499SAndroid Build Coastguard Worker if (!FindBenchmarksInternal(spec, &benchmarks, &Err)) {
620*dbb99499SAndroid Build Coastguard Worker Out.flush();
621*dbb99499SAndroid Build Coastguard Worker Err.flush();
622*dbb99499SAndroid Build Coastguard Worker return 0;
623*dbb99499SAndroid Build Coastguard Worker }
624*dbb99499SAndroid Build Coastguard Worker
625*dbb99499SAndroid Build Coastguard Worker if (benchmarks.empty()) {
626*dbb99499SAndroid Build Coastguard Worker Err << "Failed to match any benchmarks against regex: " << spec << "\n";
627*dbb99499SAndroid Build Coastguard Worker Out.flush();
628*dbb99499SAndroid Build Coastguard Worker Err.flush();
629*dbb99499SAndroid Build Coastguard Worker return 0;
630*dbb99499SAndroid Build Coastguard Worker }
631*dbb99499SAndroid Build Coastguard Worker
632*dbb99499SAndroid Build Coastguard Worker if (FLAGS_benchmark_list_tests) {
633*dbb99499SAndroid Build Coastguard Worker for (auto const& benchmark : benchmarks)
634*dbb99499SAndroid Build Coastguard Worker Out << benchmark.name().str() << "\n";
635*dbb99499SAndroid Build Coastguard Worker } else {
636*dbb99499SAndroid Build Coastguard Worker internal::RunBenchmarks(benchmarks, display_reporter, file_reporter);
637*dbb99499SAndroid Build Coastguard Worker }
638*dbb99499SAndroid Build Coastguard Worker
639*dbb99499SAndroid Build Coastguard Worker Out.flush();
640*dbb99499SAndroid Build Coastguard Worker Err.flush();
641*dbb99499SAndroid Build Coastguard Worker return benchmarks.size();
642*dbb99499SAndroid Build Coastguard Worker }
643*dbb99499SAndroid Build Coastguard Worker
644*dbb99499SAndroid Build Coastguard Worker namespace {
645*dbb99499SAndroid Build Coastguard Worker // stores the time unit benchmarks use by default
646*dbb99499SAndroid Build Coastguard Worker TimeUnit default_time_unit = kNanosecond;
647*dbb99499SAndroid Build Coastguard Worker } // namespace
648*dbb99499SAndroid Build Coastguard Worker
GetDefaultTimeUnit()649*dbb99499SAndroid Build Coastguard Worker TimeUnit GetDefaultTimeUnit() { return default_time_unit; }
650*dbb99499SAndroid Build Coastguard Worker
SetDefaultTimeUnit(TimeUnit unit)651*dbb99499SAndroid Build Coastguard Worker void SetDefaultTimeUnit(TimeUnit unit) { default_time_unit = unit; }
652*dbb99499SAndroid Build Coastguard Worker
GetBenchmarkFilter()653*dbb99499SAndroid Build Coastguard Worker std::string GetBenchmarkFilter() { return FLAGS_benchmark_filter; }
654*dbb99499SAndroid Build Coastguard Worker
SetBenchmarkFilter(std::string value)655*dbb99499SAndroid Build Coastguard Worker void SetBenchmarkFilter(std::string value) {
656*dbb99499SAndroid Build Coastguard Worker FLAGS_benchmark_filter = std::move(value);
657*dbb99499SAndroid Build Coastguard Worker }
658*dbb99499SAndroid Build Coastguard Worker
GetBenchmarkVerbosity()659*dbb99499SAndroid Build Coastguard Worker int32_t GetBenchmarkVerbosity() { return FLAGS_v; }
660*dbb99499SAndroid Build Coastguard Worker
RegisterMemoryManager(MemoryManager * manager)661*dbb99499SAndroid Build Coastguard Worker void RegisterMemoryManager(MemoryManager* manager) {
662*dbb99499SAndroid Build Coastguard Worker internal::memory_manager = manager;
663*dbb99499SAndroid Build Coastguard Worker }
664*dbb99499SAndroid Build Coastguard Worker
RegisterProfilerManager(ProfilerManager * manager)665*dbb99499SAndroid Build Coastguard Worker void RegisterProfilerManager(ProfilerManager* manager) {
666*dbb99499SAndroid Build Coastguard Worker internal::profiler_manager = manager;
667*dbb99499SAndroid Build Coastguard Worker }
668*dbb99499SAndroid Build Coastguard Worker
AddCustomContext(const std::string & key,const std::string & value)669*dbb99499SAndroid Build Coastguard Worker void AddCustomContext(const std::string& key, const std::string& value) {
670*dbb99499SAndroid Build Coastguard Worker if (internal::global_context == nullptr) {
671*dbb99499SAndroid Build Coastguard Worker internal::global_context = new std::map<std::string, std::string>();
672*dbb99499SAndroid Build Coastguard Worker }
673*dbb99499SAndroid Build Coastguard Worker if (!internal::global_context->emplace(key, value).second) {
674*dbb99499SAndroid Build Coastguard Worker std::cerr << "Failed to add custom context \"" << key << "\" as it already "
675*dbb99499SAndroid Build Coastguard Worker << "exists with value \"" << value << "\"\n";
676*dbb99499SAndroid Build Coastguard Worker }
677*dbb99499SAndroid Build Coastguard Worker }
678*dbb99499SAndroid Build Coastguard Worker
679*dbb99499SAndroid Build Coastguard Worker namespace internal {
680*dbb99499SAndroid Build Coastguard Worker
681*dbb99499SAndroid Build Coastguard Worker void (*HelperPrintf)();
682*dbb99499SAndroid Build Coastguard Worker
PrintUsageAndExit()683*dbb99499SAndroid Build Coastguard Worker void PrintUsageAndExit() {
684*dbb99499SAndroid Build Coastguard Worker HelperPrintf();
685*dbb99499SAndroid Build Coastguard Worker exit(0);
686*dbb99499SAndroid Build Coastguard Worker }
687*dbb99499SAndroid Build Coastguard Worker
SetDefaultTimeUnitFromFlag(const std::string & time_unit_flag)688*dbb99499SAndroid Build Coastguard Worker void SetDefaultTimeUnitFromFlag(const std::string& time_unit_flag) {
689*dbb99499SAndroid Build Coastguard Worker if (time_unit_flag == "s") {
690*dbb99499SAndroid Build Coastguard Worker return SetDefaultTimeUnit(kSecond);
691*dbb99499SAndroid Build Coastguard Worker }
692*dbb99499SAndroid Build Coastguard Worker if (time_unit_flag == "ms") {
693*dbb99499SAndroid Build Coastguard Worker return SetDefaultTimeUnit(kMillisecond);
694*dbb99499SAndroid Build Coastguard Worker }
695*dbb99499SAndroid Build Coastguard Worker if (time_unit_flag == "us") {
696*dbb99499SAndroid Build Coastguard Worker return SetDefaultTimeUnit(kMicrosecond);
697*dbb99499SAndroid Build Coastguard Worker }
698*dbb99499SAndroid Build Coastguard Worker if (time_unit_flag == "ns") {
699*dbb99499SAndroid Build Coastguard Worker return SetDefaultTimeUnit(kNanosecond);
700*dbb99499SAndroid Build Coastguard Worker }
701*dbb99499SAndroid Build Coastguard Worker if (!time_unit_flag.empty()) {
702*dbb99499SAndroid Build Coastguard Worker PrintUsageAndExit();
703*dbb99499SAndroid Build Coastguard Worker }
704*dbb99499SAndroid Build Coastguard Worker }
705*dbb99499SAndroid Build Coastguard Worker
ParseCommandLineFlags(int * argc,char ** argv)706*dbb99499SAndroid Build Coastguard Worker void ParseCommandLineFlags(int* argc, char** argv) {
707*dbb99499SAndroid Build Coastguard Worker using namespace benchmark;
708*dbb99499SAndroid Build Coastguard Worker BenchmarkReporter::Context::executable_name =
709*dbb99499SAndroid Build Coastguard Worker (argc && *argc > 0) ? argv[0] : "unknown";
710*dbb99499SAndroid Build Coastguard Worker for (int i = 1; argc && i < *argc; ++i) {
711*dbb99499SAndroid Build Coastguard Worker if (ParseBoolFlag(argv[i], "benchmark_list_tests",
712*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_list_tests) ||
713*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_filter", &FLAGS_benchmark_filter) ||
714*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_min_time",
715*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_min_time) ||
716*dbb99499SAndroid Build Coastguard Worker ParseDoubleFlag(argv[i], "benchmark_min_warmup_time",
717*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_min_warmup_time) ||
718*dbb99499SAndroid Build Coastguard Worker ParseInt32Flag(argv[i], "benchmark_repetitions",
719*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_repetitions) ||
720*dbb99499SAndroid Build Coastguard Worker ParseBoolFlag(argv[i], "benchmark_enable_random_interleaving",
721*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_enable_random_interleaving) ||
722*dbb99499SAndroid Build Coastguard Worker ParseBoolFlag(argv[i], "benchmark_report_aggregates_only",
723*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_report_aggregates_only) ||
724*dbb99499SAndroid Build Coastguard Worker ParseBoolFlag(argv[i], "benchmark_display_aggregates_only",
725*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_display_aggregates_only) ||
726*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_format", &FLAGS_benchmark_format) ||
727*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_out", &FLAGS_benchmark_out) ||
728*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_out_format",
729*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_out_format) ||
730*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_color", &FLAGS_benchmark_color) ||
731*dbb99499SAndroid Build Coastguard Worker ParseBoolFlag(argv[i], "benchmark_counters_tabular",
732*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_counters_tabular) ||
733*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_perf_counters",
734*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_perf_counters) ||
735*dbb99499SAndroid Build Coastguard Worker ParseKeyValueFlag(argv[i], "benchmark_context",
736*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_context) ||
737*dbb99499SAndroid Build Coastguard Worker ParseStringFlag(argv[i], "benchmark_time_unit",
738*dbb99499SAndroid Build Coastguard Worker &FLAGS_benchmark_time_unit) ||
739*dbb99499SAndroid Build Coastguard Worker ParseInt32Flag(argv[i], "v", &FLAGS_v)) {
740*dbb99499SAndroid Build Coastguard Worker for (int j = i; j != *argc - 1; ++j) argv[j] = argv[j + 1];
741*dbb99499SAndroid Build Coastguard Worker
742*dbb99499SAndroid Build Coastguard Worker --(*argc);
743*dbb99499SAndroid Build Coastguard Worker --i;
744*dbb99499SAndroid Build Coastguard Worker } else if (IsFlag(argv[i], "help")) {
745*dbb99499SAndroid Build Coastguard Worker PrintUsageAndExit();
746*dbb99499SAndroid Build Coastguard Worker }
747*dbb99499SAndroid Build Coastguard Worker }
748*dbb99499SAndroid Build Coastguard Worker for (auto const* flag :
749*dbb99499SAndroid Build Coastguard Worker {&FLAGS_benchmark_format, &FLAGS_benchmark_out_format}) {
750*dbb99499SAndroid Build Coastguard Worker if (*flag != "console" && *flag != "json" && *flag != "csv") {
751*dbb99499SAndroid Build Coastguard Worker PrintUsageAndExit();
752*dbb99499SAndroid Build Coastguard Worker }
753*dbb99499SAndroid Build Coastguard Worker }
754*dbb99499SAndroid Build Coastguard Worker SetDefaultTimeUnitFromFlag(FLAGS_benchmark_time_unit);
755*dbb99499SAndroid Build Coastguard Worker if (FLAGS_benchmark_color.empty()) {
756*dbb99499SAndroid Build Coastguard Worker PrintUsageAndExit();
757*dbb99499SAndroid Build Coastguard Worker }
758*dbb99499SAndroid Build Coastguard Worker for (const auto& kv : FLAGS_benchmark_context) {
759*dbb99499SAndroid Build Coastguard Worker AddCustomContext(kv.first, kv.second);
760*dbb99499SAndroid Build Coastguard Worker }
761*dbb99499SAndroid Build Coastguard Worker }
762*dbb99499SAndroid Build Coastguard Worker
InitializeStreams()763*dbb99499SAndroid Build Coastguard Worker int InitializeStreams() {
764*dbb99499SAndroid Build Coastguard Worker static std::ios_base::Init init;
765*dbb99499SAndroid Build Coastguard Worker return 0;
766*dbb99499SAndroid Build Coastguard Worker }
767*dbb99499SAndroid Build Coastguard Worker
768*dbb99499SAndroid Build Coastguard Worker } // end namespace internal
769*dbb99499SAndroid Build Coastguard Worker
GetBenchmarkVersion()770*dbb99499SAndroid Build Coastguard Worker std::string GetBenchmarkVersion() {
771*dbb99499SAndroid Build Coastguard Worker #ifdef BENCHMARK_VERSION
772*dbb99499SAndroid Build Coastguard Worker return {BENCHMARK_VERSION};
773*dbb99499SAndroid Build Coastguard Worker #else
774*dbb99499SAndroid Build Coastguard Worker return {""};
775*dbb99499SAndroid Build Coastguard Worker #endif
776*dbb99499SAndroid Build Coastguard Worker }
777*dbb99499SAndroid Build Coastguard Worker
PrintDefaultHelp()778*dbb99499SAndroid Build Coastguard Worker void PrintDefaultHelp() {
779*dbb99499SAndroid Build Coastguard Worker fprintf(stdout,
780*dbb99499SAndroid Build Coastguard Worker "benchmark"
781*dbb99499SAndroid Build Coastguard Worker " [--benchmark_list_tests={true|false}]\n"
782*dbb99499SAndroid Build Coastguard Worker " [--benchmark_filter=<regex>]\n"
783*dbb99499SAndroid Build Coastguard Worker " [--benchmark_min_time=`<integer>x` OR `<float>s` ]\n"
784*dbb99499SAndroid Build Coastguard Worker " [--benchmark_min_warmup_time=<min_warmup_time>]\n"
785*dbb99499SAndroid Build Coastguard Worker " [--benchmark_repetitions=<num_repetitions>]\n"
786*dbb99499SAndroid Build Coastguard Worker " [--benchmark_enable_random_interleaving={true|false}]\n"
787*dbb99499SAndroid Build Coastguard Worker " [--benchmark_report_aggregates_only={true|false}]\n"
788*dbb99499SAndroid Build Coastguard Worker " [--benchmark_display_aggregates_only={true|false}]\n"
789*dbb99499SAndroid Build Coastguard Worker " [--benchmark_format=<console|json|csv>]\n"
790*dbb99499SAndroid Build Coastguard Worker " [--benchmark_out=<filename>]\n"
791*dbb99499SAndroid Build Coastguard Worker " [--benchmark_out_format=<json|console|csv>]\n"
792*dbb99499SAndroid Build Coastguard Worker " [--benchmark_color={auto|true|false}]\n"
793*dbb99499SAndroid Build Coastguard Worker " [--benchmark_counters_tabular={true|false}]\n"
794*dbb99499SAndroid Build Coastguard Worker #if defined HAVE_LIBPFM
795*dbb99499SAndroid Build Coastguard Worker " [--benchmark_perf_counters=<counter>,...]\n"
796*dbb99499SAndroid Build Coastguard Worker #endif
797*dbb99499SAndroid Build Coastguard Worker " [--benchmark_context=<key>=<value>,...]\n"
798*dbb99499SAndroid Build Coastguard Worker " [--benchmark_time_unit={ns|us|ms|s}]\n"
799*dbb99499SAndroid Build Coastguard Worker " [--v=<verbosity>]\n");
800*dbb99499SAndroid Build Coastguard Worker }
801*dbb99499SAndroid Build Coastguard Worker
Initialize(int * argc,char ** argv,void (* HelperPrintf)())802*dbb99499SAndroid Build Coastguard Worker void Initialize(int* argc, char** argv, void (*HelperPrintf)()) {
803*dbb99499SAndroid Build Coastguard Worker internal::HelperPrintf = HelperPrintf;
804*dbb99499SAndroid Build Coastguard Worker internal::ParseCommandLineFlags(argc, argv);
805*dbb99499SAndroid Build Coastguard Worker internal::LogLevel() = FLAGS_v;
806*dbb99499SAndroid Build Coastguard Worker }
807*dbb99499SAndroid Build Coastguard Worker
Shutdown()808*dbb99499SAndroid Build Coastguard Worker void Shutdown() { delete internal::global_context; }
809*dbb99499SAndroid Build Coastguard Worker
ReportUnrecognizedArguments(int argc,char ** argv)810*dbb99499SAndroid Build Coastguard Worker bool ReportUnrecognizedArguments(int argc, char** argv) {
811*dbb99499SAndroid Build Coastguard Worker for (int i = 1; i < argc; ++i) {
812*dbb99499SAndroid Build Coastguard Worker fprintf(stderr, "%s: error: unrecognized command-line flag: %s\n", argv[0],
813*dbb99499SAndroid Build Coastguard Worker argv[i]);
814*dbb99499SAndroid Build Coastguard Worker }
815*dbb99499SAndroid Build Coastguard Worker return argc > 1;
816*dbb99499SAndroid Build Coastguard Worker }
817*dbb99499SAndroid Build Coastguard Worker
818*dbb99499SAndroid Build Coastguard Worker } // end namespace benchmark
819