Skip to content

Commit d4ee1c9

Browse files
committedNov 21, 2018
Fix BTreeSet and BTreeMap gdb pretty-printers
The BTreeSet and BTreeMap gdb pretty-printers did not take the node structure into account, and consequently only worked for shallow sets. This fixes the problem by iterating over child nodes when needed. This patch avoids the current approach of implementing some of the value manipulations in debugger-indepdendent code. This was done for convenience: a type lookup was needed for the first time, and there currently are no lldb formatters for these types. Closes #55771
1 parent 15e6613 commit d4ee1c9

File tree

3 files changed

+50
-63
lines changed

3 files changed

+50
-63
lines changed
 

‎src/etc/debugger_pretty_printers_common.py

-26
Original file line numberDiff line numberDiff line change
@@ -375,32 +375,6 @@ def extract_tail_head_ptr_and_cap_from_std_vecdeque(vec_val):
375375
assert data_ptr.type.get_dwarf_type_kind() == DWARF_TYPE_CODE_PTR
376376
return (tail, head, data_ptr, capacity)
377377

378-
379-
def extract_length_and_ptr_from_std_btreeset(vec_val):
380-
assert vec_val.type.get_type_kind() == TYPE_KIND_STD_BTREESET
381-
map = vec_val.get_child_at_index(0)
382-
root = map.get_child_at_index(0)
383-
length = map.get_child_at_index(1).as_integer()
384-
node = root.get_child_at_index(0)
385-
ptr = node.get_child_at_index(0)
386-
unique_ptr_val = ptr.get_child_at_index(0)
387-
data_ptr = unique_ptr_val.get_child_at_index(0)
388-
assert data_ptr.type.get_dwarf_type_kind() == DWARF_TYPE_CODE_PTR
389-
return (length, data_ptr)
390-
391-
392-
def extract_length_and_ptr_from_std_btreemap(vec_val):
393-
assert vec_val.type.get_type_kind() == TYPE_KIND_STD_BTREEMAP
394-
root = vec_val.get_child_at_index(0)
395-
length = vec_val.get_child_at_index(1).as_integer()
396-
node = root.get_child_at_index(0)
397-
ptr = node.get_child_at_index(0)
398-
unique_ptr_val = ptr.get_child_at_index(0)
399-
data_ptr = unique_ptr_val.get_child_at_index(0)
400-
assert data_ptr.type.get_dwarf_type_kind() == DWARF_TYPE_CODE_PTR
401-
return (length, data_ptr)
402-
403-
404378
def extract_length_and_ptr_from_slice(slice_val):
405379
assert (slice_val.type.get_type_kind() == TYPE_KIND_SLICE or
406380
slice_val.type.get_type_kind() == TYPE_KIND_STR_SLICE)

‎src/etc/gdb_rust_pretty_printing.py

+41-29
Original file line numberDiff line numberDiff line change
@@ -304,6 +304,32 @@ def children(self):
304304
yield (str(index), (gdb_ptr + index).dereference())
305305

306306

307+
# Yield each key (and optionally value) from a BoxedNode.
308+
def children_of_node(boxed_node, height, want_values):
309+
ptr = boxed_node['ptr']['pointer']
310+
# This is written oddly because we don't want to rely on the field name being `__0`.
311+
node_ptr = ptr[ptr.type.fields()[0]]
312+
if height > 0:
313+
type_name = str(node_ptr.type.target()).replace('LeafNode', 'InternalNode')
314+
node_type = gdb.lookup_type(type_name)
315+
node_ptr = node_ptr.cast(node_type.pointer())
316+
leaf = node_ptr['data']
317+
else:
318+
leaf = node_ptr.dereference()
319+
keys = leaf['keys']['value']['value']
320+
if want_values:
321+
values = leaf['vals']['value']['value']
322+
length = int(leaf['len'])
323+
for i in xrange(0, length + 1):
324+
if height > 0:
325+
for child in children_of_node(node_ptr['edges'][i], height - 1, want_values):
326+
yield child
327+
if i < length:
328+
if want_values:
329+
yield (keys[i], values[i])
330+
else:
331+
yield keys[i]
332+
307333
class RustStdBTreeSetPrinter(object):
308334
def __init__(self, val):
309335
self.__val = val
@@ -313,21 +339,16 @@ def display_hint():
313339
return "array"
314340

315341
def to_string(self):
316-
(length, data_ptr) = \
317-
rustpp.extract_length_and_ptr_from_std_btreeset(self.__val)
318342
return (self.__val.type.get_unqualified_type_name() +
319-
("(len: %i)" % length))
343+
("(len: %i)" % self.__val.get_wrapped_value()['map']['length']))
320344

