// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/metrics/statistics_recorder.h" #include <stddef.h> #include <memory> #include <utility> #include <vector> #include "base/bind.h" #include "base/json/json_reader.h" #include "base/logging.h" #include "base/memory/weak_ptr.h" #include "base/metrics/histogram_base.h" #include "base/metrics/histogram_macros.h" #include "base/metrics/persistent_histogram_allocator.h" #include "base/metrics/record_histogram_checker.h" #include "base/metrics/sparse_histogram.h" #include "base/values.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" namespace { // Class to make sure any manipulations we do to the min log level are // contained (i.e., do not affect other unit tests). class LogStateSaver { public: LogStateSaver() : old_min_log_level_(logging::GetMinLogLevel()) {} ~LogStateSaver() { logging::SetMinLogLevel(old_min_log_level_); } private: int old_min_log_level_; DISALLOW_COPY_AND_ASSIGN(LogStateSaver); }; // Test implementation of RecordHistogramChecker interface. class OddRecordHistogramChecker : public base::RecordHistogramChecker { public: ~OddRecordHistogramChecker() override = default; // base::RecordHistogramChecker: bool ShouldRecord(uint64_t histogram_hash) const override { return histogram_hash % 2; } }; } // namespace namespace base { using testing::IsEmpty; using testing::SizeIs; using testing::UnorderedElementsAre; class StatisticsRecorderTest : public testing::TestWithParam<bool> { protected: const int32_t kAllocatorMemorySize = 64 << 10; // 64 KiB StatisticsRecorderTest() : use_persistent_histogram_allocator_(GetParam()) { // Each test will have a clean state (no Histogram / BucketRanges // registered). InitializeStatisticsRecorder(); // Use persistent memory for histograms if so indicated by test parameter. if (use_persistent_histogram_allocator_) { GlobalHistogramAllocator::CreateWithLocalMemory(kAllocatorMemorySize, 0, "StatisticsRecorderTest"); } } ~StatisticsRecorderTest() override { GlobalHistogramAllocator::ReleaseForTesting(); UninitializeStatisticsRecorder(); } void InitializeStatisticsRecorder() { DCHECK(!statistics_recorder_); statistics_recorder_ = StatisticsRecorder::CreateTemporaryForTesting(); } // Deletes the global recorder if there is any. This is used by test // NotInitialized to ensure a clean global state. void UninitializeStatisticsRecorder() { statistics_recorder_.reset(); delete StatisticsRecorder::top_; DCHECK(!StatisticsRecorder::top_); } bool HasGlobalRecorder() { return StatisticsRecorder::top_ != nullptr; } Histogram* CreateHistogram(const char* name, HistogramBase::Sample min, HistogramBase::Sample max, size_t bucket_count) { BucketRanges* ranges = new BucketRanges(bucket_count + 1); Histogram::InitializeBucketRanges(min, max, ranges); const BucketRanges* registered_ranges = StatisticsRecorder::RegisterOrDeleteDuplicateRanges(ranges); return new Histogram(name, min, max, registered_ranges); } void InitLogOnShutdown() { StatisticsRecorder::InitLogOnShutdown(); } bool IsVLogInitialized() { return StatisticsRecorder::is_vlog_initialized_; } void ResetVLogInitialized() { UninitializeStatisticsRecorder(); StatisticsRecorder::is_vlog_initialized_ = false; } const bool use_persistent_histogram_allocator_; std::unique_ptr<StatisticsRecorder> statistics_recorder_; std::unique_ptr<GlobalHistogramAllocator> old_global_allocator_; private: LogStateSaver log_state_saver_; DISALLOW_COPY_AND_ASSIGN(StatisticsRecorderTest); }; // Run all HistogramTest cases with both heap and persistent memory. INSTANTIATE_TEST_CASE_P(Allocator, StatisticsRecorderTest, testing::Bool()); TEST_P(StatisticsRecorderTest, NotInitialized) { UninitializeStatisticsRecorder(); EXPECT_FALSE(HasGlobalRecorder()); HistogramBase* const histogram = CreateHistogram("TestHistogram", 1, 1000, 10); EXPECT_EQ(StatisticsRecorder::RegisterOrDeleteDuplicate(histogram), histogram); EXPECT_TRUE(HasGlobalRecorder()); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram)); UninitializeStatisticsRecorder(); EXPECT_FALSE(HasGlobalRecorder()); BucketRanges* const ranges = new BucketRanges(3); ranges->ResetChecksum(); EXPECT_EQ(StatisticsRecorder::RegisterOrDeleteDuplicateRanges(ranges), ranges); EXPECT_TRUE(HasGlobalRecorder()); EXPECT_THAT(StatisticsRecorder::GetBucketRanges(), UnorderedElementsAre(ranges)); } TEST_P(StatisticsRecorderTest, RegisterBucketRanges) { std::vector<const BucketRanges*> registered_ranges; BucketRanges* ranges1 = new BucketRanges(3); ranges1->ResetChecksum(); BucketRanges* ranges2 = new BucketRanges(4); ranges2->ResetChecksum(); // Register new ranges. EXPECT_EQ(ranges1, StatisticsRecorder::RegisterOrDeleteDuplicateRanges(ranges1)); EXPECT_EQ(ranges2, StatisticsRecorder::RegisterOrDeleteDuplicateRanges(ranges2)); EXPECT_THAT(StatisticsRecorder::GetBucketRanges(), UnorderedElementsAre(ranges1, ranges2)); // Register some ranges again. EXPECT_EQ(ranges1, StatisticsRecorder::RegisterOrDeleteDuplicateRanges(ranges1)); EXPECT_THAT(StatisticsRecorder::GetBucketRanges(), UnorderedElementsAre(ranges1, ranges2)); // Make sure the ranges is still the one we know. ASSERT_EQ(3u, ranges1->size()); EXPECT_EQ(0, ranges1->range(0)); EXPECT_EQ(0, ranges1->range(1)); EXPECT_EQ(0, ranges1->range(2)); // Register ranges with same values. BucketRanges* ranges3 = new BucketRanges(3); ranges3->ResetChecksum(); EXPECT_EQ(ranges1, // returning ranges1 StatisticsRecorder::RegisterOrDeleteDuplicateRanges(ranges3)); EXPECT_THAT(StatisticsRecorder::GetBucketRanges(), UnorderedElementsAre(ranges1, ranges2)); } TEST_P(StatisticsRecorderTest, RegisterHistogram) { // Create a Histogram that was not registered. Histogram* const histogram1 = CreateHistogram("TestHistogram1", 1, 1000, 10); EXPECT_THAT(StatisticsRecorder::GetHistograms(), IsEmpty()); // Register the Histogram. EXPECT_EQ(histogram1, StatisticsRecorder::RegisterOrDeleteDuplicate(histogram1)); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1)); // Register the same Histogram again. EXPECT_EQ(histogram1, StatisticsRecorder::RegisterOrDeleteDuplicate(histogram1)); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1)); // Register another Histogram with the same name. Histogram* const histogram2 = CreateHistogram("TestHistogram1", 1, 1000, 10); EXPECT_NE(histogram1, histogram2); EXPECT_EQ(histogram1, StatisticsRecorder::RegisterOrDeleteDuplicate(histogram2)); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1)); // Register another Histogram with a different name. Histogram* const histogram3 = CreateHistogram("TestHistogram0", 1, 1000, 10); EXPECT_NE(histogram1, histogram3); EXPECT_EQ(histogram3, StatisticsRecorder::RegisterOrDeleteDuplicate(histogram3)); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1, histogram3)); } TEST_P(StatisticsRecorderTest, FindHistogram) { HistogramBase* histogram1 = Histogram::FactoryGet( "TestHistogram1", 1, 1000, 10, HistogramBase::kNoFlags); HistogramBase* histogram2 = Histogram::FactoryGet( "TestHistogram2", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_EQ(histogram1, StatisticsRecorder::FindHistogram("TestHistogram1")); EXPECT_EQ(histogram2, StatisticsRecorder::FindHistogram("TestHistogram2")); EXPECT_FALSE(StatisticsRecorder::FindHistogram("TestHistogram")); // Create a new global allocator using the same memory as the old one. Any // old one is kept around so the memory doesn't get released. old_global_allocator_ = GlobalHistogramAllocator::ReleaseForTesting(); if (use_persistent_histogram_allocator_) { GlobalHistogramAllocator::CreateWithPersistentMemory( const_cast<void*>(old_global_allocator_->data()), old_global_allocator_->length(), 0, old_global_allocator_->Id(), old_global_allocator_->Name()); } // Reset statistics-recorder to validate operation from a clean start. UninitializeStatisticsRecorder(); InitializeStatisticsRecorder(); if (use_persistent_histogram_allocator_) { EXPECT_TRUE(StatisticsRecorder::FindHistogram("TestHistogram1")); EXPECT_TRUE(StatisticsRecorder::FindHistogram("TestHistogram2")); } else { EXPECT_FALSE(StatisticsRecorder::FindHistogram("TestHistogram1")); EXPECT_FALSE(StatisticsRecorder::FindHistogram("TestHistogram2")); } EXPECT_FALSE(StatisticsRecorder::FindHistogram("TestHistogram")); } TEST_P(StatisticsRecorderTest, WithName) { Histogram::FactoryGet("TestHistogram1", 1, 1000, 10, Histogram::kNoFlags); Histogram::FactoryGet("TestHistogram2", 1, 1000, 10, Histogram::kNoFlags); Histogram::FactoryGet("TestHistogram3", 1, 1000, 10, Histogram::kNoFlags); const auto histograms = StatisticsRecorder::GetHistograms(); EXPECT_THAT(histograms, SizeIs(3)); EXPECT_THAT(StatisticsRecorder::WithName(histograms, ""), SizeIs(3)); EXPECT_THAT(StatisticsRecorder::WithName(histograms, "Test"), SizeIs(3)); EXPECT_THAT(StatisticsRecorder::WithName(histograms, "1"), SizeIs(1)); EXPECT_THAT(StatisticsRecorder::WithName(histograms, "hello"), IsEmpty()); } TEST_P(StatisticsRecorderTest, RegisterHistogramWithFactoryGet) { EXPECT_THAT(StatisticsRecorder::GetHistograms(), IsEmpty()); // Create a histogram. HistogramBase* const histogram1 = Histogram::FactoryGet( "TestHistogram", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1)); // Get an existing histogram. HistogramBase* const histogram2 = Histogram::FactoryGet( "TestHistogram", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_EQ(histogram1, histogram2); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1)); // Create a LinearHistogram. HistogramBase* const histogram3 = LinearHistogram::FactoryGet( "TestLinearHistogram", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1, histogram3)); // Create a BooleanHistogram. HistogramBase* const histogram4 = BooleanHistogram::FactoryGet( "TestBooleanHistogram", HistogramBase::kNoFlags); EXPECT_THAT(StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1, histogram3, histogram4)); // Create a CustomHistogram. std::vector<int> custom_ranges; custom_ranges.push_back(1); custom_ranges.push_back(5); HistogramBase* const histogram5 = CustomHistogram::FactoryGet( "TestCustomHistogram", custom_ranges, HistogramBase::kNoFlags); EXPECT_THAT( StatisticsRecorder::GetHistograms(), UnorderedElementsAre(histogram1, histogram3, histogram4, histogram5)); } TEST_P(StatisticsRecorderTest, RegisterHistogramWithMacros) { // Macros cache pointers and so tests that use them can only be run once. // Stop immediately if this test has run previously. static bool already_run = false; if (already_run) return; already_run = true; StatisticsRecorder::Histograms registered_histograms; HistogramBase* histogram = Histogram::FactoryGet( "TestHistogramCounts", 1, 1000000, 50, HistogramBase::kNoFlags); // The histogram we got from macro is the same as from FactoryGet. LOCAL_HISTOGRAM_COUNTS("TestHistogramCounts", 30); registered_histograms = StatisticsRecorder::GetHistograms(); ASSERT_EQ(1u, registered_histograms.size()); EXPECT_EQ(histogram, registered_histograms[0]); LOCAL_HISTOGRAM_TIMES("TestHistogramTimes", TimeDelta::FromDays(1)); LOCAL_HISTOGRAM_ENUMERATION("TestHistogramEnumeration", 20, 200); EXPECT_THAT(StatisticsRecorder::GetHistograms(), SizeIs(3)); } TEST_P(StatisticsRecorderTest, BucketRangesSharing) { EXPECT_THAT(StatisticsRecorder::GetBucketRanges(), IsEmpty()); Histogram::FactoryGet("Histogram", 1, 64, 8, HistogramBase::kNoFlags); Histogram::FactoryGet("Histogram2", 1, 64, 8, HistogramBase::kNoFlags); EXPECT_THAT(StatisticsRecorder::GetBucketRanges(), SizeIs(1)); Histogram::FactoryGet("Histogram3", 1, 64, 16, HistogramBase::kNoFlags); EXPECT_THAT(StatisticsRecorder::GetBucketRanges(), SizeIs(2)); } TEST_P(StatisticsRecorderTest, ToJSON) { Histogram::FactoryGet("TestHistogram1", 1, 1000, 50, HistogramBase::kNoFlags) ->Add(30); Histogram::FactoryGet("TestHistogram1", 1, 1000, 50, HistogramBase::kNoFlags) ->Add(40); Histogram::FactoryGet("TestHistogram2", 1, 1000, 50, HistogramBase::kNoFlags) ->Add(30); Histogram::FactoryGet("TestHistogram2", 1, 1000, 50, HistogramBase::kNoFlags) ->Add(40); std::string json(StatisticsRecorder::ToJSON(JSON_VERBOSITY_LEVEL_FULL)); // Check for valid JSON. std::unique_ptr<Value> root = JSONReader::Read(json); ASSERT_TRUE(root.get()); DictionaryValue* root_dict = nullptr; ASSERT_TRUE(root->GetAsDictionary(&root_dict)); // No query should be set. ASSERT_FALSE(root_dict->HasKey("query")); ListValue* histogram_list = nullptr; ASSERT_TRUE(root_dict->GetList("histograms", &histogram_list)); ASSERT_EQ(2u, histogram_list->GetSize()); // Examine the first histogram. DictionaryValue* histogram_dict = nullptr; ASSERT_TRUE(histogram_list->GetDictionary(0, &histogram_dict)); int sample_count; ASSERT_TRUE(histogram_dict->GetInteger("count", &sample_count)); EXPECT_EQ(2, sample_count); ListValue* buckets_list = nullptr; ASSERT_TRUE(histogram_dict->GetList("buckets", &buckets_list)); EXPECT_EQ(2u, buckets_list->GetList().size()); // Check the serialized JSON with a different verbosity level. json = StatisticsRecorder::ToJSON(JSON_VERBOSITY_LEVEL_OMIT_BUCKETS); root = JSONReader::Read(json); ASSERT_TRUE(root.get()); root_dict = nullptr; ASSERT_TRUE(root->GetAsDictionary(&root_dict)); histogram_list = nullptr; ASSERT_TRUE(root_dict->GetList("histograms", &histogram_list)); ASSERT_EQ(2u, histogram_list->GetSize()); histogram_dict = nullptr; ASSERT_TRUE(histogram_list->GetDictionary(0, &histogram_dict)); sample_count = 0; ASSERT_TRUE(histogram_dict->GetInteger("count", &sample_count)); EXPECT_EQ(2, sample_count); buckets_list = nullptr; // Bucket information should be omitted. ASSERT_FALSE(histogram_dict->GetList("buckets", &buckets_list)); } TEST_P(StatisticsRecorderTest, IterationTest) { Histogram::FactoryGet("IterationTest1", 1, 64, 16, HistogramBase::kNoFlags); Histogram::FactoryGet("IterationTest2", 1, 64, 16, HistogramBase::kNoFlags); auto histograms = StatisticsRecorder::GetHistograms(); EXPECT_THAT(histograms, SizeIs(2)); histograms = StatisticsRecorder::NonPersistent(std::move(histograms)); EXPECT_THAT(histograms, SizeIs(use_persistent_histogram_allocator_ ? 0 : 2)); // Create a new global allocator using the same memory as the old one. Any // old one is kept around so the memory doesn't get released. old_global_allocator_ = GlobalHistogramAllocator::ReleaseForTesting(); if (use_persistent_histogram_allocator_) { GlobalHistogramAllocator::CreateWithPersistentMemory( const_cast<void*>(old_global_allocator_->data()), old_global_allocator_->length(), 0, old_global_allocator_->Id(), old_global_allocator_->Name()); } // Reset statistics-recorder to validate operation from a clean start. UninitializeStatisticsRecorder(); InitializeStatisticsRecorder(); histograms = StatisticsRecorder::GetHistograms(); EXPECT_THAT(histograms, SizeIs(use_persistent_histogram_allocator_ ? 2 : 0)); histograms = StatisticsRecorder::NonPersistent(std::move(histograms)); EXPECT_THAT(histograms, IsEmpty()); } namespace { // CallbackCheckWrapper is simply a convenient way to check and store that // a callback was actually run. struct CallbackCheckWrapper { CallbackCheckWrapper() : called(false), last_histogram_value(0) {} void OnHistogramChanged(base::HistogramBase::Sample histogram_value) { called = true; last_histogram_value = histogram_value; } bool called; base::HistogramBase::Sample last_histogram_value; }; } // namespace // Check that you can't overwrite the callback with another. TEST_P(StatisticsRecorderTest, SetCallbackFailsWithoutHistogramTest) { CallbackCheckWrapper callback_wrapper; bool result = base::StatisticsRecorder::SetCallback( "TestHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); EXPECT_TRUE(result); result = base::StatisticsRecorder::SetCallback( "TestHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); EXPECT_FALSE(result); } // Check that you can't overwrite the callback with another. TEST_P(StatisticsRecorderTest, SetCallbackFailsWithHistogramTest) { HistogramBase* histogram = Histogram::FactoryGet("TestHistogram", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_TRUE(histogram); CallbackCheckWrapper callback_wrapper; bool result = base::StatisticsRecorder::SetCallback( "TestHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); EXPECT_TRUE(result); EXPECT_EQ(histogram->flags() & base::HistogramBase::kCallbackExists, base::HistogramBase::kCallbackExists); result = base::StatisticsRecorder::SetCallback( "TestHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); EXPECT_FALSE(result); EXPECT_EQ(histogram->flags() & base::HistogramBase::kCallbackExists, base::HistogramBase::kCallbackExists); histogram->Add(1); EXPECT_TRUE(callback_wrapper.called); } // Check that you can't overwrite the callback with another. TEST_P(StatisticsRecorderTest, ClearCallbackSuceedsWithHistogramTest) { HistogramBase* histogram = Histogram::FactoryGet("TestHistogram", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_TRUE(histogram); CallbackCheckWrapper callback_wrapper; bool result = base::StatisticsRecorder::SetCallback( "TestHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); EXPECT_TRUE(result); EXPECT_EQ(histogram->flags() & base::HistogramBase::kCallbackExists, base::HistogramBase::kCallbackExists); base::StatisticsRecorder::ClearCallback("TestHistogram"); EXPECT_EQ(histogram->flags() & base::HistogramBase::kCallbackExists, 0); histogram->Add(1); EXPECT_FALSE(callback_wrapper.called); } // Check that callback is used. TEST_P(StatisticsRecorderTest, CallbackUsedTest) { { HistogramBase* histogram = Histogram::FactoryGet( "TestHistogram", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_TRUE(histogram); CallbackCheckWrapper callback_wrapper; base::StatisticsRecorder::SetCallback( "TestHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); histogram->Add(1); EXPECT_TRUE(callback_wrapper.called); EXPECT_EQ(callback_wrapper.last_histogram_value, 1); } { HistogramBase* linear_histogram = LinearHistogram::FactoryGet( "TestLinearHistogram", 1, 1000, 10, HistogramBase::kNoFlags); CallbackCheckWrapper callback_wrapper; base::StatisticsRecorder::SetCallback( "TestLinearHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); linear_histogram->Add(1); EXPECT_TRUE(callback_wrapper.called); EXPECT_EQ(callback_wrapper.last_histogram_value, 1); } { std::vector<int> custom_ranges; custom_ranges.push_back(1); custom_ranges.push_back(5); HistogramBase* custom_histogram = CustomHistogram::FactoryGet( "TestCustomHistogram", custom_ranges, HistogramBase::kNoFlags); CallbackCheckWrapper callback_wrapper; base::StatisticsRecorder::SetCallback( "TestCustomHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); custom_histogram->Add(1); EXPECT_TRUE(callback_wrapper.called); EXPECT_EQ(callback_wrapper.last_histogram_value, 1); } { HistogramBase* custom_histogram = SparseHistogram::FactoryGet( "TestSparseHistogram", HistogramBase::kNoFlags); CallbackCheckWrapper callback_wrapper; base::StatisticsRecorder::SetCallback( "TestSparseHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); custom_histogram->Add(1); EXPECT_TRUE(callback_wrapper.called); EXPECT_EQ(callback_wrapper.last_histogram_value, 1); } } // Check that setting a callback before the histogram exists works. TEST_P(StatisticsRecorderTest, CallbackUsedBeforeHistogramCreatedTest) { CallbackCheckWrapper callback_wrapper; base::StatisticsRecorder::SetCallback( "TestHistogram", base::Bind(&CallbackCheckWrapper::OnHistogramChanged, base::Unretained(&callback_wrapper))); HistogramBase* histogram = Histogram::FactoryGet("TestHistogram", 1, 1000, 10, HistogramBase::kNoFlags); EXPECT_TRUE(histogram); histogram->Add(1); EXPECT_TRUE(callback_wrapper.called); EXPECT_EQ(callback_wrapper.last_histogram_value, 1); } TEST_P(StatisticsRecorderTest, LogOnShutdownNotInitialized) { ResetVLogInitialized(); logging::SetMinLogLevel(logging::LOG_WARNING); InitializeStatisticsRecorder(); EXPECT_FALSE(VLOG_IS_ON(1)); EXPECT_FALSE(IsVLogInitialized()); InitLogOnShutdown(); EXPECT_FALSE(IsVLogInitialized()); } TEST_P(StatisticsRecorderTest, LogOnShutdownInitializedExplicitly) { ResetVLogInitialized(); logging::SetMinLogLevel(logging::LOG_WARNING); InitializeStatisticsRecorder(); EXPECT_FALSE(VLOG_IS_ON(1)); EXPECT_FALSE(IsVLogInitialized()); logging::SetMinLogLevel(logging::LOG_VERBOSE); EXPECT_TRUE(VLOG_IS_ON(1)); InitLogOnShutdown(); EXPECT_TRUE(IsVLogInitialized()); } TEST_P(StatisticsRecorderTest, LogOnShutdownInitialized) { ResetVLogInitialized(); logging::SetMinLogLevel(logging::LOG_VERBOSE); InitializeStatisticsRecorder(); EXPECT_TRUE(VLOG_IS_ON(1)); EXPECT_TRUE(IsVLogInitialized()); } class TestHistogramProvider : public StatisticsRecorder::HistogramProvider { public: TestHistogramProvider(std::unique_ptr<PersistentHistogramAllocator> allocator) : allocator_(std::move(allocator)), weak_factory_(this) { StatisticsRecorder::RegisterHistogramProvider(weak_factory_.GetWeakPtr()); } void MergeHistogramDeltas() override { PersistentHistogramAllocator::Iterator hist_iter(allocator_.get()); while (true) { std::unique_ptr<base::HistogramBase> histogram = hist_iter.GetNext(); if (!histogram) break; allocator_->MergeHistogramDeltaToStatisticsRecorder(histogram.get()); } } private: std::unique_ptr<PersistentHistogramAllocator> allocator_; WeakPtrFactory<TestHistogramProvider> weak_factory_; DISALLOW_COPY_AND_ASSIGN(TestHistogramProvider); }; TEST_P(StatisticsRecorderTest, ImportHistogramsTest) { // Create a second SR to create some histograms for later import. std::unique_ptr<StatisticsRecorder> temp_sr = StatisticsRecorder::CreateTemporaryForTesting(); // Extract any existing global allocator so a new one can be created. std::unique_ptr<GlobalHistogramAllocator> old_allocator = GlobalHistogramAllocator::ReleaseForTesting(); // Create a histogram inside a new allocator for testing. GlobalHistogramAllocator::CreateWithLocalMemory(kAllocatorMemorySize, 0, ""); HistogramBase* histogram = LinearHistogram::FactoryGet("Foo", 1, 10, 11, 0); histogram->Add(3); // Undo back to the starting point. std::unique_ptr<GlobalHistogramAllocator> new_allocator = GlobalHistogramAllocator::ReleaseForTesting(); GlobalHistogramAllocator::Set(std::move(old_allocator)); temp_sr.reset(); // Create a provider that can supply histograms to the current SR. TestHistogramProvider provider(std::move(new_allocator)); // Verify that the created histogram is no longer known. ASSERT_FALSE(StatisticsRecorder::FindHistogram(histogram->histogram_name())); // Now test that it merges. StatisticsRecorder::ImportProvidedHistograms(); HistogramBase* found = StatisticsRecorder::FindHistogram(histogram->histogram_name()); ASSERT_TRUE(found); EXPECT_NE(histogram, found); std::unique_ptr<HistogramSamples> snapshot = found->SnapshotSamples(); EXPECT_EQ(1, snapshot->TotalCount()); EXPECT_EQ(1, snapshot->GetCount(3)); // Finally, verify that updates can also be merged. histogram->Add(3); histogram->Add(5); StatisticsRecorder::ImportProvidedHistograms(); snapshot = found->SnapshotSamples(); EXPECT_EQ(3, snapshot->TotalCount()); EXPECT_EQ(2, snapshot->GetCount(3)); EXPECT_EQ(1, snapshot->GetCount(5)); } TEST_P(StatisticsRecorderTest, RecordHistogramChecker) { // Before record checker is set all histograms should be recorded. EXPECT_TRUE(StatisticsRecorder::ShouldRecordHistogram(1)); EXPECT_TRUE(StatisticsRecorder::ShouldRecordHistogram(2)); auto record_checker = std::make_unique<OddRecordHistogramChecker>(); StatisticsRecorder::SetRecordChecker(std::move(record_checker)); EXPECT_TRUE(StatisticsRecorder::ShouldRecordHistogram(1)); EXPECT_FALSE(StatisticsRecorder::ShouldRecordHistogram(2)); } } // namespace base