Add a function to make it easier to debug a test failure caused by an unexpected stop reason. This is similar to the assertState helper that was added in ce825e46743b. Before: self.assertEqual(stop_reason, lldb.eStopReasonInstrumentation) AssertionError: 5 != 10 After: self.assertStopReason(stop_reason, lldb.eStopReasonInstrumentation) AssertionError: signal (5) != instrumentation (10) Differential revision: https://reviews.llvm.org/D131083
132 lines
6.8 KiB
Python
132 lines
6.8 KiB
Python
"""
|
|
Test the "process continue -b" option.
|
|
"""
|
|
|
|
|
|
import lldb
|
|
from lldbsuite.test.decorators import *
|
|
from lldbsuite.test.lldbtest import *
|
|
from lldbsuite.test import lldbutil
|
|
|
|
|
|
class TestContinueToBkpts(TestBase):
|
|
|
|
NO_DEBUG_INFO_TESTCASE = True
|
|
|
|
@add_test_categories(['pyapi'])
|
|
def test_continue_to_breakpoints(self):
|
|
"""Test that the continue to breakpoints feature works correctly."""
|
|
self.build()
|
|
self.do_test_continue_to_breakpoint()
|
|
|
|
def setUp(self):
|
|
# Call super's setUp().
|
|
TestBase.setUp(self)
|
|
self.main_source_spec = lldb.SBFileSpec("main.c")
|
|
|
|
def continue_and_check(self, stop_list, bkpt_to_hit, loc_to_hit = 0):
|
|
"""Build up a command that will run a continue -b commands using the breakpoints on stop_list, and
|
|
ensure that we hit bkpt_to_hit.
|
|
If loc_to_hit is not 0, also verify that we hit that location."""
|
|
command = "process continue"
|
|
for elem in stop_list:
|
|
command += " -b {0}".format(elem)
|
|
self.expect(command)
|
|
self.assertStopReason(self.thread.stop_reason, lldb.eStopReasonBreakpoint, "Hit a breakpoint")
|
|
self.assertEqual(self.thread.GetStopReasonDataAtIndex(0), bkpt_to_hit, "Hit the right breakpoint")
|
|
if loc_to_hit != 0:
|
|
self.assertEqual(self.thread.GetStopReasonDataAtIndex(1), loc_to_hit, "Hit the right location")
|
|
for bkpt_id in self.bkpt_list:
|
|
bkpt = self.target.FindBreakpointByID(bkpt_id)
|
|
self.assertTrue(bkpt.IsValid(), "Breakpoint id's round trip")
|
|
if bkpt.MatchesName("disabled"):
|
|
self.assertFalse(bkpt.IsEnabled(), "Disabled breakpoints stay disabled: {0}".format(bkpt.GetID()))
|
|
else:
|
|
self.assertTrue(bkpt.IsEnabled(), "Enabled breakpoints stay enabled: {0}".format(bkpt.GetID()))
|
|
# Also do our multiple location one:
|
|
bkpt = self.target.FindBreakpointByID(self.multiple_loc_id)
|
|
self.assertTrue(bkpt.IsValid(), "Breakpoint with locations round trip")
|
|
for i in range(1,3):
|
|
loc = bkpt.FindLocationByID(i)
|
|
self.assertTrue(loc.IsValid(), "Locations round trip")
|
|
if i == 2:
|
|
self.assertTrue(loc.IsEnabled(), "Locations that were enabled stay enabled")
|
|
else:
|
|
self.assertFalse(loc.IsEnabled(), "Locations that were disabled stay disabled")
|
|
|
|
def do_test_continue_to_breakpoint(self):
|
|
"""Test the continue to breakpoint feature."""
|
|
(self.target, process, self.thread, bkpt) = lldbutil.run_to_source_breakpoint(self,
|
|
"Stop here to get started", self.main_source_spec)
|
|
|
|
# Now set up all our breakpoints:
|
|
bkpt_pattern = "This is the {0} stop"
|
|
bkpt_elements = ["zeroth", "first", "second", "third", "fourth", "fifth", "sixth", "seventh", "eighth", "nineth"]
|
|
disabled_bkpts = ["first", "eigth"]
|
|
bkpts_for_MyBKPT = ["first", "sixth", "nineth"]
|
|
self.bkpt_list = []
|
|
for elem in bkpt_elements:
|
|
bkpt = self.target.BreakpointCreateBySourceRegex(bkpt_pattern.format(elem), self.main_source_spec)
|
|
self.assertGreater(bkpt.GetNumLocations(), 0, "Found a bkpt match")
|
|
self.bkpt_list.append(bkpt.GetID())
|
|
bkpt.AddName(elem)
|
|
if elem in disabled_bkpts:
|
|
bkpt.AddName("disabled")
|
|
bkpt.SetEnabled(False)
|
|
if elem in bkpts_for_MyBKPT:
|
|
bkpt.AddName("MyBKPT")
|
|
# Also make one that has several locations, so we can test locations:
|
|
mult_bkpt = self.target.BreakpointCreateBySourceRegex(bkpt_pattern.format("(seventh|eighth|nineth)"), self.main_source_spec)
|
|
self.assertEqual(mult_bkpt.GetNumLocations(), 3, "Got three matches")
|
|
mult_bkpt.AddName("Locations")
|
|
# Disable all of these:
|
|
for i in range(1,4):
|
|
loc = mult_bkpt.FindLocationByID(i)
|
|
self.assertTrue(loc.IsValid(), "Location {0} is valid".format(i))
|
|
loc.SetEnabled(False)
|
|
self.assertFalse(loc.IsEnabled(), "Loc {0} wasn't disabled".format(i))
|
|
self.multiple_loc_id = mult_bkpt.GetID()
|
|
|
|
# First test out various error conditions
|
|
|
|
# All locations of the multiple_loc_id are disabled, so running to this should be an error:
|
|
self.expect("process continue -b {0}".format(self.multiple_loc_id), error=True, msg="Running to a disabled breakpoint by number")
|
|
|
|
# Now re-enable the middle one so we can run to it:
|
|
loc = mult_bkpt.FindLocationByID(2)
|
|
loc.SetEnabled(True)
|
|
|
|
self.expect("process continue -b {0}".format(self.bkpt_list[1]), error=True, msg="Running to a disabled breakpoint by number")
|
|
self.expect("process continue -b {0}.1".format(self.bkpt_list[1]), error=True, msg="Running to a location of a disabled breakpoint")
|
|
self.expect("process continue -b disabled", error=True, msg="Running to a disabled set of breakpoints")
|
|
self.expect("process continue -b {0}.{1}".format(self.multiple_loc_id, 1), error=True, msg="Running to a disabled breakpoint location")
|
|
self.expect("process continue -b {0}".format("THERE_ARE_NO_BREAKPOINTS_BY_THIS_NAME"), error=True, msg="Running to no such name")
|
|
self.expect("process continue -b {0}".format(1000), error=True, msg="Running to no such breakpoint")
|
|
self.expect("process continue -b {0}.{1}".format(self.multiple_loc_id, 1000), error=True, msg="Running to no such location")
|
|
|
|
# Now move forward, this time with breakpoint numbers. First time we don't skip other bkpts.
|
|
bkpt = self.bkpt_list[0]
|
|
self.continue_and_check([str(bkpt)], bkpt)
|
|
|
|
# Now skip to the third stop, do it by name and supply one of the later breakpoints as well:
|
|
# This continue has to muck with the sync mode of the debugger, so let's make sure we
|
|
# put it back. First try if it was in sync mode:
|
|
orig_async = self.dbg.GetAsync()
|
|
self.dbg.SetAsync(True)
|
|
self.continue_and_check([bkpt_elements[2], bkpt_elements[7]], self.bkpt_list[2])
|
|
after_value = self.dbg.GetAsync()
|
|
self.dbg.SetAsync(orig_async)
|
|
self.assertTrue(after_value, "Preserve async as True if it started that way")
|
|
|
|
# Now try a name that has several breakpoints.
|
|
# This time I'm also going to check that we put the debugger async mode back if
|
|
# if was False to begin with:
|
|
self.dbg.SetAsync(False)
|
|
self.continue_and_check(["MyBKPT"], self.bkpt_list[6])
|
|
after_value = self.dbg.GetAsync()
|
|
self.dbg.SetAsync(orig_async)
|
|
self.assertFalse(after_value, "Preserve async as False if it started that way")
|
|
|
|
# Now let's run to a particular location. Also specify a breakpoint we've already hit:
|
|
self.continue_and_check([self.bkpt_list[0], self.multiple_loc_id], self.multiple_loc_id, 2)
|