summaryrefslogtreecommitdiffstats
path: root/lldb/examples/python/diagnose_unwind.py
blob: b90372970e5de3533190fce96ad73f16d60f6ca6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
# This implements the "diagnose-unwind" command, usually installed
# in the debug session like
#   command script import lldb.diagnose
# it is used when lldb's backtrace fails -- it collects and prints
# information about the stack frames, and tries an alternate unwind
# algorithm, that will help to understand why lldb's unwind algorithm
# did not succeed.

import optparse
import lldb
import re
import shlex

# Print the frame number, pc, frame pointer, module UUID and function name
# Returns the SBModule that contains the PC, if it could be found


def backtrace_print_frame(target, frame_num, addr, fp):
    process = target.GetProcess()
    addr_for_printing = addr
    addr_width = process.GetAddressByteSize() * 2
    if frame_num > 0:
        addr = addr - 1

    sbaddr = lldb.SBAddress()
    try:
        sbaddr.SetLoadAddress(addr, target)
        module_description = ""
        if sbaddr.GetModule():
            module_filename = ""
            module_uuid_str = sbaddr.GetModule().GetUUIDString()
            if module_uuid_str is None:
                module_uuid_str = ""
            if sbaddr.GetModule().GetFileSpec():
                module_filename = sbaddr.GetModule().GetFileSpec().GetFilename()
                if module_filename is None:
                    module_filename = ""
            if module_uuid_str != "" or module_filename != "":
                module_description = '%s %s' % (
                    module_filename, module_uuid_str)
    except Exception:
        print '%2d: pc==0x%-*x fp==0x%-*x' % (frame_num, addr_width, addr_for_printing, addr_width, fp)
        return

    sym_ctx = target.ResolveSymbolContextForAddress(
        sbaddr, lldb.eSymbolContextEverything)
    if sym_ctx.IsValid() and sym_ctx.GetSymbol().IsValid():
        function_start = sym_ctx.GetSymbol().GetStartAddress().GetLoadAddress(target)
        offset = addr - function_start
        print '%2d: pc==0x%-*x fp==0x%-*x %s %s + %d' % (frame_num, addr_width, addr_for_printing, addr_width, fp, module_description, sym_ctx.GetSymbol().GetName(), offset)
    else:
        print '%2d: pc==0x%-*x fp==0x%-*x %s' % (frame_num, addr_width, addr_for_printing, addr_width, fp, module_description)
    return sbaddr.GetModule()

# A simple stack walk algorithm that follows the frame chain.
# Returns a two-element list; the first element is a list of modules
# seen and the second element is a list of addresses seen during the backtrace.


