#!/usr/bin/env python # # Copyright (C) 2014 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. """Analyzes the dump of initialization failures and creates a Graphviz dot file representing dependencies.""" import codecs import os import re import string import sys _CLASS_RE = re.compile(r'^L(.*);$') _ERROR_LINE_RE = re.compile(r'^dalvik.system.TransactionAbortError: (.*)') _STACK_LINE_RE = re.compile(r'^\s*at\s[^\s]*\s([^\s]*)') def Confused(filename, line_number, line): sys.stderr.write('%s:%d: confused by:\n%s\n' % (filename, line_number, line)) raise Exception("giving up!") sys.exit(1) def ProcessFile(filename): lines = codecs.open(filename, 'r', 'utf8', 'replace').read().split('\n') it = iter(lines) class_fail_class = {} class_fail_method = {} class_fail_load_library = {} class_fail_get_property = {} root_failures = set() root_errors = {} while True: try: # We start with a class descriptor. raw_line = it.next() m = _CLASS_RE.search(raw_line) # print(raw_line) if m is None: continue # Found a class. failed_clazz = m.group(1).replace('/','.') # print('Is a class %s' % failed_clazz) # The error line should be next. raw_line = it.next() m = _ERROR_LINE_RE.search(raw_line) # print(raw_line) if m is None: Confused(filename, -1, raw_line) continue # Found an error line. error = m.group(1) # print('Is an error %s' % error) # Get the top of the stack raw_line = it.next() m = _STACK_LINE_RE.search(raw_line) if m is None: continue # Found a stack line. Get the method. method = m.group(1) # print('Is a stack element %s' % method) (left_of_paren,paren,right_of_paren) = method.partition('(') (root_err_class,dot,root_method_name) = left_of_paren.rpartition('.') # print('Error class %s' % err_class) # print('Error method %s' % method_name) # Record the root error. root_failures.add(root_err_class) # Parse all the trace elements to find the "immediate" cause. immediate_class = root_err_class immediate_method = root_method_name root_errors[root_err_class] = error was_load_library = False was_get_property = False # Now go "up" the stack. while True: raw_line = it.next() m = _STACK_LINE_RE.search(raw_line) if m is None: break # Nothing more to see here. method = m.group(1) (left_of_paren,paren,right_of_paren) = method.partition('(') (err_class,dot,err_method_name) = left_of_paren.rpartition('.') if err_method_name == "<clinit>": # A class initializer is on the stack... class_fail_class[err_class] = immediate_class class_fail_method[err_class] = immediate_method class_fail_load_library[err_class] = was_load_library immediate_class = err_class immediate_method = err_method_name class_fail_get_property[err_class] = was_get_property was_get_property = False was_load_library = err_method_name == "loadLibrary" was_get_property = was_get_property or err_method_name == "getProperty" failed_clazz_norm = re.sub(r"^L", "", failed_clazz) failed_clazz_norm = re.sub(r";$", "", failed_clazz_norm) failed_clazz_norm = re.sub(r"/", "", failed_clazz_norm) if immediate_class != failed_clazz_norm: class_fail_class[failed_clazz_norm] = immediate_class class_fail_method[failed_clazz_norm] = immediate_method except StopIteration: # print('Done') break # Done # Assign IDs. fail_sources = set(class_fail_class.values()); all_classes = fail_sources | set(class_fail_class.keys()) i = 0 class_index = {} for clazz in all_classes: class_index[clazz] = i i = i + 1 # Now create the nodes. for (r_class, r_id) in class_index.items(): error_string = '' if r_class in root_failures: error_string = ',style=filled,fillcolor=Red,tooltip="' + root_errors[r_class] + '",URL="' + root_errors[r_class] + '"' elif r_class in class_fail_load_library and class_fail_load_library[r_class] == True: error_string = error_string + ',style=filled,fillcolor=Bisque' elif r_class in class_fail_get_property and class_fail_get_property[r_class] == True: error_string = error_string + ',style=filled,fillcolor=Darkseagreen' print(' n%d [shape=box,label="%s"%s];' % (r_id, r_class, error_string)) # Some space. print('') # Connections. for (failed_class,error_class) in class_fail_class.items(): print(' n%d -> n%d;' % (class_index[failed_class], class_index[error_class])) def main(): print('digraph {') print(' overlap=false;') print(' splines=true;') ProcessFile(sys.argv[1]) print('}') sys.exit(0) if __name__ == '__main__': main()