321345
def children(self):
322-
(length, data_ptr) = \
323-
rustpp.extract_length_and_ptr_from_std_btreeset(self.__val)
324-
leaf_node = GdbValue(data_ptr.get_wrapped_value().dereference())
325-
maybe_uninit_keys = leaf_node.get_child_at_index(3)
326-
manually_drop_keys = maybe_uninit_keys.get_child_at_index(1)
327-
keys = manually_drop_keys.get_child_at_index(0)
328-
gdb_ptr = keys.get_wrapped_value()
329-
for index in xrange(length):
330-
yield (str(index), gdb_ptr[index])
346+
root = self.__val.get_wrapped_value()['map']['root']
347+
node_ptr = root['node']
348+
i = 0
349+
for child in children_of_node(node_ptr, root['height'], False):
350+
yield (str(i), child)
351+
i = i + 1
331352

332353

333354
class RustStdBTreeMapPrinter(object):
@@ -339,26 +360,17 @@ def display_hint():
339360
return "map"
340361

341362
def to_string(self):
342-
(length, data_ptr) = \
343-
rustpp.extract_length_and_ptr_from_std_btreemap(self.__val)
344363
return (self.__val.type.get_unqualified_type_name() +
345-
("(len: %i)" % length))
364+
("(len: %i)" % self.__val.get_wrapped_value()['length']))
346365

347366
def children(self):
348-
(length, data_ptr) = \
349-
rustpp.extract_length_and_ptr_from_std_btreemap(self.__val)
350-
leaf_node = GdbValue(data_ptr.get_wrapped_value().dereference())
351-
maybe_uninit_keys = leaf_node.get_child_at_index(3)
352-
manually_drop_keys = maybe_uninit_keys.get_child_at_index(1)
353-
keys = manually_drop_keys.get_child_at_index(0)
354-
keys_ptr = keys.get_wrapped_value()
355-
maybe_uninit_vals = leaf_node.get_child_at_index(4)
356-
manually_drop_vals = maybe_uninit_vals.get_child_at_index(1)
357-
vals = manually_drop_vals.get_child_at_index(0)
358-
vals_ptr = vals.get_wrapped_value()
359-
for index in xrange(length):
360-
yield (str(index), keys_ptr[index])
361-
yield (str(index), vals_ptr[index])
367+
root = self.__val.get_wrapped_value()['root']
368+
node_ptr = root['node']
369+
i = 0
370+
for child in children_of_node(node_ptr, root['height'], True):
371+
yield (str(i), child[0])
372+
yield (str(i), child[1])
373+
i = i + 1
362374

363375

364376
class RustStdStringPrinter(object):

‎src/test/debuginfo/pretty-std-collections.rs

+9-8
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
// ignore-tidy-linelength
1112
// ignore-windows failing on win32 bot
1213
// ignore-freebsd: gdb package too new
1314
// ignore-android: FIXME(#10381)
@@ -20,10 +21,10 @@
2021
// gdb-command: run
2122

2223
// gdb-command: print btree_set
23-
// gdb-check:$1 = BTreeSet<i32>(len: 3) = {3, 5, 7}
24+
// gdb-check:$1 = BTreeSet<i32>(len: 15) = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14}
2425

2526
// gdb-command: print btree_map
26-
// gdb-check:$2 = BTreeMap<i32, i32>(len: 3) = {[3] = 3, [5] = 7, [7] = 4}
27+
// gdb-check:$2 = BTreeMap<i32, i32>(len: 15) = {[0] = 0, [1] = 1, [2] = 2, [3] = 3, [4] = 4, [5] = 5, [6] = 6, [7] = 7, [8] = 8, [9] = 9, [10] = 10, [11] = 11, [12] = 12, [13] = 13, [14] = 14}
2728

2829
// gdb-command: print vec_deque
2930
// gdb-check:$3 = VecDeque<i32>(len: 3, cap: 8) = {5, 3, 7}
@@ -38,15 +39,15 @@ fn main() {
3839

3940
// BTreeSet
4041
let mut btree_set = BTreeSet::new();
41-
btree_set.insert(5);
42-
btree_set.insert(3);
43-
btree_set.insert(7);
42+
for i in 0..15 {
43+
btree_set.insert(i);
44+
}
4445

4546
// BTreeMap
4647
let mut btree_map = BTreeMap::new();
47-
btree_map.insert(5, 7);
48-
btree_map.insert(3, 3);
49-
btree_map.insert(7, 4);
48+
for i in 0..15 {
49+
btree_map.insert(i, i);
50+
}
5051

5152
// VecDeque
5253
let mut vec_deque = VecDeque::new();

0 commit comments

Comments
 (0)
Please sign in to comment.