forked from OSchip/llvm-project
2745 lines
107 KiB
Python
2745 lines
107 KiB
Python
"""
|
|
LLDB module which provides the abstract base class of lldb test case.
|
|
|
|
The concrete subclass can override lldbtest.TestBase in order to inherit the
|
|
common behavior for unitest.TestCase.setUp/tearDown implemented in this file.
|
|
|
|
The subclass should override the attribute mydir in order for the python runtime
|
|
to locate the individual test cases when running as part of a large test suite
|
|
or when running each test case as a separate python invocation.
|
|
|
|
./dotest.py provides a test driver which sets up the environment to run the
|
|
entire of part of the test suite . Example:
|
|
|
|
# Exercises the test suite in the types directory....
|
|
/Volumes/data/lldb/svn/ToT/test $ ./dotest.py -A x86_64 types
|
|
...
|
|
|
|
Session logs for test failures/errors/unexpected successes will go into directory '2012-05-16-13_35_42'
|
|
Command invoked: python ./dotest.py -A x86_64 types
|
|
compilers=['clang']
|
|
|
|
Configuration: arch=x86_64 compiler=clang
|
|
----------------------------------------------------------------------
|
|
Collected 72 tests
|
|
|
|
........................................................................
|
|
----------------------------------------------------------------------
|
|
Ran 72 tests in 135.468s
|
|
|
|
OK
|
|
$
|
|
"""
|
|
|
|
from __future__ import absolute_import
|
|
from __future__ import print_function
|
|
|
|
# System modules
|
|
import abc
|
|
from distutils.version import LooseVersion
|
|
from functools import wraps
|
|
import gc
|
|
import glob
|
|
import io
|
|
import os.path
|
|
import re
|
|
import shutil
|
|
import signal
|
|
from subprocess import *
|
|
import sys
|
|
import time
|
|
import traceback
|
|
import distutils.spawn
|
|
|
|
# Third-party modules
|
|
import unittest2
|
|
from six import add_metaclass
|
|
from six import StringIO as SixStringIO
|
|
import six
|
|
|
|
# LLDB modules
|
|
import lldb
|
|
from . import configuration
|
|
from . import decorators
|
|
from . import lldbplatformutil
|
|
from . import lldbtest_config
|
|
from . import lldbutil
|
|
from . import test_categories
|
|
from lldbsuite.support import encoded_file
|
|
from lldbsuite.support import funcutils
|
|
from lldbsuite.test.builders import get_builder
|
|
|
|
# See also dotest.parseOptionsAndInitTestdirs(), where the environment variables
|
|
# LLDB_COMMAND_TRACE is set from '-t' option.
|
|
|
|
# By default, traceAlways is False.
|
|
if "LLDB_COMMAND_TRACE" in os.environ and os.environ[
|
|
"LLDB_COMMAND_TRACE"] == "YES":
|
|
traceAlways = True
|
|
else:
|
|
traceAlways = False
|
|
|
|
# By default, doCleanup is True.
|
|
if "LLDB_DO_CLEANUP" in os.environ and os.environ["LLDB_DO_CLEANUP"] == "NO":
|
|
doCleanup = False
|
|
else:
|
|
doCleanup = True
|
|
|
|
|
|
#
|
|
# Some commonly used assert messages.
|
|
#
|
|
|
|
COMMAND_FAILED_AS_EXPECTED = "Command has failed as expected"
|
|
|
|
CURRENT_EXECUTABLE_SET = "Current executable set successfully"
|
|
|
|
PROCESS_IS_VALID = "Process is valid"
|
|
|
|
PROCESS_KILLED = "Process is killed successfully"
|
|
|
|
PROCESS_EXITED = "Process exited successfully"
|
|
|
|
PROCESS_STOPPED = "Process status should be stopped"
|
|
|
|
RUN_SUCCEEDED = "Process is launched successfully"
|
|
|
|
RUN_COMPLETED = "Process exited successfully"
|
|
|
|
BACKTRACE_DISPLAYED_CORRECTLY = "Backtrace displayed correctly"
|
|
|
|
BREAKPOINT_CREATED = "Breakpoint created successfully"
|
|
|
|
BREAKPOINT_STATE_CORRECT = "Breakpoint state is correct"
|
|
|
|
BREAKPOINT_PENDING_CREATED = "Pending breakpoint created successfully"
|
|
|
|
BREAKPOINT_HIT_ONCE = "Breakpoint resolved with hit count = 1"
|
|
|
|
BREAKPOINT_HIT_TWICE = "Breakpoint resolved with hit count = 2"
|
|
|
|
BREAKPOINT_HIT_THRICE = "Breakpoint resolved with hit count = 3"
|
|
|
|
MISSING_EXPECTED_REGISTERS = "At least one expected register is unavailable."
|
|
|
|
OBJECT_PRINTED_CORRECTLY = "Object printed correctly"
|
|
|
|
SOURCE_DISPLAYED_CORRECTLY = "Source code displayed correctly"
|
|
|
|
STEP_IN_SUCCEEDED = "Thread step-in succeeded"
|
|
|
|
STEP_OUT_SUCCEEDED = "Thread step-out succeeded"
|
|
|
|
STOPPED_DUE_TO_EXC_BAD_ACCESS = "Process should be stopped due to bad access exception"
|
|
|
|
STOPPED_DUE_TO_ASSERT = "Process should be stopped due to an assertion"
|
|
|
|
STOPPED_DUE_TO_BREAKPOINT = "Process should be stopped due to breakpoint"
|
|
|
|
STOPPED_DUE_TO_BREAKPOINT_WITH_STOP_REASON_AS = "%s, %s" % (
|
|
STOPPED_DUE_TO_BREAKPOINT, "instead, the actual stop reason is: '%s'")
|
|
|
|
STOPPED_DUE_TO_BREAKPOINT_CONDITION = "Stopped due to breakpoint condition"
|
|
|
|
STOPPED_DUE_TO_BREAKPOINT_IGNORE_COUNT = "Stopped due to breakpoint and ignore count"
|
|
|
|
STOPPED_DUE_TO_BREAKPOINT_JITTED_CONDITION = "Stopped due to breakpoint jitted condition"
|
|
|
|
STOPPED_DUE_TO_SIGNAL = "Process state is stopped due to signal"
|
|
|
|
STOPPED_DUE_TO_STEP_IN = "Process state is stopped due to step in"
|
|
|
|
STOPPED_DUE_TO_WATCHPOINT = "Process should be stopped due to watchpoint"
|
|
|
|
DATA_TYPES_DISPLAYED_CORRECTLY = "Data type(s) displayed correctly"
|
|
|
|
VALID_BREAKPOINT = "Got a valid breakpoint"
|
|
|
|
VALID_BREAKPOINT_LOCATION = "Got a valid breakpoint location"
|
|
|
|
VALID_COMMAND_INTERPRETER = "Got a valid command interpreter"
|
|
|
|
VALID_FILESPEC = "Got a valid filespec"
|
|
|
|
VALID_MODULE = "Got a valid module"
|
|
|
|
VALID_PROCESS = "Got a valid process"
|
|
|
|
VALID_SYMBOL = "Got a valid symbol"
|
|
|
|
VALID_TARGET = "Got a valid target"
|
|
|
|
VALID_PLATFORM = "Got a valid platform"
|
|
|
|
VALID_TYPE = "Got a valid type"
|
|
|
|
VALID_VARIABLE = "Got a valid variable"
|
|
|
|
VARIABLES_DISPLAYED_CORRECTLY = "Variable(s) displayed correctly"
|
|
|
|
WATCHPOINT_CREATED = "Watchpoint created successfully"
|
|
|
|
|
|
def CMD_MSG(str):
|
|
'''A generic "Command '%s' did not return successfully" message generator.'''
|
|
return "Command '%s' did not return successfully" % str
|
|
|
|
|
|
def COMPLETION_MSG(str_before, str_after, completions):
|
|
'''A generic assertion failed message generator for the completion mechanism.'''
|
|
return ("'%s' successfully completes to '%s', but completions were:\n%s"
|
|
% (str_before, str_after, "\n".join(completions)))
|
|
|
|
|
|
def EXP_MSG(str, actual, exe):
|
|
'''A generic "'%s' returned unexpected result" message generator if exe.
|
|
Otherwise, it generates "'%s' does not match expected result" message.'''
|
|
|
|
return "'%s' %s result, got '%s'" % (
|
|
str, 'returned unexpected' if exe else 'does not match expected', actual.strip())
|
|
|
|
|
|
def SETTING_MSG(setting):
|
|
'''A generic "Value of setting '%s' is not correct" message generator.'''
|
|
return "Value of setting '%s' is not correct" % setting
|
|
|
|
|
|
def line_number(filename, string_to_match):
|
|
"""Helper function to return the line number of the first matched string."""
|
|
with io.open(filename, mode='r', encoding="utf-8") as f:
|
|
for i, line in enumerate(f):
|
|
if line.find(string_to_match) != -1:
|
|
# Found our match.
|
|
return i + 1
|
|
raise Exception(
|
|
"Unable to find '%s' within file %s" %
|
|
(string_to_match, filename))
|
|
|
|
def get_line(filename, line_number):
|
|
"""Return the text of the line at the 1-based line number."""
|
|
with io.open(filename, mode='r', encoding="utf-8") as f:
|
|
return f.readlines()[line_number - 1]
|
|
|
|
def pointer_size():
|
|
"""Return the pointer size of the host system."""
|
|
import ctypes
|
|
a_pointer = ctypes.c_void_p(0xffff)
|
|
return 8 * ctypes.sizeof(a_pointer)
|
|
|
|
|
|
def is_exe(fpath):
|
|
"""Returns true if fpath is an executable."""
|
|
return os.path.isfile(fpath) and os.access(fpath, os.X_OK)
|
|
|
|
|
|
def which(program):
|
|
"""Returns the full path to a program; None otherwise."""
|
|
fpath, fname = os.path.split(program)
|
|
if fpath:
|
|
if is_exe(program):
|
|
return program
|
|
else:
|
|
for path in os.environ["PATH"].split(os.pathsep):
|
|
exe_file = os.path.join(path, program)
|
|
if is_exe(exe_file):
|
|
return exe_file
|
|
return None
|
|
|
|
class ValueCheck:
|
|
def __init__(self, name=None, value=None, type=None, summary=None,
|
|
children=None):
|
|
"""
|
|
:param name: The name that the SBValue should have. None if the summary
|
|
should not be checked.
|
|
:param summary: The summary that the SBValue should have. None if the
|
|
summary should not be checked.
|
|
:param value: The value that the SBValue should have. None if the value
|
|
should not be checked.
|
|
:param type: The type that the SBValue result should have. None if the
|
|
type should not be checked.
|
|
:param children: A list of ValueChecks that need to match the children
|
|
of this SBValue. None if children shouldn't be checked.
|
|
The order of checks is the order of the checks in the
|
|
list. The number of checks has to match the number of
|
|
children.
|
|
"""
|
|
self.expect_name = name
|
|
self.expect_value = value
|
|
self.expect_type = type
|
|
self.expect_summary = summary
|
|
self.children = children
|
|
|
|
def check_value(self, test_base, val, error_msg=None):
|
|
"""
|
|
Checks that the given value matches the currently set properties
|
|
of this ValueCheck. If a match failed, the given TestBase will
|
|
be used to emit an error. A custom error message can be specified
|
|
that will be used to describe failed check for this SBValue (but
|
|
not errors in the child values).
|
|
"""
|
|
|
|
this_error_msg = error_msg if error_msg else ""
|
|
this_error_msg += "\nChecking SBValue: " + str(val)
|
|
|
|
test_base.assertSuccess(val.GetError())
|
|
|
|
if self.expect_name:
|
|
test_base.assertEqual(self.expect_name, val.GetName(),
|
|
this_error_msg)
|
|
if self.expect_value:
|
|
test_base.assertEqual(self.expect_value, val.GetValue(),
|
|
this_error_msg)
|
|
if self.expect_type:
|
|
test_base.assertEqual(self.expect_type, val.GetDisplayTypeName(),
|
|
this_error_msg)
|
|
if self.expect_summary:
|
|
test_base.assertEqual(self.expect_summary, val.GetSummary(),
|
|
this_error_msg)
|
|
if self.children is not None:
|
|
self.check_value_children(test_base, val, error_msg)
|
|
|
|
def check_value_children(self, test_base, val, error_msg=None):
|
|
"""
|
|
Checks that the children of a SBValue match a certain structure and
|
|
have certain properties.
|
|
|
|
:param test_base: The current test's TestBase object.
|
|
:param val: The SBValue to check.
|
|
"""
|
|
|
|
this_error_msg = error_msg if error_msg else ""
|
|
this_error_msg += "\nChecking SBValue: " + str(val)
|
|
|
|
test_base.assertEqual(len(self.children), val.GetNumChildren(), this_error_msg)
|
|
|
|
for i in range(0, val.GetNumChildren()):
|
|
expected_child = self.children[i]
|
|
actual_child = val.GetChildAtIndex(i)
|
|
expected_child.check_value(test_base, actual_child, error_msg)
|
|
|
|
class recording(SixStringIO):
|
|
"""
|
|
A nice little context manager for recording the debugger interactions into
|
|
our session object. If trace flag is ON, it also emits the interactions
|
|
into the stderr.
|
|
"""
|
|
|
|
def __init__(self, test, trace):
|
|
"""Create a SixStringIO instance; record the session obj and trace flag."""
|
|
SixStringIO.__init__(self)
|
|
# The test might not have undergone the 'setUp(self)' phase yet, so that
|
|
# the attribute 'session' might not even exist yet.
|
|
self.session = getattr(test, "session", None) if test else None
|
|
self.trace = trace
|
|
|
|
def __enter__(self):
|
|
"""
|
|
Context management protocol on entry to the body of the with statement.
|
|
Just return the SixStringIO object.
|
|
"""
|
|
return self
|
|
|
|
def __exit__(self, type, value, tb):
|
|
"""
|
|
Context management protocol on exit from the body of the with statement.
|
|
If trace is ON, it emits the recordings into stderr. Always add the
|
|
recordings to our session object. And close the SixStringIO object, too.
|
|
"""
|
|
if self.trace:
|
|
print(self.getvalue(), file=sys.stderr)
|
|
if self.session:
|
|
print(self.getvalue(), file=self.session)
|
|
self.close()
|
|
|
|
|
|
@add_metaclass(abc.ABCMeta)
|
|
class _BaseProcess(object):
|
|
|
|
@abc.abstractproperty
|
|
def pid(self):
|
|
"""Returns process PID if has been launched already."""
|
|
|
|
@abc.abstractmethod
|
|
def launch(self, executable, args):
|
|
"""Launches new process with given executable and args."""
|
|
|
|
@abc.abstractmethod
|
|
def terminate(self):
|
|
"""Terminates previously launched process.."""
|
|
|
|
|
|
class _LocalProcess(_BaseProcess):
|
|
|
|
def __init__(self, trace_on):
|
|
self._proc = None
|
|
self._trace_on = trace_on
|
|
self._delayafterterminate = 0.1
|
|
|
|
@property
|
|
def pid(self):
|
|
return self._proc.pid
|
|
|
|
def launch(self, executable, args):
|
|
self._proc = Popen(
|
|
[executable] + args,
|
|
stdout=open(
|
|
os.devnull) if not self._trace_on else None,
|
|
stdin=PIPE,
|
|
preexec_fn=lldbplatformutil.enable_attach)
|
|
|
|
def terminate(self):
|
|
if self._proc.poll() is None:
|
|
# Terminate _proc like it does the pexpect
|
|
signals_to_try = [
|
|
sig for sig in [
|
|
'SIGHUP',
|
|
'SIGCONT',
|
|
'SIGINT'] if sig in dir(signal)]
|
|
for sig in signals_to_try:
|
|
try:
|
|
self._proc.send_signal(getattr(signal, sig))
|
|
time.sleep(self._delayafterterminate)
|
|
if self._proc.poll() is not None:
|
|
return
|
|
except ValueError:
|
|
pass # Windows says SIGINT is not a valid signal to send
|
|
self._proc.terminate()
|
|
time.sleep(self._delayafterterminate)
|
|
if self._proc.poll() is not None:
|
|
return
|
|
self._proc.kill()
|
|
time.sleep(self._delayafterterminate)
|
|
|
|
def poll(self):
|
|
return self._proc.poll()
|
|
|
|
|
|
class _RemoteProcess(_BaseProcess):
|
|
|
|
def __init__(self, install_remote):
|
|
self._pid = None
|
|
self._install_remote = install_remote
|
|
|
|
@property
|
|
def pid(self):
|
|
return self._pid
|
|
|
|
def launch(self, executable, args):
|
|
if self._install_remote:
|
|
src_path = executable
|
|
dst_path = lldbutil.join_remote_paths(
|
|
lldb.remote_platform.GetWorkingDirectory(), os.path.basename(executable))
|
|
|
|
dst_file_spec = lldb.SBFileSpec(dst_path, False)
|
|
err = lldb.remote_platform.Install(
|
|
lldb.SBFileSpec(src_path, True), dst_file_spec)
|
|
if err.Fail():
|
|
raise Exception(
|
|
"remote_platform.Install('%s', '%s') failed: %s" %
|
|
(src_path, dst_path, err))
|
|
else:
|
|
dst_path = executable
|
|
dst_file_spec = lldb.SBFileSpec(executable, False)
|
|
|
|
launch_info = lldb.SBLaunchInfo(args)
|
|
launch_info.SetExecutableFile(dst_file_spec, True)
|
|
launch_info.SetWorkingDirectory(
|
|
lldb.remote_platform.GetWorkingDirectory())
|
|
|
|
# Redirect stdout and stderr to /dev/null
|
|
launch_info.AddSuppressFileAction(1, False, True)
|
|
launch_info.AddSuppressFileAction(2, False, True)
|
|
|
|
err = lldb.remote_platform.Launch(launch_info)
|
|
if err.Fail():
|
|
raise Exception(
|
|
"remote_platform.Launch('%s', '%s') failed: %s" %
|
|
(dst_path, args, err))
|
|
self._pid = launch_info.GetProcessID()
|
|
|
|
def terminate(self):
|
|
lldb.remote_platform.Kill(self._pid)
|
|
|
|
# From 2.7's subprocess.check_output() convenience function.
|
|
# Return a tuple (stdoutdata, stderrdata).
|
|
|
|
|
|
def system(commands, **kwargs):
|
|
r"""Run an os command with arguments and return its output as a byte string.
|
|
|
|
If the exit code was non-zero it raises a CalledProcessError. The
|
|
CalledProcessError object will have the return code in the returncode
|
|
attribute and output in the output attribute.
|
|
|
|
The arguments are the same as for the Popen constructor. Example:
|
|
|
|
>>> check_output(["ls", "-l", "/dev/null"])
|
|
'crw-rw-rw- 1 root root 1, 3 Oct 18 2007 /dev/null\n'
|
|
|
|
The stdout argument is not allowed as it is used internally.
|
|
To capture standard error in the result, use stderr=STDOUT.
|
|
|
|
>>> check_output(["/bin/sh", "-c",
|
|
... "ls -l non_existent_file ; exit 0"],
|
|
... stderr=STDOUT)
|
|
'ls: non_existent_file: No such file or directory\n'
|
|
"""
|
|
|
|
# Assign the sender object to variable 'test' and remove it from kwargs.
|
|
test = kwargs.pop('sender', None)
|
|
|
|
# [['make', 'clean', 'foo'], ['make', 'foo']] -> ['make clean foo', 'make foo']
|
|
commandList = [' '.join(x) for x in commands]
|
|
output = ""
|
|
error = ""
|
|
for shellCommand in commandList:
|
|
if 'stdout' in kwargs:
|
|
raise ValueError(
|
|
'stdout argument not allowed, it will be overridden.')
|
|
if 'shell' in kwargs and kwargs['shell'] == False:
|
|
raise ValueError('shell=False not allowed')
|
|
process = Popen(
|
|
shellCommand,
|
|
stdout=PIPE,
|
|
stderr=STDOUT,
|
|
shell=True,
|
|
**kwargs)
|
|
pid = process.pid
|
|
this_output, this_error = process.communicate()
|
|
retcode = process.poll()
|
|
|
|
if retcode:
|
|
cmd = kwargs.get("args")
|
|
if cmd is None:
|
|
cmd = shellCommand
|
|
cpe = CalledProcessError(retcode, cmd)
|
|
# Ensure caller can access the stdout/stderr.
|
|
cpe.lldb_extensions = {
|
|
"combined_output": this_output,
|
|
"command": shellCommand
|
|
}
|
|
raise cpe
|
|
output = output + this_output.decode("utf-8", errors='ignore')
|
|
return output
|
|
|
|
|
|
def getsource_if_available(obj):
|
|
"""
|
|
Return the text of the source code for an object if available. Otherwise,
|
|
a print representation is returned.
|
|
"""
|
|
import inspect
|
|
try:
|
|
return inspect.getsource(obj)
|
|
except:
|
|
return repr(obj)
|
|
|
|
|
|
def builder_module():
|
|
return get_builder(sys.platform)
|
|
|
|
|
|
class Base(unittest2.TestCase):
|
|
"""
|
|
Abstract base for performing lldb (see TestBase) or other generic tests (see
|
|
BenchBase for one example). lldbtest.Base works with the test driver to
|
|
accomplish things.
|
|
|
|
"""
|
|
|
|
# The concrete subclass should override this attribute.
|
|
mydir = None
|
|
|
|
# Keep track of the old current working directory.
|
|
oldcwd = None
|
|
|
|
@staticmethod
|
|
def compute_mydir(test_file):
|
|
'''Subclasses should call this function to correctly calculate the
|
|
required "mydir" attribute as follows:
|
|
|
|
mydir = TestBase.compute_mydir(__file__)
|
|
'''
|
|
# /abs/path/to/packages/group/subdir/mytest.py -> group/subdir
|
|
lldb_test_src = configuration.test_src_root
|
|
if not test_file.startswith(lldb_test_src):
|
|
raise Exception(
|
|
"Test file '%s' must reside within lldb_test_src "
|
|
"(which is '%s')." % (test_file, lldb_test_src))
|
|
return os.path.dirname(os.path.relpath(test_file, start=lldb_test_src))
|
|
|
|
def TraceOn(self):
|
|
"""Returns True if we are in trace mode (tracing detailed test execution)."""
|
|
return traceAlways
|
|
|
|
def trace(self, *args,**kwargs):
|
|
with recording(self, self.TraceOn()) as sbuf:
|
|
print(*args, file=sbuf, **kwargs)
|
|
|
|
@classmethod
|
|
def setUpClass(cls):
|
|
"""
|
|
Python unittest framework class setup fixture.
|
|
Do current directory manipulation.
|
|
"""
|
|
# Fail fast if 'mydir' attribute is not overridden.
|
|
if not cls.mydir or len(cls.mydir) == 0:
|
|
raise Exception("Subclasses must override the 'mydir' attribute.")
|
|
|
|
# Save old working directory.
|
|
cls.oldcwd = os.getcwd()
|
|
|
|
full_dir = os.path.join(configuration.test_src_root, cls.mydir)
|
|
if traceAlways:
|
|
print("Change dir to:", full_dir, file=sys.stderr)
|
|
os.chdir(full_dir)
|
|
lldb.SBReproducer.SetWorkingDirectory(full_dir)
|
|
|
|
# Set platform context.
|
|
cls.platformContext = lldbplatformutil.createPlatformContext()
|
|
|
|
@classmethod
|
|
def tearDownClass(cls):
|
|
"""
|
|
Python unittest framework class teardown fixture.
|
|
Do class-wide cleanup.
|
|
"""
|
|
|
|
if doCleanup:
|
|
# First, let's do the platform-specific cleanup.
|
|
module = builder_module()
|
|
module.cleanup()
|
|
|
|
# Subclass might have specific cleanup function defined.
|
|
if getattr(cls, "classCleanup", None):
|
|
if traceAlways:
|
|
print(
|
|
"Call class-specific cleanup function for class:",
|
|
cls,
|
|
file=sys.stderr)
|
|
try:
|
|
cls.classCleanup()
|
|
except:
|
|
exc_type, exc_value, exc_tb = sys.exc_info()
|
|
traceback.print_exception(exc_type, exc_value, exc_tb)
|
|
|
|
# Restore old working directory.
|
|
if traceAlways:
|
|
print("Restore dir to:", cls.oldcwd, file=sys.stderr)
|
|
os.chdir(cls.oldcwd)
|
|
|
|
def enableLogChannelsForCurrentTest(self):
|
|
if len(lldbtest_config.channels) == 0:
|
|
return
|
|
|
|
# if debug channels are specified in lldbtest_config.channels,
|
|
# create a new set of log files for every test
|
|
log_basename = self.getLogBasenameForCurrentTest()
|
|
|
|
# confirm that the file is writeable
|
|
host_log_path = "{}-host.log".format(log_basename)
|
|
open(host_log_path, 'w').close()
|
|
self.log_files.append(host_log_path)
|
|
|
|
log_enable = "log enable -Tpn -f {} ".format(host_log_path)
|
|
for channel_with_categories in lldbtest_config.channels:
|
|
channel_then_categories = channel_with_categories.split(' ', 1)
|
|
channel = channel_then_categories[0]
|
|
if len(channel_then_categories) > 1:
|
|
categories = channel_then_categories[1]
|
|
else:
|
|
categories = "default"
|
|
|
|
if channel == "gdb-remote" and lldb.remote_platform is None:
|
|
# communicate gdb-remote categories to debugserver
|
|
os.environ["LLDB_DEBUGSERVER_LOG_FLAGS"] = categories
|
|
|
|
self.ci.HandleCommand(
|
|
log_enable + channel_with_categories, self.res)
|
|
if not self.res.Succeeded():
|
|
raise Exception(
|
|
'log enable failed (check LLDB_LOG_OPTION env variable)')
|
|
|
|
# Communicate log path name to debugserver & lldb-server
|
|
# For remote debugging, these variables need to be set when starting the platform
|
|
# instance.
|
|
if lldb.remote_platform is None:
|
|
server_log_path = "{}-server.log".format(log_basename)
|
|
open(server_log_path, 'w').close()
|
|
self.log_files.append(server_log_path)
|
|
os.environ["LLDB_DEBUGSERVER_LOG_FILE"] = server_log_path
|
|
|
|
# Communicate channels to lldb-server
|
|
os.environ["LLDB_SERVER_LOG_CHANNELS"] = ":".join(
|
|
lldbtest_config.channels)
|
|
|
|
self.addTearDownHook(self.disableLogChannelsForCurrentTest)
|
|
|
|
def disableLogChannelsForCurrentTest(self):
|
|
# close all log files that we opened
|
|
for channel_and_categories in lldbtest_config.channels:
|
|
# channel format - <channel-name> [<category0> [<category1> ...]]
|
|
channel = channel_and_categories.split(' ', 1)[0]
|
|
self.ci.HandleCommand("log disable " + channel, self.res)
|
|
if not self.res.Succeeded():
|
|
raise Exception(
|
|
'log disable failed (check LLDB_LOG_OPTION env variable)')
|
|
|
|
# Retrieve the server log (if any) from the remote system. It is assumed the server log
|
|
# is writing to the "server.log" file in the current test directory. This can be
|
|
# achieved by setting LLDB_DEBUGSERVER_LOG_FILE="server.log" when starting remote
|
|
# platform.
|
|
if lldb.remote_platform:
|
|
server_log_path = self.getLogBasenameForCurrentTest() + "-server.log"
|
|
if lldb.remote_platform.Get(
|
|
lldb.SBFileSpec("server.log"),
|
|
lldb.SBFileSpec(server_log_path)).Success():
|
|
self.log_files.append(server_log_path)
|
|
|
|
def setPlatformWorkingDir(self):
|
|
if not lldb.remote_platform or not configuration.lldb_platform_working_dir:
|
|
return
|
|
|
|
components = self.mydir.split(os.path.sep) + [str(self.test_number), self.getBuildDirBasename()]
|
|
remote_test_dir = configuration.lldb_platform_working_dir
|
|
for c in components:
|
|
remote_test_dir = lldbutil.join_remote_paths(remote_test_dir, c)
|
|
error = lldb.remote_platform.MakeDirectory(
|
|
remote_test_dir, 448) # 448 = 0o700
|
|
if error.Fail():
|
|
raise Exception("making remote directory '%s': %s" % (
|
|
remote_test_dir, error))
|
|
|
|
lldb.remote_platform.SetWorkingDirectory(remote_test_dir)
|
|
|
|
# This function removes all files from the current working directory while leaving
|
|
# the directories in place. The cleanup is required to reduce the disk space required
|
|
# by the test suite while leaving the directories untouched is neccessary because
|
|
# sub-directories might belong to an other test
|
|
def clean_working_directory():
|
|
# TODO: Make it working on Windows when we need it for remote debugging support
|
|
# TODO: Replace the heuristic to remove the files with a logic what collects the
|
|
# list of files we have to remove during test runs.
|
|
shell_cmd = lldb.SBPlatformShellCommand(
|
|
"rm %s/*" % remote_test_dir)
|
|
lldb.remote_platform.Run(shell_cmd)
|
|
self.addTearDownHook(clean_working_directory)
|
|
|
|
def getSourceDir(self):
|
|
"""Return the full path to the current test."""
|
|
return os.path.join(configuration.test_src_root, self.mydir)
|
|
|
|
def getBuildDirBasename(self):
|
|
return self.__class__.__module__ + "." + self.testMethodName
|
|
|
|
def getBuildDir(self):
|
|
"""Return the full path to the current test."""
|
|
return os.path.join(configuration.test_build_dir, self.mydir,
|
|
self.getBuildDirBasename())
|
|
|
|
def getReproducerDir(self):
|
|
"""Return the full path to the reproducer if enabled."""
|
|
if configuration.capture_path:
|
|
return configuration.capture_path
|
|
if configuration.replay_path:
|
|
return configuration.replay_path
|
|
return None
|
|
|
|
def makeBuildDir(self):
|
|
"""Create the test-specific working directory, deleting any previous
|
|
contents."""
|
|
bdir = self.getBuildDir()
|
|
if os.path.isdir(bdir):
|
|
shutil.rmtree(bdir)
|
|
lldbutil.mkdir_p(bdir)
|
|
|
|
def getBuildArtifact(self, name="a.out"):
|
|
"""Return absolute path to an artifact in the test's build directory."""
|
|
return os.path.join(self.getBuildDir(), name)
|
|
|
|
def getSourcePath(self, name):
|
|
"""Return absolute path to a file in the test's source directory."""
|
|
return os.path.join(self.getSourceDir(), name)
|
|
|
|
def getReproducerArtifact(self, name):
|
|
lldbutil.mkdir_p(self.getReproducerDir())
|
|
return os.path.join(self.getReproducerDir(), name)
|
|
|
|
def getReproducerRemappedPath(self, path):
|
|
assert configuration.replay_path
|
|
assert os.path.isabs(path)
|
|
path = os.path.relpath(path, '/')
|
|
return os.path.join(configuration.replay_path, 'root', path)
|
|
|
|
@classmethod
|
|
def setUpCommands(cls):
|
|
commands = [
|
|
# First of all, clear all settings to have clean state of global properties.
|
|
"settings clear -all",
|
|
|
|
# Disable Spotlight lookup. The testsuite creates
|
|
# different binaries with the same UUID, because they only
|
|
# differ in the debug info, which is not being hashed.
|
|
"settings set symbols.enable-external-lookup false",
|
|
|
|
# Inherit the TCC permissions from the inferior's parent.
|
|
"settings set target.inherit-tcc true",
|
|
|
|
# Kill rather than detach from the inferior if something goes wrong.
|
|
"settings set target.detach-on-error false",
|
|
|
|
# Disable fix-its by default so that incorrect expressions in tests don't
|
|
# pass just because Clang thinks it has a fix-it.
|
|
"settings set target.auto-apply-fixits false",
|
|
|
|
# Testsuite runs in parallel and the host can have also other load.
|
|
"settings set plugin.process.gdb-remote.packet-timeout 60",
|
|
|
|
'settings set symbols.clang-modules-cache-path "{}"'.format(
|
|
configuration.lldb_module_cache_dir),
|
|
"settings set use-color false",
|
|
]
|
|
|
|
# Set any user-overridden settings.
|
|
for setting, value in configuration.settings:
|
|
commands.append('setting set %s %s'%(setting, value))
|
|
|
|
# Make sure that a sanitizer LLDB's environment doesn't get passed on.
|
|
if cls.platformContext and cls.platformContext.shlib_environment_var in os.environ:
|
|
commands.append('settings set target.env-vars {}='.format(
|
|
cls.platformContext.shlib_environment_var))
|
|
|
|
# Set environment variables for the inferior.
|
|
if lldbtest_config.inferior_env:
|
|
commands.append('settings set target.env-vars {}'.format(
|
|
lldbtest_config.inferior_env))
|
|
return commands
|
|
|
|
def setUp(self):
|
|
"""Fixture for unittest test case setup.
|
|
|
|
It works with the test driver to conditionally skip tests and does other
|
|
initializations."""
|
|
#import traceback
|
|
# traceback.print_stack()
|
|
|
|
if "LIBCXX_PATH" in os.environ:
|
|
self.libcxxPath = os.environ["LIBCXX_PATH"]
|
|
else:
|
|
self.libcxxPath = None
|
|
|
|
if "LLDBVSCODE_EXEC" in os.environ:
|
|
self.lldbVSCodeExec = os.environ["LLDBVSCODE_EXEC"]
|
|
else:
|
|
self.lldbVSCodeExec = None
|
|
|
|
self.lldbOption = " ".join(
|
|
"-o '" + s + "'" for s in self.setUpCommands())
|
|
|
|
# If we spawn an lldb process for test (via pexpect), do not load the
|
|
# init file unless told otherwise.
|
|
if os.environ.get("NO_LLDBINIT") != "NO":
|
|
self.lldbOption += " --no-lldbinit"
|
|
|
|
# Assign the test method name to self.testMethodName.
|
|
#
|
|
# For an example of the use of this attribute, look at test/types dir.
|
|
# There are a bunch of test cases under test/types and we don't want the
|
|
# module cacheing subsystem to be confused with executable name "a.out"
|
|
# used for all the test cases.
|
|
self.testMethodName = self._testMethodName
|
|
|
|
# This is for the case of directly spawning 'lldb'/'gdb' and interacting
|
|
# with it using pexpect.
|
|
self.child = None
|
|
self.child_prompt = "(lldb) "
|
|
# If the child is interacting with the embedded script interpreter,
|
|
# there are two exits required during tear down, first to quit the
|
|
# embedded script interpreter and second to quit the lldb command
|
|
# interpreter.
|
|
self.child_in_script_interpreter = False
|
|
|
|
# These are for customized teardown cleanup.
|
|
self.dict = None
|
|
self.doTearDownCleanup = False
|
|
# And in rare cases where there are multiple teardown cleanups.
|
|
self.dicts = []
|
|
self.doTearDownCleanups = False
|
|
|
|
# List of spawned subproces.Popen objects
|
|
self.subprocesses = []
|
|
|
|
# List of log files produced by the current test.
|
|
self.log_files = []
|
|
|
|
# Create the build directory.
|
|
# The logs are stored in the build directory, so we have to create it
|
|
# before creating the first log file.
|
|
self.makeBuildDir()
|
|
|
|
session_file = self.getLogBasenameForCurrentTest()+".log"
|
|
self.log_files.append(session_file)
|
|
|
|
# Python 3 doesn't support unbuffered I/O in text mode. Open buffered.
|
|
self.session = encoded_file.open(session_file, "utf-8", mode="w")
|
|
|
|
# Optimistically set __errored__, __failed__, __expected__ to False
|
|
# initially. If the test errored/failed, the session info
|
|
# (self.session) is then dumped into a session specific file for
|
|
# diagnosis.
|
|
self.__cleanup_errored__ = False
|
|
self.__errored__ = False
|
|
self.__failed__ = False
|
|
self.__expected__ = False
|
|
# We are also interested in unexpected success.
|
|
self.__unexpected__ = False
|
|
# And skipped tests.
|
|
self.__skipped__ = False
|
|
|
|
# See addTearDownHook(self, hook) which allows the client to add a hook
|
|
# function to be run during tearDown() time.
|
|
self.hooks = []
|
|
|
|
# See HideStdout(self).
|
|
self.sys_stdout_hidden = False
|
|
|
|
if self.platformContext:
|
|
# set environment variable names for finding shared libraries
|
|
self.dylibPath = self.platformContext.shlib_environment_var
|
|
|
|
# Create the debugger instance.
|
|
self.dbg = lldb.SBDebugger.Create()
|
|
# Copy selected platform from a global instance if it exists.
|
|
if lldb.selected_platform is not None:
|
|
self.dbg.SetSelectedPlatform(lldb.selected_platform)
|
|
|
|
if not self.dbg:
|
|
raise Exception('Invalid debugger instance')
|
|
|
|
# Retrieve the associated command interpreter instance.
|
|
self.ci = self.dbg.GetCommandInterpreter()
|
|
if not self.ci:
|
|
raise Exception('Could not get the command interpreter')
|
|
|
|
# And the result object.
|
|
self.res = lldb.SBCommandReturnObject()
|
|
|
|
self.setPlatformWorkingDir()
|
|
self.enableLogChannelsForCurrentTest()
|
|
|
|
self.lib_lldb = None
|
|
self.framework_dir = None
|
|
self.darwinWithFramework = False
|
|
|
|
if sys.platform.startswith("darwin") and configuration.lldb_framework_path:
|
|
framework = configuration.lldb_framework_path
|
|
lib = os.path.join(framework, 'LLDB')
|
|
if os.path.exists(lib):
|
|
self.framework_dir = os.path.dirname(framework)
|
|
self.lib_lldb = lib
|
|
self.darwinWithFramework = self.platformIsDarwin()
|
|
|
|
def setAsync(self, value):
|
|
""" Sets async mode to True/False and ensures it is reset after the testcase completes."""
|
|
old_async = self.dbg.GetAsync()
|
|
self.dbg.SetAsync(value)
|
|
self.addTearDownHook(lambda: self.dbg.SetAsync(old_async))
|
|
|
|
def cleanupSubprocesses(self):
|
|
# Terminate subprocesses in reverse order from how they were created.
|
|
for p in reversed(self.subprocesses):
|
|
p.terminate()
|
|
del p
|
|
del self.subprocesses[:]
|
|
|
|
def spawnSubprocess(self, executable, args=[], install_remote=True):
|
|
""" Creates a subprocess.Popen object with the specified executable and arguments,
|
|
saves it in self.subprocesses, and returns the object.
|
|
"""
|
|
proc = _RemoteProcess(
|
|
install_remote) if lldb.remote_platform else _LocalProcess(self.TraceOn())
|
|
proc.launch(executable, args)
|
|
self.subprocesses.append(proc)
|
|
return proc
|
|
|
|
def HideStdout(self):
|
|
"""Hide output to stdout from the user.
|
|
|
|
During test execution, there might be cases where we don't want to show the
|
|
standard output to the user. For example,
|
|
|
|
self.runCmd(r'''sc print("\n\n\tHello!\n")''')
|
|
|
|
tests whether command abbreviation for 'script' works or not. There is no
|
|
need to show the 'Hello' output to the user as long as the 'script' command
|
|
succeeds and we are not in TraceOn() mode (see the '-t' option).
|
|
|
|
In this case, the test method calls self.HideStdout(self) to redirect the
|
|
sys.stdout to a null device, and restores the sys.stdout upon teardown.
|
|
|
|
Note that you should only call this method at most once during a test case
|
|
execution. Any subsequent call has no effect at all."""
|
|
if self.sys_stdout_hidden:
|
|
return
|
|
|
|
self.sys_stdout_hidden = True
|
|
old_stdout = sys.stdout
|
|
sys.stdout = open(os.devnull, 'w')
|
|
|
|
def restore_stdout():
|
|
sys.stdout = old_stdout
|
|
self.addTearDownHook(restore_stdout)
|
|
|
|
# =======================================================================
|
|
# Methods for customized teardown cleanups as well as execution of hooks.
|
|
# =======================================================================
|
|
|
|
def setTearDownCleanup(self, dictionary=None):
|
|
"""Register a cleanup action at tearDown() time with a dictionary"""
|
|
self.dict = dictionary
|
|
self.doTearDownCleanup = True
|
|
|
|
def addTearDownCleanup(self, dictionary):
|
|
"""Add a cleanup action at tearDown() time with a dictionary"""
|
|
self.dicts.append(dictionary)
|
|
self.doTearDownCleanups = True
|
|
|
|
def addTearDownHook(self, hook):
|
|
"""
|
|
Add a function to be run during tearDown() time.
|
|
|
|
Hooks are executed in a first come first serve manner.
|
|
"""
|
|
if six.callable(hook):
|
|
with recording(self, traceAlways) as sbuf:
|
|
print(
|
|
"Adding tearDown hook:",
|
|
getsource_if_available(hook),
|
|
file=sbuf)
|
|
self.hooks.append(hook)
|
|
|
|
return self
|
|
|
|
def deletePexpectChild(self):
|
|
# This is for the case of directly spawning 'lldb' and interacting with it
|
|
# using pexpect.
|
|
if self.child and self.child.isalive():
|
|
import pexpect
|
|
with recording(self, traceAlways) as sbuf:
|
|
print("tearing down the child process....", file=sbuf)
|
|
try:
|
|
if self.child_in_script_interpreter:
|
|
self.child.sendline('quit()')
|
|
self.child.expect_exact(self.child_prompt)
|
|
self.child.sendline(
|
|
'settings set interpreter.prompt-on-quit false')
|
|
self.child.sendline('quit')
|
|
self.child.expect(pexpect.EOF)
|
|
except (ValueError, pexpect.ExceptionPexpect):
|
|
# child is already terminated
|
|
pass
|
|
except OSError as exception:
|
|
import errno
|
|
if exception.errno != errno.EIO:
|
|
# unexpected error
|
|
raise
|
|
# child is already terminated
|
|
finally:
|
|
# Give it one final blow to make sure the child is terminated.
|
|
self.child.close()
|
|
|
|
def tearDown(self):
|
|
"""Fixture for unittest test case teardown."""
|
|
self.deletePexpectChild()
|
|
|
|
# Check and run any hook functions.
|
|
for hook in reversed(self.hooks):
|
|
with recording(self, traceAlways) as sbuf:
|
|
print(
|
|
"Executing tearDown hook:",
|
|
getsource_if_available(hook),
|
|
file=sbuf)
|
|
if funcutils.requires_self(hook):
|
|
hook(self)
|
|
else:
|
|
hook() # try the plain call and hope it works
|
|
|
|
del self.hooks
|
|
|
|
# Perform registered teardown cleanup.
|
|
if doCleanup and self.doTearDownCleanup:
|
|
self.cleanup(dictionary=self.dict)
|
|
|
|
# In rare cases where there are multiple teardown cleanups added.
|
|
if doCleanup and self.doTearDownCleanups:
|
|
if self.dicts:
|
|
for dict in reversed(self.dicts):
|
|
self.cleanup(dictionary=dict)
|
|
|
|
# Remove subprocesses created by the test.
|
|
self.cleanupSubprocesses()
|
|
|
|
# This must be the last statement, otherwise teardown hooks or other
|
|
# lines might depend on this still being active.
|
|
lldb.SBDebugger.Destroy(self.dbg)
|
|
del self.dbg
|
|
|
|
# All modules should be orphaned now so that they can be cleared from
|
|
# the shared module cache.
|
|
lldb.SBModule.GarbageCollectAllocatedModules()
|
|
|
|
# Modules are not orphaned during reproducer replay because they're
|
|
# leaked on purpose.
|
|
if not configuration.is_reproducer():
|
|
# Assert that the global module cache is empty.
|
|
self.assertEqual(lldb.SBModule.GetNumberAllocatedModules(), 0)
|
|
|
|
|
|
# =========================================================
|
|
# Various callbacks to allow introspection of test progress
|
|
# =========================================================
|
|
|
|
def markError(self):
|
|
"""Callback invoked when an error (unexpected exception) errored."""
|
|
self.__errored__ = True
|
|
with recording(self, False) as sbuf:
|
|
# False because there's no need to write "ERROR" to the stderr twice.
|
|
# Once by the Python unittest framework, and a second time by us.
|
|
print("ERROR", file=sbuf)
|
|
|
|
def markCleanupError(self):
|
|
"""Callback invoked when an error occurs while a test is cleaning up."""
|
|
self.__cleanup_errored__ = True
|
|
with recording(self, False) as sbuf:
|
|
# False because there's no need to write "CLEANUP_ERROR" to the stderr twice.
|
|
# Once by the Python unittest framework, and a second time by us.
|
|
print("CLEANUP_ERROR", file=sbuf)
|
|
|
|
def markFailure(self):
|
|
"""Callback invoked when a failure (test assertion failure) occurred."""
|
|
self.__failed__ = True
|
|
with recording(self, False) as sbuf:
|
|
# False because there's no need to write "FAIL" to the stderr twice.
|
|
# Once by the Python unittest framework, and a second time by us.
|
|
print("FAIL", file=sbuf)
|
|
|
|
def markExpectedFailure(self, err, bugnumber):
|
|
"""Callback invoked when an expected failure/error occurred."""
|
|
self.__expected__ = True
|
|
with recording(self, False) as sbuf:
|
|
# False because there's no need to write "expected failure" to the
|
|
# stderr twice.
|
|
# Once by the Python unittest framework, and a second time by us.
|
|
if bugnumber is None:
|
|
print("expected failure", file=sbuf)
|
|
else:
|
|
print(
|
|
"expected failure (problem id:" + str(bugnumber) + ")",
|
|
file=sbuf)
|
|
|
|
def markSkippedTest(self):
|
|
"""Callback invoked when a test is skipped."""
|
|
self.__skipped__ = True
|
|
with recording(self, False) as sbuf:
|
|
# False because there's no need to write "skipped test" to the
|
|
# stderr twice.
|
|
# Once by the Python unittest framework, and a second time by us.
|
|
print("skipped test", file=sbuf)
|
|
|
|
def markUnexpectedSuccess(self, bugnumber):
|
|
"""Callback invoked when an unexpected success occurred."""
|
|
self.__unexpected__ = True
|
|
with recording(self, False) as sbuf:
|
|
# False because there's no need to write "unexpected success" to the
|
|
# stderr twice.
|
|
# Once by the Python unittest framework, and a second time by us.
|
|
if bugnumber is None:
|
|
print("unexpected success", file=sbuf)
|
|
else:
|
|
print(
|
|
"unexpected success (problem id:" + str(bugnumber) + ")",
|
|
file=sbuf)
|
|
|
|
def getRerunArgs(self):
|
|
return " -f %s.%s" % (self.__class__.__name__, self._testMethodName)
|
|
|
|
def getLogBasenameForCurrentTest(self, prefix="Incomplete"):
|
|
"""
|
|
returns a partial path that can be used as the beginning of the name of multiple
|
|
log files pertaining to this test
|
|
"""
|
|
return os.path.join(self.getBuildDir(), prefix)
|
|
|
|
def dumpSessionInfo(self):
|
|
"""
|
|
Dump the debugger interactions leading to a test error/failure. This
|
|
allows for more convenient postmortem analysis.
|
|
|
|
See also LLDBTestResult (dotest.py) which is a singlton class derived
|
|
from TextTestResult and overwrites addError, addFailure, and
|
|
addExpectedFailure methods to allow us to to mark the test instance as
|
|
such.
|
|
"""
|
|
|
|
# We are here because self.tearDown() detected that this test instance
|
|
# either errored or failed. The lldb.test_result singleton contains
|
|
# two lists (errors and failures) which get populated by the unittest
|
|
# framework. Look over there for stack trace information.
|
|
#
|
|
# The lists contain 2-tuples of TestCase instances and strings holding
|
|
# formatted tracebacks.
|
|
#
|
|
# See http://docs.python.org/library/unittest.html#unittest.TestResult.
|
|
|
|
# output tracebacks into session
|
|
pairs = []
|
|
if self.__errored__:
|
|
pairs = configuration.test_result.errors
|
|
prefix = 'Error'
|
|
elif self.__cleanup_errored__:
|
|
pairs = configuration.test_result.cleanup_errors
|
|
prefix = 'CleanupError'
|
|
elif self.__failed__:
|
|
pairs = configuration.test_result.failures
|
|
prefix = 'Failure'
|
|
elif self.__expected__:
|
|
pairs = configuration.test_result.expectedFailures
|
|
prefix = 'ExpectedFailure'
|
|
elif self.__skipped__:
|
|
prefix = 'SkippedTest'
|
|
elif self.__unexpected__:
|
|
prefix = 'UnexpectedSuccess'
|
|
else:
|
|
prefix = 'Success'
|
|
|
|
if not self.__unexpected__ and not self.__skipped__:
|
|
for test, traceback in pairs:
|
|
if test is self:
|
|
print(traceback, file=self.session)
|
|
|
|
import datetime
|
|
print(
|
|
"Session info generated @",
|
|
datetime.datetime.now().ctime(),
|
|
file=self.session)
|
|
self.session.close()
|
|
del self.session
|
|
|
|
# process the log files
|
|
if prefix != 'Success' or lldbtest_config.log_success:
|
|
# keep all log files, rename them to include prefix
|
|
src_log_basename = self.getLogBasenameForCurrentTest()
|
|
dst_log_basename = self.getLogBasenameForCurrentTest(prefix)
|
|
for src in self.log_files:
|
|
if os.path.isfile(src):
|
|
dst = src.replace(src_log_basename, dst_log_basename)
|
|
if os.name == "nt" and os.path.isfile(dst):
|
|
# On Windows, renaming a -> b will throw an exception if
|
|
# b exists. On non-Windows platforms it silently
|
|
# replaces the destination. Ultimately this means that
|
|
# atomic renames are not guaranteed to be possible on
|
|
# Windows, but we need this to work anyway, so just
|
|
# remove the destination first if it already exists.
|
|
remove_file(dst)
|
|
|
|
lldbutil.mkdir_p(os.path.dirname(dst))
|
|
os.rename(src, dst)
|
|
else:
|
|
# success! (and we don't want log files) delete log files
|
|
for log_file in self.log_files:
|
|
if os.path.isfile(log_file):
|
|
remove_file(log_file)
|
|
|
|
# ====================================================
|
|
# Config. methods supported through a plugin interface
|
|
# (enables reading of the current test configuration)
|
|
# ====================================================
|
|
|
|
def isMIPS(self):
|
|
"""Returns true if the architecture is MIPS."""
|
|
arch = self.getArchitecture()
|
|
if re.match("mips", arch):
|
|
return True
|
|
return False
|
|
|
|
def isPPC64le(self):
|
|
"""Returns true if the architecture is PPC64LE."""
|
|
arch = self.getArchitecture()
|
|
if re.match("powerpc64le", arch):
|
|
return True
|
|
return False
|
|
|
|
def isAArch64SVE(self):
|
|
triple = self.dbg.GetSelectedPlatform().GetTriple()
|
|
|
|
# TODO other platforms, please implement this function
|
|
if not re.match(".*-.*-linux", triple):
|
|
return False
|
|
|
|
# Need to do something different for non-Linux/Android targets
|
|
cpuinfo_path = self.getBuildArtifact("cpuinfo")
|
|
if configuration.lldb_platform_name:
|
|
self.runCmd('platform get-file "/proc/cpuinfo" ' + cpuinfo_path)
|
|
else:
|
|
cpuinfo_path = "/proc/cpuinfo"
|
|
|
|
try:
|
|
f = open(cpuinfo_path, 'r')
|
|
cpuinfo = f.read()
|
|
f.close()
|
|
except:
|
|
return False
|
|
|
|
return " sve " in cpuinfo
|
|
|
|
def hasLinuxVmFlags(self):
|
|
""" Check that the target machine has "VmFlags" lines in
|
|
its /proc/{pid}/smaps files."""
|
|
|
|
triple = self.dbg.GetSelectedPlatform().GetTriple()
|
|
if not re.match(".*-.*-linux", triple):
|
|
return False
|
|
|
|
self.runCmd('platform process list')
|
|
pid = None
|
|
for line in self.res.GetOutput().splitlines():
|
|
if 'lldb-server' in line:
|
|
pid = line.split(' ')[0]
|
|
break
|
|
|
|
if pid is None:
|
|
return False
|
|
|
|
smaps_path = self.getBuildArtifact('smaps')
|
|
self.runCmd('platform get-file "/proc/{}/smaps" {}'.format(pid, smaps_path))
|
|
|
|
with open(smaps_path, 'r') as f:
|
|
return "VmFlags" in f.read()
|
|
|
|
def getArchitecture(self):
|
|
"""Returns the architecture in effect the test suite is running with."""
|
|
module = builder_module()
|
|
arch = module.getArchitecture()
|
|
if arch == 'amd64':
|
|
arch = 'x86_64'
|
|
if arch in ['armv7l', 'armv8l'] :
|
|
arch = 'arm'
|
|
return arch
|
|
|
|
def getLldbArchitecture(self):
|
|
"""Returns the architecture of the lldb binary."""
|
|
if not hasattr(self, 'lldbArchitecture'):
|
|
|
|
# spawn local process
|
|
command = [
|
|
lldbtest_config.lldbExec,
|
|
"-o",
|
|
"file " + lldbtest_config.lldbExec,
|
|
"-o",
|
|
"quit"
|
|
]
|
|
|
|
output = check_output(command)
|
|
str = output.decode("utf-8")
|
|
|
|
for line in str.splitlines():
|
|
m = re.search(
|
|
"Current executable set to '.*' \\((.*)\\)\\.", line)
|
|
if m:
|
|
self.lldbArchitecture = m.group(1)
|
|
break
|
|
|
|
return self.lldbArchitecture
|
|
|
|
def getCompiler(self):
|
|
"""Returns the compiler in effect the test suite is running with."""
|
|
module = builder_module()
|
|
return module.getCompiler()
|
|
|
|
def getCompilerBinary(self):
|
|
"""Returns the compiler binary the test suite is running with."""
|
|
return self.getCompiler().split()[0]
|
|
|
|
def getCompilerVersion(self):
|
|
""" Returns a string that represents the compiler version.
|
|
Supports: llvm, clang.
|
|
"""
|
|
compiler = self.getCompilerBinary()
|
|
version_output = system([[compiler, "--version"]])
|
|
for line in version_output.split(os.linesep):
|
|
m = re.search('version ([0-9.]+)', line)
|
|
if m:
|
|
return m.group(1)
|
|
return 'unknown'
|
|
|
|
def getDwarfVersion(self):
|
|
""" Returns the dwarf version generated by clang or '0'. """
|
|
if configuration.dwarf_version:
|
|
return str(configuration.dwarf_version)
|
|
if 'clang' in self.getCompiler():
|
|
try:
|
|
driver_output = check_output(
|
|
[self.getCompiler()] + '-g -c -x c - -o - -###'.split(),
|
|
stderr=STDOUT)
|
|
driver_output = driver_output.decode("utf-8")
|
|
for line in driver_output.split(os.linesep):
|
|
m = re.search('dwarf-version=([0-9])', line)
|
|
if m:
|
|
return m.group(1)
|
|
except: pass
|
|
return '0'
|
|
|
|
def platformIsDarwin(self):
|
|
"""Returns true if the OS triple for the selected platform is any valid apple OS"""
|
|
return lldbplatformutil.platformIsDarwin()
|
|
|
|
def hasDarwinFramework(self):
|
|
return self.darwinWithFramework
|
|
|
|
def getPlatform(self):
|
|
"""Returns the target platform the test suite is running on."""
|
|
return lldbplatformutil.getPlatform()
|
|
|
|
def isIntelCompiler(self):
|
|
""" Returns true if using an Intel (ICC) compiler, false otherwise. """
|
|
return any([x in self.getCompiler() for x in ["icc", "icpc", "icl"]])
|
|
|
|
def expectedCompilerVersion(self, compiler_version):
|
|
"""Returns True iff compiler_version[1] matches the current compiler version.
|
|
Use compiler_version[0] to specify the operator used to determine if a match has occurred.
|
|
Any operator other than the following defaults to an equality test:
|
|
'>', '>=', "=>", '<', '<=', '=<', '!=', "!" or 'not'
|
|
|
|
If the current compiler version cannot be determined, we assume it is close to the top
|
|
of trunk, so any less-than or equal-to comparisons will return False, and any
|
|
greater-than or not-equal-to comparisons will return True.
|
|
"""
|
|
if compiler_version is None:
|
|
return True
|
|
operator = str(compiler_version[0])
|
|
version = compiler_version[1]
|
|
|
|
if version is None:
|
|
return True
|
|
|
|
test_compiler_version = self.getCompilerVersion()
|
|
if test_compiler_version == 'unknown':
|
|
# Assume the compiler version is at or near the top of trunk.
|
|
return operator in ['>', '>=', '!', '!=', 'not']
|
|
|
|
if operator == '>':
|
|
return LooseVersion(test_compiler_version) > LooseVersion(version)
|
|
if operator == '>=' or operator == '=>':
|
|
return LooseVersion(test_compiler_version) >= LooseVersion(version)
|
|
if operator == '<':
|
|
return LooseVersion(test_compiler_version) < LooseVersion(version)
|
|
if operator == '<=' or operator == '=<':
|
|
return LooseVersion(test_compiler_version) <= LooseVersion(version)
|
|
if operator == '!=' or operator == '!' or operator == 'not':
|
|
return str(version) not in str(test_compiler_version)
|
|
return str(version) in str(test_compiler_version)
|
|
|
|
def expectedCompiler(self, compilers):
|
|
"""Returns True iff any element of compilers is a sub-string of the current compiler."""
|
|
if (compilers is None):
|
|
return True
|
|
|
|
for compiler in compilers:
|
|
if compiler in self.getCompiler():
|
|
return True
|
|
|
|
return False
|
|
|
|
def expectedArch(self, archs):
|
|
"""Returns True iff any element of archs is a sub-string of the current architecture."""
|
|
if (archs is None):
|
|
return True
|
|
|
|
for arch in archs:
|
|
if arch in self.getArchitecture():
|
|
return True
|
|
|
|
return False
|
|
|
|
def getRunOptions(self):
|
|
"""Command line option for -A and -C to run this test again, called from
|
|
self.dumpSessionInfo()."""
|
|
arch = self.getArchitecture()
|
|
comp = self.getCompiler()
|
|
option_str = ""
|
|
if arch:
|
|
option_str = "-A " + arch
|
|
if comp:
|
|
option_str += " -C " + comp
|
|
return option_str
|
|
|
|
def getDebugInfo(self):
|
|
method = getattr(self, self.testMethodName)
|
|
return getattr(method, "debug_info", None)
|
|
|
|
# ==================================================
|
|
# Build methods supported through a plugin interface
|
|
# ==================================================
|
|
|
|
def getstdlibFlag(self):
|
|
""" Returns the proper -stdlib flag, or empty if not required."""
|
|
if self.platformIsDarwin() or self.getPlatform() == "freebsd" or self.getPlatform() == "openbsd":
|
|
stdlibflag = "-stdlib=libc++"
|
|
else: # this includes NetBSD
|
|
stdlibflag = ""
|
|
return stdlibflag
|
|
|
|
def getstdFlag(self):
|
|
""" Returns the proper stdflag. """
|
|
if "gcc" in self.getCompiler() and "4.6" in self.getCompilerVersion():
|
|
stdflag = "-std=c++0x"
|
|
else:
|
|
stdflag = "-std=c++11"
|
|
return stdflag
|
|
|
|
def buildDriver(self, sources, exe_name):
|
|
""" Platform-specific way to build a program that links with LLDB (via the liblldb.so
|
|
or LLDB.framework).
|
|
"""
|
|
stdflag = self.getstdFlag()
|
|
stdlibflag = self.getstdlibFlag()
|
|
|
|
lib_dir = configuration.lldb_libs_dir
|
|
if self.hasDarwinFramework():
|
|
d = {'CXX_SOURCES': sources,
|
|
'EXE': exe_name,
|
|
'CFLAGS_EXTRAS': "%s %s" % (stdflag, stdlibflag),
|
|
'FRAMEWORK_INCLUDES': "-F%s" % self.framework_dir,
|
|
'LD_EXTRAS': "%s -Wl,-rpath,%s" % (self.lib_lldb, self.framework_dir),
|
|
}
|
|
elif sys.platform.startswith('win'):
|
|
d = {
|
|
'CXX_SOURCES': sources,
|
|
'EXE': exe_name,
|
|
'CFLAGS_EXTRAS': "%s %s -I%s" % (stdflag,
|
|
stdlibflag,
|
|
os.path.join(
|
|
os.environ["LLDB_SRC"],
|
|
"include")),
|
|
'LD_EXTRAS': "-L%s -lliblldb" % lib_dir}
|
|
else:
|
|
d = {
|
|
'CXX_SOURCES': sources,
|
|
'EXE': exe_name,
|
|
'CFLAGS_EXTRAS': "%s %s -I%s" % (stdflag,
|
|
stdlibflag,
|
|
os.path.join(
|
|
os.environ["LLDB_SRC"],
|
|
"include")),
|
|
'LD_EXTRAS': "-L%s -llldb -Wl,-rpath,%s" % (lib_dir, lib_dir)}
|
|
if self.TraceOn():
|
|
print(
|
|
"Building LLDB Driver (%s) from sources %s" %
|
|
(exe_name, sources))
|
|
|
|
self.buildDefault(dictionary=d)
|
|
|
|
def buildLibrary(self, sources, lib_name):
|
|
"""Platform specific way to build a default library. """
|
|
|
|
stdflag = self.getstdFlag()
|
|
|
|
lib_dir = configuration.lldb_libs_dir
|
|
if self.hasDarwinFramework():
|
|
d = {'DYLIB_CXX_SOURCES': sources,
|
|
'DYLIB_NAME': lib_name,
|
|
'CFLAGS_EXTRAS': "%s -stdlib=libc++" % stdflag,
|
|
'FRAMEWORK_INCLUDES': "-F%s" % self.framework_dir,
|
|
'LD_EXTRAS': "%s -Wl,-rpath,%s -dynamiclib" % (self.lib_lldb, self.framework_dir),
|
|
}
|
|
elif self.getPlatform() == 'windows':
|
|
d = {
|
|
'DYLIB_CXX_SOURCES': sources,
|
|
'DYLIB_NAME': lib_name,
|
|
'CFLAGS_EXTRAS': "%s -I%s " % (stdflag,
|
|
os.path.join(
|
|
os.environ["LLDB_SRC"],
|
|
"include")),
|
|
'LD_EXTRAS': "-shared -l%s\liblldb.lib" % lib_dir}
|
|
else:
|
|
d = {
|
|
'DYLIB_CXX_SOURCES': sources,
|
|
'DYLIB_NAME': lib_name,
|
|
'CFLAGS_EXTRAS': "%s -I%s -fPIC" % (stdflag,
|
|
os.path.join(
|
|
os.environ["LLDB_SRC"],
|
|
"include")),
|
|
'LD_EXTRAS': "-shared -L%s -llldb -Wl,-rpath,%s" % (lib_dir, lib_dir)}
|
|
if self.TraceOn():
|
|
print(
|
|
"Building LLDB Library (%s) from sources %s" %
|
|
(lib_name, sources))
|
|
|
|
self.buildDefault(dictionary=d)
|
|
|
|
def buildProgram(self, sources, exe_name):
|
|
""" Platform specific way to build an executable from C/C++ sources. """
|
|
d = {'CXX_SOURCES': sources,
|
|
'EXE': exe_name}
|
|
self.buildDefault(dictionary=d)
|
|
|
|
def buildDefault(
|
|
self,
|
|
architecture=None,
|
|
compiler=None,
|
|
dictionary=None):
|
|
"""Platform specific way to build the default binaries."""
|
|
testdir = self.mydir
|
|
testname = self.getBuildDirBasename()
|
|
|
|
if not architecture and configuration.arch:
|
|
architecture = configuration.arch
|
|
|
|
if self.getDebugInfo():
|
|
raise Exception("buildDefault tests must set NO_DEBUG_INFO_TESTCASE")
|
|
module = builder_module()
|
|
dictionary = lldbplatformutil.finalize_build_dictionary(dictionary)
|
|
if not module.buildDefault(self, architecture, compiler,
|
|
dictionary, testdir, testname):
|
|
raise Exception("Don't know how to build default binary")
|
|
|
|
def buildDsym(
|
|
self,
|
|
architecture=None,
|
|
compiler=None,
|
|
dictionary=None):
|
|
"""Platform specific way to build binaries with dsym info."""
|
|
testdir = self.mydir
|
|
testname = self.getBuildDirBasename()
|
|
if self.getDebugInfo() != "dsym":
|
|
raise Exception("NO_DEBUG_INFO_TESTCASE must build with buildDefault")
|
|
|
|
module = builder_module()
|
|
dictionary = lldbplatformutil.finalize_build_dictionary(dictionary)
|
|
if not module.buildDsym(self, architecture, compiler,
|
|
dictionary, testdir, testname):
|
|
raise Exception("Don't know how to build binary with dsym")
|
|
|
|
def buildDwarf(
|
|
self,
|
|
architecture=None,
|
|
compiler=None,
|
|
dictionary=None):
|
|
"""Platform specific way to build binaries with dwarf maps."""
|
|
testdir = self.mydir
|
|
testname = self.getBuildDirBasename()
|
|
if self.getDebugInfo() != "dwarf":
|
|
raise Exception("NO_DEBUG_INFO_TESTCASE must build with buildDefault")
|
|
|
|
module = builder_module()
|
|
dictionary = lldbplatformutil.finalize_build_dictionary(dictionary)
|
|
if not module.buildDwarf(self, architecture, compiler,
|
|
dictionary, testdir, testname):
|
|
raise Exception("Don't know how to build binary with dwarf")
|
|
|
|
def buildDwo(
|
|
self,
|
|
architecture=None,
|
|
compiler=None,
|
|
dictionary=None):
|
|
"""Platform specific way to build binaries with dwarf maps."""
|
|
testdir = self.mydir
|
|
testname = self.getBuildDirBasename()
|
|
if self.getDebugInfo() != "dwo":
|
|
raise Exception("NO_DEBUG_INFO_TESTCASE must build with buildDefault")
|
|
|
|
module = builder_module()
|
|
dictionary = lldbplatformutil.finalize_build_dictionary(dictionary)
|
|
if not module.buildDwo(self, architecture, compiler,
|
|
dictionary, testdir, testname):
|
|
raise Exception("Don't know how to build binary with dwo")
|
|
|
|
def buildGModules(
|
|
self,
|
|
architecture=None,
|
|
compiler=None,
|
|
dictionary=None):
|
|
"""Platform specific way to build binaries with gmodules info."""
|
|
testdir = self.mydir
|
|
testname = self.getBuildDirBasename()
|
|
if self.getDebugInfo() != "gmodules":
|
|
raise Exception("NO_DEBUG_INFO_TESTCASE must build with buildDefault")
|
|
|
|
module = builder_module()
|
|
dictionary = lldbplatformutil.finalize_build_dictionary(dictionary)
|
|
if not module.buildGModules(self, architecture, compiler,
|
|
dictionary, testdir, testname):
|
|
raise Exception("Don't know how to build binary with gmodules")
|
|
|
|
def signBinary(self, binary_path):
|
|
if sys.platform.startswith("darwin"):
|
|
codesign_cmd = "codesign --force --sign \"%s\" %s" % (
|
|
lldbtest_config.codesign_identity, binary_path)
|
|
call(codesign_cmd, shell=True)
|
|
|
|
def findBuiltClang(self):
|
|
"""Tries to find and use Clang from the build directory as the compiler (instead of the system compiler)."""
|
|
paths_to_try = [
|
|
"llvm-build/Release+Asserts/x86_64/bin/clang",
|
|
"llvm-build/Debug+Asserts/x86_64/bin/clang",
|
|
"llvm-build/Release/x86_64/bin/clang",
|
|
"llvm-build/Debug/x86_64/bin/clang",
|
|
]
|
|
lldb_root_path = os.path.join(
|
|
os.path.dirname(__file__), "..", "..", "..", "..")
|
|
for p in paths_to_try:
|
|
path = os.path.join(lldb_root_path, p)
|
|
if os.path.exists(path):
|
|
return path
|
|
|
|
# Tries to find clang at the same folder as the lldb
|
|
lldb_dir = os.path.dirname(lldbtest_config.lldbExec)
|
|
path = distutils.spawn.find_executable("clang", lldb_dir)
|
|
if path is not None:
|
|
return path
|
|
|
|
return os.environ["CC"]
|
|
|
|
|
|
def yaml2obj(self, yaml_path, obj_path):
|
|
"""
|
|
Create an object file at the given path from a yaml file.
|
|
|
|
Throws subprocess.CalledProcessError if the object could not be created.
|
|
"""
|
|
yaml2obj_bin = configuration.get_yaml2obj_path()
|
|
if not yaml2obj_bin:
|
|
self.assertTrue(False, "No valid yaml2obj executable specified")
|
|
command = [yaml2obj_bin, "-o=%s" % obj_path, yaml_path]
|
|
system([command])
|
|
|
|
def getBuildFlags(
|
|
self,
|
|
use_cpp11=True,
|
|
use_libcxx=False,
|
|
use_libstdcxx=False):
|
|
""" Returns a dictionary (which can be provided to build* functions above) which
|
|
contains OS-specific build flags.
|
|
"""
|
|
cflags = ""
|
|
ldflags = ""
|
|
|
|
# On Mac OS X, unless specifically requested to use libstdc++, use
|
|
# libc++
|
|
if not use_libstdcxx and self.platformIsDarwin():
|
|
use_libcxx = True
|
|
|
|
if use_libcxx and self.libcxxPath:
|
|
cflags += "-stdlib=libc++ "
|
|
if self.libcxxPath:
|
|
libcxxInclude = os.path.join(self.libcxxPath, "include")
|
|
libcxxLib = os.path.join(self.libcxxPath, "lib")
|
|
if os.path.isdir(libcxxInclude) and os.path.isdir(libcxxLib):
|
|
cflags += "-nostdinc++ -I%s -L%s -Wl,-rpath,%s " % (
|
|
libcxxInclude, libcxxLib, libcxxLib)
|
|
|
|
if use_cpp11:
|
|
cflags += "-std="
|
|
if "gcc" in self.getCompiler() and "4.6" in self.getCompilerVersion():
|
|
cflags += "c++0x"
|
|
else:
|
|
cflags += "c++11"
|
|
if self.platformIsDarwin() or self.getPlatform() == "freebsd":
|
|
cflags += " -stdlib=libc++"
|
|
elif self.getPlatform() == "openbsd":
|
|
cflags += " -stdlib=libc++"
|
|
elif self.getPlatform() == "netbsd":
|
|
# NetBSD defaults to libc++
|
|
pass
|
|
elif "clang" in self.getCompiler():
|
|
cflags += " -stdlib=libstdc++"
|
|
|
|
return {'CFLAGS_EXTRAS': cflags,
|
|
'LD_EXTRAS': ldflags,
|
|
}
|
|
|
|
def cleanup(self, dictionary=None):
|
|
"""Platform specific way to do cleanup after build."""
|
|
module = builder_module()
|
|
if not module.cleanup(self, dictionary):
|
|
raise Exception(
|
|
"Don't know how to do cleanup with dictionary: " +
|
|
dictionary)
|
|
|
|
def invoke(self, obj, name, trace=False):
|
|
"""Use reflection to call a method dynamically with no argument."""
|
|
trace = (True if traceAlways else trace)
|
|
|
|
method = getattr(obj, name)
|
|
import inspect
|
|
self.assertTrue(inspect.ismethod(method),
|
|
name + "is a method name of object: " + str(obj))
|
|
result = method()
|
|
with recording(self, trace) as sbuf:
|
|
print(str(method) + ":", result, file=sbuf)
|
|
return result
|
|
|
|
def getLLDBLibraryEnvVal(self):
|
|
""" Returns the path that the OS-specific library search environment variable
|
|
(self.dylibPath) should be set to in order for a program to find the LLDB
|
|
library. If an environment variable named self.dylibPath is already set,
|
|
the new path is appended to it and returned.
|
|
"""
|
|
existing_library_path = os.environ[
|
|
self.dylibPath] if self.dylibPath in os.environ else None
|
|
if existing_library_path:
|
|
return "%s:%s" % (existing_library_path, configuration.lldb_libs_dir)
|
|
if sys.platform.startswith("darwin") and configuration.lldb_framework_path:
|
|
return configuration.lldb_framework_path
|
|
return configuration.lldb_libs_dir
|
|
|
|
def getLibcPlusPlusLibs(self):
|
|
if self.getPlatform() in ('freebsd', 'linux', 'netbsd', 'openbsd'):
|
|
return ['libc++.so.1']
|
|
else:
|
|
return ['libc++.1.dylib', 'libc++abi.']
|
|
|
|
def run_platform_command(self, cmd):
|
|
platform = self.dbg.GetSelectedPlatform()
|
|
shell_command = lldb.SBPlatformShellCommand(cmd)
|
|
err = platform.Run(shell_command)
|
|
return (err, shell_command.GetStatus(), shell_command.GetOutput())
|
|
|
|
# Metaclass for TestBase to change the list of test metods when a new TestCase is loaded.
|
|
# We change the test methods to create a new test method for each test for each debug info we are
|
|
# testing. The name of the new test method will be '<original-name>_<debug-info>' and with adding
|
|
# the new test method we remove the old method at the same time. This functionality can be
|
|
# supressed by at test case level setting the class attribute NO_DEBUG_INFO_TESTCASE or at test
|
|
# level by using the decorator @no_debug_info_test.
|
|
|
|
|
|
class LLDBTestCaseFactory(type):
|
|
|
|
def __new__(cls, name, bases, attrs):
|
|
original_testcase = super(
|
|
LLDBTestCaseFactory, cls).__new__(
|
|
cls, name, bases, attrs)
|
|
if original_testcase.NO_DEBUG_INFO_TESTCASE:
|
|
return original_testcase
|
|
|
|
newattrs = {}
|
|
for attrname, attrvalue in attrs.items():
|
|
if attrname.startswith("test") and not getattr(
|
|
attrvalue, "__no_debug_info_test__", False):
|
|
|
|
# If any debug info categories were explicitly tagged, assume that list to be
|
|
# authoritative. If none were specified, try with all debug
|
|
# info formats.
|
|
all_dbginfo_categories = set(test_categories.debug_info_categories)
|
|
categories = set(
|
|
getattr(
|
|
attrvalue,
|
|
"categories",
|
|
[])) & all_dbginfo_categories
|
|
if not categories:
|
|
categories = all_dbginfo_categories
|
|
|
|
for cat in categories:
|
|
@decorators.add_test_categories([cat])
|
|
@wraps(attrvalue)
|
|
def test_method(self, attrvalue=attrvalue):
|
|
return attrvalue(self)
|
|
|
|
method_name = attrname + "_" + cat
|
|
test_method.__name__ = method_name
|
|
test_method.debug_info = cat
|
|
newattrs[method_name] = test_method
|
|
|
|
else:
|
|
newattrs[attrname] = attrvalue
|
|
return super(
|
|
LLDBTestCaseFactory,
|
|
cls).__new__(
|
|
cls,
|
|
name,
|
|
bases,
|
|
newattrs)
|
|
|
|
# Setup the metaclass for this class to change the list of the test
|
|
# methods when a new class is loaded
|
|
|
|
|
|
@add_metaclass(LLDBTestCaseFactory)
|
|
class TestBase(Base):
|
|
"""
|
|
This abstract base class is meant to be subclassed. It provides default
|
|
implementations for setUpClass(), tearDownClass(), setUp(), and tearDown(),
|
|
among other things.
|
|
|
|
Important things for test class writers:
|
|
|
|
- Overwrite the mydir class attribute, otherwise your test class won't
|
|
run. It specifies the relative directory to the top level 'test' so
|
|
the test harness can change to the correct working directory before
|
|
running your test.
|
|
|
|
- The setUp method sets up things to facilitate subsequent interactions
|
|
with the debugger as part of the test. These include:
|
|
- populate the test method name
|
|
- create/get a debugger set with synchronous mode (self.dbg)
|
|
- get the command interpreter from with the debugger (self.ci)
|
|
- create a result object for use with the command interpreter
|
|
(self.res)
|
|
- plus other stuffs
|
|
|
|
- The tearDown method tries to perform some necessary cleanup on behalf
|
|
of the test to return the debugger to a good state for the next test.
|
|
These include:
|
|
- execute any tearDown hooks registered by the test method with
|
|
TestBase.addTearDownHook(); examples can be found in
|
|
settings/TestSettings.py
|
|
- kill the inferior process associated with each target, if any,
|
|
and, then delete the target from the debugger's target list
|
|
- perform build cleanup before running the next test method in the
|
|
same test class; examples of registering for this service can be
|
|
found in types/TestIntegerTypes.py with the call:
|
|
- self.setTearDownCleanup(dictionary=d)
|
|
|
|
- Similarly setUpClass and tearDownClass perform classwise setup and
|
|
teardown fixtures. The tearDownClass method invokes a default build
|
|
cleanup for the entire test class; also, subclasses can implement the
|
|
classmethod classCleanup(cls) to perform special class cleanup action.
|
|
|
|
- The instance methods runCmd and expect are used heavily by existing
|
|
test cases to send a command to the command interpreter and to perform
|
|
string/pattern matching on the output of such command execution. The
|
|
expect method also provides a mode to peform string/pattern matching
|
|
without running a command.
|
|
|
|
- The build methods buildDefault, buildDsym, and buildDwarf are used to
|
|
build the binaries used during a particular test scenario. A plugin
|
|
should be provided for the sys.platform running the test suite. The
|
|
Mac OS X implementation is located in builders/darwin.py.
|
|
"""
|
|
|
|
# Subclasses can set this to true (if they don't depend on debug info) to avoid running the
|
|
# test multiple times with various debug info types.
|
|
NO_DEBUG_INFO_TESTCASE = False
|
|
|
|
# Maximum allowed attempts when launching the inferior process.
|
|
# Can be overridden by the LLDB_MAX_LAUNCH_COUNT environment variable.
|
|
maxLaunchCount = 1
|
|
|
|
# Time to wait before the next launching attempt in second(s).
|
|
# Can be overridden by the LLDB_TIME_WAIT_NEXT_LAUNCH environment variable.
|
|
timeWaitNextLaunch = 1.0
|
|
|
|
def generateSource(self, source):
|
|
template = source + '.template'
|
|
temp = os.path.join(self.getSourceDir(), template)
|
|
with open(temp, 'r') as f:
|
|
content = f.read()
|
|
|
|
public_api_dir = os.path.join(
|
|
os.environ["LLDB_SRC"], "include", "lldb", "API")
|
|
|
|
# Look under the include/lldb/API directory and add #include statements
|
|
# for all the SB API headers.
|
|
public_headers = os.listdir(public_api_dir)
|
|
# For different platforms, the include statement can vary.
|
|
if self.hasDarwinFramework():
|
|
include_stmt = "'#include <%s>' % os.path.join('LLDB', header)"
|
|
else:
|
|
include_stmt = "'#include <%s>' % os.path.join('" + public_api_dir + "', header)"
|
|
list = [eval(include_stmt) for header in public_headers if (
|
|
header.startswith("SB") and header.endswith(".h"))]
|
|
includes = '\n'.join(list)
|
|
new_content = content.replace('%include_SB_APIs%', includes)
|
|
new_content = new_content.replace('%SOURCE_DIR%', self.getSourceDir())
|
|
src = os.path.join(self.getBuildDir(), source)
|
|
with open(src, 'w') as f:
|
|
f.write(new_content)
|
|
|
|
self.addTearDownHook(lambda: os.remove(src))
|
|
|
|
def setUp(self):
|
|
# Works with the test driver to conditionally skip tests via
|
|
# decorators.
|
|
Base.setUp(self)
|
|
|
|
for s in self.setUpCommands():
|
|
self.runCmd(s)
|
|
|
|
if "LLDB_MAX_LAUNCH_COUNT" in os.environ:
|
|
self.maxLaunchCount = int(os.environ["LLDB_MAX_LAUNCH_COUNT"])
|
|
|
|
if "LLDB_TIME_WAIT_NEXT_LAUNCH" in os.environ:
|
|
self.timeWaitNextLaunch = float(
|
|
os.environ["LLDB_TIME_WAIT_NEXT_LAUNCH"])
|
|
|
|
# We want our debugger to be synchronous.
|
|
self.dbg.SetAsync(False)
|
|
|
|
# Retrieve the associated command interpreter instance.
|
|
self.ci = self.dbg.GetCommandInterpreter()
|
|
if not self.ci:
|
|
raise Exception('Could not get the command interpreter')
|
|
|
|
# And the result object.
|
|
self.res = lldb.SBCommandReturnObject()
|
|
|
|
def registerSharedLibrariesWithTarget(self, target, shlibs):
|
|
'''If we are remotely running the test suite, register the shared libraries with the target so they get uploaded, otherwise do nothing
|
|
|
|
Any modules in the target that have their remote install file specification set will
|
|
get uploaded to the remote host. This function registers the local copies of the
|
|
shared libraries with the target and sets their remote install locations so they will
|
|
be uploaded when the target is run.
|
|
'''
|
|
if not shlibs or not self.platformContext:
|
|
return None
|
|
|
|
shlib_environment_var = self.platformContext.shlib_environment_var
|
|
shlib_prefix = self.platformContext.shlib_prefix
|
|
shlib_extension = '.' + self.platformContext.shlib_extension
|
|
|
|
dirs = []
|
|
# Add any shared libraries to our target if remote so they get
|
|
# uploaded into the working directory on the remote side
|
|
for name in shlibs:
|
|
# The path can be a full path to a shared library, or a make file name like "Foo" for
|
|
# "libFoo.dylib" or "libFoo.so", or "Foo.so" for "Foo.so" or "libFoo.so", or just a
|
|
# basename like "libFoo.so". So figure out which one it is and resolve the local copy
|
|
# of the shared library accordingly
|
|
if os.path.isfile(name):
|
|
local_shlib_path = name # name is the full path to the local shared library
|
|
else:
|
|
# Check relative names
|
|
local_shlib_path = os.path.join(
|
|
self.getBuildDir(), shlib_prefix + name + shlib_extension)
|
|
if not os.path.exists(local_shlib_path):
|
|
local_shlib_path = os.path.join(
|
|
self.getBuildDir(), name + shlib_extension)
|
|
if not os.path.exists(local_shlib_path):
|
|
local_shlib_path = os.path.join(self.getBuildDir(), name)
|
|
|
|
# Make sure we found the local shared library in the above code
|
|
self.assertTrue(os.path.exists(local_shlib_path))
|
|
|
|
|
|
# Add the shared library to our target
|
|
shlib_module = target.AddModule(local_shlib_path, None, None, None)
|
|
if lldb.remote_platform:
|
|
# We must set the remote install location if we want the shared library
|
|
# to get uploaded to the remote target
|
|
remote_shlib_path = lldbutil.append_to_process_working_directory(self,
|
|
os.path.basename(local_shlib_path))
|
|
shlib_module.SetRemoteInstallFileSpec(
|
|
lldb.SBFileSpec(remote_shlib_path, False))
|
|
dir_to_add = self.get_process_working_directory()
|
|
else:
|
|
dir_to_add = os.path.dirname(local_shlib_path)
|
|
|
|
if dir_to_add not in dirs:
|
|
dirs.append(dir_to_add)
|
|
|
|
env_value = self.platformContext.shlib_path_separator.join(dirs)
|
|
return ['%s=%s' % (shlib_environment_var, env_value)]
|
|
|
|
def registerSanitizerLibrariesWithTarget(self, target):
|
|
runtimes = []
|
|
for m in target.module_iter():
|
|
libspec = m.GetFileSpec()
|
|
if "clang_rt" in libspec.GetFilename():
|
|
runtimes.append(os.path.join(libspec.GetDirectory(),
|
|
libspec.GetFilename()))
|
|
return self.registerSharedLibrariesWithTarget(target, runtimes)
|
|
|
|
# utility methods that tests can use to access the current objects
|
|
def target(self):
|
|
if not self.dbg:
|
|
raise Exception('Invalid debugger instance')
|
|
return self.dbg.GetSelectedTarget()
|
|
|
|
def process(self):
|
|
if not self.dbg:
|
|
raise Exception('Invalid debugger instance')
|
|
return self.dbg.GetSelectedTarget().GetProcess()
|
|
|
|
def thread(self):
|
|
if not self.dbg:
|
|
raise Exception('Invalid debugger instance')
|
|
return self.dbg.GetSelectedTarget().GetProcess().GetSelectedThread()
|
|
|
|
def frame(self):
|
|
if not self.dbg:
|
|
raise Exception('Invalid debugger instance')
|
|
return self.dbg.GetSelectedTarget().GetProcess(
|
|
).GetSelectedThread().GetSelectedFrame()
|
|
|
|
def get_process_working_directory(self):
|
|
'''Get the working directory that should be used when launching processes for local or remote processes.'''
|
|
if lldb.remote_platform:
|
|
# Remote tests set the platform working directory up in
|
|
# TestBase.setUp()
|
|
return lldb.remote_platform.GetWorkingDirectory()
|
|
else:
|
|
# local tests change directory into each test subdirectory
|
|
return self.getBuildDir()
|
|
|
|
def tearDown(self):
|
|
# Ensure all the references to SB objects have gone away so that we can
|
|
# be sure that all test-specific resources have been freed before we
|
|
# attempt to delete the targets.
|
|
gc.collect()
|
|
|
|
# Delete the target(s) from the debugger as a general cleanup step.
|
|
# This includes terminating the process for each target, if any.
|
|
# We'd like to reuse the debugger for our next test without incurring
|
|
# the initialization overhead.
|
|
targets = []
|
|
for target in self.dbg:
|
|
if target:
|
|
targets.append(target)
|
|
process = target.GetProcess()
|
|
if process:
|
|
rc = self.invoke(process, "Kill")
|
|
assert rc.Success()
|
|
for target in targets:
|
|
self.dbg.DeleteTarget(target)
|
|
|
|
if not configuration.is_reproducer():
|
|
# Assert that all targets are deleted.
|
|
self.assertEqual(self.dbg.GetNumTargets(), 0)
|
|
|
|
# Do this last, to make sure it's in reverse order from how we setup.
|
|
Base.tearDown(self)
|
|
|
|
def switch_to_thread_with_stop_reason(self, stop_reason):
|
|
"""
|
|
Run the 'thread list' command, and select the thread with stop reason as
|
|
'stop_reason'. If no such thread exists, no select action is done.
|
|
"""
|
|
from .lldbutil import stop_reason_to_str
|
|
self.runCmd('thread list')
|
|
output = self.res.GetOutput()
|
|
thread_line_pattern = re.compile(
|
|
"^[ *] thread #([0-9]+):.*stop reason = %s" %
|
|
stop_reason_to_str(stop_reason))
|
|
for line in output.splitlines():
|
|
matched = thread_line_pattern.match(line)
|
|
if matched:
|
|
self.runCmd('thread select %s' % matched.group(1))
|
|
|
|
def runCmd(self, cmd, msg=None, check=True, trace=False, inHistory=False):
|
|
"""
|
|
Ask the command interpreter to handle the command and then check its
|
|
return status.
|
|
"""
|
|
# Fail fast if 'cmd' is not meaningful.
|
|
if cmd is None:
|
|
raise Exception("Bad 'cmd' parameter encountered")
|
|
|
|
trace = (True if traceAlways else trace)
|
|
|
|
if cmd.startswith("target create "):
|
|
cmd = cmd.replace("target create ", "file ")
|
|
|
|
running = (cmd.startswith("run") or cmd.startswith("process launch"))
|
|
|
|
for i in range(self.maxLaunchCount if running else 1):
|
|
self.ci.HandleCommand(cmd, self.res, inHistory)
|
|
|
|
with recording(self, trace) as sbuf:
|
|
print("runCmd:", cmd, file=sbuf)
|
|
if not check:
|
|
print("check of return status not required", file=sbuf)
|
|
if self.res.Succeeded():
|
|
print("output:", self.res.GetOutput(), file=sbuf)
|
|
else:
|
|
print("runCmd failed!", file=sbuf)
|
|
print(self.res.GetError(), file=sbuf)
|
|
|
|
if self.res.Succeeded():
|
|
break
|
|
elif running:
|
|
# For process launch, wait some time before possible next try.
|
|
time.sleep(self.timeWaitNextLaunch)
|
|
with recording(self, trace) as sbuf:
|
|
print("Command '" + cmd + "' failed!", file=sbuf)
|
|
|
|
if check:
|
|
output = ""
|
|
if self.res.GetOutput():
|
|
output += "\nCommand output:\n" + self.res.GetOutput()
|
|
if self.res.GetError():
|
|
output += "\nError output:\n" + self.res.GetError()
|
|
if msg:
|
|
msg += output
|
|
if cmd:
|
|
cmd += output
|
|
self.assertTrue(self.res.Succeeded(),
|
|
msg if (msg) else CMD_MSG(cmd))
|
|
|
|
def match(
|
|
self,
|
|
str,
|
|
patterns,
|
|
msg=None,
|
|
trace=False,
|
|
error=False,
|
|
matching=True,
|
|
exe=True):
|
|
"""run command in str, and match the result against regexp in patterns returning the match object for the first matching pattern
|
|
|
|
Otherwise, all the arguments have the same meanings as for the expect function"""
|
|
|
|
trace = (True if traceAlways else trace)
|
|
|
|
if exe:
|
|
# First run the command. If we are expecting error, set check=False.
|
|
# Pass the assert message along since it provides more semantic
|
|
# info.
|
|
self.runCmd(
|
|
str,
|
|
msg=msg,
|
|
trace=(
|
|
True if trace else False),
|
|
check=not error)
|
|
|
|
# Then compare the output against expected strings.
|
|
output = self.res.GetError() if error else self.res.GetOutput()
|
|
|
|
# If error is True, the API client expects the command to fail!
|
|
if error:
|
|
self.assertFalse(self.res.Succeeded(),
|
|
"Command '" + str + "' is expected to fail!")
|
|
else:
|
|
# No execution required, just compare str against the golden input.
|
|
output = str
|
|
with recording(self, trace) as sbuf:
|
|
print("looking at:", output, file=sbuf)
|
|
|
|
# The heading says either "Expecting" or "Not expecting".
|
|
heading = "Expecting" if matching else "Not expecting"
|
|
|
|
for pattern in patterns:
|
|
# Match Objects always have a boolean value of True.
|
|
match_object = re.search(pattern, output)
|
|
matched = bool(match_object)
|
|
with recording(self, trace) as sbuf:
|
|
print("%s pattern: %s" % (heading, pattern), file=sbuf)
|
|
print("Matched" if matched else "Not matched", file=sbuf)
|
|
if matched:
|
|
break
|
|
|
|
self.assertTrue(matched if matching else not matched,
|
|
msg if msg else EXP_MSG(str, output, exe))
|
|
|
|
return match_object
|
|
|
|
def check_completion_with_desc(self, str_input, match_desc_pairs, enforce_order=False):
|
|
"""
|
|
Checks that when the given input is completed at the given list of
|
|
completions and descriptions is returned.
|
|
:param str_input: The input that should be completed. The completion happens at the end of the string.
|
|
:param match_desc_pairs: A list of pairs that indicate what completions have to be in the list of
|
|
completions returned by LLDB. The first element of the pair is the completion
|
|
string that LLDB should generate and the second element the description.
|
|
:param enforce_order: True iff the order in which the completions are returned by LLDB
|
|
should match the order of the match_desc_pairs pairs.
|
|
"""
|
|
interp = self.dbg.GetCommandInterpreter()
|
|
match_strings = lldb.SBStringList()
|
|
description_strings = lldb.SBStringList()
|
|
num_matches = interp.HandleCompletionWithDescriptions(str_input, len(str_input), 0, -1, match_strings, description_strings)
|
|
self.assertEqual(len(description_strings), len(match_strings))
|
|
|
|
# The index of the last matched description in description_strings or
|
|
# -1 if no description has been matched yet.
|
|
last_found_index = -1
|
|
out_of_order_errors = ""
|
|
missing_pairs = []
|
|
for pair in match_desc_pairs:
|
|
found_pair = False
|
|
for i in range(num_matches + 1):
|
|
match_candidate = match_strings.GetStringAtIndex(i)
|
|
description_candidate = description_strings.GetStringAtIndex(i)
|
|
if match_candidate == pair[0] and description_candidate == pair[1]:
|
|
found_pair = True
|
|
if enforce_order and last_found_index > i:
|
|
new_err = ("Found completion " + pair[0] + " at index " +
|
|
str(i) + " in returned completion list but " +
|
|
"should have been after completion " +
|
|
match_strings.GetStringAtIndex(last_found_index) +
|
|
" (index:" + str(last_found_index) + ")\n")
|
|
out_of_order_errors += new_err
|
|
last_found_index = i
|
|
break
|
|
if not found_pair:
|
|
missing_pairs.append(pair)
|
|
|
|
error_msg = ""
|
|
got_failure = False
|
|
if len(missing_pairs):
|
|
got_failure = True
|
|
error_msg += "Missing pairs:\n"
|
|
for pair in missing_pairs:
|
|
error_msg += " [" + pair[0] + ":" + pair[1] + "]\n"
|
|
if len(out_of_order_errors):
|
|
got_failure = True
|
|
error_msg += out_of_order_errors
|
|
if got_failure:
|
|
error_msg += "Got the following " + str(num_matches) + " completions back:\n"
|
|
for i in range(num_matches + 1):
|
|
match_candidate = match_strings.GetStringAtIndex(i)
|
|
description_candidate = description_strings.GetStringAtIndex(i)
|
|
error_msg += "[" + match_candidate + ":" + description_candidate + "] index " + str(i) + "\n"
|
|
self.assertFalse(got_failure, error_msg)
|
|
|
|
def complete_exactly(self, str_input, patterns):
|
|
self.complete_from_to(str_input, patterns, True)
|
|
|
|
def complete_from_to(self, str_input, patterns, turn_off_re_match=False):
|
|
"""Test that the completion mechanism completes str_input to patterns,
|
|
where patterns could be a pattern-string or a list of pattern-strings"""
|
|
# Patterns should not be None in order to proceed.
|
|
self.assertFalse(patterns is None)
|
|
# And should be either a string or list of strings. Check for list type
|
|
# below, if not, make a list out of the singleton string. If patterns
|
|
# is not a string or not a list of strings, there'll be runtime errors
|
|
# later on.
|
|
if not isinstance(patterns, list):
|
|
patterns = [patterns]
|
|
|
|
interp = self.dbg.GetCommandInterpreter()
|
|
match_strings = lldb.SBStringList()
|
|
num_matches = interp.HandleCompletion(str_input, len(str_input), 0, -1, match_strings)
|
|
common_match = match_strings.GetStringAtIndex(0)
|
|
if num_matches == 0:
|
|
compare_string = str_input
|
|
else:
|
|
if common_match != None and len(common_match) > 0:
|
|
compare_string = str_input + common_match
|
|
else:
|
|
compare_string = ""
|
|
for idx in range(1, num_matches+1):
|
|
compare_string += match_strings.GetStringAtIndex(idx) + "\n"
|
|
|
|
for p in patterns:
|
|
if turn_off_re_match:
|
|
self.expect(
|
|
compare_string, msg=COMPLETION_MSG(
|
|
str_input, p, match_strings), exe=False, substrs=[p])
|
|
else:
|
|
self.expect(
|
|
compare_string, msg=COMPLETION_MSG(
|
|
str_input, p, match_strings), exe=False, patterns=[p])
|
|
|
|
def completions_match(self, command, completions):
|
|
"""Checks that the completions for the given command are equal to the
|
|
given list of completions"""
|
|
interp = self.dbg.GetCommandInterpreter()
|
|
match_strings = lldb.SBStringList()
|
|
interp.HandleCompletion(command, len(command), 0, -1, match_strings)
|
|
# match_strings is a 1-indexed list, so we have to slice...
|
|
self.assertItemsEqual(completions, list(match_strings)[1:],
|
|
"List of returned completion is wrong")
|
|
|
|
def completions_contain(self, command, completions):
|
|
"""Checks that the completions for the given command contain the given
|
|
list of completions."""
|
|
interp = self.dbg.GetCommandInterpreter()
|
|
match_strings = lldb.SBStringList()
|
|
interp.HandleCompletion(command, len(command), 0, -1, match_strings)
|
|
for completion in completions:
|
|
# match_strings is a 1-indexed list, so we have to slice...
|
|
self.assertIn(completion, list(match_strings)[1:],
|
|
"Couldn't find expected completion")
|
|
|
|
def filecheck(
|
|
self,
|
|
command,
|
|
check_file,
|
|
filecheck_options = '',
|
|
expect_cmd_failure = False):
|
|
# Run the command.
|
|
self.runCmd(
|
|
command,
|
|
check=(not expect_cmd_failure),
|
|
msg="FileCheck'ing result of `{0}`".format(command))
|
|
|
|
self.assertTrue((not expect_cmd_failure) == self.res.Succeeded())
|
|
|
|
# Get the error text if there was an error, and the regular text if not.
|
|
output = self.res.GetOutput() if self.res.Succeeded() \
|
|
else self.res.GetError()
|
|
|
|
# Assemble the absolute path to the check file. As a convenience for
|
|
# LLDB inline tests, assume that the check file is a relative path to
|
|
# a file within the inline test directory.
|
|
if check_file.endswith('.pyc'):
|
|
check_file = check_file[:-1]
|
|
check_file_abs = os.path.abspath(check_file)
|
|
|
|
# Run FileCheck.
|
|
filecheck_bin = configuration.get_filecheck_path()
|
|
if not filecheck_bin:
|
|
self.assertTrue(False, "No valid FileCheck executable specified")
|
|
filecheck_args = [filecheck_bin, check_file_abs]
|
|
if filecheck_options:
|
|
filecheck_args.append(filecheck_options)
|
|
subproc = Popen(filecheck_args, stdin=PIPE, stdout=PIPE, stderr=PIPE, universal_newlines = True)
|
|
cmd_stdout, cmd_stderr = subproc.communicate(input=output)
|
|
cmd_status = subproc.returncode
|
|
|
|
filecheck_cmd = " ".join(filecheck_args)
|
|
filecheck_trace = """
|
|
--- FileCheck trace (code={0}) ---
|
|
{1}
|
|
|
|
FileCheck input:
|
|
{2}
|
|
|
|
FileCheck output:
|
|
{3}
|
|
{4}
|
|
""".format(cmd_status, filecheck_cmd, output, cmd_stdout, cmd_stderr)
|
|
|
|
trace = cmd_status != 0 or traceAlways
|
|
with recording(self, trace) as sbuf:
|
|
print(filecheck_trace, file=sbuf)
|
|
|
|
self.assertTrue(cmd_status == 0)
|
|
|
|
def expect(
|
|
self,
|
|
str,
|
|
msg=None,
|
|
patterns=None,
|
|
startstr=None,
|
|
endstr=None,
|
|
substrs=None,
|
|
trace=False,
|
|
error=False,
|
|
ordered=True,
|
|
matching=True,
|
|
exe=True,
|
|
inHistory=False):
|
|
"""
|
|
Similar to runCmd; with additional expect style output matching ability.
|
|
|
|
Ask the command interpreter to handle the command and then check its
|
|
return status. The 'msg' parameter specifies an informational assert
|
|
message. We expect the output from running the command to start with
|
|
'startstr', matches the substrings contained in 'substrs', and regexp
|
|
matches the patterns contained in 'patterns'.
|
|
|
|
When matching is true and ordered is true, which are both the default,
|
|
the strings in the substrs array have to appear in the command output
|
|
in the order in which they appear in the array.
|
|
|
|
If the keyword argument error is set to True, it signifies that the API
|
|
client is expecting the command to fail. In this case, the error stream
|
|
from running the command is retrieved and compared against the golden
|
|
input, instead.
|
|
|
|
If the keyword argument matching is set to False, it signifies that the API
|
|
client is expecting the output of the command not to match the golden
|
|
input.
|
|
|
|
Finally, the required argument 'str' represents the lldb command to be
|
|
sent to the command interpreter. In case the keyword argument 'exe' is
|
|
set to False, the 'str' is treated as a string to be matched/not-matched
|
|
against the golden input.
|
|
"""
|
|
# Catch cases where `expect` has been miscalled. Specifically, prevent
|
|
# this easy to make mistake:
|
|
# self.expect("lldb command", "some substr")
|
|
# The `msg` parameter is used only when a failed match occurs. A failed
|
|
# match can only occur when one of `patterns`, `startstr`, `endstr`, or
|
|
# `substrs` has been given. Thus, if a `msg` is given, it's an error to
|
|
# not also provide one of the matcher parameters.
|
|
if msg and not (patterns or startstr or endstr or substrs or error):
|
|
assert False, "expect() missing a matcher argument"
|
|
|
|
# Check `patterns` and `substrs` are not accidentally given as strings.
|
|
assert not isinstance(patterns, six.string_types), \
|
|
"patterns must be a collection of strings"
|
|
assert not isinstance(substrs, six.string_types), \
|
|
"substrs must be a collection of strings"
|
|
|
|
trace = (True if traceAlways else trace)
|
|
|
|
if exe:
|
|
# First run the command. If we are expecting error, set check=False.
|
|
# Pass the assert message along since it provides more semantic
|
|
# info.
|
|
self.runCmd(
|
|
str,
|
|
msg=msg,
|
|
trace=(
|
|
True if trace else False),
|
|
check=not error,
|
|
inHistory=inHistory)
|
|
|
|
# Then compare the output against expected strings.
|
|
output = self.res.GetError() if error else self.res.GetOutput()
|
|
|
|
# If error is True, the API client expects the command to fail!
|
|
if error:
|
|
self.assertFalse(self.res.Succeeded(),
|
|
"Command '" + str + "' is expected to fail!")
|
|
else:
|
|
# No execution required, just compare str against the golden input.
|
|
if isinstance(str, lldb.SBCommandReturnObject):
|
|
output = str.GetOutput()
|
|
else:
|
|
output = str
|
|
with recording(self, trace) as sbuf:
|
|
print("looking at:", output, file=sbuf)
|
|
|
|
expecting_str = "Expecting" if matching else "Not expecting"
|
|
def found_str(matched):
|
|
return "was found" if matched else "was not found"
|
|
|
|
# To be used as assert fail message and/or trace content
|
|
log_lines = [
|
|
"{}:".format("Ran command" if exe else "Checking string"),
|
|
"\"{}\"".format(str),
|
|
# Space out command and output
|
|
"",
|
|
]
|
|
if exe:
|
|
# Newline before output to make large strings more readable
|
|
log_lines.append("Got output:\n{}".format(output))
|
|
|
|
# Assume that we start matched if we want a match
|
|
# Meaning if you have no conditions, matching or
|
|
# not matching will always pass
|
|
matched = matching
|
|
|
|
# We will stop checking on first failure
|
|
if startstr:
|
|
matched = output.startswith(startstr)
|
|
log_lines.append("{} start string: \"{}\" ({})".format(
|
|
expecting_str, startstr, found_str(matched)))
|
|
|
|
if endstr and matched == matching:
|
|
matched = output.endswith(endstr)
|
|
log_lines.append("{} end string: \"{}\" ({})".format(
|
|
expecting_str, endstr, found_str(matched)))
|
|
|
|
if substrs and matched == matching:
|
|
start = 0
|
|
for substr in substrs:
|
|
index = output[start:].find(substr)
|
|
start = start + index if ordered and matching else 0
|
|
matched = index != -1
|
|
log_lines.append("{} sub string: \"{}\" ({})".format(
|
|
expecting_str, substr, found_str(matched)))
|
|
|
|
if matched != matching:
|
|
break
|
|
|
|
if patterns and matched == matching:
|
|
for pattern in patterns:
|
|
matched = re.search(pattern, output)
|
|
|
|
pattern_line = "{} regex pattern: \"{}\" ({}".format(
|
|
expecting_str, pattern, found_str(matched))
|
|
if matched:
|
|
pattern_line += ", matched \"{}\"".format(
|
|
matched.group(0))
|
|
pattern_line += ")"
|
|
log_lines.append(pattern_line)
|
|
|
|
# Convert to bool because match objects
|
|
# are True-ish but != True itself
|
|
matched = bool(matched)
|
|
if matched != matching:
|
|
break
|
|
|
|
# If a check failed, add any extra assert message
|
|
if msg is not None and matched != matching:
|
|
log_lines.append(msg)
|
|
|
|
log_msg = "\n".join(log_lines)
|
|
with recording(self, trace) as sbuf:
|
|
print(log_msg, file=sbuf)
|
|
if matched != matching:
|
|
self.fail(log_msg)
|
|
|
|
def expect_expr(
|
|
self,
|
|
expr,
|
|
result_summary=None,
|
|
result_value=None,
|
|
result_type=None,
|
|
result_children=None
|
|
):
|
|
"""
|
|
Evaluates the given expression and verifies the result.
|
|
:param expr: The expression as a string.
|
|
:param result_summary: The summary that the expression should have. None if the summary should not be checked.
|
|
:param result_value: The value that the expression should have. None if the value should not be checked.
|
|
:param result_type: The type that the expression result should have. None if the type should not be checked.
|
|
:param result_children: The expected children of the expression result
|
|
as a list of ValueChecks. None if the children shouldn't be checked.
|
|
"""
|
|
self.assertTrue(expr.strip() == expr, "Expression contains trailing/leading whitespace: '" + expr + "'")
|
|
|
|
frame = self.frame()
|
|
options = lldb.SBExpressionOptions()
|
|
|
|
# Disable fix-its that tests don't pass by accident.
|
|
options.SetAutoApplyFixIts(False)
|
|
|
|
# Set the usual default options for normal expressions.
|
|
options.SetIgnoreBreakpoints(True)
|
|
|
|
if self.frame().IsValid():
|
|
options.SetLanguage(frame.GuessLanguage())
|
|
eval_result = self.frame().EvaluateExpression(expr, options)
|
|
else:
|
|
target = self.target()
|
|
# If there is no selected target, run the expression in the dummy
|
|
# target.
|
|
if not target.IsValid():
|
|
target = self.dbg.GetDummyTarget()
|
|
eval_result = target.EvaluateExpression(expr, options)
|
|
|
|
value_check = ValueCheck(type=result_type, value=result_value,
|
|
summary=result_summary, children=result_children)
|
|
value_check.check_value(self, eval_result, str(eval_result))
|
|
return eval_result
|
|
|
|
def expect_var_path(
|
|
self,
|
|
var_path,
|
|
summary=None,
|
|
value=None,
|
|
type=None,
|
|
children=None
|
|
):
|
|
"""
|
|
Evaluates the given variable path and verifies the result.
|
|
See also 'frame variable' and SBFrame.GetValueForVariablePath.
|
|
:param var_path: The variable path as a string.
|
|
:param summary: The summary that the variable should have. None if the summary should not be checked.
|
|
:param value: The value that the variable should have. None if the value should not be checked.
|
|
:param type: The type that the variable result should have. None if the type should not be checked.
|
|
:param children: The expected children of the variable as a list of ValueChecks.
|
|
None if the children shouldn't be checked.
|
|
"""
|
|
self.assertTrue(var_path.strip() == var_path,
|
|
"Expression contains trailing/leading whitespace: '" + var_path + "'")
|
|
|
|
frame = self.frame()
|
|
eval_result = frame.GetValueForVariablePath(var_path)
|
|
|
|
value_check = ValueCheck(type=type, value=value,
|
|
summary=summary, children=children)
|
|
value_check.check_value(self, eval_result, str(eval_result))
|
|
return eval_result
|
|
|
|
def build(
|
|
self,
|
|
architecture=None,
|
|
compiler=None,
|
|
dictionary=None):
|
|
"""Platform specific way to build the default binaries."""
|
|
module = builder_module()
|
|
|
|
if not architecture and configuration.arch:
|
|
architecture = configuration.arch
|
|
|
|
dictionary = lldbplatformutil.finalize_build_dictionary(dictionary)
|
|
if self.getDebugInfo() is None:
|
|
return self.buildDefault(architecture, compiler, dictionary)
|
|
elif self.getDebugInfo() == "dsym":
|
|
return self.buildDsym(architecture, compiler, dictionary)
|
|
elif self.getDebugInfo() == "dwarf":
|
|
return self.buildDwarf(architecture, compiler, dictionary)
|
|
elif self.getDebugInfo() == "dwo":
|
|
return self.buildDwo(architecture, compiler, dictionary)
|
|
elif self.getDebugInfo() == "gmodules":
|
|
return self.buildGModules(architecture, compiler, dictionary)
|
|
else:
|
|
self.fail("Can't build for debug info: %s" % self.getDebugInfo())
|
|
|
|
"""Assert that an lldb.SBError is in the "success" state."""
|
|
def assertSuccess(self, obj, msg=None):
|
|
if not obj.Success():
|
|
error = obj.GetCString()
|
|
self.fail(self._formatMessage(msg,
|
|
"'{}' is not success".format(error)))
|
|
|
|
# =================================================
|
|
# Misc. helper methods for debugging test execution
|
|
# =================================================
|
|
|
|
def DebugSBValue(self, val):
|
|
"""Debug print a SBValue object, if traceAlways is True."""
|
|
from .lldbutil import value_type_to_str
|
|
|
|
if not traceAlways:
|
|
return
|
|
|
|
err = sys.stderr
|
|
err.write(val.GetName() + ":\n")
|
|
err.write('\t' + "TypeName -> " + val.GetTypeName() + '\n')
|
|
err.write('\t' + "ByteSize -> " +
|
|
str(val.GetByteSize()) + '\n')
|
|
err.write('\t' + "NumChildren -> " +
|
|
str(val.GetNumChildren()) + '\n')
|
|
err.write('\t' + "Value -> " + str(val.GetValue()) + '\n')
|
|
err.write('\t' + "ValueAsUnsigned -> " +
|
|
str(val.GetValueAsUnsigned()) + '\n')
|
|
err.write(
|
|
'\t' +
|
|
"ValueType -> " +
|
|
value_type_to_str(
|
|
val.GetValueType()) +
|
|
'\n')
|
|
err.write('\t' + "Summary -> " + str(val.GetSummary()) + '\n')
|
|
err.write('\t' + "IsPointerType -> " +
|
|
str(val.TypeIsPointerType()) + '\n')
|
|
err.write('\t' + "Location -> " + val.GetLocation() + '\n')
|
|
|
|
def DebugSBType(self, type):
|
|
"""Debug print a SBType object, if traceAlways is True."""
|
|
if not traceAlways:
|
|
return
|
|
|
|
err = sys.stderr
|
|
err.write(type.GetName() + ":\n")
|
|
err.write('\t' + "ByteSize -> " +
|
|
str(type.GetByteSize()) + '\n')
|
|
err.write('\t' + "IsPointerType -> " +
|
|
str(type.IsPointerType()) + '\n')
|
|
err.write('\t' + "IsReferenceType -> " +
|
|
str(type.IsReferenceType()) + '\n')
|
|
|
|
def DebugPExpect(self, child):
|
|
"""Debug the spwaned pexpect object."""
|
|
if not traceAlways:
|
|
return
|
|
|
|
print(child)
|
|
|
|
@classmethod
|
|
def RemoveTempFile(cls, file):
|
|
if os.path.exists(file):
|
|
remove_file(file)
|
|
|
|
# On Windows, the first attempt to delete a recently-touched file can fail
|
|
# because of a race with antimalware scanners. This function will detect a
|
|
# failure and retry.
|
|
|
|
|
|
def remove_file(file, num_retries=1, sleep_duration=0.5):
|
|
for i in range(num_retries + 1):
|
|
try:
|
|
os.remove(file)
|
|
return True
|
|
except:
|
|
time.sleep(sleep_duration)
|
|
continue
|
|
return False
|