1*cc02d7e2SAndroid Build Coastguard Worker#!/usr/bin/env ruby 2*cc02d7e2SAndroid Build Coastguard Worker 3*cc02d7e2SAndroid Build Coastguard Worker# Copyright 2016 gRPC authors. 4*cc02d7e2SAndroid Build Coastguard Worker# 5*cc02d7e2SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 6*cc02d7e2SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 7*cc02d7e2SAndroid Build Coastguard Worker# You may obtain a copy of the License at 8*cc02d7e2SAndroid Build Coastguard Worker# 9*cc02d7e2SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 10*cc02d7e2SAndroid Build Coastguard Worker# 11*cc02d7e2SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 12*cc02d7e2SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 13*cc02d7e2SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14*cc02d7e2SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 15*cc02d7e2SAndroid Build Coastguard Worker# limitations under the License. 16*cc02d7e2SAndroid Build Coastguard Worker 17*cc02d7e2SAndroid Build Coastguard Worker# Histogram class for use in performance testing and measurement 18*cc02d7e2SAndroid Build Coastguard Worker 19*cc02d7e2SAndroid Build Coastguard Workerrequire 'thread' 20*cc02d7e2SAndroid Build Coastguard Worker 21*cc02d7e2SAndroid Build Coastguard Workerclass Histogram 22*cc02d7e2SAndroid Build Coastguard Worker # Determine the bucket index for a given value 23*cc02d7e2SAndroid Build Coastguard Worker # @param {number} value The value to check 24*cc02d7e2SAndroid Build Coastguard Worker # @return {number} The bucket index 25*cc02d7e2SAndroid Build Coastguard Worker def bucket_for(value) 26*cc02d7e2SAndroid Build Coastguard Worker (Math.log(value)/Math.log(@multiplier)).to_i 27*cc02d7e2SAndroid Build Coastguard Worker end 28*cc02d7e2SAndroid Build Coastguard Worker # Initialize an empty histogram 29*cc02d7e2SAndroid Build Coastguard Worker # @param {number} resolution The resolution of the histogram 30*cc02d7e2SAndroid Build Coastguard Worker # @param {number} max_possible The maximum value for the histogram 31*cc02d7e2SAndroid Build Coastguard Worker def initialize(resolution, max_possible) 32*cc02d7e2SAndroid Build Coastguard Worker @lock = Mutex.new 33*cc02d7e2SAndroid Build Coastguard Worker @resolution=resolution 34*cc02d7e2SAndroid Build Coastguard Worker @max_possible=max_possible 35*cc02d7e2SAndroid Build Coastguard Worker @sum=0 36*cc02d7e2SAndroid Build Coastguard Worker @sum_of_squares=0 37*cc02d7e2SAndroid Build Coastguard Worker @multiplier=1+resolution 38*cc02d7e2SAndroid Build Coastguard Worker @count=0 39*cc02d7e2SAndroid Build Coastguard Worker @min_seen=max_possible 40*cc02d7e2SAndroid Build Coastguard Worker @max_seen=0 41*cc02d7e2SAndroid Build Coastguard Worker @buckets=Array.new(bucket_for(max_possible)+1, 0) 42*cc02d7e2SAndroid Build Coastguard Worker end 43*cc02d7e2SAndroid Build Coastguard Worker # Add a value to the histogram. This updates all statistics with the new 44*cc02d7e2SAndroid Build Coastguard Worker # value. Those statistics should not be modified except with this function 45*cc02d7e2SAndroid Build Coastguard Worker # @param {number} value The value to add 46*cc02d7e2SAndroid Build Coastguard Worker def add(value) 47*cc02d7e2SAndroid Build Coastguard Worker @sum += value 48*cc02d7e2SAndroid Build Coastguard Worker @sum_of_squares += value * value 49*cc02d7e2SAndroid Build Coastguard Worker @count += 1 50*cc02d7e2SAndroid Build Coastguard Worker if value < @min_seen 51*cc02d7e2SAndroid Build Coastguard Worker @min_seen = value 52*cc02d7e2SAndroid Build Coastguard Worker end 53*cc02d7e2SAndroid Build Coastguard Worker if value > @max_seen 54*cc02d7e2SAndroid Build Coastguard Worker @max_seen = value 55*cc02d7e2SAndroid Build Coastguard Worker end 56*cc02d7e2SAndroid Build Coastguard Worker @buckets[bucket_for(value)] += 1 57*cc02d7e2SAndroid Build Coastguard Worker end 58*cc02d7e2SAndroid Build Coastguard Worker def minimum 59*cc02d7e2SAndroid Build Coastguard Worker @min_seen 60*cc02d7e2SAndroid Build Coastguard Worker end 61*cc02d7e2SAndroid Build Coastguard Worker def maximum 62*cc02d7e2SAndroid Build Coastguard Worker @max_seen 63*cc02d7e2SAndroid Build Coastguard Worker end 64*cc02d7e2SAndroid Build Coastguard Worker def sum 65*cc02d7e2SAndroid Build Coastguard Worker @sum 66*cc02d7e2SAndroid Build Coastguard Worker end 67*cc02d7e2SAndroid Build Coastguard Worker def sum_of_squares 68*cc02d7e2SAndroid Build Coastguard Worker @sum_of_squares 69*cc02d7e2SAndroid Build Coastguard Worker end 70*cc02d7e2SAndroid Build Coastguard Worker def count 71*cc02d7e2SAndroid Build Coastguard Worker @count 72*cc02d7e2SAndroid Build Coastguard Worker end 73*cc02d7e2SAndroid Build Coastguard Worker def contents 74*cc02d7e2SAndroid Build Coastguard Worker @buckets 75*cc02d7e2SAndroid Build Coastguard Worker end 76*cc02d7e2SAndroid Build Coastguard Worker 77*cc02d7e2SAndroid Build Coastguard Worker def merge(hist) 78*cc02d7e2SAndroid Build Coastguard Worker @lock.synchronize do 79*cc02d7e2SAndroid Build Coastguard Worker @min_seen = hist.min_seen 80*cc02d7e2SAndroid Build Coastguard Worker @max_seen = hist.max_seen 81*cc02d7e2SAndroid Build Coastguard Worker @sum += hist.sum 82*cc02d7e2SAndroid Build Coastguard Worker @sum_of_squares += hist.sum_of_squares 83*cc02d7e2SAndroid Build Coastguard Worker @count += hist.count 84*cc02d7e2SAndroid Build Coastguard Worker received_bucket = hist.bucket.to_a 85*cc02d7e2SAndroid Build Coastguard Worker @buckets = @buckets.map.with_index{ |m,i| m + received_bucket[i].to_i } 86*cc02d7e2SAndroid Build Coastguard Worker end 87*cc02d7e2SAndroid Build Coastguard Worker end 88*cc02d7e2SAndroid Build Coastguard Workerend 89