|  | #!/usr/bin/env python | 
|  |  | 
|  | # Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
|  | # Use of this source code is governed by a BSD-style license that can be | 
|  | # found in the LICENSE file. | 
|  |  | 
|  | from third_party import asan_symbolize | 
|  |  | 
|  | import os | 
|  | import sys | 
|  |  | 
|  | class LineBuffered(object): | 
|  | """Disable buffering on a file object.""" | 
|  | def __init__(self, stream): | 
|  | self.stream = stream | 
|  |  | 
|  | def write(self, data): | 
|  | self.stream.write(data) | 
|  | if '\n' in data: | 
|  | self.stream.flush() | 
|  |  | 
|  | def __getattr__(self, attr): | 
|  | return getattr(self.stream, attr) | 
|  |  | 
|  |  | 
|  | def disable_buffering(): | 
|  | """Makes this process and child processes stdout unbuffered.""" | 
|  | if not os.environ.get('PYTHONUNBUFFERED'): | 
|  | # Since sys.stdout is a C++ object, it's impossible to do | 
|  | # sys.stdout.write = lambda... | 
|  | sys.stdout = LineBuffered(sys.stdout) | 
|  | os.environ['PYTHONUNBUFFERED'] = 'x' | 
|  |  | 
|  |  | 
|  | def set_symbolizer_path(): | 
|  | """Set the path to the llvm-symbolize binary in the Chromium source tree.""" | 
|  | if not os.environ.get('LLVM_SYMBOLIZER_PATH'): | 
|  | script_dir = os.path.dirname(os.path.abspath(__file__)) | 
|  | # Assume this script resides three levels below src/ (i.e. | 
|  | # src/tools/valgrind/asan/). | 
|  | src_root = os.path.join(script_dir, "..", "..", "..") | 
|  | symbolizer_path = os.path.join(src_root, 'third_party', | 
|  | 'llvm-build', 'Release+Asserts', 'bin', 'llvm-symbolizer') | 
|  | assert(os.path.isfile(symbolizer_path)) | 
|  | os.environ['LLVM_SYMBOLIZER_PATH'] = os.path.abspath(symbolizer_path) | 
|  |  | 
|  |  | 
|  | # Construct a path to the .dSYM bundle for the given binary. | 
|  | # There are three possible cases for binary location in Chromium: | 
|  | # 1. The binary is a standalone executable or dynamic library in the product | 
|  | #    dir, the debug info is in "binary.dSYM" in the product dir. | 
|  | # 2. The binary is a standalone framework or .app bundle, the debug info is in | 
|  | #    "Framework.framework.dSYM" or "App.app.dSYM" in the product dir. | 
|  | # 3. The binary is a framework or an .app bundle within another .app bundle | 
|  | #    (e.g. Outer.app/Contents/Versions/1.2.3.4/Inner.app), and the debug info | 
|  | #    is in Inner.app.dSYM in the product dir. | 
|  | # The first case is handled by llvm-symbolizer, so we only need to construct | 
|  | # .dSYM paths for .app bundles and frameworks. | 
|  | # We're assuming that there're no more than two nested bundles in the binary | 
|  | # path. Only one of these bundles may be a framework and frameworks cannot | 
|  | # contain other bundles. | 
|  | def chrome_dsym_hints(binary): | 
|  | path_parts = binary.split(os.path.sep) | 
|  | app_positions = [] | 
|  | framework_positions = [] | 
|  | for index, part in enumerate(path_parts): | 
|  | if part.endswith('.app'): | 
|  | app_positions.append(index) | 
|  | elif part.endswith('.framework'): | 
|  | framework_positions.append(index) | 
|  | bundle_positions = app_positions + framework_positions | 
|  | bundle_positions.sort() | 
|  | assert len(bundle_positions) <= 2, \ | 
|  | "The path contains more than two nested bundles: %s" % binary | 
|  | if len(bundle_positions) == 0: | 
|  | # Case 1: this is a standalone executable or dylib. | 
|  | return [] | 
|  | assert (not (len(app_positions) == 1 and | 
|  | len(framework_positions) == 1 and | 
|  | app_positions[0] > framework_positions[0])), \ | 
|  | "The path contains an app bundle inside a framework: %s" % binary | 
|  | # Cases 2 and 3. The outermost bundle (which is the only bundle in the case 2) | 
|  | # is located in the product dir. | 
|  | outermost_bundle = bundle_positions[0] | 
|  | product_dir = path_parts[:outermost_bundle] | 
|  | # In case 2 this is the same as |outermost_bundle|. | 
|  | innermost_bundle = bundle_positions[-1] | 
|  | dsym_path = product_dir + [path_parts[innermost_bundle]] | 
|  | result = '%s.dSYM' % os.path.sep.join(dsym_path) | 
|  | return [result] | 
|  |  | 
|  |  | 
|  | def main(): | 
|  | disable_buffering() | 
|  | set_symbolizer_path() | 
|  | asan_symbolize.demangle = True | 
|  | asan_symbolize.fix_filename_patterns = sys.argv[1:] | 
|  | asan_symbolize.logfile = sys.stdin | 
|  | loop = asan_symbolize.SymbolizationLoop(dsym_hint_producer=chrome_dsym_hints) | 
|  | loop.process_logfile() | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | main() |