1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
|
"""Test printing ivars and ObjC objects captured in blocks that are made in methods of an ObjC class."""
import lldb
from lldbsuite.test.decorators import *
from lldbsuite.test.lldbtest import *
from lldbsuite.test import lldbutil
class TestObjCIvarsInBlocks(TestBase):
def setUp(self):
# Call super's setUp().
TestBase.setUp(self)
# Find the line numbers to break inside main().
self.main_source = "main.m"
self.class_source = "ivars-in-blocks.m"
self.class_source_file_spec = lldb.SBFileSpec(self.class_source)
@add_test_categories(["pyapi"])
@skipIf(dwarf_version=["<", "4"])
@expectedFailureAll(
archs=["i[3-6]86"],
bugnumber="This test requires the 2.0 runtime, so it will fail on i386",
)
def test_with_python_api(self):
"""Test printing the ivars of the self when captured in blocks"""
self.build()
exe = self.getBuildArtifact("a.out")
target = self.dbg.CreateTarget(exe)
self.assertTrue(target, VALID_TARGET)
breakpoint = target.BreakpointCreateBySourceRegex(
"// Break here inside the block.", self.class_source_file_spec
)
self.assertTrue(breakpoint, VALID_BREAKPOINT)
breakpoint_two = target.BreakpointCreateBySourceRegex(
"// Break here inside the class method block.", self.class_source_file_spec
)
self.assertTrue(breakpoint, VALID_BREAKPOINT)
process = target.LaunchSimple(None, None, self.get_process_working_directory())
self.assertTrue(process, "Created a process.")
self.assertEqual(process.GetState(), lldb.eStateStopped, "Stopped it too.")
self.runCmd("settings set target.prefer-dynamic-value no-dynamic-values")
thread_list = lldbutil.get_threads_stopped_at_breakpoint(process, breakpoint)
self.assertEqual(len(thread_list), 1)
thread = thread_list[0]
frame = thread.GetFrameAtIndex(0)
self.assertTrue(frame, "frame 0 is valid")
# First use the FindVariable API to see if we can find the ivar by
# undecorated name:
direct_blocky = frame.GetValueForVariablePath("blocky_ivar")
self.assertTrue(direct_blocky, "Found direct access to blocky_ivar.")
# Now get it as a member of "self" and make sure the two values are
# equal:
self_var = frame.GetValueForVariablePath("self")
self.assertTrue(self_var, "Found self in block.")
indirect_blocky = self_var.GetChildMemberWithName("blocky_ivar")
self.assertTrue(indirect_blocky, "Found blocky_ivar through self")
error = lldb.SBError()
direct_value = direct_blocky.GetValueAsSigned(error)
self.assertSuccess(error, "Got direct value for blocky_ivar")
indirect_value = indirect_blocky.GetValueAsSigned(error)
self.assertSuccess(error, "Got indirect value for blocky_ivar")
self.assertEqual(
direct_value, indirect_value, "Direct and indirect values are equal."
)
# Now make sure that we can get at the captured ivar through the expression parser.
# Doing a little trivial math will force this into the real expression
# parser:
direct_expr = frame.EvaluateExpression("blocky_ivar + 10")
self.assertTrue(direct_expr, "Got blocky_ivar through the expression parser")
# Again, get the value through self directly and make sure they are the
# same:
indirect_expr = frame.EvaluateExpression("self->blocky_ivar + 10")
self.assertTrue(
indirect_expr, "Got blocky ivar through expression parser using self."
)
direct_value = direct_expr.GetValueAsSigned(error)
self.assertTrue(
error.Success(), "Got value from direct use of expression parser"
)
indirect_value = indirect_expr.GetValueAsSigned(error)
self.assertTrue(
error.Success(),
"Got value from indirect access using the expression parser",
)
self.assertEqual(
direct_value,
indirect_value,
"Direct ivar access and indirect through expression parser produce same value.",
)
process.Continue()
self.assertEqual(
process.GetState(), lldb.eStateStopped, "Stopped at the second breakpoint."
)
thread_list = lldbutil.get_threads_stopped_at_breakpoint(
process, breakpoint_two
)
self.assertEqual(len(thread_list), 1)
thread = thread_list[0]
frame = thread.GetFrameAtIndex(0)
self.assertTrue(frame, "frame 0 is valid")
expr = frame.EvaluateExpression("(ret)")
self.assertTrue(
expr, "Successfully got a local variable in a block in a class method."
)
ret_value_signed = expr.GetValueAsSigned(error)
self.trace("ret_value_signed = %i" % (ret_value_signed))
self.assertEqual(
ret_value_signed, 5, "The local variable in the block was what we expected."
)
|