From f965eab5083010ea7c3a8298176a837e5bff3016 Mon Sep 17 00:00:00 2001 From: Dominic Hamon Date: Tue, 24 Jul 2018 15:57:15 +0100 Subject: [PATCH] Memory management and reporting hooks (#625) * Introduce memory manager interface * Add memory stats to JSON reporter and a test * Add comments and switch json output test to int --- include/benchmark/benchmark.h | 43 +++++++++++++++++++++++++++++------ src/benchmark.cc | 35 ++++++++++++++++++++++++++-- src/json_reporter.cc | 6 +++++ test/memory_manager_test.cc | 40 ++++++++++++++++++++++++++++++++ 4 files changed, 115 insertions(+), 9 deletions(-) create mode 100644 test/memory_manager_test.cc diff --git a/include/benchmark/benchmark.h b/include/benchmark/benchmark.h index 193fffc4be..aef995ecdd 100644 --- a/include/benchmark/benchmark.h +++ b/include/benchmark/benchmark.h @@ -243,6 +243,7 @@ BENCHMARK(BM_test)->Unit(benchmark::kMillisecond); namespace benchmark { class BenchmarkReporter; +class MemoryManager; void Initialize(int* argc, char** argv); @@ -267,12 +268,9 @@ size_t RunSpecifiedBenchmarks(BenchmarkReporter* console_reporter); size_t RunSpecifiedBenchmarks(BenchmarkReporter* console_reporter, BenchmarkReporter* file_reporter); -// If this routine is called, peak memory allocation past this point in the -// benchmark is reported at the end of the benchmark report line. (It is -// computed by running the benchmark once with a single iteration and a memory -// tracer.) -// TODO(dominic) -// void MemoryUsage(); +// Register a MemoryManager instance that will be used to collect and report +// allocation measurements for benchmark runs. +void RegisterMemoryManager(MemoryManager* memory_manager); namespace internal { class Benchmark; @@ -1280,7 +1278,10 @@ class BenchmarkReporter { complexity_n(0), report_big_o(false), report_rms(false), - counters() {} + counters(), + has_memory_result(false), + allocs_per_iter(0.0), + max_bytes_used(0) {} std::string benchmark_name; std::string report_label; // Empty if not set by benchmark. @@ -1324,6 +1325,11 @@ class BenchmarkReporter { bool report_rms; UserCounters counters; + + // Memory metrics. + bool has_memory_result; + double allocs_per_iter; + int64_t max_bytes_used; }; // Construct a BenchmarkReporter with the output stream set to 'std::cout' @@ -1438,6 +1444,29 @@ class BENCHMARK_DEPRECATED_MSG("The CSV Reporter will be removed in a future rel std::set user_counter_names_; }; +// If a MemoryManager is registered, it can be used to collect and report +// allocation metrics for a run of the benchmark. +class MemoryManager { + public: + struct Result { + Result() : num_allocs(0), max_bytes_used(0) {} + + // The number of allocations made in total between Start and Stop. + int64_t num_allocs; + + // The peak memory use between Start and Stop. + int64_t max_bytes_used; + }; + + virtual ~MemoryManager() {} + + // Implement this to start recording allocation information. + virtual void Start() = 0; + + // Implement this to stop recording and fill out the given Result structure. + virtual void Stop(Result* result) = 0; +}; + inline const char* GetTimeUnitString(TimeUnit unit) { switch (unit) { case kMillisecond: diff --git a/src/benchmark.cc b/src/benchmark.cc index b14bc62914..0f2e198a8f 100644 --- a/src/benchmark.cc +++ b/src/benchmark.cc @@ -105,6 +105,8 @@ namespace benchmark { namespace { static const size_t kMaxIterations = 1000000000; + +static MemoryManager* memory_manager = nullptr; } // end namespace namespace internal { @@ -115,7 +117,8 @@ namespace { BenchmarkReporter::Run CreateRunReport( const benchmark::internal::Benchmark::Instance& b, - const internal::ThreadManager::Result& results, double seconds) { + const internal::ThreadManager::Result& results, size_t memory_iterations, + const MemoryManager::Result& memory_result, double seconds) { // Create report about this benchmark run. BenchmarkReporter::Run report; @@ -150,6 +153,16 @@ BenchmarkReporter::Run CreateRunReport( report.complexity_lambda = b.complexity_lambda; report.statistics = b.statistics; report.counters = results.counters; + + if (memory_iterations > 0) { + report.has_memory_result = true; + report.allocs_per_iter = + memory_iterations ? static_cast(memory_result.num_allocs) / + memory_iterations + : 0; + report.max_bytes_used = memory_result.max_bytes_used; + } + internal::Finish(&report.counters, results.iterations, seconds, b.threads); } return report; @@ -249,7 +262,23 @@ std::vector RunBenchmark( // clang-format on if (should_report) { - BenchmarkReporter::Run report = CreateRunReport(b, results, seconds); + MemoryManager::Result memory_result; + size_t memory_iterations = 0; + if (memory_manager != nullptr) { + // Only run a few iterations to reduce the impact of one-time + // allocations in benchmarks that are not properly managed. + memory_iterations = std::min(16, iters); + memory_manager->Start(); + manager.reset(new internal::ThreadManager(1)); + RunInThread(&b, memory_iterations, 0, manager.get()); + manager->WaitForAllThreads(); + manager.reset(); + + memory_manager->Stop(&memory_result); + } + + BenchmarkReporter::Run report = CreateRunReport( + b, results, memory_iterations, memory_result, seconds); if (!report.error_occurred && b.complexity != oNone) complexity_reports->push_back(report); reports.push_back(report); @@ -555,6 +584,8 @@ size_t RunSpecifiedBenchmarks(BenchmarkReporter* console_reporter, return benchmarks.size(); } +void RegisterMemoryManager(MemoryManager* manager) { memory_manager = manager; } + namespace internal { void PrintUsageAndExit() { diff --git a/src/json_reporter.cc b/src/json_reporter.cc index 611605af6b..6d0706f1e4 100644 --- a/src/json_reporter.cc +++ b/src/json_reporter.cc @@ -186,6 +186,12 @@ void JSONReporter::PrintRunData(Run const& run) { for (auto& c : run.counters) { out << ",\n" << indent << FormatKV(c.first, c.second); } + + if (run.has_memory_result) { + out << ",\n" << indent << FormatKV("allocs_per_iter", run.allocs_per_iter); + out << ",\n" << indent << FormatKV("max_bytes_used", run.max_bytes_used); + } + if (!run.report_label.empty()) { out << ",\n" << indent << FormatKV("label", run.report_label); } diff --git a/test/memory_manager_test.cc b/test/memory_manager_test.cc new file mode 100644 index 0000000000..da735938ae --- /dev/null +++ b/test/memory_manager_test.cc @@ -0,0 +1,40 @@ +#include + +#include "../src/check.h" +#include "benchmark/benchmark.h" +#include "output_test.h" + +class TestMemoryManager : public benchmark::MemoryManager { + void Start() {} + void Stop(Result* result) { + result->num_allocs = 42; + result->max_bytes_used = 42000; + } +}; + +void BM_empty(benchmark::State& state) { + for (auto _ : state) { + benchmark::DoNotOptimize(state.iterations()); + } +} +BENCHMARK(BM_empty); + +ADD_CASES(TC_ConsoleOut, {{"^BM_empty %console_report$"}}); +ADD_CASES(TC_JSONOut, {{"\"name\": \"BM_empty\",$"}, + {"\"iterations\": %int,$", MR_Next}, + {"\"real_time\": %float,$", MR_Next}, + {"\"cpu_time\": %float,$", MR_Next}, + {"\"time_unit\": \"ns\",$", MR_Next}, + {"\"allocs_per_iter\": %float,$", MR_Next}, + {"\"max_bytes_used\": 42000$", MR_Next}, + {"}", MR_Next}}); +ADD_CASES(TC_CSVOut, {{"^\"BM_empty\",%csv_report$"}}); + + +int main(int argc, char *argv[]) { + std::unique_ptr mm(new TestMemoryManager()); + + benchmark::RegisterMemoryManager(mm.get()); + RunOutputTests(argc, argv); + benchmark::RegisterMemoryManager(nullptr); +}