llvm/lldb/test/API/lang/c/stepping/TestThreadStepping.py

"""
Test thread stepping features in combination with frame select.
"""


import lldb
from lldbsuite.test.lldbtest import *
import lldbsuite.test.lldbutil as lldbutil


class ThreadSteppingTestCase(TestBase):
    def setUp(self):
        # Call super's setUp().
        TestBase.setUp(self)
        # Find the line number to of function 'c'.
        self.line1 = line_number(
            "main.c", '// Find the line number of function "c" here.'
        )
        self.line2 = line_number(
            "main.c", '// frame select 2, thread step-out while stopped at "c(1)"'
        )
        self.line3 = line_number("main.c", '// thread step-out while stopped at "c(2)"')
        self.line4 = line_number(
            "main.c", '// frame select 1, thread step-out while stopped at "c(3)"'
        )

    def test_step_out_with_run_command(self):
        """Exercise thread step-out and frame select followed by thread step-out."""
        self.build()
        exe = self.getBuildArtifact("a.out")
        self.runCmd("file " + exe, CURRENT_EXECUTABLE_SET)

        # Create a breakpoint inside function 'c'.
        lldbutil.run_break_set_by_file_and_line(
            self, "main.c", self.line1, num_expected_locations=1, loc_exact=True
        )

        # Now run the program.
        self.runCmd("run", RUN_SUCCEEDED)

        # The process should be stopped at this point.
        self.expect("process status", PROCESS_STOPPED, patterns=["Process .* stopped"])

        # The frame #0 should correspond to main.c:32, the executable statement
        # in function name 'c'.  And frame #3 should point to main.c:37.
        self.expect(
            "thread backtrace",
            STOPPED_DUE_TO_BREAKPOINT,
            substrs=["stop reason = breakpoint"],
            patterns=[
                "frame #0.*main.c:%d" % self.line1,
                "frame #3.*main.c:%d" % self.line2,
            ],
        )

        # We want to move the pc to frame #3.  This can be accomplished by
        # 'frame select 2', followed by 'thread step-out'.
        self.runCmd("frame select 2")
        self.runCmd("thread step-out")
        self.expect(
            "thread backtrace",
            STEP_OUT_SUCCEEDED,
            substrs=["stop reason = step out"],
            patterns=["frame #0.*main.c:%d" % self.line2],
        )

        # Let's move on to a single step-out case.
        self.runCmd("process continue")

        # The process should be stopped at this point.
        self.expect("process status", PROCESS_STOPPED, patterns=["Process .* stopped"])
        self.runCmd("thread step-out")
        self.expect(
            "thread backtrace",
            STEP_OUT_SUCCEEDED,
            substrs=["stop reason = step out"],
            patterns=["frame #0.*main.c:%d" % self.line3],
        )

        # Do another frame selct, followed by thread step-out.
        self.runCmd("process continue")

        # The process should be stopped at this point.
        self.expect("process status", PROCESS_STOPPED, patterns=["Process .* stopped"])
        self.runCmd("frame select 1")
        self.runCmd("thread step-out")
        self.expect(
            "thread backtrace",
            STEP_OUT_SUCCEEDED,
            substrs=["stop reason = step out"],
            patterns=["frame #0.*main.c:%d" % self.line4],
        )