gecko-dev/tools/rb/fix_stacks.py

100 строки
3.4 KiB
Python
Executable File

#!/usr/bin/python
# vim:sw=4:ts=4:et:
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
# This script uses `fix-stacks` to post-process the entries produced by
# MozFormatCodeAddress().
from __future__ import absolute_import, print_function
from subprocess import Popen, PIPE
import atexit
import os
import platform
import re
import sys
import six
# Matches lines produced by MozFormatCodeAddress(), e.g.
# `#01: ???[tests/example +0x43a0]`.
line_re = re.compile("#\d+: .+\[.+ \+0x[0-9A-Fa-f]+\]")
fix_stacks = None
def fixSymbols(line, jsonMode=False, slowWarning=False, breakpadSymsDir=None, hide_errors=False):
global fix_stacks
line = six.ensure_str(line)
result = line_re.search(line)
if result is None:
return line
if not fix_stacks:
# Look in MOZ_FETCHES_DIR (for automation), then in MOZBUILD_STATE_PATH
# (for a local build where the user has that set), then in ~/.mozbuild
# (for a local build with default settings).
base = os.environ.get(
'MOZ_FETCHES_DIR',
os.environ.get(
'MOZBUILD_STATE_PATH',
os.path.expanduser('~/.mozbuild')
)
)
fix_stacks_exe = base + '/fix-stacks/fix-stacks'
if platform.system() == 'Windows':
fix_stacks_exe = fix_stacks_exe + '.exe'
if not (os.path.isfile(fix_stacks_exe) and os.access(fix_stacks_exe, os.X_OK)):
raise Exception('cannot find `fix-stacks`; please run `./mach bootstrap`')
args = [fix_stacks_exe]
if jsonMode:
args.append('-j')
if breakpadSymsDir:
# `fileid` should be packaged next to `fix_stacks.py`.
here = os.path.dirname(__file__)
fileid_exe = os.path.join(here, 'fileid')
if platform.system() == 'Windows':
fileid_exe = fileid_exe + '.exe'
args.append('-b')
args.append(breakpadSymsDir + "," + fileid_exe)
# Sometimes we need to prevent errors from going to stderr.
stderr = open(os.devnull) if hide_errors else None
fix_stacks = Popen(args, stdin=PIPE, stdout=PIPE, stderr=stderr, universal_newlines=True)
# Shut down the fix_stacks process on exit. We use `terminate()`
# because it is more forceful than `wait()`, and the Python docs warn
# about possible deadlocks with `wait()`.
def cleanup(fix_stacks):
fix_stacks.stdin.close()
fix_stacks.terminate()
atexit.register(cleanup, fix_stacks)
if slowWarning:
print("Initializing stack-fixing for the first stack frame, this may take a while...")
# Sometimes `line` is lacking a trailing newline. If we pass such a `line`
# to `fix-stacks` it will wait until it receives a newline, causing this
# script to hang. So we add a newline if one is missing and then remove it
# from the output.
is_missing_newline = not line.endswith('\n')
if is_missing_newline:
line = line + "\n"
fix_stacks.stdin.write(line)
fix_stacks.stdin.flush()
out = fix_stacks.stdout.readline()
if is_missing_newline:
out = out[:-1]
return out
if __name__ == "__main__":
for line in sys.stdin:
sys.stdout.write(fixSymbols(line))