• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1"""
2Test retrieval of SBAddress from function/symbol, disassembly, and SBAddress APIs.
3"""
4
5import os, time
6import re
7import unittest2
8import lldb, lldbutil
9from lldbtest import *
10
11class 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
123if __name__ == '__main__':
124    import atexit
125    lldb.SBDebugger.Initialize()
126    atexit.register(lambda: lldb.SBDebugger.Terminate())
127    unittest2.main()
128