summaryrefslogtreecommitdiff
path: root/packages/Python/lldbsuite/test/test_runner
diff options
context:
space:
mode:
authorDimitry Andric <dim@FreeBSD.org>2016-01-06 20:12:03 +0000
committerDimitry Andric <dim@FreeBSD.org>2016-01-06 20:12:03 +0000
commit9e6d35490a6542f9c97607f93c2ef8ca8e03cbcc (patch)
treedd2a1ddf0476664c2b823409c36cbccd52662ca7 /packages/Python/lldbsuite/test/test_runner
parent3bd2e91faeb9eeec1aae82c64a3253afff551cfd (diff)
Notes
Diffstat (limited to 'packages/Python/lldbsuite/test/test_runner')
-rw-r--r--packages/Python/lldbsuite/test/test_runner/README.txt5
-rw-r--r--packages/Python/lldbsuite/test/test_runner/lib/lldb_utils.py66
-rw-r--r--packages/Python/lldbsuite/test/test_runner/lib/process_control.py705
-rwxr-xr-xpackages/Python/lldbsuite/test/test_runner/test/inferior.py146
-rwxr-xr-xpackages/Python/lldbsuite/test/test_runner/test/process_control_tests.py235
5 files changed, 1157 insertions, 0 deletions
diff --git a/packages/Python/lldbsuite/test/test_runner/README.txt b/packages/Python/lldbsuite/test/test_runner/README.txt
new file mode 100644
index 0000000000000..bb40870e79651
--- /dev/null
+++ b/packages/Python/lldbsuite/test/test_runner/README.txt
@@ -0,0 +1,5 @@
+This directory contains source and tests for the lldb test runner
+architecture. This directory is not for lldb python tests. It
+is the test runner. The tests under this diretory are test-runner
+tests (i.e. tests that verify the test runner itself runs properly).
+
diff --git a/packages/Python/lldbsuite/test/test_runner/lib/lldb_utils.py b/packages/Python/lldbsuite/test/test_runner/lib/lldb_utils.py
new file mode 100644
index 0000000000000..e469bbf122072
--- /dev/null
+++ b/packages/Python/lldbsuite/test/test_runner/lib/lldb_utils.py
@@ -0,0 +1,66 @@
+"""
+The LLVM Compiler Infrastructure
+
+This file is distributed under the University of Illinois Open Source
+License. See LICENSE.TXT for details.
+
+Provides classes used by the test results reporting infrastructure
+within the LLDB test suite.
+
+
+This module contains utilities used by the lldb test framwork.
+"""
+
+
+class OptionalWith(object):
+ # pylint: disable=too-few-public-methods
+ # This is a wrapper - it is not meant to provide any extra methods.
+ """Provides a wrapper for objects supporting "with", allowing None.
+
+ This lets a user use the "with object" syntax for resource usage
+ (e.g. locks) even when the wrapped with object is None.
+
+ e.g.
+
+ wrapped_lock = OptionalWith(thread.Lock())
+ with wrapped_lock:
+ # Do something while the lock is obtained.
+ pass
+
+ might_be_none = None
+ wrapped_none = OptionalWith(might_be_none)
+ with wrapped_none:
+ # This code here still works.
+ pass
+
+ This prevents having to write code like this when
+ a lock is optional:
+
+ if lock:
+ lock.acquire()
+
+ try:
+ code_fragament_always_run()
+ finally:
+ if lock:
+ lock.release()
+
+ And I'd posit it is safer, as it becomes impossible to
+ forget the try/finally using OptionalWith(), since
+ the with syntax can be used.
+ """
+ def __init__(self, wrapped_object):
+ self.wrapped_object = wrapped_object
+
+ def __enter__(self):
+ if self.wrapped_object is not None:
+ return self.wrapped_object.__enter__()
+ else:
+ return self
+
+ def __exit__(self, the_type, value, traceback):
+ if self.wrapped_object is not None:
+ return self.wrapped_object.__exit__(the_type, value, traceback)
+ else:
+ # Don't suppress any exceptions
+ return False
diff --git a/packages/Python/lldbsuite/test/test_runner/lib/process_control.py b/packages/Python/lldbsuite/test/test_runner/lib/process_control.py
new file mode 100644
index 0000000000000..a7e639e4b8b65
--- /dev/null
+++ b/packages/Python/lldbsuite/test/test_runner/lib/process_control.py
@@ -0,0 +1,705 @@
+"""
+The LLVM Compiler Infrastructure
+
+This file is distributed under the University of Illinois Open Source
+License. See LICENSE.TXT for details.
+
+Provides classes used by the test results reporting infrastructure
+within the LLDB test suite.
+
+
+This module provides process-management support for the LLDB test
+running infrasructure.
+"""
+
+# System imports
+import os
+import re
+import signal
+import subprocess
+import sys
+import threading
+
+
+class CommunicatorThread(threading.Thread):
+ """Provides a thread class that communicates with a subprocess."""
+ def __init__(self, process, event, output_file):
+ super(CommunicatorThread, self).__init__()
+ # Don't let this thread prevent shutdown.
+ self.daemon = True
+ self.process = process
+ self.pid = process.pid
+ self.event = event
+ self.output_file = output_file
+ self.output = None
+
+ def run(self):
+ try:
+ # Communicate with the child process.
+ # This will not complete until the child process terminates.
+ self.output = self.process.communicate()
+ except Exception as exception: # pylint: disable=broad-except
+ if self.output_file:
+ self.output_file.write(
+ "exception while using communicate() for pid: {}\n".format(
+ exception))
+ finally:
+ # Signal that the thread's run is complete.
+ self.event.set()
+
+
+# Provides a regular expression for matching gtimeout-based durations.
+TIMEOUT_REGEX = re.compile(r"(^\d+)([smhd])?$")
+
+
+def timeout_to_seconds(timeout):
+ """Converts timeout/gtimeout timeout values into seconds.
+
+ @param timeout a timeout in the form of xm representing x minutes.
+
+ @return None if timeout is None, or the number of seconds as a float
+ if a valid timeout format was specified.
+ """
+ if timeout is None:
+ return None
+ else:
+ match = TIMEOUT_REGEX.match(timeout)
+ if match:
+ value = float(match.group(1))
+ units = match.group(2)
+ if units is None:
+ # default is seconds. No conversion necessary.
+ return value
+ elif units == 's':
+ # Seconds. No conversion necessary.
+ return value
+ elif units == 'm':
+ # Value is in minutes.
+ return 60.0 * value
+ elif units == 'h':
+ # Value is in hours.
+ return (60.0 * 60.0) * value
+ elif units == 'd':
+ # Value is in days.
+ return 24 * (60.0 * 60.0) * value
+ else:
+ raise Exception("unexpected units value '{}'".format(units))
+ else:
+ raise Exception("could not parse TIMEOUT spec '{}'".format(
+ timeout))
+
+
+class ProcessHelper(object):
+ """Provides an interface for accessing process-related functionality.
+
+ This class provides a factory method that gives the caller a
+ platform-specific implementation instance of the class.
+
+ Clients of the class should stick to the methods provided in this
+ base class.
+
+ @see ProcessHelper.process_helper()
+ """
+ def __init__(self):
+ super(ProcessHelper, self).__init__()
+
+ @classmethod
+ def process_helper(cls):
+ """Returns a platform-specific ProcessHelper instance.
+ @return a ProcessHelper instance that does the right thing for
+ the current platform.
+ """
+
+ # If you add a new platform, create an instance here and
+ # return it.
+ if os.name == "nt":
+ return WindowsProcessHelper()
+ else:
+ # For all POSIX-like systems.
+ return UnixProcessHelper()
+
+ def create_piped_process(self, command, new_process_group=True):
+ # pylint: disable=no-self-use,unused-argument
+ # As expected. We want derived classes to implement this.
+ """Creates a subprocess.Popen-based class with I/O piped to the parent.
+
+ @param command the command line list as would be passed to
+ subprocess.Popen(). Use the list form rather than the string form.
+
+ @param new_process_group indicates if the caller wants the
+ process to be created in its own process group. Each OS handles
+ this concept differently. It provides a level of isolation and
+ can simplify or enable terminating the process tree properly.
+
+ @return a subprocess.Popen-like object.
+ """
+ raise Exception("derived class must implement")
+
+ def supports_soft_terminate(self):
+ # pylint: disable=no-self-use
+ # As expected. We want derived classes to implement this.
+ """Indicates if the platform supports soft termination.
+
+ Soft termination is the concept of a terminate mechanism that
+ allows the target process to shut down nicely, but with the
+ catch that the process might choose to ignore it.
+
+ Platform supporter note: only mark soft terminate as supported
+ if the target process has some way to evade the soft terminate
+ request; otherwise, just support the hard terminate method.
+
+ @return True if the platform supports a soft terminate mechanism.
+ """
+ # By default, we do not support a soft terminate mechanism.
+ return False
+
+ def soft_terminate(self, popen_process, log_file=None, want_core=True):
+ # pylint: disable=no-self-use,unused-argument
+ # As expected. We want derived classes to implement this.
+ """Attempts to terminate the process in a polite way.
+
+ This terminate method is intended to give the child process a
+ chance to clean up and exit on its own, possibly with a request
+ to drop a core file or equivalent (i.e. [mini-]crashdump, crashlog,
+ etc.) If new_process_group was set in the process creation method
+ and the platform supports it, this terminate call will attempt to
+ kill the whole process tree rooted in this child process.
+
+ @param popen_process the subprocess.Popen-like object returned
+ by one of the process-creation methods of this class.
+
+ @param log_file file-like object used to emit error-related
+ logging info. May be None if no error-related info is desired.
+
+ @param want_core True if the caller would like to get a core
+ dump (or the analogous crash report) from the terminated process.
+ """
+ popen_process.terminate()
+
+ def hard_terminate(self, popen_process, log_file=None):
+ # pylint: disable=no-self-use,unused-argument
+ # As expected. We want derived classes to implement this.
+ """Attempts to terminate the process immediately.
+
+ This terminate method is intended to kill child process in
+ a manner in which the child process has no ability to block,
+ and also has no ability to clean up properly. If new_process_group
+ was specified when creating the process, and if the platform
+ implementation supports it, this will attempt to kill the
+ whole process tree rooted in the child process.
+
+ @param popen_process the subprocess.Popen-like object returned
+ by one of the process-creation methods of this class.
+
+ @param log_file file-like object used to emit error-related
+ logging info. May be None if no error-related info is desired.
+ """
+ popen_process.kill()
+
+ def was_soft_terminate(self, returncode, with_core):
+ # pylint: disable=no-self-use,unused-argument
+ # As expected. We want derived classes to implement this.
+ """Returns if Popen-like object returncode matches soft terminate.
+
+ @param returncode the returncode from the Popen-like object that
+ terminated with a given return code.
+
+ @param with_core indicates whether the returncode should match
+ a core-generating return signal.
+
+ @return True when the returncode represents what the system would
+ issue when a soft_terminate() with the given with_core arg occurred;
+ False otherwise.
+ """
+ if not self.supports_soft_terminate():
+ # If we don't support soft termination on this platform,
+ # then this should always be False.
+ return False
+ else:
+ # Once a platform claims to support soft terminate, it
+ # needs to be able to identify it by overriding this method.
+ raise Exception("platform needs to implement")
+
+ def was_hard_terminate(self, returncode):
+ # pylint: disable=no-self-use,unused-argument
+ # As expected. We want derived classes to implement this.
+ """Returns if Popen-like object returncode matches that of a hard
+ terminate attempt.
+
+ @param returncode the returncode from the Popen-like object that
+ terminated with a given return code.
+
+ @return True when the returncode represents what the system would
+ issue when a hard_terminate() occurred; False
+ otherwise.
+ """
+ raise Exception("platform needs to implement")
+
+ def soft_terminate_signals(self):
+ # pylint: disable=no-self-use
+ """Retrieve signal numbers that can be sent to soft terminate.
+ @return a list of signal numbers that can be sent to soft terminate
+ a process, or None if not applicable.
+ """
+ return None
+
+ def is_exceptional_exit(self, popen_status):
+ """Returns whether the program exit status is exceptional.
+
+ Returns whether the return code from a Popen process is exceptional
+ (e.g. signals on POSIX systems).
+
+ Derived classes should override this if they can detect exceptional
+ program exit.
+
+ @return True if the given popen_status represents an exceptional
+ program exit; False otherwise.
+ """
+ return False
+
+ def exceptional_exit_details(self, popen_status):
+ """Returns the normalized exceptional exit code and a description.
+
+ Given an exceptional exit code, returns the integral value of the
+ exception (e.g. signal number for POSIX) and a description (e.g.
+ signal name on POSIX) for the result.
+
+ Derived classes should override this if they can detect exceptional
+ program exit.
+
+ It is fine to not implement this so long as is_exceptional_exit()
+ always returns False.
+
+ @return (normalized exception code, symbolic exception description)
+ """
+ raise Exception("exception_exit_details() called on unsupported class")
+
+
+class UnixProcessHelper(ProcessHelper):
+ """Provides a ProcessHelper for Unix-like operating systems.
+
+ This implementation supports anything that looks Posix-y
+ (e.g. Darwin, Linux, *BSD, etc.)
+ """
+ def __init__(self):
+ super(UnixProcessHelper, self).__init__()
+
+ @classmethod
+ def _create_new_process_group(cls):
+ """Creates a new process group for the calling process."""
+ os.setpgid(os.getpid(), os.getpid())
+
+ def create_piped_process(self, command, new_process_group=True):
+ # Determine what to run after the fork but before the exec.
+ if new_process_group:
+ preexec_func = self._create_new_process_group
+ else:
+ preexec_func = None
+
+ # Create the process.
+ process = subprocess.Popen(
+ command,
+ stdin=subprocess.PIPE,
+ stdout=subprocess.PIPE,
+ stderr=subprocess.PIPE,
+ universal_newlines=True, # Elicits automatic byte -> string decoding in Py3
+ close_fds=True,
+ preexec_fn=preexec_func)
+
+ # Remember whether we're using process groups for this
+ # process.
+ process.using_process_groups = new_process_group
+ return process
+
+ def supports_soft_terminate(self):
+ # POSIX does support a soft terminate via:
+ # * SIGTERM (no core requested)
+ # * SIGQUIT (core requested if enabled, see ulimit -c)
+ return True
+
+ @classmethod
+ def _validate_pre_terminate(cls, popen_process, log_file):
+ # Validate args.
+ if popen_process is None:
+ raise ValueError("popen_process is None")
+
+ # Ensure we have something that looks like a valid process.
+ if popen_process.pid < 1:
+ if log_file:
+ log_file.write("skipping soft_terminate(): no process id")
+ return False
+
+ # We only do the process liveness check if we're not using
+ # process groups. With process groups, checking if the main
+ # inferior process is dead and short circuiting here is no
+ # good - children of it in the process group could still be
+ # alive, and they should be killed during a timeout.
+ if not popen_process.using_process_groups:
+ # Don't kill if it's already dead.
+ popen_process.poll()
+ if popen_process.returncode is not None:
+ # It has a returncode. It has already stopped.
+ if log_file:
+ log_file.write(
+ "requested to terminate pid {} but it has already "
+ "terminated, returncode {}".format(
+ popen_process.pid, popen_process.returncode))
+ # Move along...
+ return False
+
+ # Good to go.
+ return True
+
+ def _kill_with_signal(self, popen_process, log_file, signum):
+ # Validate we're ready to terminate this.
+ if not self._validate_pre_terminate(popen_process, log_file):
+ return
+
+ # Choose kill mechanism based on whether we're targeting
+ # a process group or just a process.
+ if popen_process.using_process_groups:
+ # if log_file:
+ # log_file.write(
+ # "sending signum {} to process group {} now\n".format(
+ # signum, popen_process.pid))
+ os.killpg(popen_process.pid, signum)
+ else:
+ # if log_file:
+ # log_file.write(
+ # "sending signum {} to process {} now\n".format(
+ # signum, popen_process.pid))
+ os.kill(popen_process.pid, signum)
+
+ def soft_terminate(self, popen_process, log_file=None, want_core=True):
+ # Choose signal based on desire for core file.
+ if want_core:
+ # SIGQUIT will generate core by default. Can be caught.
+ signum = signal.SIGQUIT
+ else:
+ # SIGTERM is the traditional nice way to kill a process.
+ # Can be caught, doesn't generate a core.
+ signum = signal.SIGTERM
+
+ self._kill_with_signal(popen_process, log_file, signum)
+
+ def hard_terminate(self, popen_process, log_file=None):
+ self._kill_with_signal(popen_process, log_file, signal.SIGKILL)
+
+ def was_soft_terminate(self, returncode, with_core):
+ if with_core:
+ return returncode == -signal.SIGQUIT
+ else:
+ return returncode == -signal.SIGTERM
+
+ def was_hard_terminate(self, returncode):
+ return returncode == -signal.SIGKILL
+
+ def soft_terminate_signals(self):
+ return [signal.SIGQUIT, signal.SIGTERM]
+
+ def is_exceptional_exit(self, popen_status):
+ return popen_status < 0
+
+ @classmethod
+ def _signal_names_by_number(cls):
+ return dict(
+ (k, v) for v, k in reversed(sorted(signal.__dict__.items()))
+ if v.startswith('SIG') and not v.startswith('SIG_'))
+
+ def exceptional_exit_details(self, popen_status):
+ signo = -popen_status
+ signal_names_by_number = self._signal_names_by_number()
+ signal_name = signal_names_by_number.get(signo, "")
+ return (signo, signal_name)
+
+class WindowsProcessHelper(ProcessHelper):
+ """Provides a Windows implementation of the ProcessHelper class."""
+ def __init__(self):
+ super(WindowsProcessHelper, self).__init__()
+
+ def create_piped_process(self, command, new_process_group=True):
+ if new_process_group:
+ # We need this flag if we want os.kill() to work on the subprocess.
+ creation_flags = subprocess.CREATE_NEW_PROCESS_GROUP
+ else:
+ creation_flags = 0
+
+ return subprocess.Popen(
+ command,
+ stdin=subprocess.PIPE,
+ stdout=subprocess.PIPE,
+ stderr=subprocess.PIPE,
+ universal_newlines=True, # Elicits automatic byte -> string decoding in Py3
+ creationflags=creation_flags)
+
+ def was_hard_terminate(self, returncode):
+ return returncode != 0
+
+
+class ProcessDriver(object):
+ """Drives a child process, notifies on important events, and can timeout.
+
+ Clients are expected to derive from this class and override the
+ on_process_started and on_process_exited methods if they want to
+ hook either of those.
+
+ This class supports timing out the child process in a platform-agnostic
+ way. The on_process_exited method is informed if the exit was natural
+ or if it was due to a timeout.
+ """
+ def __init__(self, soft_terminate_timeout=10.0):
+ super(ProcessDriver, self).__init__()
+ self.process_helper = ProcessHelper.process_helper()
+ self.pid = None
+ # Create the synchronization event for notifying when the
+ # inferior dotest process is complete.
+ self.done_event = threading.Event()
+ self.io_thread = None
+ self.process = None
+ # Number of seconds to wait for the soft terminate to
+ # wrap up, before moving to more drastic measures.
+ # Might want this longer if core dumps are generated and
+ # take a long time to write out.
+ self.soft_terminate_timeout = soft_terminate_timeout
+ # Number of seconds to wait for the hard terminate to
+ # wrap up, before giving up on the io thread. This should
+ # be fast.
+ self.hard_terminate_timeout = 5.0
+ self.returncode = None
+
+ # =============================================
+ # Methods for subclasses to override if desired.
+ # =============================================
+
+ def on_process_started(self):
+ pass
+
+ def on_process_exited(self, command, output, was_timeout, exit_status):
+ pass
+
+ def write(self, content):
+ # pylint: disable=no-self-use
+ # Intended - we want derived classes to be able to override
+ # this and use any self state they may contain.
+ sys.stdout.write(content)
+
+ # ==============================================================
+ # Operations used to drive processes. Clients will want to call
+ # one of these.
+ # ==============================================================
+
+ def run_command(self, command):
+ # Start up the child process and the thread that does the
+ # communication pump.
+ self._start_process_and_io_thread(command)
+
+ # Wait indefinitely for the child process to finish
+ # communicating. This indicates it has closed stdout/stderr
+ # pipes and is done.
+ self.io_thread.join()
+ self.returncode = self.process.wait()
+ if self.returncode is None:
+ raise Exception(
+ "no exit status available for pid {} after the "
+ " inferior dotest.py should have completed".format(
+ self.process.pid))
+
+ # Notify of non-timeout exit.
+ self.on_process_exited(
+ command,
+ self.io_thread.output,
+ False,
+ self.returncode)
+
+ def run_command_with_timeout(self, command, timeout, want_core):
+ # Figure out how many seconds our timeout description is requesting.
+ timeout_seconds = timeout_to_seconds(timeout)
+
+ # Start up the child process and the thread that does the
+ # communication pump.
+ self._start_process_and_io_thread(command)
+
+ self._wait_with_timeout(timeout_seconds, command, want_core)
+
+ # ================
+ # Internal details.
+ # ================
+
+ def _start_process_and_io_thread(self, command):
+ # Create the process.
+ self.process = self.process_helper.create_piped_process(command)
+ self.pid = self.process.pid
+ self.on_process_started()
+
+ # Ensure the event is cleared that is used for signaling
+ # from the communication() thread when communication is
+ # complete (i.e. the inferior process has finished).
+ self.done_event.clear()
+
+ self.io_thread = CommunicatorThread(
+ self.process, self.done_event, self.write)
+ self.io_thread.start()
+
+ def _attempt_soft_kill(self, want_core):
+ # The inferior dotest timed out. Attempt to clean it
+ # with a non-drastic method (so it can clean up properly
+ # and/or generate a core dump). Often the OS can't guarantee
+ # that the process will really terminate after this.
+ self.process_helper.soft_terminate(
+ self.process,
+ want_core=want_core,
+ log_file=self)
+
+ # Now wait up to a certain timeout period for the io thread
+ # to say that the communication ended. If that wraps up
+ # within our soft terminate timeout, we're all done here.
+ self.io_thread.join(self.soft_terminate_timeout)
+ if not self.io_thread.is_alive():
+ # stdout/stderr were closed on the child process side. We
+ # should be able to wait and reap the child process here.
+ self.returncode = self.process.wait()
+ # We terminated, and the done_trying result is n/a
+ terminated = True
+ done_trying = None
+ else:
+ self.write("soft kill attempt of process {} timed out "
+ "after {} seconds\n".format(
+ self.process.pid, self.soft_terminate_timeout))
+ terminated = False
+ done_trying = False
+ return terminated, done_trying
+
+ def _attempt_hard_kill(self):
+ # Instruct the process to terminate and really force it to
+ # happen. Don't give the process a chance to ignore.
+ self.process_helper.hard_terminate(
+ self.process,
+ log_file=self)
+
+ # Reap the child process. This should not hang as the
+ # hard_kill() mechanism is supposed to really kill it.
+ # Improvement option:
+ # If this does ever hang, convert to a self.process.poll()
+ # loop checking on self.process.returncode until it is not
+ # None or the timeout occurs.
+ self.returncode = self.process.wait()
+
+ # Wait a few moments for the io thread to finish...
+ self.io_thread.join(self.hard_terminate_timeout)
+ if self.io_thread.is_alive():
+ # ... but this is not critical if it doesn't end for some
+ # reason.
+ self.write(
+ "hard kill of process {} timed out after {} seconds waiting "
+ "for the io thread (ignoring)\n".format(
+ self.process.pid, self.hard_terminate_timeout))
+
+ # Set if it terminated. (Set up for optional improvement above).
+ terminated = self.returncode is not None
+ # Nothing else to try.
+ done_trying = True
+
+ return terminated, done_trying
+
+ def _attempt_termination(self, attempt_count, want_core):
+ if self.process_helper.supports_soft_terminate():
+ # When soft termination is supported, we first try to stop
+ # the process with a soft terminate. Failing that, we try
+ # the hard terminate option.
+ if attempt_count == 1:
+ return self._attempt_soft_kill(want_core)
+ elif attempt_count == 2:
+ return self._attempt_hard_kill()
+ else:
+ # We don't have anything else to try.
+ terminated = self.returncode is not None
+ done_trying = True
+ return terminated, done_trying
+ else:
+ # We only try the hard terminate option when there
+ # is no soft terminate available.
+ if attempt_count == 1:
+ return self._attempt_hard_kill()
+ else:
+ # We don't have anything else to try.
+ terminated = self.returncode is not None
+ done_trying = True
+ return terminated, done_trying
+
+ def _wait_with_timeout(self, timeout_seconds, command, want_core):
+ # Allow up to timeout seconds for the io thread to wrap up.
+ # If that completes, the child process should be done.
+ completed_normally = self.done_event.wait(timeout_seconds)
+ if completed_normally:
+ # Reap the child process here.
+ self.returncode = self.process.wait()
+ else:
+ # Prepare to stop the process
+ process_terminated = completed_normally
+ terminate_attempt_count = 0
+
+ # Try as many attempts as we support for trying to shut down
+ # the child process if it's not already shut down.
+ while not process_terminated:
+ terminate_attempt_count += 1
+ # Attempt to terminate.
+ process_terminated, done_trying = self._attempt_termination(
+ terminate_attempt_count, want_core)
+ # Check if there's nothing more to try.
+ if done_trying:
+ # Break out of our termination attempt loop.
+ break
+
+ # At this point, we're calling it good. The process
+ # finished gracefully, was shut down after one or more
+ # attempts, or we failed but gave it our best effort.
+ self.on_process_exited(
+ command,
+ self.io_thread.output,
+ not completed_normally,
+ self.returncode)
+
+
+def patched_init(self, *args, **kwargs):
+ self.original_init(*args, **kwargs)
+ # Initialize our condition variable that protects wait()/poll().
+ self.wait_condition = threading.Condition()
+
+
+def patched_wait(self, *args, **kwargs):
+ self.wait_condition.acquire()
+ try:
+ result = self.original_wait(*args, **kwargs)
+ # The process finished. Signal the condition.
+ self.wait_condition.notify_all()
+ return result
+ finally:
+ self.wait_condition.release()
+
+
+def patched_poll(self, *args, **kwargs):
+ self.wait_condition.acquire()
+ try:
+ result = self.original_poll(*args, **kwargs)
+ if self.returncode is not None:
+ # We did complete, and we have the return value.
+ # Signal the event to indicate we're done.
+ self.wait_condition.notify_all()
+ return result
+ finally:
+ self.wait_condition.release()
+
+
+def patch_up_subprocess_popen():
+ subprocess.Popen.original_init = subprocess.Popen.__init__
+ subprocess.Popen.__init__ = patched_init
+
+ subprocess.Popen.original_wait = subprocess.Popen.wait
+ subprocess.Popen.wait = patched_wait
+
+ subprocess.Popen.original_poll = subprocess.Popen.poll
+ subprocess.Popen.poll = patched_poll
+
+# Replace key subprocess.Popen() threading-unprotected methods with
+# threading-protected versions.
+patch_up_subprocess_popen()
diff --git a/packages/Python/lldbsuite/test/test_runner/test/inferior.py b/packages/Python/lldbsuite/test/test_runner/test/inferior.py
new file mode 100755
index 0000000000000..4207bac300fdb
--- /dev/null
+++ b/packages/Python/lldbsuite/test/test_runner/test/inferior.py
@@ -0,0 +1,146 @@
+#!/usr/bin/env python
+"""Inferior program used by process control tests."""
+
+from __future__ import print_function
+
+import argparse
+import datetime
+import signal
+import subprocess
+import sys
+import time
+
+
+def parse_args(command_line):
+ """Parses the command line arguments given to it.
+
+ @param command_line a list of command line arguments to be parsed.
+
+ @return the argparse options dictionary.
+ """
+ parser = argparse.ArgumentParser()
+ parser.add_argument(
+ "--ignore-signal",
+ "-i",
+ dest="ignore_signals",
+ metavar="SIGNUM",
+ action="append",
+ type=int,
+ default=[],
+ help="ignore the given signal number (if possible)")
+ parser.add_argument(
+ "--launch-child-share-handles",
+ action="store_true",
+ help=("launch a child inferior.py that shares stdout/stderr/stdio and "
+ "never returns"))
+ parser.add_argument(
+ "--never-return",
+ action="store_true",
+ help="run in an infinite loop, never return")
+ parser.add_argument(
+ "--return-code",
+ "-r",
+ type=int,
+ default=0,
+ help="specify the return code for the inferior upon exit")
+ parser.add_argument(
+ "--sleep",
+ "-s",
+ metavar="SECONDS",
+ dest="sleep_seconds",
+ type=float,
+ help="sleep for SECONDS seconds before returning")
+ parser.add_argument(
+ "--verbose", "-v", action="store_true",
+ help="log verbose operation details to stdout")
+ return parser.parse_args(command_line)
+
+
+def handle_ignore_signals(options, signals):
+ """Ignores any signals provided to it.
+
+ @param options the command line options parsed by the program.
+ General used to check flags for things like verbosity.
+
+ @param signals the list of signals to ignore. Can be None or zero-length.
+ Entries should be type int.
+ """
+ if signals is None:
+ return
+
+ for signum in signals:
+ if options.verbose:
+ print("disabling signum {}".format(signum))
+ signal.signal(signum, signal.SIG_IGN)
+
+
+def handle_sleep(options, sleep_seconds):
+ """Sleeps the number of seconds specified, restarting as needed.
+
+ @param options the command line options parsed by the program.
+ General used to check flags for things like verbosity.
+
+ @param sleep_seconds the number of seconds to sleep. If None
+ or <= 0, no sleeping will occur.
+ """
+ if sleep_seconds is None:
+ return
+
+ if sleep_seconds <= 0:
+ return
+
+ end_time = datetime.datetime.now() + datetime.timedelta(0, sleep_seconds)
+ if options.verbose:
+ print("sleep end time: {}".format(end_time))
+
+ # Do sleep in a loop: signals can interrupt.
+ while datetime.datetime.now() < end_time:
+ # We'll wrap this in a try/catch so we don't encounter
+ # a race if a signal (ignored) knocks us out of this
+ # loop and causes us to return.
+ try:
+ sleep_interval = end_time - datetime.datetime.now()
+ sleep_seconds = sleep_interval.total_seconds()
+ if sleep_seconds > 0:
+ time.sleep(sleep_seconds)
+ except: # pylint: disable=bare-except
+ pass
+
+
+def handle_launch_children(options):
+ if options.launch_child_share_handles:
+ # Launch the child, share our file handles.
+ # We won't bother reaping it since it will likely outlive us.
+ subprocess.Popen([sys.executable, __file__, "--never-return"])
+
+
+def handle_never_return(options):
+ if not options.never_return:
+ return
+
+ # Loop forever.
+ while True:
+ try:
+ time.sleep(10)
+ except: # pylint: disable=bare-except
+ # Ignore
+ pass
+
+
+def main(command_line):
+ """Drives the main operation of the inferior test program.
+
+ @param command_line the command line options to process.
+
+ @return the exit value (program return code) for the process.
+ """
+ options = parse_args(command_line)
+ handle_ignore_signals(options, options.ignore_signals)
+ handle_launch_children(options)
+ handle_sleep(options, options.sleep_seconds)
+ handle_never_return(options)
+
+ return options.return_code
+
+if __name__ == "__main__":
+ sys.exit(main(sys.argv[1:]))
diff --git a/packages/Python/lldbsuite/test/test_runner/test/process_control_tests.py b/packages/Python/lldbsuite/test/test_runner/test/process_control_tests.py
new file mode 100755
index 0000000000000..354506d658125
--- /dev/null
+++ b/packages/Python/lldbsuite/test/test_runner/test/process_control_tests.py
@@ -0,0 +1,235 @@
+#!/usr/bin/env python
+"""
+The LLVM Compiler Infrastructure
+
+This file is distributed under the University of Illinois Open Source
+License. See LICENSE.TXT for details.
+
+Provides classes used by the test results reporting infrastructure
+within the LLDB test suite.
+
+
+Tests the process_control module.
+"""
+
+# System imports.
+import os
+import platform
+import unittest
+import sys
+import threading
+
+# Add lib dir to pythonpath
+sys.path.append(os.path.join(os.path.dirname(__file__), '..', 'lib'))
+
+# Our imports.
+import process_control
+
+
+class TestInferiorDriver(process_control.ProcessDriver):
+ def __init__(self, soft_terminate_timeout=None):
+ super(TestInferiorDriver, self).__init__(
+ soft_terminate_timeout=soft_terminate_timeout)
+ self.started_event = threading.Event()
+ self.started_event.clear()
+
+ self.completed_event = threading.Event()
+ self.completed_event.clear()
+
+ self.was_timeout = False
+ self.returncode = None
+ self.output = None
+
+ def write(self, content):
+ # We'll swallow this to keep tests non-noisy.
+ # Uncomment the following line if you want to see it.
+ # sys.stdout.write(content)
+ pass
+
+ def on_process_started(self):
+ self.started_event.set()
+
+ def on_process_exited(self, command, output, was_timeout, exit_status):
+ self.returncode = exit_status
+ self.was_timeout = was_timeout
+ self.output = output
+ self.returncode = exit_status
+ self.completed_event.set()
+
+
+class ProcessControlTests(unittest.TestCase):
+ @classmethod
+ def _suppress_soft_terminate(cls, command):
+ # Do the right thing for your platform here.
+ # Right now only POSIX-y systems are reporting
+ # soft terminate support, so this is set up for
+ # those.
+ helper = process_control.ProcessHelper.process_helper()
+ signals = helper.soft_terminate_signals()
+ if signals is not None:
+ for signum in helper.soft_terminate_signals():
+ command.extend(["--ignore-signal", str(signum)])
+
+ @classmethod
+ def inferior_command(
+ cls,
+ ignore_soft_terminate=False,
+ options=None):
+
+ # Base command.
+ command = ([sys.executable, "inferior.py"])
+
+ if ignore_soft_terminate:
+ cls._suppress_soft_terminate(command)
+
+ # Handle options as string or list.
+ if isinstance(options, str):
+ command.extend(options.split())
+ elif isinstance(options, list):
+ command.extend(options)
+
+ # Return full command.
+ return command
+
+
+class ProcessControlNoTimeoutTests(ProcessControlTests):
+ """Tests the process_control module."""
+ def test_run_completes(self):
+ """Test that running completes and gets expected stdout/stderr."""
+ driver = TestInferiorDriver()
+ driver.run_command(self.inferior_command())
+ self.assertTrue(
+ driver.completed_event.wait(5), "process failed to complete")
+ self.assertEqual(driver.returncode, 0, "return code does not match")
+
+ def test_run_completes_with_code(self):
+ """Test that running completes and gets expected stdout/stderr."""
+ driver = TestInferiorDriver()
+ driver.run_command(self.inferior_command(options="-r10"))
+ self.assertTrue(
+ driver.completed_event.wait(5), "process failed to complete")
+ self.assertEqual(driver.returncode, 10, "return code does not match")
+
+
+class ProcessControlTimeoutTests(ProcessControlTests):
+ def test_run_completes(self):
+ """Test that running completes and gets expected return code."""
+ driver = TestInferiorDriver()
+ timeout_seconds = 5
+ driver.run_command_with_timeout(
+ self.inferior_command(),
+ "{}s".format(timeout_seconds),
+ False)
+ self.assertTrue(
+ driver.completed_event.wait(2*timeout_seconds),
+ "process failed to complete")
+ self.assertEqual(driver.returncode, 0)
+
+ def _soft_terminate_works(self, with_core):
+ # Skip this test if the platform doesn't support soft ti
+ helper = process_control.ProcessHelper.process_helper()
+ if not helper.supports_soft_terminate():
+ self.skipTest("soft terminate not supported by platform")
+
+ driver = TestInferiorDriver()
+ timeout_seconds = 5
+
+ driver.run_command_with_timeout(
+ # Sleep twice as long as the timeout interval. This
+ # should force a timeout.
+ self.inferior_command(
+ options="--sleep {}".format(timeout_seconds*2)),
+ "{}s".format(timeout_seconds),
+ with_core)
+
+ # We should complete, albeit with a timeout.
+ self.assertTrue(
+ driver.completed_event.wait(2*timeout_seconds),
+ "process failed to complete")
+
+ # Ensure we received a timeout.
+ self.assertTrue(driver.was_timeout, "expected to end with a timeout")
+
+ self.assertTrue(
+ helper.was_soft_terminate(driver.returncode, with_core),
+ ("timeout didn't return expected returncode "
+ "for soft terminate with core: {}").format(driver.returncode))
+
+ def test_soft_terminate_works_core(self):
+ """Driver uses soft terminate (with core request) when process times out.
+ """
+ self._soft_terminate_works(True)
+
+ def test_soft_terminate_works_no_core(self):
+ """Driver uses soft terminate (no core request) when process times out.
+ """
+ self._soft_terminate_works(False)
+
+ def test_hard_terminate_works(self):
+ """Driver falls back to hard terminate when soft terminate is ignored.
+ """
+
+ driver = TestInferiorDriver(soft_terminate_timeout=2.0)
+ timeout_seconds = 1
+
+ driver.run_command_with_timeout(
+ # Sleep much longer than the timeout interval,forcing a
+ # timeout. Do whatever is needed to have the inferior
+ # ignore soft terminate calls.
+ self.inferior_command(
+ ignore_soft_terminate=True,
+ options="--never-return"),
+ "{}s".format(timeout_seconds),
+ True)
+
+ # We should complete, albeit with a timeout.
+ self.assertTrue(
+ driver.completed_event.wait(60),
+ "process failed to complete")
+
+ # Ensure we received a timeout.
+ self.assertTrue(driver.was_timeout, "expected to end with a timeout")
+
+ helper = process_control.ProcessHelper.process_helper()
+ self.assertTrue(
+ helper.was_hard_terminate(driver.returncode),
+ ("timeout didn't return expected returncode "
+ "for hard teriminate: {} ({})").format(
+ driver.returncode,
+ driver.output))
+
+ def test_inferior_exits_with_live_child_shared_handles(self):
+ """inferior exit detected when inferior children are live with shared
+ stdout/stderr handles.
+ """
+ # Requires review D13362 or equivalent to be implemented.
+ self.skipTest("http://reviews.llvm.org/D13362")
+
+ driver = TestInferiorDriver()
+
+ # Create the inferior (I1), and instruct it to create a child (C1)
+ # that shares the stdout/stderr handles with the inferior.
+ # C1 will then loop forever.
+ driver.run_command_with_timeout(
+ self.inferior_command(
+ options="--launch-child-share-handles --return-code 3"),
+ "5s",
+ False)
+
+ # We should complete without a timetout. I1 should end
+ # immediately after launching C1.
+ self.assertTrue(
+ driver.completed_event.wait(5),
+ "process failed to complete")
+
+ # Ensure we didn't receive a timeout.
+ self.assertFalse(
+ driver.was_timeout, "inferior should have completed normally")
+
+ self.assertEqual(
+ driver.returncode, 3,
+ "expected inferior process to end with expected returncode")
+
+
+if __name__ == "__main__":
+ unittest.main()