]> git.lizzy.rs Git - rust.git/blob - src/etc/lldb_rust_formatters.py
Auto merge of #31077 - nagisa:mir-temp-promotion, r=dotdash
[rust.git] / src / etc / lldb_rust_formatters.py
1 # Copyright 2014 The Rust Project Developers. See the COPYRIGHT
2 # file at the top-level directory of this distribution and at
3 # http://rust-lang.org/COPYRIGHT.
4 #
5 # Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6 # http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7 # <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8 # option. This file may not be copied, modified, or distributed
9 # except according to those terms.
10
11 import lldb
12 import re
13 import debugger_pretty_printers_common as rustpp
14
15 #===============================================================================
16 # LLDB Pretty Printing Module for Rust
17 #===============================================================================
18
19 class LldbType(rustpp.Type):
20
21     def __init__(self, ty):
22         super(LldbType, self).__init__()
23         self.ty = ty
24         self.fields = None
25
26     def get_unqualified_type_name(self):
27         qualified_name = self.ty.GetName()
28
29         if qualified_name is None:
30             return qualified_name
31
32         return extract_type_name(qualified_name).replace("&'static ", "&")
33
34     def get_dwarf_type_kind(self):
35         type_class = self.ty.GetTypeClass()
36
37         if type_class == lldb.eTypeClassStruct:
38             return rustpp.DWARF_TYPE_CODE_STRUCT
39
40         if type_class == lldb.eTypeClassUnion:
41             return rustpp.DWARF_TYPE_CODE_UNION
42
43         if type_class == lldb.eTypeClassPointer:
44             return rustpp.DWARF_TYPE_CODE_PTR
45
46         if type_class == lldb.eTypeClassArray:
47             return rustpp.DWARF_TYPE_CODE_ARRAY
48
49         if type_class == lldb.eTypeClassEnumeration:
50             return rustpp.DWARF_TYPE_CODE_ENUM
51
52         return None
53
54     def get_fields(self):
55         assert ((self.get_dwarf_type_kind() == rustpp.DWARF_TYPE_CODE_STRUCT) or
56                 (self.get_dwarf_type_kind() == rustpp.DWARF_TYPE_CODE_UNION))
57         if self.fields is None:
58             self.fields = list(self.ty.fields)
59         return self.fields
60
61     def get_wrapped_value(self):
62         return self.ty
63
64
65 class LldbValue(rustpp.Value):
66     def __init__(self, lldb_val):
67         ty = lldb_val.type
68         wty = LldbType(ty)
69         super(LldbValue, self).__init__(wty)
70         self.lldb_val = lldb_val
71         self.children = {}
72
73     def get_child_at_index(self, index):
74         child = self.children.get(index)
75         if child is None:
76             lldb_field = self.lldb_val.GetChildAtIndex(index)
77             child = LldbValue(lldb_field)
78             self.children[index] = child
79         return child
80
81     def as_integer(self):
82         return self.lldb_val.GetValueAsUnsigned()
83
84     def get_wrapped_value(self):
85         return self.lldb_val
86
87
88 def print_val(lldb_val, internal_dict):
89     val = LldbValue(lldb_val)
90     type_kind = val.type.get_type_kind()
91
92     if (type_kind == rustpp.TYPE_KIND_REGULAR_STRUCT or
93         type_kind == rustpp.TYPE_KIND_EMPTY):
94         return print_struct_val(val,
95                                 internal_dict,
96                                 omit_first_field = False,
97                                 omit_type_name = False,
98                                 is_tuple_like = False)
99
100     if type_kind == rustpp.TYPE_KIND_STRUCT_VARIANT:
101         return print_struct_val(val,
102                                 internal_dict,
103                                 omit_first_field = True,
104                                 omit_type_name = False,
105                                 is_tuple_like = False)
106
107     if type_kind == rustpp.TYPE_KIND_SLICE:
108         return print_vec_slice_val(val, internal_dict)
109
110     if type_kind == rustpp.TYPE_KIND_STR_SLICE:
111         return print_str_slice_val(val, internal_dict)
112
113     if type_kind == rustpp.TYPE_KIND_STD_VEC:
114         return print_std_vec_val(val, internal_dict)
115
116     if type_kind == rustpp.TYPE_KIND_STD_STRING:
117         return print_std_string_val(val, internal_dict)
118
119     if type_kind == rustpp.TYPE_KIND_TUPLE:
120         return print_struct_val(val,
121                                 internal_dict,
122                                 omit_first_field = False,
123                                 omit_type_name = True,
124                                 is_tuple_like = True)
125
126     if type_kind == rustpp.TYPE_KIND_TUPLE_STRUCT:
127         return print_struct_val(val,
128                                 internal_dict,
129                                 omit_first_field = False,
130                                 omit_type_name = False,
131                                 is_tuple_like = True)
132
133     if type_kind == rustpp.TYPE_KIND_CSTYLE_VARIANT:
134         return val.type.get_unqualified_type_name()
135
136     if type_kind == rustpp.TYPE_KIND_TUPLE_VARIANT:
137         return print_struct_val(val,
138                                 internal_dict,
139                                 omit_first_field = True,
140                                 omit_type_name = False,
141                                 is_tuple_like = True)
142
143     if type_kind == rustpp.TYPE_KIND_SINGLETON_ENUM:
144         return print_val(lldb_val.GetChildAtIndex(0), internal_dict)
145
146     if type_kind == rustpp.TYPE_KIND_PTR:
147         return print_pointer_val(val, internal_dict)
148
149     if type_kind == rustpp.TYPE_KIND_FIXED_SIZE_VEC:
150         return print_fixed_size_vec_val(val, internal_dict)
151
152     if type_kind == rustpp.TYPE_KIND_REGULAR_ENUM:
153         # This is a regular enum, extract the discriminant
154         discriminant_val = rustpp.get_discriminant_value_as_integer(val)
155         return print_val(lldb_val.GetChildAtIndex(discriminant_val), internal_dict)
156
157     if type_kind == rustpp.TYPE_KIND_COMPRESSED_ENUM:
158         encoded_enum_info = rustpp.EncodedEnumInfo(val)
159         if encoded_enum_info.is_null_variant():
160             return encoded_enum_info.get_null_variant_name()
161
162         non_null_val = encoded_enum_info.get_non_null_variant_val()
163         return print_val(non_null_val.get_wrapped_value(), internal_dict)
164
165     # No pretty printer has been found
166     return lldb_val.GetValue()
167
168
169 #=--------------------------------------------------------------------------------------------------
170 # Type-Specialized Printing Functions
171 #=--------------------------------------------------------------------------------------------------
172
173 def print_struct_val(val, internal_dict, omit_first_field, omit_type_name, is_tuple_like):
174     '''
175     Prints a struct, tuple, or tuple struct value with Rust syntax.
176     Ignores any fields before field_start_index.
177     '''
178     assert val.type.get_dwarf_type_kind() == rustpp.DWARF_TYPE_CODE_STRUCT
179
180     if omit_type_name:
181         type_name = ""
182     else:
183         type_name = val.type.get_unqualified_type_name()
184
185     if is_tuple_like:
186         template = "%(type_name)s(%(body)s)"
187         separator = ", "
188     else:
189         template = "%(type_name)s {\n%(body)s\n}"
190         separator = ", \n"
191
192     fields = val.type.get_fields()
193
194     def render_child(child_index):
195         this = ""
196         if not is_tuple_like:
197             field_name = fields[child_index].name
198             this += field_name + ": "
199
200         field_val = val.get_child_at_index(child_index)
201
202         if not field_val.get_wrapped_value().IsValid():
203             field = fields[child_index]
204             # LLDB is not good at handling zero-sized values, so we have to help
205             # it a little
206             if field.GetType().GetByteSize() == 0:
207                 return this + extract_type_name(field.GetType().GetName())
208             else:
209                 return this + "<invalid value>"
210
211         return this + print_val(field_val.get_wrapped_value(), internal_dict)
212
213     if omit_first_field:
214         field_start_index = 1
215     else:
216         field_start_index = 0
217
218     body = separator.join([render_child(idx) for idx in range(field_start_index, len(fields))])
219
220     return template % {"type_name": type_name,
221                        "body": body}
222
223 def print_pointer_val(val, internal_dict):
224     '''Prints a pointer value with Rust syntax'''
225     assert val.type.get_dwarf_type_kind() == rustpp.DWARF_TYPE_CODE_PTR
226     sigil = "&"
227     type_name = val.type.get_unqualified_type_name()
228     if type_name and type_name[0:1] in ["&", "*"]:
229         sigil = type_name[0:1]
230
231     return sigil + hex(val.as_integer())
232
233
234 def print_fixed_size_vec_val(val, internal_dict):
235     assert val.type.get_dwarf_type_kind() == rustpp.DWARF_TYPE_CODE_ARRAY
236     lldb_val = val.get_wrapped_value()
237
238     output = "["
239
240     for i in range(lldb_val.num_children):
241         output += print_val(lldb_val.GetChildAtIndex(i), internal_dict)
242         if i != lldb_val.num_children - 1:
243             output += ", "
244
245     output += "]"
246     return output
247
248
249 def print_vec_slice_val(val, internal_dict):
250     (length, data_ptr) = rustpp.extract_length_and_ptr_from_slice(val)
251     return "&[%s]" % print_array_of_values(val.get_wrapped_value().GetName(),
252                                            data_ptr,
253                                            length,
254                                            internal_dict)
255
256
257 def print_std_vec_val(val, internal_dict):
258     (length, data_ptr, cap) = rustpp.extract_length_ptr_and_cap_from_std_vec(val)
259     return "vec![%s]" % print_array_of_values(val.get_wrapped_value().GetName(),
260                                               data_ptr,
261                                               length,
262                                               internal_dict)
263
264 def print_str_slice_val(val, internal_dict):
265     (length, data_ptr) = rustpp.extract_length_and_ptr_from_slice(val)
266     return read_utf8_string(data_ptr, length)
267
268 def print_std_string_val(val, internal_dict):
269     vec = val.get_child_at_index(0)
270     (length, data_ptr, cap) = rustpp.extract_length_ptr_and_cap_from_std_vec(vec)
271     return read_utf8_string(data_ptr, length)
272
273 #=--------------------------------------------------------------------------------------------------
274 # Helper Functions
275 #=--------------------------------------------------------------------------------------------------
276
277 UNQUALIFIED_TYPE_MARKERS = frozenset(["(", "[", "&", "*"])
278
279 def extract_type_name(qualified_type_name):
280     '''Extracts the type name from a fully qualified path'''
281     if qualified_type_name[0] in UNQUALIFIED_TYPE_MARKERS:
282         return qualified_type_name
283
284     end_of_search = qualified_type_name.find("<")
285     if end_of_search < 0:
286         end_of_search = len(qualified_type_name)
287
288     index = qualified_type_name.rfind("::", 0, end_of_search)
289     if index < 0:
290         return qualified_type_name
291     else:
292         return qualified_type_name[index + 2:]
293
294 def print_array_of_values(array_name, data_ptr_val, length, internal_dict):
295     '''Prints a contigous memory range, interpreting it as values of the
296        pointee-type of data_ptr_val.'''
297
298     data_ptr_type = data_ptr_val.type
299     assert data_ptr_type.get_dwarf_type_kind() == rustpp.DWARF_TYPE_CODE_PTR
300
301     element_type = data_ptr_type.get_wrapped_value().GetPointeeType()
302     element_type_size = element_type.GetByteSize()
303
304     start_address = data_ptr_val.as_integer()
305     raw_value = data_ptr_val.get_wrapped_value()
306
307     def render_element(i):
308         address = start_address + i * element_type_size
309         element_val = raw_value.CreateValueFromAddress(array_name + ("[%s]" % i),
310                                                        address,
311                                                        element_type)
312         return print_val(element_val, internal_dict)
313
314     return ', '.join([render_element(i) for i in range(length)])
315
316
317 def read_utf8_string(ptr_val, byte_count):
318     error = lldb.SBError()
319     process = ptr_val.get_wrapped_value().GetProcess()
320     data = process.ReadMemory(ptr_val.as_integer(), byte_count, error)
321     if error.Success():
322         return '"%s"' % data.decode(encoding='UTF-8')
323     else:
324         return '<error: %s>' % error.GetCString()