| """ |
| Use lldb Python API to verify that expression evaluation for property references uses the correct getters and setters |
| """ |
| |
| |
| |
| import lldb |
| from lldbsuite.test.decorators import * |
| from lldbsuite.test.lldbtest import * |
| from lldbsuite.test import lldbutil |
| |
| |
| class ObjCPropertyTestCase(TestBase): |
| |
| mydir = TestBase.compute_mydir(__file__) |
| |
| def setUp(self): |
| # Call super's setUp(). |
| TestBase.setUp(self) |
| |
| # Find the line number to break for main.c. |
| self.source_name = 'main.m' |
| |
| @add_test_categories(['pyapi']) |
| def test_objc_properties(self): |
| """Test that expr uses the correct property getters and setters""" |
| if self.getArchitecture() == 'i386': |
| self.skipTest("requires modern objc runtime") |
| |
| self.build() |
| exe = self.getBuildArtifact("a.out") |
| |
| # Create a target from the debugger. |
| |
| target = self.dbg.CreateTarget(exe) |
| self.assertTrue(target, VALID_TARGET) |
| |
| # Set up our breakpoints: |
| |
| main_bkpt = target.BreakpointCreateBySourceRegex( |
| "Set a breakpoint here.", lldb.SBFileSpec(self.source_name)) |
| self.assertTrue(main_bkpt and |
| main_bkpt.GetNumLocations() == 1, |
| VALID_BREAKPOINT) |
| |
| # Now launch the process, and do not stop at the entry point. |
| process = target.LaunchSimple( |
| None, None, self.get_process_working_directory()) |
| |
| self.assertEquals(process.GetState(), lldb.eStateStopped, |
| PROCESS_STOPPED) |
| |
| threads = lldbutil.get_threads_stopped_at_breakpoint( |
| process, main_bkpt) |
| self.assertEquals(len(threads), 1) |
| thread = threads[0] |
| frame = thread.GetFrameAtIndex(0) |
| |
| mine = frame.FindVariable("mine") |
| self.assertTrue(mine.IsValid()) |
| access_count = mine.GetChildMemberWithName("_access_count") |
| self.assertTrue(access_count.IsValid()) |
| start_access_count = access_count.GetValueAsUnsigned(123456) |
| self.assertNotEqual(start_access_count, 123456) |
| |
| # |
| # The first set of tests test calling the getter & setter of |
| # a property that actually only has a getter & setter and no |
| # @property. |
| # |
| nonexistant_value = frame.EvaluateExpression( |
| "mine.nonexistantInt", False) |
| nonexistant_error = nonexistant_value.GetError() |
| self.assertTrue(nonexistant_error.Success()) |
| nonexistant_int = nonexistant_value.GetValueAsUnsigned(123456) |
| self.assertEquals(nonexistant_int, 6) |
| |
| # Calling the getter function would up the access count, so make sure |
| # that happened. |
| |
| new_access_count = access_count.GetValueAsUnsigned(123456) |
| self.assertEquals(new_access_count - start_access_count, 1) |
| start_access_count = new_access_count |
| |
| # |
| # Now call the setter, then make sure that |
| nonexistant_change = frame.EvaluateExpression( |
| "mine.nonexistantInt = 10", False) |
| nonexistant_error = nonexistant_change.GetError() |
| self.assertTrue(nonexistant_error.Success()) |
| |
| # Calling the setter function would up the access count, so make sure |
| # that happened. |
| |
| new_access_count = access_count.GetValueAsUnsigned(123456) |
| self.assertEquals(new_access_count - start_access_count, 1) |
| start_access_count = new_access_count |
| |
| # |
| # Now we call the getter of a property that is backed by an ivar, |
| # make sure it works and that we actually update the backing ivar. |
| # |
| |
| backed_value = frame.EvaluateExpression("mine.backedInt", False) |
| backed_error = backed_value.GetError() |
| self.assertTrue(backed_error.Success()) |
| backing_value = mine.GetChildMemberWithName("_backedInt") |
| self.assertTrue(backing_value.IsValid()) |
| self.assertTrue(backed_value.GetValueAsUnsigned(12345) |
| == backing_value.GetValueAsUnsigned(23456)) |
| |
| value_from_typedef = frame.EvaluateExpression("typedefd.backedInt", False) |
| self.assertTrue(value_from_typedef.GetError().Success()) |
| self.assertEqual(value_from_typedef.GetValueAsUnsigned(12345), |
| backing_value.GetValueAsUnsigned(23456)) |
| |
| unbacked_value = frame.EvaluateExpression("mine.unbackedInt", False) |
| unbacked_error = unbacked_value.GetError() |
| self.assertTrue(unbacked_error.Success()) |
| |
| idWithProtocol_value = frame.EvaluateExpression( |
| "mine.idWithProtocol", False) |
| idWithProtocol_error = idWithProtocol_value.GetError() |
| self.assertTrue(idWithProtocol_error.Success()) |
| self.assertEquals(idWithProtocol_value.GetTypeName(), "id") |
| |
| # Make sure that class property getter works as expected |
| value = frame.EvaluateExpression("BaseClass.classInt", False) |
| self.assertTrue(value.GetError().Success()) |
| self.assertEquals(value.GetValueAsUnsigned(11111), 123) |
| |
| # Make sure that class property setter works as expected |
| value = frame.EvaluateExpression("BaseClass.classInt = 234", False) |
| self.assertTrue(value.GetError().Success()) |
| |
| # Verify that setter above actually worked |
| value = frame.EvaluateExpression("BaseClass.classInt", False) |
| self.assertTrue(value.GetError().Success()) |
| self.assertEquals(value.GetValueAsUnsigned(11111), 234) |
| |
| # Test that accessing two distinct class and instance properties that |
| # share the same name works. |
| self.expect_expr("mine.propConflict", result_value="4") |
| self.expect_expr("BaseClass.propConflict", result_value="6") |