def simple_backtrace(debugger):
    target = debugger.GetSelectedTarget()
    process = target.GetProcess()
    cur_thread = process.GetSelectedThread()

    initial_fp = cur_thread.GetFrameAtIndex(0).GetFP()

    # If the pseudoreg "fp" isn't recognized, on arm hardcode to r7 which is
    # correct for Darwin programs.
    if initial_fp == lldb.LLDB_INVALID_ADDRESS and target.triple[0:3] == "arm":
        for reggroup in cur_thread.GetFrameAtIndex(1).registers:
            if reggroup.GetName() == "General Purpose Registers":
                for reg in reggroup:
                    if reg.GetName() == "r7":
                        initial_fp = int(reg.GetValue(), 16)

    module_list = []
    address_list = [cur_thread.GetFrameAtIndex(0).GetPC()]
    this_module = backtrace_print_frame(
        target, 0, cur_thread.GetFrameAtIndex(0).GetPC(), initial_fp)
    print_stack_frame(process, initial_fp)
    print ""
    if this_module is not None:
        module_list.append(this_module)
    if cur_thread.GetNumFrames() < 2:
        return [module_list, address_list]

    cur_fp = process.ReadPointerFromMemory(initial_fp, lldb.SBError())
    cur_pc = process.ReadPointerFromMemory(
        initial_fp + process.GetAddressByteSize(), lldb.SBError())

    frame_num = 1

    while cur_pc != 0 and cur_fp != 0 and cur_pc != lldb.LLDB_INVALID_ADDRESS and cur_fp != lldb.LLDB_INVALID_ADDRESS:
        address_list.append(cur_pc)
        this_module = backtrace_print_frame(target, frame_num, cur_pc, cur_fp)
        print_stack_frame(process, cur_fp)
        print ""
        if this_module is not None:
            module_list.append(this_module)
        frame_num = frame_num + 1
        next_pc = 0
        next_fp = 0
        if target.triple[
            0:6] == "x86_64" or target.triple[
            0:4] == "i386" or target.triple[
                0:3] == "arm":
            error = lldb.SBError()
            next_pc = process.ReadPointerFromMemory(
                cur_fp + process.GetAddressByteSize(), error)
            if not error.Success():
                next_pc = 0
            next_fp = process.ReadPointerFromMemory(cur_fp, error)
            if not error.Success():
                next_fp = 0
        # Clear the 0th bit for arm frames - this indicates it is a thumb frame
        if target.triple[0:3] == "arm" and (next_pc & 1) == 1:
            next_pc = next_pc & ~1
        cur_pc = next_pc
        cur_fp = next_fp
    this_module = backtrace_print_frame(target, frame_num, cur_pc, cur_fp)
    print_stack_frame(process, cur_fp)
    print ""
    if this_module is not None:
        module_list.append(this_module)
    return [module_list, address_list]


def print_stack_frame(process, fp):
    if fp == 0 or fp == lldb.LLDB_INVALID_ADDRESS or fp == 1:
        return
    addr_size = process.GetAddressByteSize()
    addr = fp - (2 * addr_size)
    i = 0
    outline = "Stack frame from $fp-%d: " % (2 * addr_size)
    error = lldb.SBError()
    try:
        while i < 5 and error.Success():
            address = process.ReadPointerFromMemory(
                addr + (i * addr_size), error)
            outline += " 0x%x" % address
            i += 1
        print outline
    except Exception:
        return


