Andreas Gampe | dbfe254 | 2014-11-25 22:21:42 -0800 | [diff] [blame] | 1 | #!/usr/bin/env python |
| 2 | # |
| 3 | # Copyright (C) 2014 The Android Open Source Project |
| 4 | # |
| 5 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 6 | # you may not use this file except in compliance with the License. |
| 7 | # You may obtain a copy of the License at |
| 8 | # |
| 9 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 10 | # |
| 11 | # Unless required by applicable law or agreed to in writing, software |
| 12 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 13 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 14 | # See the License for the specific language governing permissions and |
| 15 | # limitations under the License. |
| 16 | |
| 17 | """Analyzes the dump of initialization failures and creates a Graphviz dot file |
| 18 | representing dependencies.""" |
| 19 | |
| 20 | import codecs |
| 21 | import os |
| 22 | import re |
| 23 | import string |
| 24 | import sys |
| 25 | |
| 26 | |
| 27 | _CLASS_RE = re.compile(r'^L(.*);$') |
Andreas Gampe | e9b160e | 2015-04-10 13:06:22 -0700 | [diff] [blame] | 28 | _ERROR_LINE_RE = re.compile(r'^dalvik.system.TransactionAbortError: (.*)') |
Andreas Gampe | dbfe254 | 2014-11-25 22:21:42 -0800 | [diff] [blame] | 29 | _STACK_LINE_RE = re.compile(r'^\s*at\s[^\s]*\s([^\s]*)') |
| 30 | |
| 31 | def Confused(filename, line_number, line): |
| 32 | sys.stderr.write('%s:%d: confused by:\n%s\n' % (filename, line_number, line)) |
| 33 | raise Exception("giving up!") |
| 34 | sys.exit(1) |
| 35 | |
| 36 | |
| 37 | def ProcessFile(filename): |
| 38 | lines = codecs.open(filename, 'r', 'utf8', 'replace').read().split('\n') |
| 39 | it = iter(lines) |
| 40 | |
| 41 | class_fail_class = {} |
| 42 | class_fail_method = {} |
Andreas Gampe | 869c2dfa | 2015-03-09 10:36:54 -0700 | [diff] [blame] | 43 | class_fail_load_library = {} |
| 44 | class_fail_get_property = {} |
Andreas Gampe | dbfe254 | 2014-11-25 22:21:42 -0800 | [diff] [blame] | 45 | root_failures = set() |
| 46 | root_errors = {} |
| 47 | |
| 48 | while True: |
| 49 | try: |
| 50 | # We start with a class descriptor. |
| 51 | raw_line = it.next() |
| 52 | m = _CLASS_RE.search(raw_line) |
| 53 | # print(raw_line) |
| 54 | if m is None: |
| 55 | continue |
| 56 | # Found a class. |
| 57 | failed_clazz = m.group(1).replace('/','.') |
| 58 | # print('Is a class %s' % failed_clazz) |
| 59 | # The error line should be next. |
| 60 | raw_line = it.next() |
| 61 | m = _ERROR_LINE_RE.search(raw_line) |
| 62 | # print(raw_line) |
| 63 | if m is None: |
| 64 | Confused(filename, -1, raw_line) |
| 65 | continue |
| 66 | # Found an error line. |
| 67 | error = m.group(1) |
| 68 | # print('Is an error %s' % error) |
| 69 | # Get the top of the stack |
| 70 | raw_line = it.next() |
| 71 | m = _STACK_LINE_RE.search(raw_line) |
| 72 | if m is None: |
| 73 | continue |
| 74 | # Found a stack line. Get the method. |
| 75 | method = m.group(1) |
| 76 | # print('Is a stack element %s' % method) |
| 77 | (left_of_paren,paren,right_of_paren) = method.partition('(') |
| 78 | (root_err_class,dot,root_method_name) = left_of_paren.rpartition('.') |
| 79 | # print('Error class %s' % err_class) |
| 80 | # print('Error method %s' % method_name) |
| 81 | # Record the root error. |
| 82 | root_failures.add(root_err_class) |
| 83 | # Parse all the trace elements to find the "immediate" cause. |
| 84 | immediate_class = root_err_class |
| 85 | immediate_method = root_method_name |
| 86 | root_errors[root_err_class] = error |
Andreas Gampe | 869c2dfa | 2015-03-09 10:36:54 -0700 | [diff] [blame] | 87 | was_load_library = False |
| 88 | was_get_property = False |
Andreas Gampe | dbfe254 | 2014-11-25 22:21:42 -0800 | [diff] [blame] | 89 | # Now go "up" the stack. |
| 90 | while True: |
| 91 | raw_line = it.next() |
| 92 | m = _STACK_LINE_RE.search(raw_line) |
| 93 | if m is None: |
| 94 | break # Nothing more to see here. |
| 95 | method = m.group(1) |
| 96 | (left_of_paren,paren,right_of_paren) = method.partition('(') |
| 97 | (err_class,dot,err_method_name) = left_of_paren.rpartition('.') |
| 98 | if err_method_name == "<clinit>": |
| 99 | # A class initializer is on the stack... |
| 100 | class_fail_class[err_class] = immediate_class |
| 101 | class_fail_method[err_class] = immediate_method |
Andreas Gampe | 869c2dfa | 2015-03-09 10:36:54 -0700 | [diff] [blame] | 102 | class_fail_load_library[err_class] = was_load_library |
Andreas Gampe | dbfe254 | 2014-11-25 22:21:42 -0800 | [diff] [blame] | 103 | immediate_class = err_class |
| 104 | immediate_method = err_method_name |
Andreas Gampe | 869c2dfa | 2015-03-09 10:36:54 -0700 | [diff] [blame] | 105 | class_fail_get_property[err_class] = was_get_property |
| 106 | was_get_property = False |
| 107 | was_load_library = err_method_name == "loadLibrary" |
| 108 | was_get_property = was_get_property or err_method_name == "getProperty" |
| 109 | failed_clazz_norm = re.sub(r"^L", "", failed_clazz) |
| 110 | failed_clazz_norm = re.sub(r";$", "", failed_clazz_norm) |
| 111 | failed_clazz_norm = re.sub(r"/", "", failed_clazz_norm) |
| 112 | if immediate_class != failed_clazz_norm: |
| 113 | class_fail_class[failed_clazz_norm] = immediate_class |
| 114 | class_fail_method[failed_clazz_norm] = immediate_method |
Andreas Gampe | dbfe254 | 2014-11-25 22:21:42 -0800 | [diff] [blame] | 115 | except StopIteration: |
| 116 | # print('Done') |
| 117 | break # Done |
| 118 | |
| 119 | # Assign IDs. |
| 120 | fail_sources = set(class_fail_class.values()); |
| 121 | all_classes = fail_sources | set(class_fail_class.keys()) |
| 122 | i = 0 |
| 123 | class_index = {} |
| 124 | for clazz in all_classes: |
| 125 | class_index[clazz] = i |
| 126 | i = i + 1 |
| 127 | |
| 128 | # Now create the nodes. |
| 129 | for (r_class, r_id) in class_index.items(): |
| 130 | error_string = '' |
| 131 | if r_class in root_failures: |
Andreas Gampe | 869c2dfa | 2015-03-09 10:36:54 -0700 | [diff] [blame] | 132 | error_string = ',style=filled,fillcolor=Red,tooltip="' + root_errors[r_class] + '",URL="' + root_errors[r_class] + '"' |
| 133 | elif r_class in class_fail_load_library and class_fail_load_library[r_class] == True: |
| 134 | error_string = error_string + ',style=filled,fillcolor=Bisque' |
| 135 | elif r_class in class_fail_get_property and class_fail_get_property[r_class] == True: |
| 136 | error_string = error_string + ',style=filled,fillcolor=Darkseagreen' |
Andreas Gampe | dbfe254 | 2014-11-25 22:21:42 -0800 | [diff] [blame] | 137 | print(' n%d [shape=box,label="%s"%s];' % (r_id, r_class, error_string)) |
| 138 | |
| 139 | # Some space. |
| 140 | print('') |
| 141 | |
| 142 | # Connections. |
| 143 | for (failed_class,error_class) in class_fail_class.items(): |
| 144 | print(' n%d -> n%d;' % (class_index[failed_class], class_index[error_class])) |
| 145 | |
| 146 | |
| 147 | def main(): |
| 148 | print('digraph {') |
| 149 | print(' overlap=false;') |
| 150 | print(' splines=true;') |
| 151 | ProcessFile(sys.argv[1]) |
| 152 | print('}') |
| 153 | sys.exit(0) |
| 154 | |
| 155 | |
| 156 | if __name__ == '__main__': |
| 157 | main() |