| // 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/histogram.h" | 
 |  | 
 | #include <climits> | 
 | #include <algorithm> | 
 | #include <vector> | 
 |  | 
 | #include "base/logging.h" | 
 | #include "base/memory/scoped_ptr.h" | 
 | #include "base/metrics/bucket_ranges.h" | 
 | #include "base/metrics/sample_vector.h" | 
 | #include "base/metrics/statistics_recorder.h" | 
 | #include "base/pickle.h" | 
 | #include "base/time/time.h" | 
 | #include "testing/gtest/include/gtest/gtest.h" | 
 |  | 
 | namespace base { | 
 |  | 
 | class HistogramTest : public testing::Test { | 
 |  protected: | 
 |   void SetUp() override { | 
 |     // Each test will have a clean state (no Histogram / BucketRanges | 
 |     // registered). | 
 |     InitializeStatisticsRecorder(); | 
 |   } | 
 |  | 
 |   void TearDown() override { UninitializeStatisticsRecorder(); } | 
 |  | 
 |   void InitializeStatisticsRecorder() { | 
 |     statistics_recorder_ = new StatisticsRecorder(); | 
 |   } | 
 |  | 
 |   void UninitializeStatisticsRecorder() { | 
 |     delete statistics_recorder_; | 
 |     statistics_recorder_ = NULL; | 
 |   } | 
 |  | 
 |   StatisticsRecorder* statistics_recorder_; | 
 | }; | 
 |  | 
 | // Check for basic syntax and use. | 
 | TEST_F(HistogramTest, BasicTest) { | 
 |   // Try basic construction | 
 |   HistogramBase* histogram = Histogram::FactoryGet( | 
 |       "TestHistogram", 1, 1000, 10, HistogramBase::kNoFlags); | 
 |   EXPECT_TRUE(histogram); | 
 |  | 
 |   HistogramBase* linear_histogram = LinearHistogram::FactoryGet( | 
 |       "TestLinearHistogram", 1, 1000, 10, HistogramBase::kNoFlags); | 
 |   EXPECT_TRUE(linear_histogram); | 
 |  | 
 |   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); | 
 |   EXPECT_TRUE(custom_histogram); | 
 |  | 
 |   // Use standard macros (but with fixed samples) | 
 |   LOCAL_HISTOGRAM_TIMES("Test2Histogram", TimeDelta::FromDays(1)); | 
 |   LOCAL_HISTOGRAM_COUNTS("Test3Histogram", 30); | 
 |  | 
 |   LOCAL_HISTOGRAM_ENUMERATION("Test6Histogram", 129, 130); | 
 | } | 
 |  | 
 | // Check that the macro correctly matches histograms by name and records their | 
 | // data together. | 
 | TEST_F(HistogramTest, NameMatchTest) { | 
 |   LOCAL_HISTOGRAM_PERCENTAGE("DuplicatedHistogram", 10); | 
 |   LOCAL_HISTOGRAM_PERCENTAGE("DuplicatedHistogram", 10); | 
 |   HistogramBase* histogram = LinearHistogram::FactoryGet( | 
 |       "DuplicatedHistogram", 1, 101, 102, HistogramBase::kNoFlags); | 
 |  | 
 |   scoped_ptr<HistogramSamples> samples = histogram->SnapshotSamples(); | 
 |   EXPECT_EQ(2, samples->TotalCount()); | 
 |   EXPECT_EQ(2, samples->GetCount(10)); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, ExponentialRangesTest) { | 
 |   // Check that we got a nice exponential when there was enough rooom. | 
 |   BucketRanges ranges(9); | 
 |   Histogram::InitializeBucketRanges(1, 64, &ranges); | 
 |   EXPECT_EQ(0, ranges.range(0)); | 
 |   int power_of_2 = 1; | 
 |   for (int i = 1; i < 8; i++) { | 
 |     EXPECT_EQ(power_of_2, ranges.range(i)); | 
 |     power_of_2 *= 2; | 
 |   } | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges.range(8)); | 
 |  | 
 |   // Check the corresponding Histogram will use the correct ranges. | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       Histogram::FactoryGet("Histogram", 1, 64, 8, HistogramBase::kNoFlags)); | 
 |   EXPECT_TRUE(ranges.Equals(histogram->bucket_ranges())); | 
 |  | 
 |   // When bucket count is limited, exponential ranges will partially look like | 
 |   // linear. | 
 |   BucketRanges ranges2(16); | 
 |   Histogram::InitializeBucketRanges(1, 32, &ranges2); | 
 |  | 
 |   EXPECT_EQ(0, ranges2.range(0)); | 
 |   EXPECT_EQ(1, ranges2.range(1)); | 
 |   EXPECT_EQ(2, ranges2.range(2)); | 
 |   EXPECT_EQ(3, ranges2.range(3)); | 
 |   EXPECT_EQ(4, ranges2.range(4)); | 
 |   EXPECT_EQ(5, ranges2.range(5)); | 
 |   EXPECT_EQ(6, ranges2.range(6)); | 
 |   EXPECT_EQ(7, ranges2.range(7)); | 
 |   EXPECT_EQ(9, ranges2.range(8)); | 
 |   EXPECT_EQ(11, ranges2.range(9)); | 
 |   EXPECT_EQ(14, ranges2.range(10)); | 
 |   EXPECT_EQ(17, ranges2.range(11)); | 
 |   EXPECT_EQ(21, ranges2.range(12)); | 
 |   EXPECT_EQ(26, ranges2.range(13)); | 
 |   EXPECT_EQ(32, ranges2.range(14)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges2.range(15)); | 
 |  | 
 |   // Check the corresponding Histogram will use the correct ranges. | 
 |   Histogram* histogram2 = static_cast<Histogram*>( | 
 |       Histogram::FactoryGet("Histogram2", 1, 32, 15, HistogramBase::kNoFlags)); | 
 |   EXPECT_TRUE(ranges2.Equals(histogram2->bucket_ranges())); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, LinearRangesTest) { | 
 |   BucketRanges ranges(9); | 
 |   LinearHistogram::InitializeBucketRanges(1, 7, &ranges); | 
 |   // Gets a nice linear set of bucket ranges. | 
 |   for (int i = 0; i < 8; i++) | 
 |     EXPECT_EQ(i, ranges.range(i)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges.range(8)); | 
 |  | 
 |   // The correspoding LinearHistogram should use the correct ranges. | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       LinearHistogram::FactoryGet("Linear", 1, 7, 8, HistogramBase::kNoFlags)); | 
 |   EXPECT_TRUE(ranges.Equals(histogram->bucket_ranges())); | 
 |  | 
 |   // Linear ranges are not divisible. | 
 |   BucketRanges ranges2(6); | 
 |   LinearHistogram::InitializeBucketRanges(1, 6, &ranges2); | 
 |   EXPECT_EQ(0, ranges2.range(0)); | 
 |   EXPECT_EQ(1, ranges2.range(1)); | 
 |   EXPECT_EQ(3, ranges2.range(2)); | 
 |   EXPECT_EQ(4, ranges2.range(3)); | 
 |   EXPECT_EQ(6, ranges2.range(4)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges2.range(5)); | 
 |   // The correspoding LinearHistogram should use the correct ranges. | 
 |   Histogram* histogram2 = static_cast<Histogram*>( | 
 |       LinearHistogram::FactoryGet("Linear2", 1, 6, 5, HistogramBase::kNoFlags)); | 
 |   EXPECT_TRUE(ranges2.Equals(histogram2->bucket_ranges())); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, ArrayToCustomRangesTest) { | 
 |   const HistogramBase::Sample ranges[3] = {5, 10, 20}; | 
 |   std::vector<HistogramBase::Sample> ranges_vec = | 
 |       CustomHistogram::ArrayToCustomRanges(ranges, 3); | 
 |   ASSERT_EQ(6u, ranges_vec.size()); | 
 |   EXPECT_EQ(5, ranges_vec[0]); | 
 |   EXPECT_EQ(6, ranges_vec[1]); | 
 |   EXPECT_EQ(10, ranges_vec[2]); | 
 |   EXPECT_EQ(11, ranges_vec[3]); | 
 |   EXPECT_EQ(20, ranges_vec[4]); | 
 |   EXPECT_EQ(21, ranges_vec[5]); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, CustomHistogramTest) { | 
 |   // A well prepared custom ranges. | 
 |   std::vector<HistogramBase::Sample> custom_ranges; | 
 |   custom_ranges.push_back(1); | 
 |   custom_ranges.push_back(2); | 
 |  | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       CustomHistogram::FactoryGet("TestCustomHistogram1", custom_ranges, | 
 |                                   HistogramBase::kNoFlags)); | 
 |   const BucketRanges* ranges = histogram->bucket_ranges(); | 
 |   ASSERT_EQ(4u, ranges->size()); | 
 |   EXPECT_EQ(0, ranges->range(0));  // Auto added. | 
 |   EXPECT_EQ(1, ranges->range(1)); | 
 |   EXPECT_EQ(2, ranges->range(2)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges->range(3));  // Auto added. | 
 |  | 
 |   // A unordered custom ranges. | 
 |   custom_ranges.clear(); | 
 |   custom_ranges.push_back(2); | 
 |   custom_ranges.push_back(1); | 
 |   histogram = static_cast<Histogram*>( | 
 |       CustomHistogram::FactoryGet("TestCustomHistogram2", custom_ranges, | 
 |                                   HistogramBase::kNoFlags)); | 
 |   ranges = histogram->bucket_ranges(); | 
 |   ASSERT_EQ(4u, ranges->size()); | 
 |   EXPECT_EQ(0, ranges->range(0)); | 
 |   EXPECT_EQ(1, ranges->range(1)); | 
 |   EXPECT_EQ(2, ranges->range(2)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges->range(3)); | 
 |  | 
 |   // A custom ranges with duplicated values. | 
 |   custom_ranges.clear(); | 
 |   custom_ranges.push_back(4); | 
 |   custom_ranges.push_back(1); | 
 |   custom_ranges.push_back(4); | 
 |   histogram = static_cast<Histogram*>( | 
 |       CustomHistogram::FactoryGet("TestCustomHistogram3", custom_ranges, | 
 |                                   HistogramBase::kNoFlags)); | 
 |   ranges = histogram->bucket_ranges(); | 
 |   ASSERT_EQ(4u, ranges->size()); | 
 |   EXPECT_EQ(0, ranges->range(0)); | 
 |   EXPECT_EQ(1, ranges->range(1)); | 
 |   EXPECT_EQ(4, ranges->range(2)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges->range(3)); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, CustomHistogramWithOnly2Buckets) { | 
 |   // This test exploits the fact that the CustomHistogram can have 2 buckets, | 
 |   // while the base class Histogram is *supposed* to have at least 3 buckets. | 
 |   // We should probably change the restriction on the base class (or not inherit | 
 |   // the base class!). | 
 |  | 
 |   std::vector<HistogramBase::Sample> custom_ranges; | 
 |   custom_ranges.push_back(4); | 
 |  | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       CustomHistogram::FactoryGet("2BucketsCustomHistogram", custom_ranges, | 
 |                                   HistogramBase::kNoFlags)); | 
 |   const BucketRanges* ranges = histogram->bucket_ranges(); | 
 |   ASSERT_EQ(3u, ranges->size()); | 
 |   EXPECT_EQ(0, ranges->range(0)); | 
 |   EXPECT_EQ(4, ranges->range(1)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges->range(2)); | 
 | } | 
 |  | 
 | // Make sure histogram handles out-of-bounds data gracefully. | 
 | TEST_F(HistogramTest, BoundsTest) { | 
 |   const size_t kBucketCount = 50; | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       Histogram::FactoryGet("Bounded", 10, 100, kBucketCount, | 
 |                             HistogramBase::kNoFlags)); | 
 |  | 
 |   // Put two samples "out of bounds" above and below. | 
 |   histogram->Add(5); | 
 |   histogram->Add(-50); | 
 |  | 
 |   histogram->Add(100); | 
 |   histogram->Add(10000); | 
 |  | 
 |   // Verify they landed in the underflow, and overflow buckets. | 
 |   scoped_ptr<SampleVector> samples = histogram->SnapshotSampleVector(); | 
 |   EXPECT_EQ(2, samples->GetCountAtIndex(0)); | 
 |   EXPECT_EQ(0, samples->GetCountAtIndex(1)); | 
 |   size_t array_size = histogram->bucket_count(); | 
 |   EXPECT_EQ(kBucketCount, array_size); | 
 |   EXPECT_EQ(0, samples->GetCountAtIndex(array_size - 2)); | 
 |   EXPECT_EQ(2, samples->GetCountAtIndex(array_size - 1)); | 
 |  | 
 |   std::vector<int> custom_ranges; | 
 |   custom_ranges.push_back(10); | 
 |   custom_ranges.push_back(50); | 
 |   custom_ranges.push_back(100); | 
 |   Histogram* test_custom_histogram = static_cast<Histogram*>( | 
 |       CustomHistogram::FactoryGet("TestCustomRangeBoundedHistogram", | 
 |                                   custom_ranges, HistogramBase::kNoFlags)); | 
 |  | 
 |   // Put two samples "out of bounds" above and below. | 
 |   test_custom_histogram->Add(5); | 
 |   test_custom_histogram->Add(-50); | 
 |   test_custom_histogram->Add(100); | 
 |   test_custom_histogram->Add(1000); | 
 |   test_custom_histogram->Add(INT_MAX); | 
 |  | 
 |   // Verify they landed in the underflow, and overflow buckets. | 
 |   scoped_ptr<SampleVector> custom_samples = | 
 |       test_custom_histogram->SnapshotSampleVector(); | 
 |   EXPECT_EQ(2, custom_samples->GetCountAtIndex(0)); | 
 |   EXPECT_EQ(0, custom_samples->GetCountAtIndex(1)); | 
 |   size_t bucket_count = test_custom_histogram->bucket_count(); | 
 |   EXPECT_EQ(0, custom_samples->GetCountAtIndex(bucket_count - 2)); | 
 |   EXPECT_EQ(3, custom_samples->GetCountAtIndex(bucket_count - 1)); | 
 | } | 
 |  | 
 | // Check to be sure samples land as expected is "correct" buckets. | 
 | TEST_F(HistogramTest, BucketPlacementTest) { | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       Histogram::FactoryGet("Histogram", 1, 64, 8, HistogramBase::kNoFlags)); | 
 |  | 
 |   // Add i+1 samples to the i'th bucket. | 
 |   histogram->Add(0); | 
 |   int power_of_2 = 1; | 
 |   for (int i = 1; i < 8; i++) { | 
 |     for (int j = 0; j <= i; j++) | 
 |       histogram->Add(power_of_2); | 
 |     power_of_2 *= 2; | 
 |   } | 
 |  | 
 |   // Check to see that the bucket counts reflect our additions. | 
 |   scoped_ptr<SampleVector> samples = histogram->SnapshotSampleVector(); | 
 |   for (int i = 0; i < 8; i++) | 
 |     EXPECT_EQ(i + 1, samples->GetCountAtIndex(i)); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, CorruptSampleCounts) { | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       Histogram::FactoryGet("Histogram", 1, 64, 8, HistogramBase::kNoFlags)); | 
 |  | 
 |   // Add some samples. | 
 |   histogram->Add(20); | 
 |   histogram->Add(40); | 
 |  | 
 |   scoped_ptr<SampleVector> snapshot = histogram->SnapshotSampleVector(); | 
 |   EXPECT_EQ(HistogramBase::NO_INCONSISTENCIES, | 
 |             histogram->FindCorruption(*snapshot)); | 
 |   EXPECT_EQ(2, snapshot->redundant_count()); | 
 |   EXPECT_EQ(2, snapshot->TotalCount()); | 
 |  | 
 |   snapshot->counts_[3] += 100;  // Sample count won't match redundant count. | 
 |   EXPECT_EQ(HistogramBase::COUNT_LOW_ERROR, | 
 |             histogram->FindCorruption(*snapshot)); | 
 |   snapshot->counts_[2] -= 200; | 
 |   EXPECT_EQ(HistogramBase::COUNT_HIGH_ERROR, | 
 |             histogram->FindCorruption(*snapshot)); | 
 |  | 
 |   // But we can't spot a corruption if it is compensated for. | 
 |   snapshot->counts_[1] += 100; | 
 |   EXPECT_EQ(HistogramBase::NO_INCONSISTENCIES, | 
 |             histogram->FindCorruption(*snapshot)); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, CorruptBucketBounds) { | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       Histogram::FactoryGet("Histogram", 1, 64, 8, HistogramBase::kNoFlags)); | 
 |  | 
 |   scoped_ptr<SampleVector> snapshot = histogram->SnapshotSampleVector(); | 
 |   EXPECT_EQ(HistogramBase::NO_INCONSISTENCIES, | 
 |             histogram->FindCorruption(*snapshot)); | 
 |  | 
 |   BucketRanges* bucket_ranges = | 
 |       const_cast<BucketRanges*>(histogram->bucket_ranges()); | 
 |   HistogramBase::Sample tmp = bucket_ranges->range(1); | 
 |   bucket_ranges->set_range(1, bucket_ranges->range(2)); | 
 |   bucket_ranges->set_range(2, tmp); | 
 |   EXPECT_EQ( | 
 |       HistogramBase::BUCKET_ORDER_ERROR | HistogramBase::RANGE_CHECKSUM_ERROR, | 
 |       histogram->FindCorruption(*snapshot)); | 
 |  | 
 |   bucket_ranges->set_range(2, bucket_ranges->range(1)); | 
 |   bucket_ranges->set_range(1, tmp); | 
 |   EXPECT_EQ(0, histogram->FindCorruption(*snapshot)); | 
 |  | 
 |   // Show that two simple changes don't offset each other | 
 |   bucket_ranges->set_range(3, bucket_ranges->range(3) + 1); | 
 |   EXPECT_EQ(HistogramBase::RANGE_CHECKSUM_ERROR, | 
 |             histogram->FindCorruption(*snapshot)); | 
 |  | 
 |   bucket_ranges->set_range(4, bucket_ranges->range(4) - 1); | 
 |   EXPECT_EQ(HistogramBase::RANGE_CHECKSUM_ERROR, | 
 |             histogram->FindCorruption(*snapshot)); | 
 |  | 
 |   // Repair histogram so that destructor won't DCHECK(). | 
 |   bucket_ranges->set_range(3, bucket_ranges->range(3) - 1); | 
 |   bucket_ranges->set_range(4, bucket_ranges->range(4) + 1); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, HistogramSerializeInfo) { | 
 |   Histogram* histogram = static_cast<Histogram*>( | 
 |       Histogram::FactoryGet("Histogram", 1, 64, 8, | 
 |                             HistogramBase::kIPCSerializationSourceFlag)); | 
 |   Pickle pickle; | 
 |   histogram->SerializeInfo(&pickle); | 
 |  | 
 |   PickleIterator iter(pickle); | 
 |  | 
 |   int type; | 
 |   EXPECT_TRUE(iter.ReadInt(&type)); | 
 |   EXPECT_EQ(HISTOGRAM, type); | 
 |  | 
 |   std::string name; | 
 |   EXPECT_TRUE(iter.ReadString(&name)); | 
 |   EXPECT_EQ("Histogram", name); | 
 |  | 
 |   int flag; | 
 |   EXPECT_TRUE(iter.ReadInt(&flag)); | 
 |   EXPECT_EQ(HistogramBase::kIPCSerializationSourceFlag, flag); | 
 |  | 
 |   int min; | 
 |   EXPECT_TRUE(iter.ReadInt(&min)); | 
 |   EXPECT_EQ(1, min); | 
 |  | 
 |   int max; | 
 |   EXPECT_TRUE(iter.ReadInt(&max)); | 
 |   EXPECT_EQ(64, max); | 
 |  | 
 |   int64 bucket_count; | 
 |   EXPECT_TRUE(iter.ReadInt64(&bucket_count)); | 
 |   EXPECT_EQ(8, bucket_count); | 
 |  | 
 |   uint32 checksum; | 
 |   EXPECT_TRUE(iter.ReadUInt32(&checksum)); | 
 |   EXPECT_EQ(histogram->bucket_ranges()->checksum(), checksum); | 
 |  | 
 |   // No more data in the pickle. | 
 |   EXPECT_FALSE(iter.SkipBytes(1)); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, CustomHistogramSerializeInfo) { | 
 |   std::vector<int> custom_ranges; | 
 |   custom_ranges.push_back(10); | 
 |   custom_ranges.push_back(100); | 
 |  | 
 |   HistogramBase* custom_histogram = CustomHistogram::FactoryGet( | 
 |       "TestCustomRangeBoundedHistogram", | 
 |       custom_ranges, | 
 |       HistogramBase::kNoFlags); | 
 |   Pickle pickle; | 
 |   custom_histogram->SerializeInfo(&pickle); | 
 |  | 
 |   // Validate the pickle. | 
 |   PickleIterator iter(pickle); | 
 |  | 
 |   int i; | 
 |   std::string s; | 
 |   int64 bucket_count; | 
 |   uint32 ui32; | 
 |   EXPECT_TRUE(iter.ReadInt(&i) && iter.ReadString(&s) && iter.ReadInt(&i) && | 
 |               iter.ReadInt(&i) && iter.ReadInt(&i) && | 
 |               iter.ReadInt64(&bucket_count) && iter.ReadUInt32(&ui32)); | 
 |   EXPECT_EQ(3, bucket_count); | 
 |  | 
 |   int range; | 
 |   EXPECT_TRUE(iter.ReadInt(&range)); | 
 |   EXPECT_EQ(10, range); | 
 |   EXPECT_TRUE(iter.ReadInt(&range)); | 
 |   EXPECT_EQ(100, range); | 
 |  | 
 |   // No more data in the pickle. | 
 |   EXPECT_FALSE(iter.SkipBytes(1)); | 
 | } | 
 |  | 
 | TEST_F(HistogramTest, BadConstruction) { | 
 |   HistogramBase* histogram = Histogram::FactoryGet( | 
 |       "BadConstruction", 0, 100, 8, HistogramBase::kNoFlags); | 
 |   EXPECT_TRUE(histogram->HasConstructionArguments(1, 100, 8)); | 
 |  | 
 |   // Try to get the same histogram name with different arguments. | 
 |   HistogramBase* bad_histogram = Histogram::FactoryGet( | 
 |       "BadConstruction", 0, 100, 7, HistogramBase::kNoFlags); | 
 |   EXPECT_EQ(NULL, bad_histogram); | 
 |   bad_histogram = Histogram::FactoryGet( | 
 |       "BadConstruction", 0, 99, 8, HistogramBase::kNoFlags); | 
 |   EXPECT_EQ(NULL, bad_histogram); | 
 |  | 
 |   HistogramBase* linear_histogram = LinearHistogram::FactoryGet( | 
 |       "BadConstructionLinear", 0, 100, 8, HistogramBase::kNoFlags); | 
 |   EXPECT_TRUE(linear_histogram->HasConstructionArguments(1, 100, 8)); | 
 |  | 
 |   // Try to get the same histogram name with different arguments. | 
 |   bad_histogram = LinearHistogram::FactoryGet( | 
 |       "BadConstructionLinear", 0, 100, 7, HistogramBase::kNoFlags); | 
 |   EXPECT_EQ(NULL, bad_histogram); | 
 |   bad_histogram = LinearHistogram::FactoryGet( | 
 |       "BadConstructionLinear", 10, 100, 8, HistogramBase::kNoFlags); | 
 |   EXPECT_EQ(NULL, bad_histogram); | 
 | } | 
 |  | 
 | #if GTEST_HAS_DEATH_TEST | 
 | // For Histogram, LinearHistogram and CustomHistogram, the minimum for a | 
 | // declared range is 1, while the maximum is (HistogramBase::kSampleType_MAX - | 
 | // 1). But we accept ranges exceeding those limits, and silently clamped to | 
 | // those limits. This is for backwards compatibility. | 
 | TEST(HistogramDeathTest, BadRangesTest) { | 
 |   HistogramBase* histogram = Histogram::FactoryGet( | 
 |       "BadRanges", 0, HistogramBase::kSampleType_MAX, 8, | 
 |       HistogramBase::kNoFlags); | 
 |   EXPECT_TRUE( | 
 |       histogram->HasConstructionArguments( | 
 |           1, HistogramBase::kSampleType_MAX - 1, 8)); | 
 |  | 
 |   HistogramBase* linear_histogram = LinearHistogram::FactoryGet( | 
 |       "BadRangesLinear", 0, HistogramBase::kSampleType_MAX, 8, | 
 |       HistogramBase::kNoFlags); | 
 |   EXPECT_TRUE( | 
 |       linear_histogram->HasConstructionArguments( | 
 |           1, HistogramBase::kSampleType_MAX - 1, 8)); | 
 |  | 
 |   std::vector<int> custom_ranges; | 
 |   custom_ranges.push_back(0); | 
 |   custom_ranges.push_back(5); | 
 |   Histogram* custom_histogram = static_cast<Histogram*>( | 
 |       CustomHistogram::FactoryGet( | 
 |           "BadRangesCustom", custom_ranges, HistogramBase::kNoFlags)); | 
 |   const BucketRanges* ranges = custom_histogram->bucket_ranges(); | 
 |   ASSERT_EQ(3u, ranges->size()); | 
 |   EXPECT_EQ(0, ranges->range(0)); | 
 |   EXPECT_EQ(5, ranges->range(1)); | 
 |   EXPECT_EQ(HistogramBase::kSampleType_MAX, ranges->range(2)); | 
 |  | 
 |   // CustomHistogram does not accepts kSampleType_MAX as range. | 
 |   custom_ranges.push_back(HistogramBase::kSampleType_MAX); | 
 |   EXPECT_DEATH(CustomHistogram::FactoryGet("BadRangesCustom2", custom_ranges, | 
 |                                            HistogramBase::kNoFlags), | 
 |                ""); | 
 |  | 
 |   // CustomHistogram needs at least 1 valid range. | 
 |   custom_ranges.clear(); | 
 |   custom_ranges.push_back(0); | 
 |   EXPECT_DEATH(CustomHistogram::FactoryGet("BadRangesCustom3", custom_ranges, | 
 |                                            HistogramBase::kNoFlags), | 
 |                ""); | 
 | } | 
 | #endif | 
 |  | 
 | }  // namespace base |