| """Test that we handle inferiors that send signals to themselves""" |
| |
| |
| import lldb |
| import re |
| from lldbsuite.test.lldbplatformutil import getDarwinOSTriples |
| from lldbsuite.test.decorators import * |
| from lldbsuite.test.lldbtest import * |
| from lldbsuite.test import lldbutil |
| |
| |
| @skipIfWindows # signals do not exist on Windows |
| class RaiseTestCase(TestBase): |
| NO_DEBUG_INFO_TESTCASE = True |
| |
| @skipIfNetBSD # Hangs on NetBSD |
| def test_sigstop(self): |
| self.build() |
| self.signal_test("SIGSTOP", False) |
| # passing of SIGSTOP is not correctly handled, so not testing that |
| # scenario: https://llvm.org/bugs/show_bug.cgi?id=23574 |
| |
| @skipIfDarwin # darwin does not support real time signals |
| @skipIfTargetAndroid() |
| def test_sigsigrtmin(self): |
| self.build() |
| self.signal_test("SIGRTMIN", True) |
| |
| @skipIfNetBSD # Hangs on NetBSD |
| def test_sigtrap(self): |
| self.build() |
| self.signal_test("SIGTRAP", True) |
| |
| def launch(self, target, signal): |
| # launch the process, do not stop at entry point. |
| # If we have gotten the default for this signal, reset that as well. |
| if len(self.default_pass) != 0: |
| lldbutil.set_actions_for_signal( |
| self, signal, self.default_pass, self.default_stop, self.default_notify |
| ) |
| |
| process = target.LaunchSimple( |
| [signal], None, self.get_process_working_directory() |
| ) |
| self.assertTrue(process, PROCESS_IS_VALID) |
| self.assertState(process.GetState(), lldb.eStateStopped) |
| thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonBreakpoint) |
| self.assertTrue( |
| thread.IsValid(), "Thread should be stopped due to a breakpoint" |
| ) |
| return process |
| |
| def set_handle(self, signal, pass_signal, stop_at_signal, notify_signal): |
| return_obj = lldb.SBCommandReturnObject() |
| self.dbg.GetCommandInterpreter().HandleCommand( |
| "process handle %s -p %s -s %s -n %s" |
| % (signal, pass_signal, stop_at_signal, notify_signal), |
| return_obj, |
| ) |
| self.assertTrue(return_obj.Succeeded(), "Setting signal handling failed") |
| |
| def signal_test(self, signal, test_passing): |
| """Test that we handle inferior raising signals""" |
| exe = self.getBuildArtifact("a.out") |
| |
| # Create a target by the debugger. |
| target = self.dbg.CreateTarget(exe) |
| self.assertTrue(target, VALID_TARGET) |
| lldbutil.run_break_set_by_symbol(self, "main") |
| self.default_pass = "" |
| self.default_stop = "" |
| self.default_notify = "" |
| |
| # launch |
| process = self.launch(target, signal) |
| signo = process.GetUnixSignals().GetSignalNumberFromName(signal) |
| |
| # retrieve default signal disposition |
| ( |
| self.default_pass, |
| self.default_stop, |
| self.default_notify, |
| ) = lldbutil.get_actions_for_signal(self, signal) |
| |
| # Make sure we stop at the signal |
| lldbutil.set_actions_for_signal(self, signal, "false", "true", "true") |
| process.Continue() |
| self.assertState(process.GetState(), lldb.eStateStopped) |
| thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonSignal) |
| self.assertTrue(thread.IsValid(), "Thread should be stopped due to a signal") |
| self.assertTrue( |
| thread.GetStopReasonDataCount() >= 1, "There was data in the event." |
| ) |
| self.assertEqual( |
| thread.GetStopReasonDataAtIndex(0), signo, "The stop signal was %s" % signal |
| ) |
| |
| # Continue until we exit. |
| process.Continue() |
| self.assertState(process.GetState(), lldb.eStateExited) |
| self.assertEqual(process.GetExitStatus(), 0) |
| |
| process = self.launch(target, signal) |
| |
| # Make sure we do not stop at the signal. We should still get the |
| # notification. |
| lldbutil.set_actions_for_signal(self, signal, "false", "false", "true") |
| self.expect("process continue", substrs=["stopped and restarted", signal]) |
| self.assertState(process.GetState(), lldb.eStateExited) |
| self.assertEqual(process.GetExitStatus(), 0) |
| |
| # launch again |
| process = self.launch(target, signal) |
| |
| # Make sure we do not stop at the signal, and we do not get the |
| # notification. |
| lldbutil.set_actions_for_signal(self, signal, "false", "false", "false") |
| self.expect( |
| "process continue", substrs=["stopped and restarted"], matching=False |
| ) |
| self.assertState(process.GetState(), lldb.eStateExited) |
| self.assertEqual(process.GetExitStatus(), 0) |
| |
| if not test_passing: |
| # reset signal handling to default |
| lldbutil.set_actions_for_signal( |
| self, signal, self.default_pass, self.default_stop, self.default_notify |
| ) |
| return |
| |
| # launch again |
| process = self.launch(target, signal) |
| |
| # Make sure we stop at the signal |
| lldbutil.set_actions_for_signal(self, signal, "true", "true", "true") |
| process.Continue() |
| self.assertState(process.GetState(), lldb.eStateStopped) |
| thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonSignal) |
| self.assertTrue(thread.IsValid(), "Thread should be stopped due to a signal") |
| self.assertTrue( |
| thread.GetStopReasonDataCount() >= 1, "There was data in the event." |
| ) |
| self.assertEqual( |
| thread.GetStopReasonDataAtIndex(0), |
| process.GetUnixSignals().GetSignalNumberFromName(signal), |
| "The stop signal was %s" % signal, |
| ) |
| |
| # Continue until we exit. The process should receive the signal. |
| process.Continue() |
| self.assertState(process.GetState(), lldb.eStateExited) |
| self.assertEqual(process.GetExitStatus(), signo) |
| |
| # launch again |
| process = self.launch(target, signal) |
| |
| # Make sure we do not stop at the signal. We should still get the notification. Process |
| # should receive the signal. |
| lldbutil.set_actions_for_signal(self, signal, "true", "false", "true") |
| self.expect("process continue", substrs=["stopped and restarted", signal]) |
| self.assertState(process.GetState(), lldb.eStateExited) |
| self.assertEqual(process.GetExitStatus(), signo) |
| |
| # launch again |
| process = self.launch(target, signal) |
| |
| # Make sure we do not stop at the signal, and we do not get the notification. Process |
| # should receive the signal. |
| lldbutil.set_actions_for_signal(self, signal, "true", "false", "false") |
| self.expect( |
| "process continue", substrs=["stopped and restarted"], matching=False |
| ) |
| self.assertState(process.GetState(), lldb.eStateExited) |
| self.assertEqual(process.GetExitStatus(), signo) |
| |
| # reset signal handling to default |
| lldbutil.set_actions_for_signal( |
| self, signal, self.default_pass, self.default_stop, self.default_notify |
| ) |