OLD | NEW |
1 // Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 /* | 5 /* |
6 Usage: | 6 Usage: |
7 | 7 |
8 $ tools/test.py -m release \ | 8 $ tools/test.py -m release \ |
9 -c dart2js -r d8 --dart2js-batch --report \ | 9 -c dart2js -r d8 --dart2js-batch --report \ |
10 --host-checked \ | 10 --host-checked \ |
(...skipping 19 matching lines...) Expand all Loading... |
30 print(why); | 30 print(why); |
31 print('Usage:\n' | 31 print('Usage:\n' |
32 'dart rank_stacks.dart [options] test-logs\n\n' | 32 'dart rank_stacks.dart [options] test-logs\n\n' |
33 '${argParser.usage}'); | 33 '${argParser.usage}'); |
34 exit(1); | 34 exit(1); |
35 } | 35 } |
36 | 36 |
37 ArgParser argParser = new ArgParser() | 37 ArgParser argParser = new ArgParser() |
38 ..addOption('stacks', | 38 ..addOption('stacks', |
39 abbr: 's', | 39 abbr: 's', |
40 defaultsTo: '30', | 40 defaultsTo: '0', |
41 help: 'Number of highest ranking stacks to print.') | 41 help: 'Number of highest ranking stacks to print (0 for all).') |
42 ..addOption('length', | 42 ..addOption('length', |
43 abbr: 'l', defaultsTo: '12', help: 'Number of stack frames printed.'); | 43 abbr: 'l', defaultsTo: '12', help: 'Number of stack frames printed.'); |
44 | 44 |
45 int intOption(ArgResults args, String name) { | 45 int intOption(ArgResults args, String name) { |
46 onError(String text) { | 46 onError(String text) { |
47 die("Value '$text' is not an integer. " | 47 die("Value '$text' is not an integer. " |
48 "Option '$name' requires an integer value."); | 48 "Option '$name' requires an integer value."); |
49 } | 49 } |
50 | 50 |
51 return int.parse(args[name], onError: onError); | 51 return int.parse(args[name], onError: onError); |
(...skipping 15 matching lines...) Expand all Loading... |
67 for (String input in rest) { | 67 for (String input in rest) { |
68 var uri = Uri.base.resolve(input); | 68 var uri = Uri.base.resolve(input); |
69 var file = new File.fromUri(uri); | 69 var file = new File.fromUri(uri); |
70 if (!file.existsSync()) { | 70 if (!file.existsSync()) { |
71 die("File not found: '$input'."); | 71 die("File not found: '$input'."); |
72 } | 72 } |
73 String text = file.readAsStringSync(); | 73 String text = file.readAsStringSync(); |
74 records.addAll(parse(text)); | 74 records.addAll(parse(text)); |
75 } | 75 } |
76 | 76 |
77 var trie = new TrieNode(null); | 77 var trie = new TrieNode(null, 0); |
78 for (var record in records) { | 78 for (var record in records) { |
79 enter(record, 0, trie); | 79 enter(record, 0, trie); |
80 } | 80 } |
81 | 81 |
82 var leaves = trieLeaves(trie).toList(); | 82 var leaves = trieLeaves(trie).toList(); |
83 leaves.sort(compareNodesByCountAndStack); | 83 leaves.sort(compareNodesByCountAndStack); |
84 for (var leaf in howManyStacks == 0 ? leaves : leaves.take(howManyStacks)) { | 84 for (var leaf in howManyStacks == 0 ? leaves : leaves.take(howManyStacks)) { |
85 print(''); | 85 print(''); |
86 var examples = leaf.members.map((r) => r.fullReason).toSet().toList(); | 86 var examples = leaf.members.map(fullReasonOf).toSet().toList(); |
87 examples.sort(); | 87 examples.sort(); |
88 print('${leaf.length} of:'); | 88 print('${leaf.length} of:'); |
89 for (var example in examples) { | 89 for (var example in examples) { |
90 var count = leaf.members.where((r) => r.fullReason == example).length; | 90 var count = leaf.members.where((r) => fullReasonOf(r) == example).length; |
91 var countAligned = '$count'.padLeft(6); | 91 var countAligned = '$count'.padLeft(6); |
92 if (examples.length == 1) countAligned = ' .'; | 92 if (examples.length == 1) countAligned = ' .'; |
93 var indentedExample = '\t' + example.replaceAll('\n', '\n\t'); | 93 var indentedExample = '\t' + example.replaceAll('\n', '\n\t'); |
94 print('${countAligned}${indentedExample}'); | 94 print('${countAligned}${indentedExample}'); |
95 } | 95 } |
96 | 96 |
97 for (var line in leaf.members.first.stack.take(stackPrintLength)) { | 97 for (var line in leaf.members.first.stack.take(stackPrintLength)) { |
98 print(' $line'); | 98 print(' $line'); |
99 } | 99 } |
100 } | 100 } |
101 } | 101 } |
102 | 102 |
| 103 String fullReasonOf(Record r) { |
| 104 // Some records have no matched reason, so default to test status. |
| 105 return r.fullReason ?? r.actual; |
| 106 } |
| 107 |
103 int compareNodesByCountAndStack(TrieNode a, TrieNode b) { | 108 int compareNodesByCountAndStack(TrieNode a, TrieNode b) { |
104 int r = b.length.compareTo(a.length); | 109 int r = b.length.compareTo(a.length); |
105 if (r != 0) return r; | 110 if (r != 0) return r; |
106 List<String> stackA = a.members.first.stack; | 111 List<String> stackA = a.members.first.stack; |
107 List<String> stackB = b.members.first.stack; | 112 List<String> stackB = b.members.first.stack; |
108 int lengthA = stackA.length; | 113 int lengthA = stackA.length; |
109 int lengthB = stackB.length; | 114 int lengthB = stackB.length; |
110 for (int i = 0; i < lengthA && i < lengthB; i++) { | 115 for (int i = 0; i < lengthA && i < lengthB; i++) { |
111 r = stackA[i].compareTo(stackB[i]); | 116 r = stackA[i].compareTo(stackB[i]); |
112 if (r != 0) return r; | 117 if (r != 0) return r; |
113 } | 118 } |
114 return lengthA.compareTo(lengthB); | 119 return lengthA.compareTo(lengthB); |
115 } | 120 } |
116 | 121 |
117 class TrieNode { | 122 class TrieNode { |
118 final int depth; | 123 final int depth; |
119 final String key; | 124 final String key; |
120 final Map<String, TrieNode> map = <String, TrieNode>{}; | 125 final Map<String, TrieNode> map = <String, TrieNode>{}; |
121 final List<Record> members = <Record>[]; | 126 final List<Record> members = <Record>[]; |
122 | 127 |
123 int get length => members.length; | 128 int get length => members.length; |
124 | 129 |
125 TrieNode(this.key, [this.depth = 0]); | 130 TrieNode(this.key, this.depth); |
126 | 131 |
127 String toString() => 'TrieNode(#$length)'; | 132 String toString() => 'TrieNode(#$length)'; |
128 } | 133 } |
129 | 134 |
130 void enter(Record record, int depth, TrieNode root) { | 135 void enter(Record record, int depth, TrieNode root) { |
131 root.members.add(record); | 136 // Cluster on printed stack. |
132 if (depth >= record.stack.length) return; | 137 if (depth >= stackPrintLength || depth >= record.stack.length) { |
| 138 root.members.add(record); |
| 139 return; |
| 140 } |
133 var key = record.stack[depth]; | 141 var key = record.stack[depth]; |
134 var node = root.map[key] ??= new TrieNode(key, depth + 1); | 142 var node = root.map[key] ??= new TrieNode(key, depth + 1); |
135 enter(record, depth + 1, node); | 143 enter(record, depth + 1, node); |
136 } | 144 } |
137 | 145 |
138 void printTrie(TrieNode node) { | 146 void printTrie(TrieNode node) { |
139 var indent = ' ' * node.depth; | 147 var indent = ' ' * node.depth; |
140 print('${indent} ${node.length} ${node.key}'); | 148 print('${indent} ${node.length} ${node.key}'); |
141 for (var key in node.map.keys) { | 149 for (var key in node.map.keys) { |
142 printTrie(node.map[key]); | 150 printTrie(node.map[key]); |
143 } | 151 } |
144 } | 152 } |
145 | 153 |
146 trieLeaves(node) sync* { | 154 trieLeaves(node) sync* { |
147 if (node.map.isEmpty) { | 155 if (node.members.isNotEmpty) { |
148 yield node; | 156 yield node; |
149 } else { | 157 } |
150 for (var v in node.map.values) { | 158 for (var v in node.map.values) { |
151 yield* trieLeaves(v); | 159 yield* trieLeaves(v); |
152 } | |
153 } | 160 } |
154 } | 161 } |
OLD | NEW |