199451b44SJordan Rupprecht"""Test passing structs to Objective-C methods."""
299451b44SJordan Rupprecht
399451b44SJordan Rupprecht
499451b44SJordan Rupprecht
599451b44SJordan Rupprechtimport lldb
699451b44SJordan Rupprechtfrom lldbsuite.test.decorators import *
799451b44SJordan Rupprechtfrom lldbsuite.test.lldbtest import *
899451b44SJordan Rupprechtfrom lldbsuite.test import lldbutil
999451b44SJordan Rupprecht
1099451b44SJordan Rupprecht
1199451b44SJordan Rupprechtclass TestObjCStructArgument(TestBase):
1299451b44SJordan Rupprecht
1399451b44SJordan Rupprecht    def setUp(self):
1499451b44SJordan Rupprecht        # Call super's setUp().
1599451b44SJordan Rupprecht        TestBase.setUp(self)
1699451b44SJordan Rupprecht        # Find the line numbers to break inside main().
1799451b44SJordan Rupprecht        self.main_source = "test.m"
1899451b44SJordan Rupprecht        self.break_line = line_number(
1999451b44SJordan Rupprecht            self.main_source, '// Set breakpoint here.')
2099451b44SJordan Rupprecht
2199451b44SJordan Rupprecht    @add_test_categories(['pyapi'])
2299451b44SJordan Rupprecht    @skipIf(debug_info=no_match(["gmodules"]), oslist=['ios', 'watchos', 'tvos', 'bridgeos'], archs=['armv7', 'arm64'])  # this test program only builds for ios with -gmodules
2399451b44SJordan Rupprecht    def test_with_python_api(self):
2499451b44SJordan Rupprecht        """Test passing structs to Objective-C methods."""
2599451b44SJordan Rupprecht        self.build()
2699451b44SJordan Rupprecht        exe = self.getBuildArtifact("a.out")
2799451b44SJordan Rupprecht
2899451b44SJordan Rupprecht        target = self.dbg.CreateTarget(exe)
2999451b44SJordan Rupprecht        self.assertTrue(target, VALID_TARGET)
3099451b44SJordan Rupprecht
3199451b44SJordan Rupprecht        bpt = target.BreakpointCreateByLocation(
3299451b44SJordan Rupprecht            self.main_source, self.break_line)
3399451b44SJordan Rupprecht        self.assertTrue(bpt, VALID_BREAKPOINT)
3499451b44SJordan Rupprecht
3599451b44SJordan Rupprecht        # Now launch the process, and do not stop at entry point.
3699451b44SJordan Rupprecht        process = target.LaunchSimple(
3799451b44SJordan Rupprecht            None, None, self.get_process_working_directory())
3899451b44SJordan Rupprecht
3999451b44SJordan Rupprecht        self.assertTrue(process, PROCESS_IS_VALID)
4099451b44SJordan Rupprecht
4199451b44SJordan Rupprecht        # The stop reason of the thread should be breakpoint.
4299451b44SJordan Rupprecht        thread_list = lldbutil.get_threads_stopped_at_breakpoint(process, bpt)
4399451b44SJordan Rupprecht
4499451b44SJordan Rupprecht        # Make sure we stopped at the first breakpoint.
4599451b44SJordan Rupprecht        self.assertTrue(
4699451b44SJordan Rupprecht            len(thread_list) != 0,
4799451b44SJordan Rupprecht            "No thread stopped at our breakpoint.")
48*b3a0c4d7SRaphael Isemann        self.assertEquals(len(thread_list), 1,
4999451b44SJordan Rupprecht                        "More than one thread stopped at our breakpoint.")
5099451b44SJordan Rupprecht
5199451b44SJordan Rupprecht        frame = thread_list[0].GetFrameAtIndex(0)
5299451b44SJordan Rupprecht        self.assertTrue(frame, "Got a valid frame 0 frame.")
5399451b44SJordan Rupprecht
5499451b44SJordan Rupprecht        self.expect("p [summer sumThings:tts]", substrs=['9'])
5599451b44SJordan Rupprecht
5699451b44SJordan Rupprecht        self.expect(
5799451b44SJordan Rupprecht            "po [NSValue valueWithRect:rect]",
5899451b44SJordan Rupprecht            substrs=['NSRect: {{0, 0}, {10, 20}}'])
5999451b44SJordan Rupprecht
6099451b44SJordan Rupprecht        # Now make sure we can call a method that returns a struct without
6199451b44SJordan Rupprecht        # crashing.
6299451b44SJordan Rupprecht        cmd_value = frame.EvaluateExpression("[provider getRange]")
6399451b44SJordan Rupprecht        self.assertTrue(cmd_value.IsValid())
64