The implementation has an optimization which detects the range of line table entries covered by the function and then only searches for a matching line between them. This optimization was interfering with the logic for detecting whether a line belongs to the function because the first call to FindLineEntry was made with exact=false, which meant that if the function did not contain any exact matches, we would just pick the closest line number in that range, even if it was very far away. This patch fixes that by first attempting an inexact search across the entire line table, and then use the (potentially inexact) result of that for searching within the function. This makes the optimization a less effective, but I don't think we can differentiate between a line that belongs to the function (but doesn't have any code) and a line outside the function without that. The patch also avoids the use of (deprecated) Function::GetAddressRange by iterating over the GetAddressRanges result to find the full range of line entries for the function.
127 lines
4.6 KiB
Python
127 lines
4.6 KiB
Python
"""Test stepping over vrs. hitting breakpoints & subsequent stepping in various forms."""
|
|
|
|
|
|
import lldb
|
|
from lldbsuite.test.decorators import *
|
|
from lldbsuite.test.lldbtest import *
|
|
from lldbsuite.test import lldbutil
|
|
|
|
|
|
class StepUntilTestCase(TestBase):
|
|
def setUp(self):
|
|
# Call super's setUp().
|
|
TestBase.setUp(self)
|
|
# Find the line numbers that we will step to in main:
|
|
self.main_source = "main.c"
|
|
self.less_than_two = line_number("main.c", "Less than 2")
|
|
self.greater_than_two = line_number("main.c", "Greater than or equal to 2.")
|
|
self.back_out_in_main = line_number("main.c", "Back out in main")
|
|
self.in_foo = line_number("main.c", "In foo")
|
|
|
|
def _build_dict_for_discontinuity(self):
|
|
return dict(
|
|
CFLAGS_EXTRAS="-funique-basic-block-section-names "
|
|
+ "-ffunction-sections -fbasic-block-sections=list="
|
|
+ self.getSourcePath("function.list"),
|
|
LD_EXTRAS="-Wl,--script=" + self.getSourcePath("symbol.order"),
|
|
)
|
|
|
|
def _common_setup(self, build_dict, args):
|
|
self.build(dictionary=build_dict)
|
|
exe = self.getBuildArtifact("a.out")
|
|
|
|
target = self.dbg.CreateTarget(exe)
|
|
self.assertTrue(target, VALID_TARGET)
|
|
|
|
main_source_spec = lldb.SBFileSpec(self.main_source)
|
|
break_before = target.BreakpointCreateBySourceRegex(
|
|
"At the start", main_source_spec
|
|
)
|
|
self.assertTrue(break_before, VALID_BREAKPOINT)
|
|
|
|
# Now launch the process, and do not stop at entry point.
|
|
process = target.LaunchSimple(args, None, self.get_process_working_directory())
|
|
|
|
self.assertTrue(process, PROCESS_IS_VALID)
|
|
|
|
# The stop reason of the thread should be breakpoint.
|
|
threads = lldbutil.get_threads_stopped_at_breakpoint(process, break_before)
|
|
|
|
if len(threads) != 1:
|
|
self.fail("Failed to stop at first breakpoint in main.")
|
|
|
|
thread = threads[0]
|
|
return thread
|
|
|
|
def do_until(self, args, until_lines, expected_linenum):
|
|
thread = self._common_setup(None, args)
|
|
|
|
cmd_interp = self.dbg.GetCommandInterpreter()
|
|
ret_obj = lldb.SBCommandReturnObject()
|
|
|
|
cmd_line = "thread until"
|
|
for line_num in until_lines:
|
|
cmd_line += " %d" % (line_num)
|
|
|
|
cmd_interp.HandleCommand(cmd_line, ret_obj)
|
|
self.assertTrue(
|
|
ret_obj.Succeeded(), "'%s' failed: %s." % (cmd_line, ret_obj.GetError())
|
|
)
|
|
|
|
frame = thread.frames[0]
|
|
line = frame.GetLineEntry().GetLine()
|
|
self.assertEqual(
|
|
line, expected_linenum, "Did not get the expected stop line number"
|
|
)
|
|
|
|
def test_hitting_one(self):
|
|
"""Test thread step until - targeting one line and hitting it."""
|
|
self.do_until(None, [self.less_than_two], self.less_than_two)
|
|
|
|
def test_targetting_two_hitting_first(self):
|
|
"""Test thread step until - targeting two lines and hitting one."""
|
|
self.do_until(
|
|
["foo", "bar", "baz"],
|
|
[self.less_than_two, self.greater_than_two],
|
|
self.greater_than_two,
|
|
)
|
|
|
|
def test_targetting_two_hitting_second(self):
|
|
"""Test thread step until - targeting two lines and hitting the other one."""
|
|
self.do_until(
|
|
None, [self.less_than_two, self.greater_than_two], self.less_than_two
|
|
)
|
|
|
|
def test_missing_one(self):
|
|
"""Test thread step until - targeting one line and missing it by stepping out to call site"""
|
|
self.do_until(
|
|
["foo", "bar", "baz"], [self.less_than_two], self.back_out_in_main
|
|
)
|
|
|
|
@no_debug_info_test
|
|
def test_bad_line(self):
|
|
"""Test that we get an error if attempting to step outside the current
|
|
function"""
|
|
thread = self._common_setup(None, None)
|
|
self.expect(
|
|
f"thread until {self.in_foo}",
|
|
substrs=["Until target outside of the current function"],
|
|
error=True,
|
|
)
|
|
|
|
@no_debug_info_test
|
|
@skipIf(oslist=lldbplatformutil.getDarwinOSTriples() + ["windows"])
|
|
@skipIf(archs=no_match(["x86_64", "aarch64"]))
|
|
def test_bad_line_discontinuous(self):
|
|
"""Test that we get an error if attempting to step outside the current
|
|
function -- and the function is discontinuous"""
|
|
self.build(dictionary=self._build_dict_for_discontinuity())
|
|
_, _, thread, _ = lldbutil.run_to_source_breakpoint(
|
|
self, "At the start", lldb.SBFileSpec(self.main_source)
|
|
)
|
|
self.expect(
|
|
f"thread until {self.in_foo}",
|
|
substrs=["Until target outside of the current function"],
|
|
error=True,
|
|
)
|