# Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT # file at the top-level directory of this distribution and at # http://rust-lang.org/COPYRIGHT. # # Licensed under the Apache License, Version 2.0 or the MIT license # , at your # option. This file may not be copied, modified, or distributed # except according to those terms. import gdb import re #=============================================================================== # GDB Pretty Printing Module for Rust #=============================================================================== def register_printers(objfile): "Registers Rust pretty printers for the given objfile" objfile.pretty_printers.append(rust_pretty_printer_lookup_function) def rust_pretty_printer_lookup_function(val): "Returns the correct Rust pretty printer for the given value if there is one" type_code = val.type.code if type_code == gdb.TYPE_CODE_STRUCT: struct_kind = classify_struct(val.type) if struct_kind == STRUCT_KIND_SLICE: return RustSlicePrinter(val) if struct_kind == STRUCT_KIND_STR_SLICE: return RustStringSlicePrinter(val) if struct_kind == STRUCT_KIND_STD_VEC: return RustStdVecPrinter(val) if struct_kind == STRUCT_KIND_STD_STRING: return RustStdStringPrinter(val) if struct_kind == STRUCT_KIND_TUPLE: return RustTuplePrinter(val) if struct_kind == STRUCT_KIND_TUPLE_STRUCT: return RustTupleStructPrinter(val, False) if struct_kind == STRUCT_KIND_CSTYLE_VARIANT: return RustCStyleEnumPrinter(val[get_field_at_index(val, 0)]) if struct_kind == STRUCT_KIND_TUPLE_VARIANT: return RustTupleStructPrinter(val, True) if struct_kind == STRUCT_KIND_STRUCT_VARIANT: return RustStructPrinter(val, True) return RustStructPrinter(val, False) # Enum handling if type_code == gdb.TYPE_CODE_UNION: enum_members = list(val.type.fields()) enum_member_count = len(enum_members) if enum_member_count == 0: return RustStructPrinter(val, False) if enum_member_count == 1: first_variant_name = enum_members[0].name if first_variant_name is None: # This is a singleton enum return rust_pretty_printer_lookup_function(val[enum_members[0]]) else: assert first_variant_name.startswith("RUST$ENCODED$ENUM$") # This is a space-optimized enum. # This means this enum has only two states, and Rust uses one # of the fields somewhere in the struct to determine which of # the two states it's in. The location of the field is encoded # in the name as something like # RUST$ENCODED$ENUM$(num$)*name_of_zero_state last_separator_index = first_variant_name.rfind("$") start_index = len("RUST$ENCODED$ENUM$") disr_field_indices = first_variant_name[start_index:last_separator_index].split("$") disr_field_indices = [int(index) for index in disr_field_indices] sole_variant_val = val[enum_members[0]] discriminant = sole_variant_val for disr_field_index in disr_field_indices: disr_field = get_field_at_index(discriminant, disr_field_index) discriminant = discriminant[disr_field] # If the discriminant field is a fat pointer we have to consider the # first word as the true discriminant if discriminant.type.code == gdb.TYPE_CODE_STRUCT: discriminant = discriminant[get_field_at_index(discriminant, 0)] if discriminant == 0: null_variant_name = first_variant_name[last_separator_index + 1:] return IdentityPrinter(null_variant_name) return rust_pretty_printer_lookup_function(sole_variant_val) # This is a regular enum, extract the discriminant discriminant_name, discriminant_val = extract_discriminant_value(val) return rust_pretty_printer_lookup_function(val[enum_members[discriminant_val]]) # No pretty printer has been found return None #=------------------------------------------------------------------------------ # Pretty Printer Classes #=------------------------------------------------------------------------------ class RustStructPrinter: def __init__(self, val, hide_first_field): self.val = val self.hide_first_field = hide_first_field def to_string(self): return self.val.type.tag def children(self): cs = [] for field in self.val.type.fields(): field_name = field.name # Normally the field name is used as a key to access the field # value, because that's also supported in older versions of GDB... field_key = field_name if field_name is None: field_name = "" # ... but for fields without a name (as in tuples), we have to # fall back to the newer method of using the field object # directly as key. In older versions of GDB, this will just # fail. field_key = field name_value_tuple = (field_name, self.val[field_key]) cs.append(name_value_tuple) if self.hide_first_field: cs = cs[1:] return cs class RustTuplePrinter: def __init__(self, val): self.val = val def to_string(self): return None def children(self): cs = [] for field in self.val.type.fields(): cs.append(("", self.val[field])) return cs def display_hint(self): return "array" class RustTupleStructPrinter: def __init__(self, val, hide_first_field): self.val = val self.hide_first_field = hide_first_field def to_string(self): return self.val.type.tag def children(self): cs = [] for field in self.val.type.fields(): cs.append(("", self.val[field])) if self.hide_first_field: cs = cs[1:] return cs def display_hint(self): return "array" class RustSlicePrinter: def __init__(self, val): self.val = val def display_hint(self): return "array" def to_string(self): length = int(self.val["length"]) return self.val.type.tag + ("(len: %i)" % length) def children(self): cs = [] length = int(self.val["length"]) data_ptr = self.val["data_ptr"] assert data_ptr.type.code == gdb.TYPE_CODE_PTR pointee_type = data_ptr.type.target() for index in range(0, length): cs.append((str(index), (data_ptr + index).dereference())) return cs class RustStringSlicePrinter: def __init__(self, val): self.val = val def to_string(self): slice_byte_len = self.val["length"] return '"%s"' % self.val["data_ptr"].string(encoding="utf-8", length=slice_byte_len) class RustStdVecPrinter: def __init__(self, val): self.val = val def display_hint(self): return "array" def to_string(self): length = int(self.val["len"]) cap = int(self.val["cap"]) return self.val.type.tag + ("(len: %i, cap: %i)" % (length, cap)) def children(self): cs = [] (length, data_ptr) = extract_length_and_data_ptr_from_std_vec(self.val) pointee_type = data_ptr.type.target() for index in range(0, length): cs.append((str(index), (data_ptr + index).dereference())) return cs class RustStdStringPrinter: def __init__(self, val): self.val = val def to_string(self): (length, data_ptr) = extract_length_and_data_ptr_from_std_vec(self.val["vec"]) return '"%s"' % data_ptr.string(encoding="utf-8", length=length) class RustCStyleEnumPrinter: def __init__(self, val): assert val.type.code == gdb.TYPE_CODE_ENUM self.val = val def to_string(self): return str(self.val) class IdentityPrinter: def __init__(self, string): self.string = string def to_string(self): return self.string STRUCT_KIND_REGULAR_STRUCT = 0 STRUCT_KIND_TUPLE_STRUCT = 1 STRUCT_KIND_TUPLE = 2 STRUCT_KIND_TUPLE_VARIANT = 3 STRUCT_KIND_STRUCT_VARIANT = 4 STRUCT_KIND_CSTYLE_VARIANT = 5 STRUCT_KIND_SLICE = 6 STRUCT_KIND_STR_SLICE = 7 STRUCT_KIND_STD_VEC = 8 STRUCT_KIND_STD_STRING = 9 def classify_struct(type): # print("\nclassify_struct: tag=%s\n" % type.tag) if type.tag == "&str": return STRUCT_KIND_STR_SLICE if type.tag.startswith("&[") and type.tag.endswith("]"): return STRUCT_KIND_SLICE fields = list(type.fields()) field_count = len(fields) if field_count == 0: return STRUCT_KIND_REGULAR_STRUCT if (field_count == 3 and fields[0].name == "ptr" and fields[1].name == "len" and fields[2].name == "cap" and type.tag.startswith("Vec<")): return STRUCT_KIND_STD_VEC if (field_count == 1 and fields[0].name == "vec" and type.tag == "String"): return STRUCT_KIND_STD_STRING if fields[0].name == "RUST$ENUM$DISR": if field_count == 1: return STRUCT_KIND_CSTYLE_VARIANT elif all_fields_conform_to_tuple_field_naming(fields, 1): return STRUCT_KIND_TUPLE_VARIANT else: return STRUCT_KIND_STRUCT_VARIANT if all_fields_conform_to_tuple_field_naming(fields, 0): if type.tag.startswith("("): return STRUCT_KIND_TUPLE else: return STRUCT_KIND_TUPLE_STRUCT return STRUCT_KIND_REGULAR_STRUCT def extract_discriminant_value(enum_val): assert enum_val.type.code == gdb.TYPE_CODE_UNION for variant_descriptor in enum_val.type.fields(): variant_val = enum_val[variant_descriptor] for field in variant_val.type.fields(): return (field.name, int(variant_val[field])) def first_field(val): for field in val.type.fields(): return field def get_field_at_index(val, index): i = 0 for field in val.type.fields(): if i == index: return field i += 1 return None def all_fields_conform_to_tuple_field_naming(fields, start_index): for i in range(start_index, len(fields)): if (fields[i].name is None) or (re.match(r"__\d+$", fields[i].name) is None): return False return True def extract_length_and_data_ptr_from_std_vec(vec_val): length = int(vec_val["len"]) vec_ptr_val = vec_val["ptr"] unique_ptr_val = vec_ptr_val[first_field(vec_ptr_val)] data_ptr = unique_ptr_val[first_field(unique_ptr_val)] assert data_ptr.type.code == gdb.TYPE_CODE_PTR return (length, data_ptr)