""" Test that addresses of functions compiled for Arm Thumb include the Thumb mode bit (bit 0 of the address) when resolved and used in expressions. """ import lldb from lldbsuite.test.decorators import * from lldbsuite.test.lldbtest import * from lldbsuite.test import lldbutil class TestThumbFunctionAddr(TestBase): def do_thumb_function_test(self, language): self.build(dictionary={"CFLAGS_EXTRAS": f"-x {language} -mthumb"}) exe = self.getBuildArtifact("a.out") line = line_number("main.c", "// Set break point at this line.") self.runCmd("target create %s" % exe) bpid = lldbutil.run_break_set_by_file_and_line(self, "main.c", line) self.runCmd("run") self.assertIsNotNone( lldbutil.get_one_thread_stopped_at_breakpoint_id(self.process(), bpid), "Process is not stopped at breakpoint", ) # The compiler set this, so the mode bit will be included here. a_function_addr_var = ( self.thread().GetFrameAtIndex(0).FindVariable("a_function_addr") ) self.assertTrue(a_function_addr_var.IsValid()) a_function_addr = a_function_addr_var.GetValueAsUnsigned() self.assertTrue(a_function_addr & 1) self.expect("p/x a_function_addr", substrs=[f"0x{a_function_addr:08x}"]) # If lldb did not pay attention to the mode bit this would SIGILL trying # to execute Thumb encodings in Arm mode. self.expect("expression -- a_function()", substrs=["= 123"]) # We cannot call GetCallableLoadAdress via. the API, so we expect this # to not have the bit set as it's treating it as a non-function symbol. found_function = self.target().FindFunctions("a_function")[0] self.assertTrue(found_function.IsValid()) found_function = found_function.GetFunction() self.assertTrue(found_function.IsValid()) found_function_addr = found_function.GetStartAddress() a_function_load_addr = found_function_addr.GetLoadAddress(self.target()) self.assertEqual(a_function_load_addr, a_function_addr & ~1) # image lookup should not include the mode bit. a_function_file_addr = found_function_addr.GetFileAddress() self.expect( "image lookup -n a_function", substrs=[f"0x{a_function_file_addr:08x}"] ) # This test is run for C and C++ because the two will take different paths # trying to resolve the function's address. @skipIf(archs=no_match(["arm$"])) @skipIf(archs=["arm64"]) def test_function_addr_c(self): self.do_thumb_function_test("c") @skipIf(archs=no_match(["arm$"])) @skipIf(archs=["arm64"]) def test_function_addr_cpp(self): self.do_thumb_function_test("c++")