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
|
# Copyright (c) 2010 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""GDB support for WebKit types.
Add this to your gdb by amending your ~/.gdbinit as follows:
python
import sys
sys.path.insert(0, "/path/to/tools/gdb/")
import webkit
"""
import gdb
import struct
def ustring_to_string(ptr, length=None):
"""Convert a pointer to UTF-16 data into a Python Unicode string.
ptr and length are both gdb.Value objects.
If length is unspecified, will guess at the length."""
extra = ''
if length is None:
# Try to guess at the length.
for i in xrange(0, 2048):
if int((ptr + i).dereference()) == 0:
length = i
break
if length is None:
length = 256
extra = u' (no trailing NUL found)'
else:
length = int(length)
char_vals = [int((ptr + i).dereference()) for i in xrange(length)]
string = struct.pack('H' * length, *char_vals).decode('utf-16', 'replace')
return string + extra
class StringPrinter(object):
"Shared code between different string-printing classes"
def __init__(self, val):
self.val = val
def display_hint(self):
return 'string'
class UCharStringPrinter(StringPrinter):
"Print a UChar*; we must guess at the length"
def to_string(self):
return ustring_to_string(self.val)
class WebCoreAtomicStringPrinter(StringPrinter):
"Print a WebCore::AtomicString"
def to_string(self):
return self.val['m_string']
class WebCoreStringPrinter(StringPrinter):
"Print a WebCore::String"
def get_length(self):
if not self.val['m_impl']['m_ptr']:
return 0
return self.val['m_impl']['m_ptr']['m_length']
def to_string(self):
if self.get_length() == 0:
return '(null)'
return ustring_to_string(self.val['m_impl']['m_ptr']['m_data'],
self.get_length())
class WebCoreQualifiedNamePrinter(StringPrinter):
"Print a WebCore::QualifiedName"
def __init__(self, val):
super(WebCoreQualifiedNamePrinter, self).__init__(val)
self.prefix_length = 0
self.length = 0
if self.val['m_impl']:
self.prefix_printer = WebCoreStringPrinter(
self.val['m_impl']['m_prefix']['m_string'])
self.local_name_printer = WebCoreStringPrinter(
self.val['m_impl']['m_localName']['m_string'])
self.prefix_length = self.prefix_printer.get_length()
if self.prefix_length > 0:
self.length = (self.prefix_length + 1 +
self.local_name_printer.get_length())
else:
self.length = self.local_name_printer.get_length()
def get_length(self):
return self.length
def to_string(self):
if self.get_length() == 0:
return "(null)"
else:
if self.prefix_length > 0:
return (self.prefix_printer.to_string() + ":" +
self.local_name_printer.to_string())
else:
return self.local_name_printer.to_string()
def lookup_function(val):
"""Function used to load pretty printers; will be passed to GDB."""
lookup_tag = val.type.tag
printers = {
"WebCore::AtomicString": WebCoreAtomicStringPrinter,
"WebCore::String": WebCoreStringPrinter,
"WebCore::QualifiedName": WebCoreQualifiedNamePrinter,
}
name = val.type.tag
if name in printers:
return printers[name](val)
if val.type.code == gdb.TYPE_CODE_PTR:
name = str(val.type.target().unqualified())
if name == 'UChar':
return UCharStringPrinter(val)
return None
gdb.pretty_printers.append(lookup_function)
class PrintPathToRootCommand(gdb.Command):
"""Command for printing WebKit Node trees.
Usage: printpathtoroot variable_name
"""
def __init__(self):
super(PrintPathToRootCommand, self).__init__("printpathtoroot",
gdb.COMMAND_SUPPORT,
gdb.COMPLETE_NONE)
def invoke(self, arg, from_tty):
element_type = gdb.lookup_type('WebCore::Element')
node_type = gdb.lookup_type('WebCore::Node')
frame = gdb.selected_frame()
try:
val = gdb.Frame.read_var(frame, arg)
except:
print "No such variable, or invalid type"
return
target_type = str(val.type.target().strip_typedefs())
if target_type == str(node_type):
stack = []
while val:
stack.append([val,
val.cast(element_type.pointer()).dereference()['m_tagName']])
val = val.dereference()['m_parent']
padding = ''
while len(stack) > 0:
pair = stack.pop()
print padding, pair[1], pair[0]
padding = padding + ' '
else:
print 'Sorry: I don\'t know how to deal with %s yet.' % target_type
PrintPathToRootCommand()
|