1 """ 2 Test retrieval of SBAddress from function/symbol, disassembly, and SBAddress APIs. 3 """ 4 5 import os, time 6 import re 7 import unittest2 8 import lldb, lldbutil 9 from lldbtest import * 10 11 class DisasmAPITestCase(TestBase): 12 13 mydir = os.path.join("python_api", "function_symbol") 14 15 @unittest2.skipUnless(sys.platform.startswith("darwin"), "requires Darwin") 16 @python_api_test 17 @dsym_test 18 def test_with_dsym(self): 19 """Exercise getting SBAddress objects, disassembly, and SBAddress APIs.""" 20 self.buildDsym() 21 self.disasm_and_address_api() 22 23 @python_api_test 24 @dwarf_test 25 def test_with_dwarf(self): 26 """Exercise getting SBAddress objects, disassembly, and SBAddress APIs.""" 27 self.buildDwarf() 28 self.disasm_and_address_api() 29 30 def setUp(self): 31 # Call super's setUp(). 32 TestBase.setUp(self) 33 # Find the line number to of function 'c'. 34 self.line1 = line_number('main.c', '// Find the line number for breakpoint 1 here.') 35 self.line2 = line_number('main.c', '// Find the line number for breakpoint 2 here.') 36 37 def disasm_and_address_api(self): 38 """Exercise getting SBAddress objects, disassembly, and SBAddress APIs.""" 39 exe = os.path.join(os.getcwd(), "a.out") 40 41 # Create a target by the debugger. 42 target = self.dbg.CreateTarget(exe) 43 self.assertTrue(target, VALID_TARGET) 44 45 # Now create the two breakpoints inside function 'a'. 46 breakpoint1 = target.BreakpointCreateByLocation('main.c', self.line1) 47 breakpoint2 = target.BreakpointCreateByLocation('main.c', self.line2) 48 #print "breakpoint1:", breakpoint1 49 #print "breakpoint2:", breakpoint2 50 self.assertTrue(breakpoint1 and 51 breakpoint1.GetNumLocations() == 1, 52 VALID_BREAKPOINT) 53 self.assertTrue(breakpoint2 and 54 breakpoint2.GetNumLocations() == 1, 55 VALID_BREAKPOINT) 56 57 # Now launch the process, and do not stop at entry point. 58 process = target.LaunchSimple(None, None, os.getcwd()) 59 self.assertTrue(process, PROCESS_IS_VALID) 60 61 # Frame #0 should be on self.line1. 62 self.assertTrue(process.GetState() == lldb.eStateStopped) 63 thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonBreakpoint) 64 self.assertTrue(thread.IsValid(), "There should be a thread stopped due to breakpoint condition") 65 frame0 = thread.GetFrameAtIndex(0) 66 lineEntry = frame0.GetLineEntry() 67 self.assertTrue(lineEntry.GetLine() == self.line1) 68 69 address1 = lineEntry.GetStartAddress() 70 #print "address1:", address1 71 72 # Now call SBTarget.ResolveSymbolContextForAddress() with address1. 73 context1 = target.ResolveSymbolContextForAddress(address1, lldb.eSymbolContextEverything) 74 75 self.assertTrue(context1) 76 if self.TraceOn(): 77 print "context1:", context1 78 79 # Continue the inferior, the breakpoint 2 should be hit. 80 process.Continue() 81 self.assertTrue(process.GetState() == lldb.eStateStopped) 82 thread = lldbutil.get_stopped_thread(process, lldb.eStopReasonBreakpoint) 83 self.assertTrue(thread.IsValid(), "There should be a thread stopped due to breakpoint condition") 84 frame0 = thread.GetFrameAtIndex(0) 85 lineEntry = frame0.GetLineEntry() 86 self.assertTrue(lineEntry.GetLine() == self.line2) 87 88 # Verify that the symbol and the function has the same address range per function 'a'. 89 symbol = context1.GetSymbol() 90 function = frame0.GetFunction() 91 self.assertTrue(symbol and function) 92 93 disasm_output = lldbutil.disassemble(target, symbol) 94 if self.TraceOn(): 95 print "symbol:", symbol 96 print "disassembly=>\n", disasm_output 97 98 disasm_output = lldbutil.disassemble(target, function) 99 if self.TraceOn(): 100 print "function:", function 101 print "disassembly=>\n", disasm_output 102 103 sa1 = symbol.GetStartAddress() 104 #print "sa1:", sa1 105 #print "sa1.GetFileAddress():", hex(sa1.GetFileAddress()) 106 #ea1 = symbol.GetEndAddress() 107 #print "ea1:", ea1 108 sa2 = function.GetStartAddress() 109 #print "sa2:", sa2 110 #print "sa2.GetFileAddress():", hex(sa2.GetFileAddress()) 111 #ea2 = function.GetEndAddress() 112 #print "ea2:", ea2 113 self.assertTrue(sa1 and sa2 and sa1 == sa2, 114 "The two starting addresses should be the same") 115 116 from lldbutil import get_description 117 desc1 = get_description(sa1) 118 desc2 = get_description(sa2) 119 self.assertTrue(desc1 and desc2 and desc1 == desc2, 120 "SBAddress.GetDescription() API of sa1 and sa2 should return the same string") 121 122 123 if __name__ == '__main__': 124 import atexit 125 lldb.SBDebugger.Initialize() 126 atexit.register(lambda: lldb.SBDebugger.Terminate()) 127 unittest2.main() 128