def diagnose_unwind(debugger, command, result, dict):
    """
  Gather diagnostic information to help debug incorrect unwind (backtrace)
  behavior in lldb.  When there is a backtrace that doesn't look
  correct, run this command with the correct thread selected and a
  large amount of diagnostic information will be printed, it is likely
  to be helpful when reporting the problem.
    """

    command_args = shlex.split(command)
    parser = create_diagnose_unwind_options()
    try:
        (options, args) = parser.parse_args(command_args)
    except:
        return
    target = debugger.GetSelectedTarget()
    if target:
        process = target.GetProcess()
        if process:
            thread = process.GetSelectedThread()
            if thread:
                lldb_versions_match = re.search(
                    r'[lL][lL][dD][bB]-(\d+)([.](\d+))?([.](\d+))?',
                    debugger.GetVersionString())
                lldb_version = 0
                lldb_minor = 0
                if len(lldb_versions_match.groups()
                       ) >= 1 and lldb_versions_match.groups()[0]:
                    lldb_major = int(lldb_versions_match.groups()[0])
                if len(lldb_versions_match.groups()
                       ) >= 5 and lldb_versions_match.groups()[4]:
                    lldb_minor = int(lldb_versions_match.groups()[4])

                modules_seen = []
                addresses_seen = []

                print 'LLDB version %s' % debugger.GetVersionString()
                print 'Unwind diagnostics for thread %d' % thread.GetIndexID()
                print ""
                print "============================================================================================="
                print ""
                print "OS plugin setting:"
                debugger.HandleCommand(
                    "settings show target.process.python-os-plugin-path")
                print ""
                print "Live register context:"
                thread.SetSelectedFrame(0)
                debugger.HandleCommand("register read")
                print ""
                print "============================================================================================="
                print ""
                print "lldb's unwind algorithm:"
                print ""
                frame_num = 0
                for frame in thread.frames:
                    if not frame.IsInlined():
                        this_module = backtrace_print_frame(
                            target, frame_num, frame.GetPC(), frame.GetFP())
                        print_stack_frame(process, frame.GetFP())
                        print ""
                        if this_module is not None:
                            modules_seen.append(this_module)
                        addresses_seen.append(frame.GetPC())
                        frame_num = frame_num + 1
                print ""
                print "============================================================================================="
                print ""
                print "Simple stack walk algorithm:"
                print ""
                (module_list, address_list) = simple_backtrace(debugger)
                if module_list and module_list is not None:
                    modules_seen += module_list
                if address_list and address_list is not None:
                    addresses_seen = set(addresses_seen)
                    addresses_seen.update(set(address_list))

                print ""
                print "============================================================================================="
                print ""
                print "Modules seen in stack walks:"
                print ""
                modules_already_seen = set()
                for module in modules_seen:
                    if module is not None and module.GetFileSpec().GetFilename() is not None:
                        if not module.GetFileSpec().GetFilename() in modules_already_seen:
                            debugger.HandleCommand(
                                'image list %s' %
                                module.GetFileSpec().GetFilename())
                            modules_already_seen.add(
                                module.GetFileSpec().GetFilename())

                print ""
                print "============================================================================================="
                print ""
                print "Disassembly ofaddresses seen in stack walks:"
                print ""
                additional_addresses_to_disassemble = addresses_seen
                for frame in thread.frames:
                    if not frame.IsInlined():
                        print "--------------------------------------------------------------------------------------"
                        print ""
                        print "Disassembly of %s, frame %d, address 0x%x" % (frame.GetFunctionName(), frame.GetFrameID(), frame.GetPC())
                        print ""
                        if target.triple[
                                0:6] == "x86_64" or target.triple[
                                0:4] == "i386":
                            debugger.HandleCommand(
                                'disassemble -F att -a 0x%x' % frame.GetPC())
                        else:
                            debugger.HandleCommand(
                                'disassemble -a 0x%x' %
                                frame.GetPC())
                        if frame.GetPC() in additional_addresses_to_disassemble:
                            additional_addresses_to_disassemble.remove(
                                frame.GetPC())

                for address in list(additional_addresses_to_disassemble):
                    print "--------------------------------------------------------------------------------------"
                    print ""
                    print "Disassembly of 0x%x" % address
                    print ""
                    if target.triple[
                            0:6] == "x86_64" or target.triple[
                            0:4] == "i386":
                        debugger.HandleCommand(
                            'disassemble -F att -a 0x%x' % address)
                    else:
                        debugger.HandleCommand('disassemble -a 0x%x' % address)

                print ""
                print "============================================================================================="
                print ""
                additional_addresses_to_show_unwind = addresses_seen
                for frame in thread.frames:
                    if not frame.IsInlined():
                        print "--------------------------------------------------------------------------------------"
                        print ""
                        print "Unwind instructions for %s, frame %d" % (frame.GetFunctionName(), frame.GetFrameID())
                        print ""
                        debugger.HandleCommand(
                            'image show-unwind -a "0x%x"' % frame.GetPC())
                        if frame.GetPC() in additional_addresses_to_show_unwind:
                            additional_addresses_to_show_unwind.remove(
                                frame.GetPC())

                for address in list(additional_addresses_to_show_unwind):
                    print "--------------------------------------------------------------------------------------"
                    print ""
                    print "Unwind instructions for 0x%x" % address
                    print ""
                    debugger.HandleCommand(
                        'image show-unwind -a "0x%x"' % address)


def create_diagnose_unwind_options():
    usage = "usage: %prog"
    description = '''Print diagnostic information about a thread backtrace which will help to debug unwind problems'''
    parser = optparse.OptionParser(
        description=description,
        prog='diagnose_unwind',
        usage=usage)
    return parser

lldb.debugger.HandleCommand(
    'command script add -f %s.diagnose_unwind diagnose-unwind' %
    __name__)
print 'The "diagnose-unwind" command has been installed, type "help diagnose-unwind" for detailed help.'
OpenPOWER on IntegriCloud