blob: f18a579391c2f95bdfb2bd2e03ef94b7953acc19 [file] [log] [blame]
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001#!/usr/bin/env python
2#
3# Copyright 2015 the V8 project authors. All rights reserved.
4# Use of this source code is governed by a BSD-style license that can be
5# found in the LICENSE file.
6
7"""This script is used to analyze GCTracer's NVP output."""
8
9
10from argparse import ArgumentParser
11from copy import deepcopy
12from gc_nvp_common import split_nvp
13from math import log
14from sys import stdin
15
16
17class LinearBucket:
18 def __init__(self, granularity):
19 self.granularity = granularity
20
21 def value_to_bucket(self, value):
22 return int(value / self.granularity)
23
24 def bucket_to_range(self, bucket):
25 return (bucket * self.granularity, (bucket + 1) * self.granularity)
26
27
28class Log2Bucket:
29 def __init__(self, start):
30 self.start = int(log(start, 2)) - 1
31
32 def value_to_bucket(self, value):
33 index = int(log(value, 2))
34 index -= self.start
35 if index < 0:
36 index = 0
37 return index
38
39 def bucket_to_range(self, bucket):
40 if bucket == 0:
41 return (0, 2 ** (self.start + 1))
42 bucket += self.start
43 return (2 ** bucket, 2 ** (bucket + 1))
44
45
46class Histogram:
47 def __init__(self, bucket_trait, fill_empty):
48 self.histogram = {}
49 self.fill_empty = fill_empty
50 self.bucket_trait = bucket_trait
51
52 def add(self, key):
53 index = self.bucket_trait.value_to_bucket(key)
54 if index not in self.histogram:
55 self.histogram[index] = 0
56 self.histogram[index] += 1
57
58 def __str__(self):
59 ret = []
60 keys = self.histogram.keys()
61 keys.sort()
62 last = keys[len(keys) - 1]
63 for i in range(0, last + 1):
64 (min_value, max_value) = self.bucket_trait.bucket_to_range(i)
65 if i == keys[0]:
66 keys.pop(0)
67 ret.append(" [{0},{1}[: {2}".format(
68 str(min_value), str(max_value), self.histogram[i]))
69 else:
70 if self.fill_empty:
71 ret.append(" [{0},{1}[: {2}".format(
72 str(min_value), str(max_value), 0))
73 return "\n".join(ret)
74
75
76class Category:
77 def __init__(self, key, histogram):
78 self.key = key
79 self.values = []
80 self.histogram = histogram
81
82 def process_entry(self, entry):
83 if self.key in entry:
84 self.values.append(float(entry[self.key]))
85 if self.histogram:
86 self.histogram.add(float(entry[self.key]))
87
88 def min(self):
89 return min(self.values)
90
91 def max(self):
92 return max(self.values)
93
94 def avg(self):
95 return sum(self.values) / len(self.values)
96
97 def __str__(self):
98 ret = [self.key]
99 ret.append(" len: {0}".format(len(self.values)))
100 if len(self.values) > 0:
101 ret.append(" min: {0}".format(min(self.values)))
102 ret.append(" max: {0}".format(max(self.values)))
103 ret.append(" avg: {0}".format(sum(self.values) / len(self.values)))
104 if self.histogram:
105 ret.append(str(self.histogram))
106 return "\n".join(ret)
107
108 def __repr__(self):
109 return "<Category: {0}>".format(self.key)
110
111
112def make_key_func(cmp_metric):
113 def key_func(a):
114 return getattr(a, cmp_metric)()
115 return key_func
116
117
118def main():
119 parser = ArgumentParser(description="Process GCTracer's NVP output")
120 parser.add_argument('keys', metavar='KEY', type=str, nargs='+',
121 help='the keys of NVPs to process')
122 parser.add_argument('--histogram-type', metavar='<linear|log2>',
123 type=str, nargs='?', default="linear",
124 help='histogram type to use (default: linear)')
125 linear_group = parser.add_argument_group('linear histogram specific')
126 linear_group.add_argument('--linear-histogram-granularity',
127 metavar='GRANULARITY', type=int, nargs='?',
128 default=5,
129 help='histogram granularity (default: 5)')
130 log2_group = parser.add_argument_group('log2 histogram specific')
131 log2_group.add_argument('--log2-histogram-init-bucket', metavar='START',
132 type=int, nargs='?', default=64,
133 help='initial buck size (default: 64)')
134 parser.add_argument('--histogram-omit-empty-buckets',
135 dest='histogram_omit_empty',
136 action='store_true',
137 help='omit empty histogram buckets')
138 parser.add_argument('--no-histogram', dest='histogram',
139 action='store_false', help='do not print histogram')
140 parser.set_defaults(histogram=True)
141 parser.set_defaults(histogram_omit_empty=False)
142 parser.add_argument('--rank', metavar='<no|min|max|avg>',
143 type=str, nargs='?',
144 default="no",
145 help="rank keys by metric (default: no)")
146 args = parser.parse_args()
147
148 histogram = None
149 if args.histogram:
150 bucket_trait = None
151 if args.histogram_type == "log2":
152 bucket_trait = Log2Bucket(args.log2_histogram_init_bucket)
153 else:
154 bucket_trait = LinearBucket(args.linear_histogram_granularity)
155 histogram = Histogram(bucket_trait, not args.histogram_omit_empty)
156
157 categories = [ Category(key, deepcopy(histogram))
158 for key in args.keys ]
159
160 while True:
161 line = stdin.readline()
162 if not line:
163 break
164 obj = split_nvp(line)
165 for category in categories:
166 category.process_entry(obj)
167
168 if args.rank != "no":
169 categories = sorted(categories, key=make_key_func(args.rank), reverse=True)
170
171 for category in categories:
172 print(category)
173
174
175if __name__ == '__main__':
176 main()