|  | // Copyright 2020 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package runtime_test | 
|  |  | 
|  | import ( | 
|  | "math" | 
|  | . "runtime" | 
|  | "testing" | 
|  | ) | 
|  |  | 
|  | var dummyTimeHistogram TimeHistogram | 
|  |  | 
|  | func TestTimeHistogram(t *testing.T) { | 
|  | // We need to use a global dummy because this | 
|  | // could get stack-allocated with a non-8-byte alignment. | 
|  | // The result of this bad alignment is a segfault on | 
|  | // 32-bit platforms when calling Record. | 
|  | h := &dummyTimeHistogram | 
|  |  | 
|  | // Record exactly one sample in each bucket. | 
|  | for j := 0; j < TimeHistNumSubBuckets; j++ { | 
|  | v := int64(j) << (TimeHistMinBucketBits - 1 - TimeHistSubBucketBits) | 
|  | for k := 0; k < j; k++ { | 
|  | // Record a number of times equal to the bucket index. | 
|  | h.Record(v) | 
|  | } | 
|  | } | 
|  | for i := TimeHistMinBucketBits; i < TimeHistMaxBucketBits; i++ { | 
|  | base := int64(1) << (i - 1) | 
|  | for j := 0; j < TimeHistNumSubBuckets; j++ { | 
|  | v := int64(j) << (i - 1 - TimeHistSubBucketBits) | 
|  | for k := 0; k < (i+1-TimeHistMinBucketBits)*TimeHistNumSubBuckets+j; k++ { | 
|  | // Record a number of times equal to the bucket index. | 
|  | h.Record(base + v) | 
|  | } | 
|  | } | 
|  | } | 
|  | // Hit the underflow and overflow buckets. | 
|  | h.Record(int64(-1)) | 
|  | h.Record(math.MaxInt64) | 
|  | h.Record(math.MaxInt64) | 
|  |  | 
|  | // Check to make sure there's exactly one count in each | 
|  | // bucket. | 
|  | for i := 0; i < TimeHistNumBuckets; i++ { | 
|  | for j := 0; j < TimeHistNumSubBuckets; j++ { | 
|  | c, ok := h.Count(i, j) | 
|  | if !ok { | 
|  | t.Errorf("unexpected invalid bucket: (%d, %d)", i, j) | 
|  | } else if idx := uint64(i*TimeHistNumSubBuckets + j); c != idx { | 
|  | t.Errorf("bucket (%d, %d) has count that is not %d: %d", i, j, idx, c) | 
|  | } | 
|  | } | 
|  | } | 
|  | c, ok := h.Count(-1, 0) | 
|  | if ok { | 
|  | t.Errorf("expected to hit underflow bucket: (%d, %d)", -1, 0) | 
|  | } | 
|  | if c != 1 { | 
|  | t.Errorf("overflow bucket has count that is not 1: %d", c) | 
|  | } | 
|  |  | 
|  | c, ok = h.Count(TimeHistNumBuckets+1, 0) | 
|  | if ok { | 
|  | t.Errorf("expected to hit overflow bucket: (%d, %d)", TimeHistNumBuckets+1, 0) | 
|  | } | 
|  | if c != 2 { | 
|  | t.Errorf("overflow bucket has count that is not 2: %d", c) | 
|  | } | 
|  |  | 
|  | dummyTimeHistogram = TimeHistogram{} | 
|  | } | 
|  |  | 
|  | func TestTimeHistogramMetricsBuckets(t *testing.T) { | 
|  | buckets := TimeHistogramMetricsBuckets() | 
|  |  | 
|  | nonInfBucketsLen := TimeHistNumSubBuckets * TimeHistNumBuckets | 
|  | expBucketsLen := nonInfBucketsLen + 3 // Count -Inf, the edge for the overflow bucket, and +Inf. | 
|  | if len(buckets) != expBucketsLen { | 
|  | t.Fatalf("unexpected length of buckets: got %d, want %d", len(buckets), expBucketsLen) | 
|  | } | 
|  | // Check some values. | 
|  | idxToBucket := map[int]float64{ | 
|  | 0:                 math.Inf(-1), | 
|  | 1:                 0.0, | 
|  | 2:                 float64(0x040) / 1e9, | 
|  | 3:                 float64(0x080) / 1e9, | 
|  | 4:                 float64(0x0c0) / 1e9, | 
|  | 5:                 float64(0x100) / 1e9, | 
|  | 6:                 float64(0x140) / 1e9, | 
|  | 7:                 float64(0x180) / 1e9, | 
|  | 8:                 float64(0x1c0) / 1e9, | 
|  | 9:                 float64(0x200) / 1e9, | 
|  | 10:                float64(0x280) / 1e9, | 
|  | 11:                float64(0x300) / 1e9, | 
|  | 12:                float64(0x380) / 1e9, | 
|  | 13:                float64(0x400) / 1e9, | 
|  | 15:                float64(0x600) / 1e9, | 
|  | 81:                float64(0x8000000) / 1e9, | 
|  | 82:                float64(0xa000000) / 1e9, | 
|  | 108:               float64(0x380000000) / 1e9, | 
|  | expBucketsLen - 2: float64(0x1<<47) / 1e9, | 
|  | expBucketsLen - 1: math.Inf(1), | 
|  | } | 
|  | for idx, bucket := range idxToBucket { | 
|  | if got, want := buckets[idx], bucket; got != want { | 
|  | t.Errorf("expected bucket %d to have value %e, got %e", idx, want, got) | 
|  | } | 
|  | } | 
|  | } |